Гистограмма Nivo вызывает функцию метки сотни раз - PullRequest
0 голосов
/ 14 марта 2020

Я использую панель Nivo для отображения прогресса пользователя в бюджете. Я нормализовал данные, разделив баланс категории на цель категории. Пример данных.

[{
    "category": "Gas",
    "budget": 0.24,
    "over_budget": 0.0
},
{
    "category": "Groceries",
    "budget": 1.0,
    "over_budget": 0.26
}]

Я не хочу, чтобы эти значения использовались в качестве метки на графике. Я планирую использовать фактическое значение баланса в качестве метки. У меня есть конечная точка, которая будет возвращать баланс для категории, и я попытался использовать это значение следующим образом:

<ResponsiveBar
...
label={d => this.getDollarAmount(d.value)}
...
>

С функцией PO C как:

getDollarAmount(value) {
    console.log("hitting getDollarAmount")
    return 1
  };

сообщение журнала регистрируется более 500 раз. Я ожидаю, что функция будет нажата только один раз для каждого бара на графике.

Я все еще учусь реагировать, так что это может быть чем-то очевидным. Заранее спасибо!

РЕДАКТИРОВАТЬ - Вот весь компонент BarChart:

import axios from 'axios';

import React, { Component } from "react";

import { ResponsiveBar } from '@nivo/bar'

// Nivo theming
const theme = {
  axis: {
    ticks: {
      line: {
        stroke: "#e9ecee",
        strokeWidth: 40
      },
      text: {
        // fill: "#919eab",
        fill: "black",
        fontFamily: "BlinkMacSystemFont",
        fontSize: 16
      }
    }
  },
  grid: {
    line: {
      stroke: "#e9ecee",
      strokeWidth: 5
    }
  },
  legends: {
    text: {
      fontFamily: "BlinkMacSystemFont"
    }
  }
};

let budgetStatusAPI = 'http://127.0.0.1:8000/api/budget_status/?auth_user=1&month=2020-02-01';

class BarChart extends Component {

  constructor(props) {
    super(props);

    this.state = {
      data: [],
    }

    this.getDollarAmount = this.getDollarAmount.bind(this);
  }


  componentDidMount() {
    console.log("component did mount")

    axios.get(budgetStatusAPI).then(response => {
      this.setState({
        data: response.data
      }, function () {
        console.log(this.state.data);
      })
    });
  }

  componentDidUpdate() {
    console.log("component did update")
  }

  getDollarAmount(value) {
    console.log("hitting getDollarAmount")
    console.log(value)
    return 1
  };


  render() {

    const hard_data = [
        {
          "category": "Groceries",
          "budget_status": 1.0,
          "over_budget": .26,
        },
        {
          "category": "Gas",
          "budget_status": .24,
          "over_budget": 0.0,
        }]

    return(

      <ResponsiveBar
        maxValue={1.5}
        markers={[
            {
                axis: 'x',
                value: 1,
                lineStyle: { stroke: 'rgba(0, 0, 0, .35)', strokeWidth: 2 },
                legend: 'Goal',
                legendOrientation: 'horizontal',
                legendPosition: 'top'
            },
        ]}
        enableGridX={false}
        gridXValues={[1]}
        enableGridY={false}
        data={this.state.data}
        // data={hard_data}
        keys={['budget_status', 'over_budget']}
        indexBy="category"
        margin={{ top: 25, right: 130, bottom: 50, left: 125 }}
        padding={0.3}
        layout="horizontal"
        colors={{ scheme: 'set2' }}
        theme={theme}
        defs={[
            {
                id: 'dots',
                type: 'patternDots',
                background: 'inherit',
                color: '#38bcb2',
                size: 4,
                padding: 1,
                stagger: true
            },
            {
                id: 'lines',
                type: 'patternLines',
                background: 'inherit',
                color: '#eed312',
                rotation: -45,
                lineWidth: 6,
                spacing: 10
            }
        ]}
        borderColor={{ from: 'color', modifiers: [ [ 'darker', 1.6 ] ] }}
        axisBottom={null}
        label={d => this.getDollarAmount(d.value)}
        isInteractive={false}
        animate={true}
        motionStiffness={90}
        motionDamping={15}
    />
    )
  }
}


export default BarChart;

Воспроизводится здесь: https://codesandbox.io/s/nivo-bar-label-issue-k4qek

1 Ответ

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

Многократный вызов происходит, потому что гистограмма вызывает функцию label для каждого анимационного тика / рендера кадра. Если мы настроим счетчик, мы увидим, что animate prop установлен в true, он будет рендериться от 450+ до 550+ раз, но если мы установим prop animate в false, мы он рендерит 6 раз, что является точным значением цены > 0.0.

Если вы хотите избежать всех этих рендеров, вам придется отключить анимацию, используя animate={false} prop следующим образом:

getDollarAmount(value) {
  // Remove every console.log inside this function

  return `$${value}`
}

render() {
  return (
    <ResponsiveBar
      animate={false}
      label={d => this.getDollarAmount(d.value)}
      ...
  );
}

Вы можете проверить, работает ли он на клонированном CodeSandbox . Я установил animate на false и журнал counter внутри getDollarAmount звонит 6 раз. Попробуйте изменить animate на true, и вы увидите 500+- рендеры.

Кроме того, вам не нужно создавать функцию для каждого вызова label, вы можете просто передать getDollarAmount и пусть он обрабатывает весь параметр d следующим образом:

getDollarAmount(d) {
  // Remove every console.log inside this function

  return `$${d.value}`
}

render() {
  return (
    <ResponsiveBar
      animate={false}
      label={this.getDollarAmount}
      ...
  );
}
...