import React, { Component } from 'react';
import PropTypes from 'prop-types';
import './css/saveClick.css';
import CoreGraph from './_core/CoreGraph.react';
import CoreDataTable from './_core/CoreDataTable.react';
import Configurator from './Configurator.react';
import Modal from 'react-bootstrap/Modal';
import Button from 'react-bootstrap/Button';
const EMPTY_DATA = [];
const TABLE_PNG = "";
//generates random id;
let guid = () => {
let s4 = () => {
return Math.floor((1 + Math.random()) * 0x10000)
.toString(16)
.substring(1);
}
//return id of format 'aaaaaaaa'-'aaaa'-'aaaa'-'aaaa'-'aaaaaaaaaaaa'
return s4() + s4() + '-' + s4() + '-' + s4() + '-' + s4() + '-' + s4() + s4() + s4();
}
/**
* <div style="width:450px; margin-left: 20px; float: right; margin-top: -150px;">
* <img src="https://raw.githubusercontent.com/VK/dash-express-components/main/.media/graph.png"/>
* <img src="https://raw.githubusercontent.com/VK/dash-express-components/main/.media/graph-table.png"/>
* <img src="https://raw.githubusercontent.com/VK/dash-express-components/main/.media/graph-modal.png"/>
* </div>
*
*
* The `Graph` component is a combination of the original dash `Graph` and the dash `data_table`.
*
* It can not only be used to render a plotly.js-powered data visualization,
* but also shows a searchable table, if only data is submitted.
*
* In addition, there is the possibility to add plot parameters as `defParams` and
* the dataframe `meta` data.
* This automatically adds a configurator modal, which can be opened via a button
* at the bottom right.
*
*
* @hideconstructor
*
* @example
* import dash_express_components as dxc
* import plotly.express as px
*
* meta = dxc.get_meta(px.data.gapminder())
*
* dxc.Graph(
* id="fig",
* meta=meta,
* defParams={}
* )
* @public
*/
class Graph extends Component {
constructor(props) {
super(props);
this.state = {
prependData: [],
extendData: [],
//page_current: 0,
sort_by: [],
is_loading: false,
filter_query: "",
config_modal_open: false,
defParams: props.defParams,
hiddenColumns: props.hiddenColumns,
meta: this.filterMeta(props.meta),
internalFigure: { data: [] }
};
this.clearState = this.clearState.bind(this);
this.config_in_modal_ref = React.createRef();
this.config_modal_id = guid();
}
filterMeta(meta) {
// remove the hiddenColums from the meta
const hiddenColumns = (this.state && this.state.hiddenColumns) ? this.state.hiddenColumns : this.props.hiddenColumns;
if (typeof meta === "object") {
for (let key in meta) {
if (hiddenColumns.includes(key)) {
delete meta[key];
}
}
}
return meta;
}
componentDidMount() {
if ("plotApi" in this.props && "defParams" in this.props) {
this.setState({ is_loading: true });
let that = this;
setTimeout(() => {
that.update_figure_from_defParams(that.props.defParams, true);
}, 200);
}
}
isGraph() {
try {
if ("plotApi" in this.props && this.props.plotApi !== "") {
return ("internalFigure" in this.state && "data" in this.state.internalFigure)
} else {
return ("data" in this.props.figure)
}
} catch (e) {
return ("data" in this.props.figure)
}
}
sendSavedData(image) {
let messageData = { thumbnail: image, defs: this.props.defParams, app: window.appName, href: location.href };
try {
window.parent.postMessage(messageData, "*");
} catch (e) {
console.log(e);
}
}
saveClick() {
if (this.isGraph()) {
//compute a thumpnail
let imagePromise = Plotly.toImage(document.getElementById(this.props.id).children[1], { format: 'png', height: 400, width: 800 });
//send the image data once created
imagePromise.then((img) => this.sendSavedData(img));
} else {
this.sendSavedData(TABLE_PNG);
}
}
handleOpen() {
this.setState({ config_modal_open: true });
let that = this;
setTimeout(() => {
that.config_in_modal_ref.current.update_config(this.props.defParams);
}, 200);
}
handleClose() {
this.setState({ config_modal_open: false });
}
inIframe() {
try {
return window.self !== window.top;
} catch (e) {
return true;
}
}
usePlotApi() {
try {
return "plotApi" in this.props && this.props.plotApi !== "";
} catch (e) {
return false;
}
}
update_figure_from_defParams(input_params, initial = true) {
let defParams = JSON.parse(JSON.stringify(input_params));
if (!initial) {
this.setState({ defParams: defParams });
}
this.setState({ is_loading: true });
const handleResponse = (xhr) => {
if (xhr.status === 200) {
if (xhr.responseText !== "") {
try {
var data = JSON.parse(xhr.responseText);
// Handling the response data accordingly
if ("plots" in data) {
data = data.plots;
if (Array.isArray(data)) {
data = data[0];
}
}
if ("meta" in data) {
const { meta, ...figdata } = data;
this.setState({ internalFigure: figdata, meta: this.filterMeta(data.meta) });
} else {
this.setState({ internalFigure: data });
}
} catch (e) {
console.log(e);
}
this.setState({ is_loading: false });
}
} else if (xhr.status === 202) {
// Retry the request if status is 202 (Accepted)
setTimeout(() => {
sendRequest(defParams);
}, 1000); // Adjust the retry interval as needed
} else {
this.setState({ is_loading: false });
}
};
const handleTimeout = () => {
// Handle timeout situations if needed
this.setState({ is_loading: false });
};
const handleLongCallback = (xhr) => {
if (this.props.longCallback) {
xhr.setRequestHeader('X-Longcallback', 'true');
}
};
const sortedObject = (obj) => {
if (typeof obj !== "object" || obj === null) return obj;
if (Array.isArray(obj)) return obj.map(sortedObject);
const ordered = {};
Object.keys(obj).sort().forEach(function (key) {
ordered[key] = sortedObject(obj[key]);
});
return ordered;
}
const sortedStringify = (obj) => {
return JSON.stringify(sortedObject(obj));
}
const sendRequest = (send_data) => {
var xhr = new XMLHttpRequest();
xhr.open("POST", this.props.plotApi, true);
xhr.setRequestHeader('Content-Type', 'application/json');
handleLongCallback(xhr);
xhr.onreadystatechange = function () {
if (xhr.readyState === XMLHttpRequest.DONE) {
handleResponse(xhr);
}
};
xhr.ontimeout = handleTimeout;
xhr.send(sortedStringify(send_data));
};
let send_data = JSON.parse(JSON.stringify(defParams));
if (["auto", "png"].includes(send_data["plot"]["params"]["render"])) {
if (send_data["plot"]["params"]["render_size"] === undefined) {
try {
send_data["plot"]["params"]["render_size"] = [this.graphDiv.clientWidth, this.graphDiv.clientHeight];
} catch (e) { }
}
}
sendRequest(send_data);
}
/**
* if the plot config changes and the extra plotApi should be used
* Then we have to update the content
* @private
*/
UNSAFE_componentWillReceiveProps(newProps) {
if (this.usePlotApi()) {
if (newProps.defParams !== this.state.defParams) {
if (JSON.stringify(newProps.defParams) !== JSON.stringify(this.state.defParams)) {
this.update_figure_from_defParams(newProps.defParams, false);
}
}
}
if (newProps.figure !== this.state.internalFigure) {
// remove meta from figure
if (newProps.figure && newProps.figure.meta) {
let { meta, ...figure } = newProps.figure;
this.setState({ figure: figure });
} else {
this.setState({ figure: newProps.figure });
}
}
if (newProps.meta !== this.state.meta && newProps.meta) {
this.setState({ meta: this.filterMeta(newProps.meta) });
}
}
clearState(dataKey) {
this.setState(props => {
var data = props[dataKey];
const res =
data && data.length
? {
[dataKey]: EMPTY_DATA,
}
: undefined;
return res;
});
}
render() {
const spinnerClass = this.state.is_loading ? "dxc-spinner-container" : "dxc-spinner-container dxc-spinner-hidden";
/*Start VK addon*/
let save_button = "";
let edit_button = "";
if (this.props.defParams && this.inIframe() && this.props.saveClick) {
save_button = <a className="saveClickButton p-1" onClick={this.saveClick.bind(this)} key={this.props.id + "-save-button"}>
<svg viewBox="0 -35 576 512" className="icon" height="1.5em" width="1.5em">
<path fill="currentColor" d="M416 448h-84c-6.6 0-12-5.4-12-12v-40c0-6.6 5.4-12 12-12h84c17.7 0 32-14.3 32-32V160c0-17.7-14.3-32-32-32h-84c-6.6 0-12-5.4-12-12V76c0-6.6 5.4-12 12-12h84c53 0 96 43 96 96v192c0 53-43 96-96 96zm-47-201L201 79c-15-15-41-4.5-41 17v96H24c-13.3 0-24 10.7-24 24v96c0 13.3 10.7 24 24 24h136v96c0 21.5 26 32 41 17l168-168c9.3-9.4 9.3-24.6 0-34z" transform="matrix(-1 0 0 1 576 0 )"></path>
</svg>
</a>
}
if (this.props.defParams && this.state.meta && this.props.editButton) {
edit_button = <a className="saveClickButton" onClick={e => this.handleOpen()} key={this.props.id + "-edit-button"}>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 576 512" height="1.3em" width="1.3em">
<path fill="currentColor" d="M402.6 83.2l90.2 90.2c3.8 3.8 3.8 10 0 13.8L274.4 405.6l-92.8 10.3c-12.4 1.4-22.9-9.1-21.5-21.5l10.3-92.8L388.8 83.2c3.8-3.8 10-3.8 13.8 0zm162-22.9l-48.8-48.8c-15.2-15.2-39.9-15.2-55.2 0l-35.4 35.4c-3.8 3.8-3.8 10 0 13.8l90.2 90.2c3.8 3.8 10 3.8 13.8 0l35.4-35.4c15.2-15.3 15.2-40 0-55.2zM384 346.2V448H64V128h229.8c3.2 0 6.2-1.3 8.5-3.5l40-40c7.6-7.6 2.2-20.5-8.5-20.5H48C21.5 64 0 85.5 0 112v352c0 26.5 21.5 48 48 48h352c26.5 0 48-21.5 48-48V306.2c0-10.7-12.9-16-20.5-8.5l-40 40c-2.2 2.3-3.5 5.3-3.5 8.5z" />
</svg>
</a>
}
let config_meta = (this.state.meta) ? this.state.meta : this.props.meta
config_meta = (config_meta) ? config_meta : {}
let configurator_modal = (
<Modal
centered
backdrop="static"
animation={false}
show={this.state.config_modal_open}
onHide={() => this.handleClose()}>
<Modal.Header closeButton>
<Modal.Title> Edit Plot Config</Modal.Title>
</Modal.Header>
<Modal.Body>
<Configurator ref={this.config_in_modal_ref}
id={this.config_modal_id}
config={this.props.defParams}
meta={config_meta}
showUpdate={false}
showFilter={this.props.showFilter}
showTransform={this.props.showTransform}
/>
</Modal.Body>
<Modal.Footer>
<Button variant="secondary" onClick={() => this.handleClose()}>
Close
</Button>
<Button variant="primary" onClick={(e) => {
this.props.setProps({
defParams:
this.config_in_modal_ref.current.state.config
});
this.handleClose();
}}>
Update Plot
</Button>
</Modal.Footer>
</Modal>
)
/*End VK addon*/
if (this.isGraph()) {
if (this.usePlotApi()) {
let inner_props = {
...this.props,
figure: this.state.internalFigure
}
return (
<div className='pxc-graph-container' ref={(divElement) => { this.graphDiv = divElement }}>
<div className={spinnerClass}><div className="dxc-spinner-border" role="status"><span className="sr-only"></span></div>
</div>
<CoreGraph
{...inner_props}
clearState={this.clearState}
/>
<div className="saveClickContainer" >{save_button}{edit_button}</div>
{configurator_modal}
</div>
);
} else {
return (
<div className='pxc-graph-container' ref={(divElement) => { this.graphDiv = divElement }}>
<div className={spinnerClass}><div className="dxc-spinner-border" role="status"><span className="sr-only"></span></div>
</div>
<CoreGraph
{...this.props}
clearState={this.clearState}
/>
<div className="saveClickContainer" >{save_button}{edit_button}</div>
{configurator_modal}
</div>
);
}
} else {
let columns = [];
if (this.usePlotApi()) {
columns = Object.keys(this.state.internalFigure).map(k => { return { name: k, id: k, hideable:false } });
} else {
columns = Object.keys(this.props.figure).map(k => { return { name: k, id: k, hideable:false } });
}
if ("defParams" in this.props) {
try {
columns = this.props.defParams.plot.params.dimensions.map(k => { return { name: k, id: k, hideable:false } })
} catch { }
}
const hiddenColumns = (this.state && this.state.hiddenColumns) ? this.state.hiddenColumns : this.props.hiddenColumns;
let props = {
id: this.props.id,
className: this.props.className,
data: this.props.figure,
columns: columns,
//page_current: this.state.page_current,
sort_by: this.state.sort_by,
filter_query: this.state.filter_query,
virtualization: true,
page_action: 'none',
fixed_rows: { headers: true, data: 0 },
style_table: { height: '300px', overflowY: 'auto' },
style_cell: { 'minWidth': '50px', fontSize: "14px" },
hidden_columns: hiddenColumns,
}
if (this.usePlotApi()) {
props.data = this.state.internalFigure;
}
return (
<div className='pxc-graph-container' style={{ padding: "5px" }}>
<div className={spinnerClass}><div className="dxc-spinner-border" role="status"><span className="sr-only"></span></div>
</div>
{(!this.state.is_loading) && <CoreDataTable {...props} setProps={
el => {
if (("selectedData" in el) || ("prependData" in el) || ("extendData" in el)) {
this.props.setProps(el);
} else {
if (
("page_current" in el && el.page_current !== this.state.page_current) ||
("sort_by" in el && el.sort_by !== this.state.sort_by) ||
("filter_query" in el && el.filter_query !== this.state.filter_query)
) {
this.setState(el);
}
}
}
}
/>}
<div className="saveClickContainer" style={{ position: "relative", left: "-20px", bottom: "0px" }}>{save_button}{edit_button}</div>
{configurator_modal}
</div>
);
}
}
}
/**
* @typedef
* @public
* @enum {}
*/
Graph.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.
* @type {string}
*/
id: PropTypes.string.isRequired,
/**
* Configuration to describe the plot features
*/
defParams: PropTypes.object,
/**
* The metadata the plotter selection is based on.
*/
meta: PropTypes.any,
/**
* Url to the plot Api
*/
plotApi: PropTypes.string,
/**
* Plotly `figure` object. See schema:
* https://plotly.com/javascript/reference
*
* `config` is set separately by the `config` property
*/
figure: PropTypes.any,
/**
* Generic style overrides on the plot div
*/
style: PropTypes.object,
/**
* The data selected in the plot or in the table
*/
selectedData: PropTypes.any,
/**
* className of the parent div
*/
className: PropTypes.string,
/**
* enable/disable saveClick button
*/
saveClick: PropTypes.bool,
/**
* enable/disable long callbacks
*/
longCallback: PropTypes.bool,
/**
* enable/disable edit button
*/
editButton: PropTypes.bool,
/**
* The current configuration of the plot.
* @type {Object}
*/
currentConfig: PropTypes.any,
/**
* Prop to define the visibility of the Filter panel
* @type {boolean}
*/
showFilter: PropTypes.bool,
/**
* Prop to define the visibility of the Transform panel
* @type {boolean}
*/
showTransform: PropTypes.bool,
/**
* Function that updates the state tree.
*/
setProps: PropTypes.func,
/**
* hidden column names (array of strings)
*/
hiddenColumns: PropTypes.array,
};
Graph.defaultProps = {
meta: null,
figure: {
data: [],
layout: {},
frames: [],
},
style: null,
saveClick: false,
editButton: true,
longCallback: false,
showFilter: true,
showTransform: true,
className: "",
plotApi: "",
hiddenColumns: ["_id", "index"],
};
/**
* @private
*/
export default Graph;