Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improve performance of context components re-rendering #3066

Merged
merged 38 commits into from
Jan 28, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
38 commits
Select commit Hold shift + click to select a range
8a5ab9a
Improve performance of context components re-rendering
T4rk1n Nov 7, 2024
378fec3
Fix list as layout
T4rk1n Nov 7, 2024
65ffabd
memo wrapper, merge loading state selector, add back _dashprivate_layout
T4rk1n Dec 10, 2024
011ded9
Add get_props, extras to wrapper, set_props takes path
T4rk1n Dec 12, 2024
603cbce
get_props -> get_layout
T4rk1n Dec 13, 2024
e1cbc68
take loading_state out of selector
T4rk1n Dec 16, 2024
5424fe0
Rework loading component
T4rk1n Dec 19, 2024
1a56d4f
Add useLoading
T4rk1n Dec 19, 2024
1f56612
useLoading in html components
T4rk1n Dec 19, 2024
9741a69
Add LoadingDiv
T4rk1n Dec 19, 2024
957ec39
replace loading for data-dash-is-loading
T4rk1n Dec 20, 2024
19167a8
Fix forwardRef for LoadingElement
T4rk1n Jan 7, 2025
84700ea
Fix children not undefined rcap003
T4rk1n Jan 7, 2025
9cc9627
Fix multioutput loading
T4rk1n Jan 8, 2025
d5dddff
fix rdarp001
T4rk1n Jan 8, 2025
61b8ea6
Fix clipboard context
T4rk1n Jan 9, 2025
e616001
Fix confirm dialog provider
T4rk1n Jan 9, 2025
9954a2a
Fix loading target selector
T4rk1n Jan 9, 2025
e023e4f
Takes extras props directly from DashWrapper rest props
T4rk1n Jan 10, 2025
232c3c0
Refactor wrapper selectors to use path sums as equality comparator
T4rk1n Jan 14, 2025
5a68ecd
fix cbmo002
T4rk1n Jan 15, 2025
2bcb14d
Refactor TextArea to functional + new loading
T4rk1n Jan 16, 2025
f0b4908
Refactor upload to LoadingElement
T4rk1n Jan 16, 2025
fbdd794
Remove leftover loading_state from graph.
T4rk1n Jan 16, 2025
c520a82
Fix table loading.
T4rk1n Jan 17, 2025
5dce073
Fix missing export for Textarea.
T4rk1n Jan 20, 2025
71e70c3
Fix table.
T4rk1n Jan 20, 2025
b00668e
Merge branch 'dev' into fix/wrapper
T4rk1n Jan 21, 2025
a1f6b5e
Merge branch 'dash-3.0' into fix/wrapper
T4rk1n Jan 22, 2025
6af9973
Take persistence from dashPersistence on element
T4rk1n Jan 22, 2025
8b3c153
Remove defaultProps from internal DataTable.
T4rk1n Jan 22, 2025
9874f8a
Mock dash_component_api in table tests.
T4rk1n Jan 22, 2025
51dfddf
build
T4rk1n Jan 23, 2025
eb65854
Merge branch 'dash-3.0' into fix/wrapper
T4rk1n Jan 23, 2025
664e6dd
Fix tab new loading
T4rk1n Jan 24, 2025
5194e18
Move getLayout to dash_component_api
T4rk1n Jan 24, 2025
efbb5b1
fix prop typing generation
T4rk1n Jan 27, 2025
3249e22
fix table generation
T4rk1n Jan 27, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ export type TypescriptComponentProps = {
array_obj?: {a: string}[];
array_any?: any[];
enum_string?: 'one' | 'two';
enum_number?: 2 | 3 | 4 | 5 | 6;
union?: number | string;
union_shape?: {a: string} | string;
array_union_shape?: ({a: string} | string)[];
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,8 @@ const AddPropsComponent = (props) => {

return (
<div id={id}>
{React.cloneElement(children, {
{React.cloneElement(children, {
receive: `Element #${id} pass`,
id: id,
})}
</div>
);
Expand Down
32 changes: 4 additions & 28 deletions components/dash-core-components/src/components/Checklist.react.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
import PropTypes from 'prop-types';
import {append, includes, without} from 'ramda';
import React, {Component} from 'react';

import {sanitizeOptions} from '../utils/optionTypes';
import LoadingElement from '../utils/LoadingElement';

/**
* Checklist is a component that encapsulates several checkboxes.
Expand All @@ -21,19 +23,11 @@ export default class Checklist extends Component {
options,
setProps,
style,
loading_state,
value,
inline,
} = this.props;
return (
<div
data-dash-is-loading={
(loading_state && loading_state.is_loading) || undefined
}
id={id}
style={style}
className={className}
>
<LoadingElement id={id} style={style} className={className}>
{sanitizeOptions(options).map(option => {
return (
<label
Expand Down Expand Up @@ -67,7 +61,7 @@ export default class Checklist extends Component {
</label>
);
})}
</div>
</LoadingElement>
);
}
}
Expand Down Expand Up @@ -192,24 +186,6 @@ Checklist.propTypes = {
*/
setProps: PropTypes.func,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),

/**
* Used to allow user interactions in this component to be persisted when
* the component - or the page - is refreshed. If `persisted` is truthy and
Expand Down
33 changes: 8 additions & 25 deletions components/dash-core-components/src/components/Clipboard.react.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,8 @@ import PropTypes from 'prop-types';
import {FontAwesomeIcon} from '@fortawesome/react-fontawesome';
import {faCopy, faCheckCircle} from '@fortawesome/free-regular-svg-icons';

import LoadingElement from '../utils/LoadingElement';

const clipboardAPI = navigator.clipboard;

function wait(ms) {
Expand All @@ -14,6 +16,8 @@ function wait(ms) {
*/

export default class Clipboard extends React.Component {
static contextType = window.dash_component_api.DashContext;

constructor(props) {
super(props);
this.copyToClipboard = this.copyToClipboard.bind(this);
Expand Down Expand Up @@ -96,7 +100,7 @@ export default class Clipboard extends React.Component {
}

async loading() {
while (this.props.loading_state?.is_loading) {
while (this.context.isLoading()) {
await wait(100);
}
}
Expand Down Expand Up @@ -124,24 +128,21 @@ export default class Clipboard extends React.Component {
}

render() {
const {id, title, className, style, loading_state} = this.props;
const {id, title, className, style} = this.props;
const copyIcon = <FontAwesomeIcon icon={faCopy} />;
const copiedIcon = <FontAwesomeIcon icon={faCheckCircle} />;
const btnIcon = this.state.copied ? copiedIcon : copyIcon;

return clipboardAPI ? (
<div
<LoadingElement
id={id}
title={title}
style={style}
className={className}
onClick={this.onClickHandler}
data-dash-is-loading={
(loading_state && loading_state.is_loading) || undefined
}
>
<i> {btnIcon}</i>
</div>
</LoadingElement>
) : null;
}
}
Expand Down Expand Up @@ -196,24 +197,6 @@ Clipboard.propTypes = {
*/
className: PropTypes.string,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),

/**
* Dash-assigned callback that gets fired when the value changes.
*/
Expand Down
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import {clone} from 'ramda';
import React from 'react';
import PropTypes from 'prop-types';

import ConfirmDialog from './ConfirmDialog.react';
import LoadingElement from '../utils/LoadingElement';

/**
* A wrapper component that will display a confirmation dialog
Expand All @@ -18,30 +18,21 @@ import ConfirmDialog from './ConfirmDialog.react';
*/
export default class ConfirmDialogProvider extends React.Component {
render() {
const {displayed, id, setProps, children, loading_state} = this.props;
const {displayed, id, setProps, children} = this.props;

// Will lose the previous onClick of the child
const wrapClick = child => {
const props = clone(child.props);
props._dashprivate_layout.props.onClick = () => {
setProps({displayed: true});
};

return React.cloneElement(child, props);
};
const wrapClick = child =>
React.cloneElement(child, {
onClick: () => setProps({displayed: true}),
});

return (
<div
id={id}
data-dash-is-loading={
(loading_state && loading_state.is_loading) || undefined
}
>
<LoadingElement id={id}>
{Array.isArray(children)
? children.map(wrapClick)
: wrapClick(children)}
<ConfirmDialog {...this.props} displayed={displayed} />
</div>
</LoadingElement>
);
}
}
Expand Down Expand Up @@ -94,22 +85,4 @@ ConfirmDialogProvider.propTypes = {
* The children to hijack clicks from and display the popup.
*/
children: PropTypes.any,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),
};
Original file line number Diff line number Diff line change
Expand Up @@ -222,24 +222,6 @@ DatePickerRange.propTypes = {
*/
setProps: PropTypes.func,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),

/**
* Used to allow user interactions in this component to be persisted when
* the component - or the page - is refreshed. If `persisted` is truthy and
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -181,24 +181,6 @@ DatePickerSingle.propTypes = {
*/
setProps: PropTypes.func,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),

/**
* Used to allow user interactions in this component to be persisted when
* the component - or the page - is refreshed. If `persisted` is truthy and
Expand Down
18 changes: 0 additions & 18 deletions components/dash-core-components/src/components/Dropdown.react.js
Original file line number Diff line number Diff line change
Expand Up @@ -177,24 +177,6 @@ Dropdown.propTypes = {
*/
setProps: PropTypes.func,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),

/**
* Used to allow user interactions in this component to be persisted when
* the component - or the page - is refreshed. If `persisted` is truthy and
Expand Down
18 changes: 0 additions & 18 deletions components/dash-core-components/src/components/Graph.react.js
Original file line number Diff line number Diff line change
Expand Up @@ -556,24 +556,6 @@ PlotlyGraph.propTypes = {
* Function that updates the state tree.
*/
setProps: PropTypes.func,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),
};

ControlledPlotlyGraph.propTypes = PlotlyGraph.propTypes;
Expand Down
26 changes: 3 additions & 23 deletions components/dash-core-components/src/components/Input.react.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import React, {PureComponent} from 'react';
import PropTypes from 'prop-types';
import isNumeric from 'fast-isnumeric';
import './css/input.css';
import LoadingElement from '../utils/LoadingElement';

// eslint-disable-next-line no-implicit-coercion
const convert = val => (isNumeric(val) ? +val : NaN);
Expand Down Expand Up @@ -94,14 +95,11 @@ export default class Input extends PureComponent {
render() {
const valprops =
this.props.type === 'number' ? {} : {value: this.state.value};
const {loading_state} = this.props;
let {className} = this.props;
className = 'dash-input' + (className ? ` ${className}` : '');
return (
<input
data-dash-is-loading={
(loading_state && loading_state.is_loading) || undefined
}
<LoadingElement
elementType={'input'}
className={className}
ref={this.input}
onBlur={this.onBlur}
Expand Down Expand Up @@ -471,24 +469,6 @@ Input.propTypes = {
*/
setProps: PropTypes.func,

/**
* Object that holds the loading state object coming from dash-renderer
*/
loading_state: PropTypes.shape({
/**
* Determines if the component is loading or not
*/
is_loading: PropTypes.bool,
/**
* Holds which property is loading
*/
prop_name: PropTypes.string,
/**
* Holds the name of the component that is loading
*/
component_name: PropTypes.string,
}),

/**
* Used to allow user interactions in this component to be persisted when
* the component - or the page - is refreshed. If `persisted` is truthy and
Expand Down
Loading