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

Add Loading component #322

Merged
merged 1 commit into from
May 4, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
4 changes: 4 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@

All notable changes to this project will be documented in this file.

## [1.0.15] - UNRELEASED

- Add new `Loading` component

## [1.0.14] - 05-03-24

- Add new `validate` module, which adds an `assert_no_random_ids` that assets that Dash didn't generate any random component ids
Expand Down
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ The `dash-extensions` package is a collection of utility functions, syntax exten
* A number of custom components, e.g. the `Websocket` component, which enables real-time communication and push notifications
* The `javascript` module, which contains functionality to ease the interplay between Dash and JavaScript
* The `pages` module, which extends the functionality of [Dash Pages](https://dash.plotly.com/urls)
* The `snippets/validation` modules, which contains a collection of utility functions (documentation limited to source code comments)
* The `snippets/validation` modules, which contain a collection of utility functions (documentation limited to source code comments)

The `enrich` module enables a number of _transforms_ that add functionality and/or syntactic sugar to Dash. Examples include

Expand Down
2 changes: 2 additions & 0 deletions dash_extensions/_imports_.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
from .EventListener import EventListener
from .EventSource import EventSource
from .Keyboard import Keyboard
from .Loading import Loading
from .Lottie import Lottie
from .Mermaid import Mermaid
from .Purify import Purify
Expand All @@ -16,6 +17,7 @@
"EventListener",
"EventSource",
"Keyboard",
"Loading",
"Lottie",
"Mermaid",
"Purify",
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "dash-extensions",
"version": "1.0.14",
"version": "1.0.15rc2",
"description": "Extensions for Plotly Dash.",
"main": "build/index.js",
"scripts": {
Expand Down
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[tool.poetry]
name = "dash-extensions"
version = "1.0.14"
version = "1.0.15rc2"
description = "Extensions for Plotly Dash."
authors = ["emher <[email protected]>"]
license = "MIT"
Expand Down
93 changes: 93 additions & 0 deletions src/lib/components/Loading.react.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,93 @@
import { useEffect, useRef } from "react";
import PropTypes from 'prop-types';

/**
* The Loading component makes it possible to stop event propagation during loading.
*/

const Loading = (props) => {
const {
children,
loading_state,
preventDefault
} = props;
const container = useRef();

useEffect(() => {
const mounted = container && container.current;
const handlePreventDefault = (e) => {
e.preventDefault();
}
if(mounted){
const ref = container.current;
const loading = loading_state && loading_state.is_loading;
if(!loading){
preventDefault.forEach(event => {ref.removeEventListener(event, handlePreventDefault, true);})
}
else{
preventDefault.forEach(event => {ref.addEventListener(event, handlePreventDefault, true);})
}
}
return () => {
const mounted = container && container.current;
if(mounted){
const ref = container.current;
preventDefault.forEach(event => {ref.removeEventListener(event, handlePreventDefault, true);})
}
};
}, [loading_state]);

return (
<div ref={container}>
{children}
</div>
);
};

Loading._dashprivate_isLoadingComponent = true;

Loading.defaultProps = {
preventDefault: ["keydown"],
};

Loading.propTypes = {
/**
* The ID of this component, used to identify dash components
* in callbacks. The ID needs to be unique across all of the
* components in an app.
*/
id: PropTypes.string,

/**
* Events for which to call preventDefault() during loading.
*/
preventDefault: PropTypes.arrayOf(PropTypes.string),

/**
* Array that holds components to render
*/
children: PropTypes.oneOfType([
PropTypes.arrayOf(PropTypes.node),
PropTypes.node,
]),

/**
* 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,
}),
};

export default Loading;
4 changes: 3 additions & 1 deletion src/lib/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import EventListener from "./components/EventListener.react";
import EventSource from "./components/EventSource.react";
import CycleBreaker from "./components/CycleBreaker.react";
import Keyboard from "./components/Keyboard.react";
import Loading from "./components/Loading.react";

export {
Lottie,
Expand All @@ -20,5 +21,6 @@ export {
EventListener,
EventSource,
CycleBreaker,
Keyboard
Keyboard,
Loading
};
Loading