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…