Graph.react.js

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;