preface

At the end of 2017, WHEN G2 3.0 was officially released as open source, our department decided to unify the underlying framework of the visual chart library into G2, replacing Echarts and Highcharts which had grown wild inside.

G2 is a set of high interactive graphics syntax driven by data for general statistical charts, with a high degree of ease of use and scalability.

Our department is oriented towards middle and background business, and visual chart is a strong requirement. Developers cover front-end development, back-end development, partner development, and so on that are unfamiliar with visualization. It would be too expensive for everyone to learn G2’s graphical syntax from scratch and understand professional visualization concepts like “visual channel” and “data mapping.” Therefore, it is necessary to build a real chart library out of the box based on G2, enabling development, improving efficiency and unifying visual effects.

This is the origin of CloudCharts. After three years of internal incubation, it is officially open source. Interested students can experience it online on the official website: Portal.

Today, we’ll talk about how we built the easy to use React charting library with G2, which can be drawn with a few configuration items and serves nearly 200 products internally.

SanBanFu

We provide detailed solutions to the three most common problems in G2 usage: React, configuration, and data mapping:

1. Establish the chart life cycle

2, out of the box packaging

3. Data format mapping

Establish the life cycle

First, build the overall architecture of the component library to ensure long-term maintenance and development of the project, minimize possible historical burden, and convert G2 native graphics syntax into React component.

Let’s list the entire process of charting a component:

Looking at the process of diagramming, we can divide it into several parts of the component lifecycle: creation, existence, and destruction. This is basically the same as the React lifecycle.

We found that a large part of these drawing processes can be reused, with most components differing only in the “declare drawing logic” step to establish a lifecycle management mechanism.

The component code, implemented by native code, is responsible for declaring G2’s graphical syntax. The intermediate factory function is responsible for creating the diagram instance, calling the lifecycle, and some reusable logic, and finally returning the React component.

Example factory function

The factory function is the core of the management lifecycle, and the code for the factory function is shown briefly here. The actual logic will be much more, interested students can go to Github to check the source code: portal

import G2 from '@antv/g2';
import React from 'react';

function factory(name, component) {
  class CloudCharts extends React.Component {
    componentDidMount() {
      // Set the initial width and height
      this.initSize();
      // Initialize the chart
      this.initChart(this.props);
    }

    componentDidUpdate(prevProps) {
      // The configuration item has changed
      this.rerender();
      // Update the data
      component.changeData(data);
    }

    componentWillUnmount() {
      this.destroy();
    }

    // Initializes the height and width of the adaptation
    initSize(props) {}

    // Initialize the chart
    initChart(props) {
      this.chart = new G2.Chart({
        container: this.chartDom,
      });
      // Call component code
      component.init(this.chart);
    }

    destroy() {
      component.destroy();
    }

    render() {
      return (
        <div ref={dom= > (this.chartDom = dom)} />
      );
    }
  }

  CloudCharts.displayName = `CloudCharts${name}`;

  return CloudCharts;
}
Copy the code

Out of the box

With lifecycle management, components can focus on graphical syntax translation, allowing users to pass in simple configuration items to get diagrams.

Let’s start by looking at the parts of the G2 graph syntax:

  • Top-level configuration βœ…
  • Data measurement Scale
  • Coordinate Crood
  • Axis βœ…
  • Tooltip βœ…
  • Legend Legend βœ…
  • Auxiliary tag Guide βœ…
  • The graphical element Geom
  • Text labels Label βœ…

Some of the logic at βœ… is easily reusable and can be configured with function abstraction.

Component code examples

// Line
const lineComponent = {
  init(chart, config, data) {
    / / the column definitions
    constdefs = {... }// Pass in data
    chart.source(data, defs);
    
    // Set the X-axis
    rectXAxis.call(this, chart, config);
    
    // Set the Y axis
    rectYAxis.call(this, chart, config);
    
    // Set the legend
    rectLegend.call(this, chart, config);

    / / set the tooltip
    rectTooltip.call(this, chart, config);

    // Set the auxiliary line, auxiliary background area
    guide.call(this, chart, config);
    
    // Sets the graphic element
    drawLine.call(this, chart, config);
    
    // Other logic},... };Copy the code

Configure item translator

We collectively refer to the logic within a component that accepts configuration items and declares graphical syntax as configuration item translators.

An example of a line line configuration item in drawLine:

if (config.area) {
  chart.area()
    .position(['x'.'y'])
    .color('type', config.colors);
  chart.line()
    .position(['x'.'y'])
    .color('type', config.colors)
    .style('x*y*type*extra', {
      lineJoin: 'round'}); }else{... }Copy the code

Intelligent configuration item

In addition to G2’s built-in configuration items, we also made a lot of intelligent configuration packaging, reducing the difficulty of use.

  • Automatic sizing

  • Automatic data update

  • Automatic update configuration

  • Automatically calculates the time format

  • The number of axis labels is automatically calculated

  • Too many legends fold automatically

Due to the lack of space, I will not expand the details here, and there will be a follow-up article.

Data format mapping

G2 data mapping is also a puzzle for many beginners, so we made a layer of data conversion in the factory function, allowing users to pass in data in HighCharts format. So how does that work internally?

Let’s look at this chart, which consists of just one broken line and point:

The corresponding G2 data format is a very simple JSON array, where each entry corresponds to a data point on the chart:

const data = [
  { year: '1995'.value: 4.9 },
  { year: '1996'.value: 6 },
  { year: '1997'.value: 7 },
  { year: '1998'.value: 9 },
  { year: '1999'.value: 13},];Copy the code

The Highcharts format is a little more complicated, but the key points are the same:

const data = [
  {
    // name is used to distinguish multiple groups of data
    name: 'Fold one'.data: [['1995'.5],
      ['1996'.6],
      ['1997'.7],
      ['1998'.9],
      ['1999'.13]],},];Copy the code

So it can be converted to G2 format through loop processing:

const newData = [];

data.forEach(oneData= > {
  const { name: dataName } = oneData;

  oneData.data.forEach((d, i) = > {
    const [x, y, ...extra] = d;
    newData.push({
      x,
      y,
      extra,
      type: dataName,
    });
  });
});
Copy the code

The x, y, and type returned here correspond to fields in the graph syntax:

chart.geom().position(['x'.'y']).color('type')
Copy the code

Data fields are fixed inside the charting library so that users can use data in HighCharts format, which also facilitates the migration of businesses using HighCharts to the new charting library.

Write in the last

With the three axes above, the framework of the entire chart library is basically complete, and can be used happily in the business. You are welcome to use our full version:

CloudCharts welcome everyone Star, PR and brick

We have also added “Theme Designer” and “visual configuration” to the official website, which will be covered in future articles.

πŸ† technology project phase iii | data visualization of those things…