logo

G2

  • 文档
  • 图表介绍
  • API
  • 图表示例
  • 主题
  • 周边生态
  • 所有产品antv logo arrow
  • 5.3.3
  • 快速上手
  • 简介
    • 什么是 G2
    • 在前端框架中使用
    • Spec 和 API
  • 核心概念
    • 图表(Chart)
      • G2 图表组成
      • 如何使用图表
    • 标记(Mark)
      • 概览
      • area
      • box
      • boxplot
      • cell
      • chord
      • density
      • gauge
      • heatmap
      • image
      • interval
      • line
      • lineX
      • lineY
      • link
      • liquid
      • sunburst
      • point
      • polygon
      • range
      • rangeX
      • rangeY
      • rect
      • shape
      • text
      • vector
      • wordCloud
    • 视图(View)
    • 数据(Data)
      • 概览
      • custom
      • ema
      • fetch
      • filter
      • fold
      • inline
      • join
      • kde
      • log
      • map
      • pick
      • rename
      • slice
      • sort
      • sortBy
    • 编码(Encode)
    • 比例尺(Scale)
      • 概览
      • band
      • linear
      • log
      • ordinal
      • point
      • quantile
      • quantize
      • sqrt
      • threshold
      • time
      • pow
    • 转换(Transform)
      • 概览
      • bin
      • binX
      • diffY
      • dodgeX
      • flexX
      • group
      • groupColor
      • groupX
      • groupY
      • jitter
      • jitterX
      • jitterY
      • normalizeY
      • pack
      • sample
      • select
      • selectX
      • selectY
      • sortColor
      • sortX
      • sortY
      • stackEnter
      • stackY
      • symmetryY
    • 坐标系(Coordinate)
      • 概览
      • fisheye
      • parallel
      • polar
      • theta
      • transpose
      • radial
      • cartesian3D
      • helix
    • 样式(Style)
    • 动画(Animate)
      • 概览
      • fadeIn
      • fadeOut
      • growInX
      • growInY
      • morphing
      • pathIn
      • scaleInX
      • scaleInY
      • scaleOutX
      • scaleOutY
      • waveIn
      • zoomIn
      • zoomOut
    • 状态(State)
    • 交互(Interaction)
      • 概览
      • brushAxisHighlight
      • brushHighlight
      • brushXHighlight
      • brushYHighlight
      • brushFilter
      • brushXFilter
      • brushYFilter
      • chartIndex
      • elementHighlight
      • elementHighlightByColor
      • elementHighlightByX
      • elementSelect
      • elementSelectByColor
      • elementSelectByX
      • fisheye
      • legendFilter
      • legendHighlight
      • poptip
      • scrollbarFilter
      • sliderFilter
    • 复合(Composition)
      • 概览
      • facetCircle
      • facetRect
      • repeatMatrix
      • spaceFlex
      • spaceLayer
      • timingKeyframe
    • 主题(Theme)
      • 概览
      • academy
      • classic
      • classicDark
    • 事件(Event)
    • 颜色映射(Color)
  • 图表 API
  • 图表组件
    • 标题(Title)
    • 坐标轴(Axis)
    • 图例(Legend)
    • 滚动条(Scrollbar)
    • 缩略轴(Slider)
    • 提示信息(Tooltip)
    • 数据标签(Label)
  • 进阶主题
    • 关系图(Graph)
      • forceGraph
      • pack
      • sankey
      • tree
      • treemap
    • 地图(Geo)
      • geoPath
      • geoView
    • 3D 图表(3D Chart)
      • 绘制 3D 图表
      • point3D
      • line3D
      • interval3D
      • surface3D
    • 插件扩展(Plugin)
      • renderer
      • rough
      • lottie
      • a11y
    • 按需打包
    • 设置纹理
    • 服务端渲染(SSR)
    • Spec 函数表达式支持 (5.3.0 支持)
  • 版本特性
    • 新版本特性
    • v4 升级 v5 指南
  • 常见问题 FAQ

编码(Encode)

上一篇
sortBy
下一篇
概览

Resources

Ant Design
Galacea Effects
Umi-React 应用开发框架
Dumi-组件/文档研发工具
ahooks-React Hooks 库

社区

体验科技专栏
seeconfSEE Conf-蚂蚁体验科技大会

帮助

GitHub
StackOverflow

more products更多产品

Ant DesignAnt Design-企业级 UI 设计语言
yuque语雀-知识创作与分享工具
EggEgg-企业级 Node 开发框架
kitchenKitchen-Sketch 工具集
GalaceanGalacean-互动图形解决方案
xtech蚂蚁体验科技
© Copyright 2025 Ant Group Co., Ltd..备案号:京ICP备15032932号-38

Loading...

视觉通道(Channel)

数据可视化的核心内容是可视化编码,是将数据信息映射成可视化元素的技术。就像数据包含属性和值,可视化编码也由两部分组成:图形标记(Mark) 和 视觉通道(Channel)。

  • 数据属性 -> 标记:直观的代表数据的性质分类,通常是几何图形元素,例如:点、线、面、体。
  • 数据值 -> 视觉通道:用于表现数据属性的定量信息,包括标记的位置、大小、形状、方向、色调、饱和度、亮度等。

视觉通道也可以称为图表属性,原始数据与视觉通道之间的映射关系使我们能迅速获取信息。所以我们可以把图表看成一组图形标记的组合,这些图形标记中携带了一些信息,我们称作它 编码 了一些信息。而当人们从这些符号中读取信息时,我们称作我们 解码 了一些信息。

视觉通道的类型

人类对视觉通道的识别有两种基本的感知模式。第一种感知模式得到的信息是关于对象本身的特征和位置等,对应视觉通道的定性性质和分类性质;第二种感知模式得到的信息是对象某一属性在数值上的大小,对应视觉通道的定量性质或者定序性质。因此我们将视觉通道分为 两大类:

  • 定性(分类) 的视觉通道,如形状、颜色的色调、控件位置。
  • 定量(连续、有序) 的视觉通道,如直线的长度、区域的面积、空间的体积、斜度、角度、颜色的饱和度和亮度等。

然而两种分类不是绝对的,例如位置信息,既可以区分不同的分类,又可以分辨连续数据的差异。

视觉通道的表现力

进行可视化编码时我们需要考虑不同视觉通道的表现力和有效性,主要体现在下面几个方面:

  • 准确性,是否能够准确的表达视觉数据之间的变化
  • 可辨认性,同一个视觉通道能够编码的分类个数,即可辨识的分类个数上限
  • 可分离性,不同视觉通道的编码对象放置到一起,是否容易分辨
  • 视觉突出,重要的信息,是否用更加突出的视觉通道进行编码

以下是视觉通道的表现力从最好到最差的示意图:

G2 视觉通道的设计

对应视觉编码中的视觉通道,G2 中的视觉通道可以分为以下五种图形属性:

位置相关视觉通道

视觉通道描述
x定义元素在水平方向的位置
y定义元素在垂直方向的位置
z定义元素在三维空间中的深度位置
x1定义元素水平方向的终止位置
y1定义元素垂直方向的终止位置,堆叠柱状图的内部实现就是计算了 y1 通道的值
series通过空间位置区分不同数据系列
position自定义复合图表显示位置,用于平行坐标系和圆形分面图等
position${number}自定义复合图表显示位置,用于平行坐标系和圆形分面图等

颜色相关视觉通道


视觉通道描述
color通过色调映射数据分类属性
opacity控制元素的透明度,常用于突出焦点或表达数值强度(如热力图的渐变透明度)

形状相关视觉通道

视觉通道描述
shape通过几何形态区分数据类别,适合分类数据编码

大小相关视觉通道

视觉通道描述
size控制元素的物理尺寸(点半径/线宽/柱宽),适合编码定量数据

动画相关视觉通道

视觉通道描述
key动画元素唯一标识符,用于数据更新
groupKey动画分组标识符,用于数据更新
enterType元素进入场景的动画类型(如缩放/淡入)
enterEasing进入动画的缓动函数
enterDuration进入动画持续时间
enterDelay进入动画延迟时间
updateType数据更新时的过渡动画类型
updateEasing更新动画的缓动函数
updateDuration更新动画持续时间
updateDelay更新动画延迟时间
exitType元素退出场景的动画类型(如滑动消失)
exitEasing退出动画的缓动函数
exitDuration退出动画持续时间
exitDelay退出动画延迟时间

编码(Encode)

为了控制标记的样式,我们往往会把一列数据和它的某一项视觉通道绑定。

下面是一组筹备活动的时间数据:

table([
{ name: '活动策划', startTime: 1, endTime: 4 },
{ name: '场地物流规划', startTime: 3, endTime: 13 },
{ name: '选择供应商', startTime: 5, endTime: 8 },
{ name: '租赁场地', startTime: 9, endTime: 13 },
{ name: '预定餐饮服务商', startTime: 10, endTime: 14 },
{ name: '租赁活动装饰团队', startTime: 12, endTime: 17 },
{ name: '彩排', startTime: 14, endTime: 16 },
{ name: '活动庆典', startTime: 17, endTime: 18 },
]);

在下面的例子中,我们把数据中 name 一列数据和 x 通道绑定,endTime 列数据减去 startTime 列数据的值和 y 通道绑定。这种绑定的过程被称为编码(Encode),我们常常说图形的某个视觉属性编码了一列数据,这种数据驱动的属性被称为通道(Channel)。比如下面的 interval 标记 的 x,y 通道都分别编码了对应列的数据,可以清晰地显示不同事项和消耗时间的关系。

import { Chart } from '@antv/g2';
const chart = new Chart({
container: 'container',
});
chart.options({
type: 'interval',
autoFit: true,
data: [
{ name: '活动策划', startTime: 1, endTime: 4 },
{ name: '场地物流规划', startTime: 3, endTime: 13 },
{ name: '选择供应商', startTime: 5, endTime: 8 },
{ name: '租赁场地', startTime: 9, endTime: 13 },
{ name: '预定餐饮服务商', startTime: 10, endTime: 14 },
{ name: '租赁活动装饰团队', startTime: 12, endTime: 17 },
{ name: '彩排', startTime: 14, endTime: 16 },
{ name: '活动庆典', startTime: 17, endTime: 18 },
],
encode: {
x: 'name', // 事项名称字段映射X轴位置
y: (datum) => datum.endTime - datum.startTime, // 结束时间减去开始时间映射Y轴位置
},
coordinate: { transform: [{ type: 'transpose' }] },
axis: {
x: {
title: '事项',
},
y: {
title: '消耗时间',
},
},
});
chart.render();

当我们想进一步分析事项的时序关系的时候,就需要借助其他的视觉通道。

在下面的例子中,我们把数据中 startTime 一列数据和 y 通道绑定,endTime 一列数据和 y1 通道绑定(当然,你也可以将[startTime, endTime]绑定到 y 通道,详见数组通道),通过这样的数据编码方式来实现一个甘特图的效果。为了更好的区分不同事项,在 x 通道编码了 name 列的数据以外,额外把数据中 name 一列数据和 color 通道也进行绑定,这样通过不同的颜色以及不同的 x 轴位置,我们能够更好地区分不同事项。

import { Chart } from '@antv/g2';
const chart = new Chart({
container: 'container',
});
chart.options({
type: 'interval',
autoFit: true,
data: [
{ name: '活动策划', startTime: 1, endTime: 4 },
{ name: '场地物流规划', startTime: 3, endTime: 13 },
{ name: '选择供应商', startTime: 5, endTime: 8 },
{ name: '租赁场地', startTime: 9, endTime: 13 },
{ name: '预定餐饮服务商', startTime: 10, endTime: 14 },
{ name: '租赁活动装饰团队', startTime: 12, endTime: 17 },
{ name: '彩排', startTime: 14, endTime: 16 },
{ name: '活动庆典', startTime: 17, endTime: 18 },
],
encode: {
x: 'name', // 事项名称字段映射X轴位置
y: 'startTime', // 事项开始时间字段映射y位置
y1: 'endTime', // 事项结束时间字段映射y1轴位置
color: 'name', // 事项名称字段映射颜色
},
coordinate: { transform: [{ type: 'transpose' }] },
axis: {
x: {
title: '事项',
},
y: {
title: '时间',
},
},
});
chart.render();

这已经是一个较为完整的甘特图了,如果想强化时间顺序,可以借助 G2 中动画相关的视觉通道来实现时序动画的效果,下面的例子对 enterDuration 通道和 enterDelay 通道进行了编码,使得不同事项对应的标记的进入动画延迟时间和持续时间跟数据里的开始时间和结束时间相关联。

import { Chart } from '@antv/g2';
const chart = new Chart({
container: 'container',
});
chart.options({
type: 'interval',
autoFit: true,
data: [
{ name: '活动策划', startTime: 1, endTime: 4 },
{ name: '场地物流规划', startTime: 3, endTime: 13 },
{ name: '选择供应商', startTime: 5, endTime: 8 },
{ name: '租赁场地', startTime: 9, endTime: 13 },
{ name: '预定餐饮服务商', startTime: 10, endTime: 14 },
{ name: '租赁活动装饰团队', startTime: 12, endTime: 17 },
{ name: '彩排', startTime: 14, endTime: 16 },
{ name: '活动庆典', startTime: 17, endTime: 18 },
],
encode: {
x: 'name', // 事项名称字段映射X轴位置
y: 'startTime', // 事项开始时间字段映射y位置
y1: 'endTime', // 事项结束时间字段映射y1轴位置
color: 'name', // 事项名称字段映射颜色
enterDuration: (datum) => datum.endTime - datum.startTime, // 事项结束时间减去开始时间映射进入动画持续时间
enterDelay: 'startTime', // 事项开始时间映射进入动画延迟时间
},
scale: { enterDuration: { zero: true, range: [0, 3000] } },
coordinate: { transform: [{ type: 'transpose' }] },
axis: {
x: {
title: '事项',
},
y: {
title: '时间',
},
},
});
chart.render();

配置层级

G2 中编码(Encode) 主要用于指定视觉元素属性和数据之间的关系,可以在 Mark 层级指定编码:

({
type: 'interval',
encode: {
x: 'name',
y: 'value',
},
});
// API
// 第一种
chart.interval().encode('x', 'name').encode('y', 'value');
// 第二种
chart.interval().encode({ x: 'name', y: 'value' });

也可以在 View 层级指定编码:

({
type: 'view',
encode: {
x: 'name',
y: 'value',
},
});
// API
// 第一种
chart.encode('x', 'name').encode('y', 'value');
// 第二种
chart.encode({ x: 'name', y: 'value' });

通道编码具有传递性,视图的编码会传递给 children 指定的标记,如果该标记没有对应通道的编码,那么就设置,否则不做任何事情。比如绘制一个点线图:

import { Chart } from '@antv/g2';
const chart = new Chart({
container: 'container',
});
chart
.data([
{ year: '1991', value: 3 },
{ year: '1992', value: 4 },
{ year: '1993', value: 3.5 },
{ year: '1994', value: 5 },
{ year: '1995', value: 4.9 },
{ year: '1996', value: 6 },
{ year: '1997', value: 7 },
{ year: '1998', value: 9 },
{ year: '1999', value: 13 },
])
.encode('x', 'year') // 视图层级的编码
.encode('y', 'value');
chart.line();
chart.point();
chart.render();

配置项

一个完整的编码是由 encode.type 和 encode.value 组成的。

编码后会按照编码方式生成指定通道下的数组。

编码方式typevalue描述
字段编码fieldstring数据某一列的字段名
函数编码transform(value, index, array) => number | string | boolean | Date通过回调函数指定通道和新生成的一些绑定
常量编码constantany指定通道值为常量
数据编码columnnumber[] | string[] | boolean[] | Date[]指定通道值为一个数组

在 G2 中提供了语法糖,在大部分情况下 encode.type可以省略。

({
type: 'point',
encode: {
x: { type: 'field', value: 'weight' },
y: { type: 'field', value: 'height' },
color: { type: 'field', value: 'gender' },
},
});

也可以利用语法糖简化写法:

// 语法糖
({
type: 'point',
encode: {
x: 'weight',
y: 'height',
color: 'gender',
},
});

字段编码

指定通道和数据的某一列绑定:

(data) => data.map((d) => d[value]);
({
type: 'interval',
encode: { x: { type: 'field', value: 'name' } },
});
// 语法糖
({
type: 'interval',
encode: { x: 'name' },
});

函数编码

指定通道和新生成的一些绑定:

(data) => data.map(value);
({
type: 'interval',
encode: {
color: {
type: 'transform',
value: (d) => (d.value > 100 ? 'high' : 'low'),
},
},
});
// 语法糖
({
type: 'interval',
encode: {
color: (d) => (d.value > 100 ? 'high' : 'low'),
},
});

还可以通过函数编码的方式在多轴图中生成对应通道的图例。

import { Chart } from '@antv/g2';
const chart = new Chart({
container: 'container',
});
chart.options({
type: 'view',
autoFit: true,
data: [
{ time: '10:10', call: 4, waiting: 2, people: 2 },
{ time: '10:15', call: 2, waiting: 6, people: 3 },
{ time: '10:20', call: 13, waiting: 2, people: 5 },
{ time: '10:25', call: 9, waiting: 9, people: 1 },
{ time: '10:30', call: 5, waiting: 2, people: 3 },
{ time: '10:35', call: 8, waiting: 2, people: 1 },
{ time: '10:40', call: 13, waiting: 1, people: 2 },
],
children: [
{
type: 'interval',
encode: {
x: 'time',
y: 'waiting',
color: () => 'waiting',
series: () => 'waiting',
},
scale: { y: { nice: true } },
axis: { y: { title: 'Waiting' } },
},
{
type: 'interval',
encode: {
x: 'time',
y: 'people',
color: () => 'people',
series: () => 'people',
},
scale: { y: { independent: true } },
axis: { y: { position: 'right', grid: null, title: 'People' } },
},
{
type: 'line',
encode: { x: 'time', y: 'call', color: () => 'call' },
scale: { series: { independent: true } },
},
],
});
chart.render();

通过函数编码还可以在某些时候禁用某些通道。在 G2 中,当未定义 series 通道,且 color 通道已定义的情况下,会复制一份值到 series 通道,以实现分类的效果。 下面的例子中 color 通道被映射到连续字段上,此时如果 series 再被映射到连续字段上会影响渐变折线的显示,需要通过 series: () => undefined 的方式禁用。

import { Chart } from '@antv/g2';
const chart = new Chart({
container: 'container',
});
chart.options({
type: 'area',
autoFit: true,
data: {
type: 'fetch',
value: 'https://assets.antv.antgroup.com/g2/temperatures3.json',
},
encode: {
x: (d) => new Date(d.date),
y: ['low', 'high'],
color: (d) => d.high - d.low,
series: () => undefined,
},
scale: { color: { palette: 'reds' } },
style: { gradient: 'x' },
axis: { x: { title: 'date' } },
});
chart.render();

常量编码

指定通道值为常量,常常用于指定颜色通道或者大小通道,指定标记为某种颜色或者某个尺寸,这种编码方式不会生成对应通道的图例。

(data) => data.map(() => value);
({
type: 'point',
encode: {
color: { type: 'constant', value: 'steelblue' },
size: { type: 'constant', value: 10 },
},
});
// 语法糖
({
type: 'interval',
encode: {
color: 'steelblue',
size: 10,
},
});

数据编码

指定某个通道的值为一个数组。

() => value;
({
type: 'line',
encode: {
x: { type: 'column', value: [0, 1, , 2, 3] },
},
});

对于一些大数据的场景,使用数组列会更适合,下面是一个简单的例子。

import { Chart } from '@antv/g2';
const I = [0, 1, 2, 3, 4];
const X = I.map((i) => ((i - 2) * Math.PI) / 2);
const Y = X.map((x) => Math.sin(x));
const chart = new Chart({
container: 'container',
});
chart.options({
type: 'line',
data: I,
encode: {
x: {
type: 'column',
value: X,
},
y: {
type: 'column',
value: Y,
},
shape: 'smooth',
},
});
chart.render();

数组通道

当然,对于位置相关的视觉通道来说:例如 x 和 y 通道,往往不只需要一列数据,比如一个瀑布图,这个时候可以通过数组给一个通道指定多个列。

import { Chart } from '@antv/g2';
const chart = new Chart({
container: 'container',
});
chart.options({
type: 'interval',
data: [
{ month: 'Jan.', profit: 387264, start: 0, end: 387264 },
{ month: 'Feb.', profit: 772096, start: 387264, end: 1159360 },
{ month: 'Mar.', profit: 638075, start: 1159360, end: 1797435 },
{ month: 'Apr.', profit: -211386, start: 1797435, end: 1586049 },
{ month: 'May', profit: -138135, start: 1586049, end: 1447914 },
{ month: 'Jun', profit: -267238, start: 1447914, end: 1180676 },
{ month: 'Jul.', profit: 431406, start: 1180676, end: 1612082 },
{ month: 'Aug.', profit: 363018, start: 1612082, end: 1975100 },
{ month: 'Sep.', profit: -224638, start: 1975100, end: 1750462 },
{ month: 'Oct.', profit: -299867, start: 1750462, end: 1450595 },
{ month: 'Nov.', profit: 607365, start: 1450595, end: 2057960 },
{ month: 'Dec.', profit: 1106986, start: 2057960, end: 3164946 },
{ month: 'Total', start: 0, end: 3164946 },
],
encode: {
x: 'month',
y: ['end', 'start'], // 等价于 y:'end', y1:'start'
color: (d) =>
d.month === 'Total' ? 'Total' : d.profit > 0 ? 'Increase' : 'Decrease',
},
axis: { y: { labelFormatter: '~s' } },
});
chart.render();

同时也可以通过 ${channel}${index} 的形式去分别指定:

// 和上面的形式等价
chart.encode('y', 'end').encode('y1', 'start');