Попытка составить таблицы мастер-деталей в ReactJS - PullRequest
1 голос
/ 06 марта 2020

Я пытаюсь повторить этот пример: https://www.highcharts.com/demo/dynamic-master-detail, но с React (+ Hooks).
Моя проблема в detailChart, потому что мастер уже работает, я знаю, что detailChart будет генерироваться обратным вызовом (я все еще не очень хорош в концепции обратных вызовов).

В документации Highcharts говорится, что я могу сделать функцию обратного вызова в chart.events.load, что я и решил сделать в masterChart.
Однако я получаю сообщение об ошибке в функции createDetails (функция обратного вызова), говорящее

undefined is not an object (evaluating 'masterChart.series[0]')

, что странно, потому что я делаю то же самое при создании masterChart, и это сработало так что проблема не может быть в моем объекте RData, я думаю.

Это мой код на данный момент:

import React, {useEffect,useState} from 'react';
import Highcharts from 'highcharts';
import HighchartsReact from 'highcharts-react-official';
import RData from '../data/RData'

const RChart = () =>  {
    const[masterChart,setMasterChart] = useState([]);
    const[detailChart,setDetailChart] = useState([]);

    //updating masterChart
    useEffect(()=>{
        setMasterChart({
            chart: {
                reflow: false,
                borderWidth: 0,
                backgroundColor: "#ffffff",
                marginLeft: 50,
                marginRight: 20,
                zoomType: 'x',
                events: {

                    // listen to the selection event on the master chart to update the
                    // extremes of the detail chart
                    selection: function (event) {
                        var extremesObject = event.xAxis[0],
                            min = extremesObject.min,
                            max = extremesObject.max,
                            detailData = [],
                            xAxis = this.xAxis[0];

                        // reverse engineer the last part of the data
                        RData.each(this.series[0].RData, function () {
                            if (this.x > min && this.x < max) {
                                detailData.push([this.x, this.y]);
                            }
                        });

                        // move the plot bands to reflect the new detail span
                        xAxis.removePlotBand('mask-before');
                        xAxis.addPlotBand({
                            id: 'mask-before',
                            from: RData[0][0],
                            to: min,
                            color: 'rgba(0, 0, 0, 0.2)'
                        });

                        xAxis.removePlotBand('mask-after');
                        xAxis.addPlotBand({
                            id: 'mask-after',
                            from: max,
                            to: RData[RData.length - 1][0],
                            color: 'rgba(255, 255, 0, 0.2)'
                        });


                        detailChart.series[0].setData(detailData);

                        return false;
                    },
                    load:createDetail(masterChart) //callback!!
                }
            },
            title: {
                text: null
            },
            accessibility: {
                enabled: false
            },
            xAxis: {
                type: 'datetime',
                showLastTickLabel: true,
                maxZoom: 14 * 24 * 3600000, // fourteen days
                plotBands: [{
                    id: 'mask-before',
                    from: RData[0][0],
                    to: RData[RData.length - 1][0],
                    color: 'rgba(0, 0, 0, 0.2)'
                }],
                title: {
                    text: null
                }
            },
            yAxis: {
                gridLineWidth: 0,
                labels: {
                    enabled: false
                },
                title: {
                    text: null
                },
                min: 0.6,
                showFirstLabel: false
            },
            tooltip: {
                formatter: function () {
                    return false;
                }
            },
            legend: {
                enabled: false
            },
            credits: {
                enabled: false
            },
            plotOptions: {
                series: {
                    fillColor: {
                        linearGradient: [0, 0, 0, 255],
                        stops: [
                            [0, Highcharts.getOptions().colors[0]],
                            [1, 'rgba(0,255,255,0)']
                        ]
                    },
                    lineWidth: 1,
                    marker: {
                        enabled: false
                    },
                    shadow: false,
                    states: {
                        hover: {
                            lineWidth: 1
                        }
                    },
                    enableMouseTracking: false
                }
            },

            series: [{
                type: 'area',
                name: 'USD to EUR',
                pointInterval: 24 * 3600 * 1000,
                pointStart: RData[0][0],
                data: RData
            }],

            exporting: {
                enabled: false
            }
        })
    },[]);

    //callback function to create detail chart
    const createDetail = (masterChart) => {
        // prepare the detail chart
        var detailData = [],
        detailStart = RData[0][0];

        //Error here
        RData.each(masterChart.series[0].RData, function () {
            if (this.x >= detailStart) {
                detailData.push(this.y);
            }
        });

        setDetailChart({
            chart: {
                marginBottom: 120,
                reflow: false,
                marginLeft: 50,
                marginRight: 20,
                style: {
                    position: 'absolute'
                }
            },
            credits: {
                enabled: false
            },
            title: {
                text: 'Historical USD to EUR Exchange Rate',
                align: 'left'
            },
            subtitle: {
                text: 'Select an area by dragging across the lower chart',
                align: 'left'
            },
            xAxis: {
                type: 'datetime'
            },
            yAxis: {
                title: {
                    text: null
                },
                maxZoom: 0.1
            },
            tooltip: {
                formatter: function () {
                    var point = this.points[0];
                    return '<b>' + point.series.name + '</b><br/>' + Highcharts.dateFormat('%A %B %e %Y', this.x) + ':<br/>' +
                        '1 USD = ' + Highcharts.numberFormat(point.y, 2) + ' EUR';
                },
                shared: true
            },
            legend: {
                enabled: false
            },
            plotOptions: {
                series: {
                    marker: {
                        enabled: false,
                        states: {
                            hover: {
                                enabled: true,
                                radius: 3
                            }
                        }
                    }
                }
            },
            series: [{
                name: 'USD to EUR',
                pointStart: detailStart,
                pointInterval: 24 * 3600 * 1000,
                data: detailData
            }],

            exporting: {
                enabled: false
            }
        });
    }


    return(
        <React.Fragment>
            <HighchartsReact highcharts={Highcharts} options={masterChart} />
            <HighchartsReact highcharts={Highcharts} options={detailChart} />
        </React.Fragment>
    );
} 

export default RChart;

Я открыт для любой критики и лучших способов для этого.

Ответы [ 2 ]

1 голос
/ 06 марта 2020

(Отредактировано) Попробуйте добавить setMasterChart и setDetailChart в пустой массив в качестве второго аргумента вашей ловушки эффектов. Просто идея.

0 голосов
/ 10 марта 2020

Я думаю, что лучший подход состоит в том, чтобы установить параметры хуков в один функциональный компонент и передать их в качестве реквизита в функции, которые будут отображать каждый график.

Демонстрация: https://stackblitz.com/edit/react-lwkgkv?file=index.js

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...