123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823 |
- import * as Attr from '../attr/index';
- const Util = require('../util/common');
- const Base = require('../base');
- const GROUP_ATTRS = [ 'color', 'size', 'shape' ];
- const FIELD_ORIGIN = '_origin';
- const FIELD_ORIGIN_Y = '_originY';
- const Global = require('../global');
- const GeometryShape = require('./shape/shape');
- const Adjust = require('@antv/adjust/lib/base');
- function parseFields(field) {
- if (Util.isArray(field)) {
- return field;
- }
- if (Util.isString(field)) {
- return field.split('*');
- }
- return [ field ];
- }
- /**
- * The parent class for Geometry
- * @class Geom
- */
- class Geom extends Base {
- getDefaultCfg() {
- return {
- /**
- * geometry type
- * @type {String}
- */
- type: null,
- /**
- * the data of geometry
- * @type {Array}
- */
- data: null,
- /**
- * the attrs of geo,etry
- * @type {Object}
- */
- attrs: {},
- scales: {},
- /**
- * group for storing the shapes
- * @type {Canvas}
- */
- container: null,
- /**
- * style options
- * @type {Object}
- */
- styleOptions: null,
- chart: null,
- shapeType: '',
- /**
- * wether to generate key points for each shape
- * @protected
- * @type {Boolean}
- */
- generatePoints: false,
- attrOptions: {},
- sortable: false,
- startOnZero: true,
- visible: true,
- connectNulls: false,
- // 是否丢弃没有值的分组。
- ignoreEmptyGroup: false
- };
- }
- init() {
- const self = this;
- self._initAttrs();
- self._processData();
- }
- _getGroupScales() {
- const self = this;
- const scales = [];
- Util.each(GROUP_ATTRS, function(attrName) {
- const attr = self.getAttr(attrName);
- if (attr) {
- const attrScales = attr.scales;
- Util.each(attrScales, function(scale) {
- if (scale && scale.isCategory && scales.indexOf(scale) === -1) {
- scales.push(scale);
- }
- });
- }
- });
- return scales;
- }
- _groupData(data) {
- const self = this;
- const colDefs = self.get('colDefs');
- const groupScales = self._getGroupScales();
- if (groupScales.length) {
- const appendConditions = {};
- const names = [];
- Util.each(groupScales, scale => {
- const field = scale.field;
- names.push(field);
- if (colDefs && colDefs[field] && colDefs[field].values) { // users have defined
- appendConditions[scale.field] = colDefs[field].values;
- }
- });
- return Util.Array.group(data, names, appendConditions);
- }
- return [ data ];
- }
- _setAttrOptions(attrName, attrCfg) {
- const options = this.get('attrOptions');
- options[attrName] = attrCfg;
- const attrs = this.get('attrs');
- // 说明已经初始化过了
- if (Object.keys(attrs).length) {
- this._createAttr(attrName, attrCfg);
- }
- }
- _createAttrOption(attrName, field, cfg, defaultValues) {
- const attrCfg = {};
- attrCfg.field = field;
- if (cfg) {
- if (Util.isFunction(cfg)) {
- attrCfg.callback = cfg;
- } else {
- attrCfg.values = cfg;
- }
- } else {
- attrCfg.values = defaultValues;
- }
- this._setAttrOptions(attrName, attrCfg);
- }
- _createAttr(type, option) {
- const self = this;
- const attrs = self.get('attrs');
- const coord = self.get('coord');
- const className = Util.upperFirst(type);
- const fields = parseFields(option.field);
- if (type === 'position') {
- option.coord = coord;
- }
- const scales = [];
- for (let i = 0, len = fields.length; i < len; i++) {
- const field = fields[i];
- const scale = self._createScale(field);
- scales.push(scale);
- }
- if (type === 'position') {
- const yScale = scales[1];
- // 饼图的处理,但是还不知道为啥
- if (coord.type === 'polar' && coord.transposed && self.hasAdjust('stack')) {
- if (yScale.values.length) {
- yScale.change({
- nice: false,
- min: 0,
- max: Math.max.apply(null, yScale.values)
- });
- }
- }
- }
- option.scales = scales;
- const attr = new Attr[className](option);
- attrs[type] = attr;
- return attr;
- }
- _initAttrs() {
- const self = this;
- const attrOptions = self.get('attrOptions');
- for (const type in attrOptions) {
- if (attrOptions.hasOwnProperty(type)) {
- this._createAttr(type, attrOptions[type]);
- }
- }
- }
- _createScale(field) {
- const scales = this.get('scales');
- let scale = scales[field];
- if (!scale) {
- scale = this.get('chart').createScale(field);
- scales[field] = scale;
- }
- return scale;
- }
- _processData() {
- const self = this;
- const data = this.get('data');
- const dataArray = [];
- let groupedArray = this._groupData(data);
- if (this.get('ignoreEmptyGroup')) {
- const yScale = this.getYScale();
- groupedArray = groupedArray.filter(group =>
- group.some(item => typeof item[yScale.field] !== 'undefined')
- );
- }
- for (let i = 0, len = groupedArray.length; i < len; i++) {
- const subData = groupedArray[i];
- const tempData = self._saveOrigin(subData);
- if (this.hasAdjust('dodge')) {
- self._numberic(tempData);
- }
- dataArray.push(tempData);
- }
- if (self.get('adjust')) {
- self._adjustData(dataArray);
- }
- if (self.get('sortable')) {
- self._sort(dataArray);
- }
- self.set('dataArray', dataArray);
- return dataArray;
- }
- _saveOrigin(data) {
- const rst = [];
- for (let i = 0, len = data.length; i < len; i++) {
- const origin = data[i];
- const obj = {};
- for (const k in origin) {
- obj[k] = origin[k];
- }
- obj[FIELD_ORIGIN] = origin;
- rst.push(obj);
- }
- return rst;
- }
- _numberic(data) {
- const positionAttr = this.getAttr('position');
- const scales = positionAttr.scales;
- for (let j = 0, len = data.length; j < len; j++) {
- const obj = data[j];
- const count = Math.min(2, scales.length);
- for (let i = 0; i < count; i++) {
- const scale = scales[i];
- if (scale.isCategory) {
- const field = scale.field;
- obj[field] = scale.translate(obj[field]);
- }
- }
- }
- }
- _adjustData(dataArray) {
- const self = this;
- const adjust = self.get('adjust');
- if (adjust) {
- const adjustType = Util.upperFirst(adjust.type);
- if (!Adjust[adjustType]) {
- throw new Error('not support such adjust : ' + adjust);
- }
- const xScale = self.getXScale();
- const yScale = self.getYScale();
- const cfg = Util.mix({
- xField: xScale.field,
- yField: yScale.field
- }, adjust);
- const adjustObject = new Adjust[adjustType](cfg);
- adjustObject.processAdjust(dataArray);
- if (adjustType === 'Stack') {
- self._updateStackRange(yScale.field, yScale, dataArray);
- }
- }
- }
- _updateStackRange(field, scale, dataArray) {
- const mergeArray = Util.Array.merge(dataArray);
- let min = scale.min;
- let max = scale.max;
- for (let i = 0, len = mergeArray.length; i < len; i++) {
- const obj = mergeArray[i];
- const tmpMin = Math.min.apply(null, obj[field]);
- const tmpMax = Math.max.apply(null, obj[field]);
- if (tmpMin < min) {
- min = tmpMin;
- }
- if (tmpMax > max) {
- max = tmpMax;
- }
- }
- if (min < scale.min || max > scale.max) {
- scale.change({
- min,
- max
- });
- }
- }
- _sort(mappedArray) {
- const self = this;
- const xScale = self.getXScale();
- const { field, type } = xScale;
- if (type !== 'identity' && xScale.values.length > 1) {
- Util.each(mappedArray, itemArr => {
- itemArr.sort((obj1, obj2) => {
- if (type === 'timeCat') {
- return xScale._toTimeStamp(obj1[FIELD_ORIGIN][field]) - xScale._toTimeStamp(obj2[FIELD_ORIGIN][field]);
- }
- return xScale.translate(obj1[FIELD_ORIGIN][field]) - xScale.translate(obj2[FIELD_ORIGIN][field]);
- });
- });
- }
- self.set('hasSorted', true);
- self.set('dataArray', mappedArray);
- }
- paint() {
- const self = this;
- const dataArray = self.get('dataArray');
- const mappedArray = [];
- const shapeFactory = self.getShapeFactory();
- shapeFactory.setCoord(self.get('coord'));
- self._beforeMapping(dataArray);
- for (let i = 0, len = dataArray.length; i < len; i++) {
- let data = dataArray[i];
- if (data.length) {
- data = self._mapping(data);
- mappedArray.push(data);
- self.draw(data, shapeFactory);
- }
- }
- self.set('dataArray', mappedArray);
- }
- getShapeFactory() {
- let shapeFactory = this.get('shapeFactory');
- if (!shapeFactory) {
- const shapeType = this.get('shapeType');
- shapeFactory = GeometryShape.getShapeFactory(shapeType);
- this.set('shapeFactory', shapeFactory);
- }
- return shapeFactory;
- }
- _mapping(data) {
- const self = this;
- const attrs = self.get('attrs');
- const yField = self.getYScale().field;
- // 用来缓存转换的值,减少mapping耗时
- const mappedCache = {};
- for (const k in attrs) {
- if (attrs.hasOwnProperty(k)) {
- const attr = attrs[k];
- const names = attr.names;
- const scales = attr.scales;
- for (let i = 0, len = data.length; i < len; i++) {
- const record = data[i];
- record[FIELD_ORIGIN_Y] = record[yField];
- // 获取视觉属性对应的value值
- // 位置的缓存命中率低,还是每次单独计算
- if (attr.type === 'position') {
- const values = self._getAttrValues(attr, record);
- for (let j = 0, len = values.length; j < len; j++) {
- const val = values[j];
- const name = names[j];
- record[name] = (Util.isArray(val) && val.length === 1) ? val[0] : val;
- }
- } else {
- // 除了position其他都只有一项
- const name = names[0];
- const field = scales[0].field;
- const value = record[field];
- const key = `${name}${value}`;
- let values = mappedCache[key];
- if (!values) {
- values = self._getAttrValues(attr, record);
- mappedCache[key] = values;
- }
- record[name] = values[0];
- }
- }
- }
- }
- return data;
- }
- _getAttrValues(attr, record) {
- const scales = attr.scales;
- const params = [];
- for (let i = 0, len = scales.length; i < len; i++) {
- const scale = scales[i];
- const field = scale.field;
- if (scale.type === 'identity') {
- params.push(scale.value);
- } else {
- params.push(record[field]);
- }
- }
- const values = attr.mapping(...params);
- return values;
- }
- getAttrValue(attrName, record) {
- const attr = this.getAttr(attrName);
- let rst = null;
- if (attr) {
- const values = this._getAttrValues(attr, record);
- rst = values[0];
- }
- return rst;
- }
- _beforeMapping(dataArray) {
- const self = this;
- if (self.get('generatePoints')) {
- self._generatePoints(dataArray);
- }
- }
- isInCircle() {
- const coord = this.get('coord');
- return coord && coord.isPolar;
- }
- getCallbackCfg(fields, cfg, origin) {
- if (!fields) {
- return cfg;
- }
- const tmpCfg = {};
- const params = fields.map(function(field) {
- return origin[field];
- });
- Util.each(cfg, function(v, k) {
- if (Util.isFunction(v)) {
- tmpCfg[k] = v.apply(null, params);
- } else {
- tmpCfg[k] = v;
- }
- });
- return tmpCfg;
- }
- getDrawCfg(obj) {
- const self = this;
- const isInCircle = self.isInCircle();
- const cfg = {
- origin: obj,
- x: obj.x,
- y: obj.y,
- color: obj.color,
- size: obj.size,
- shape: obj.shape,
- isInCircle,
- opacity: obj.opacity
- };
- const styleOptions = self.get('styleOptions');
- if (styleOptions && styleOptions.style) {
- cfg.style = self.getCallbackCfg(styleOptions.fields, styleOptions.style, obj[FIELD_ORIGIN]);
- }
- if (self.get('generatePoints')) {
- cfg.points = obj.points;
- cfg.nextPoints = obj.nextPoints;
- }
- if (isInCircle) {
- cfg.center = self.get('coord').center;
- }
- return cfg;
- }
- draw(data, shapeFactory) {
- const self = this;
- const container = self.get('container');
- const yScale = self.getYScale();
- Util.each(data, function(obj, index) {
- if (yScale && Util.isNil(obj._origin[yScale.field])) {
- return;
- }
- obj.index = index;
- const cfg = self.getDrawCfg(obj);
- const shape = obj.shape;
- self.drawShape(shape, obj, cfg, container, shapeFactory);
- });
- }
- drawShape(shape, shapeData, cfg, container, shapeFactory) {
- const gShape = shapeFactory.drawShape(shape, cfg, container);
- if (gShape) {
- Util.each([].concat(gShape), s => {
- s.set('origin', shapeData);
- });
- }
- }
- _generatePoints(dataArray) {
- const self = this;
- const shapeFactory = self.getShapeFactory();
- const shapeAttr = self.getAttr('shape');
- Util.each(dataArray, function(data) {
- for (let i = 0, len = data.length; i < len; i++) {
- const obj = data[i];
- const cfg = self.createShapePointsCfg(obj);
- const shape = shapeAttr ? self._getAttrValues(shapeAttr, obj) : null;
- const points = shapeFactory.getShapePoints(shape, cfg);
- obj.points = points;
- }
- });
- // 添加nextPoints
- Util.each(dataArray, (data, index) => {
- const nextData = dataArray[index + 1];
- if (nextData) {
- data[0].nextPoints = nextData[0].points;
- }
- });
- }
- /**
- * get the info of each shape
- * @protected
- * @param {Object} obj the data item
- * @return {Object} cfg return the result
- */
- createShapePointsCfg(obj) {
- const xScale = this.getXScale();
- const yScale = this.getYScale();
- const x = this._normalizeValues(obj[xScale.field], xScale);
- let y;
- if (yScale) {
- y = this._normalizeValues(obj[yScale.field], yScale);
- } else {
- y = obj.y ? obj.y : 0.1;
- }
- return {
- x,
- y,
- y0: yScale ? yScale.scale(this.getYMinValue()) : undefined
- };
- }
- getYMinValue() {
- const yScale = this.getYScale();
- const { min, max } = yScale;
- let value;
- if (this.get('startOnZero')) {
- if (max <= 0 && min <= 0) {
- value = max;
- } else {
- value = min >= 0 ? min : 0;
- }
- } else {
- value = min;
- }
- return value;
- }
- _normalizeValues(values, scale) {
- let rst = [];
- if (Util.isArray(values)) {
- for (let i = 0, len = values.length; i < len; i++) {
- const v = values[i];
- rst.push(scale.scale(v));
- }
- } else {
- rst = scale.scale(values);
- }
- return rst;
- }
- getAttr(name) {
- return this.get('attrs')[name];
- }
- getXScale() {
- return this.getAttr('position').scales[0];
- }
- getYScale() {
- return this.getAttr('position').scales[1];
- }
- hasAdjust(adjust) {
- return this.get('adjust') && (this.get('adjust').type === adjust);
- }
- _getSnap(scale, item, arr) {
- let i = 0;
- let values;
- const yField = this.getYScale().field; // 叠加的维度
- if (this.hasAdjust('stack') && scale.field === yField) {
- values = [];
- arr.forEach(function(obj) {
- values.push(obj[FIELD_ORIGIN_Y]);
- });
- for (let len = values.length; i < len; i++) {
- if (values[0][0] > item) {
- break;
- }
- if (values[values.length - 1][1] <= item) {
- i = values.length - 1;
- break;
- }
- if (values[i][0] <= item && values[i][1] > item) {
- break;
- }
- }
- } else {
- values = scale.values;
- values.sort((a, b) => {
- return a - b;
- });
- for (let len = values.length; i < len; i++) {
- // 如果只有1个点直接返回第1个点
- if (len <= 1) {
- break;
- }
- // 第1个点和第2个点之间
- if ((values[0] + values[1]) / 2 > item) {
- break;
- }
- // 中间的点
- if ((values[i - 1] + values[i]) / 2 <= item && (values[i + 1] + values[i]) / 2 > item) {
- break;
- }
- // 最后2个点
- if ((values[values.length - 2] + values[values.length - 1]) / 2 <= item) {
- i = values.length - 1;
- break;
- }
- }
- }
- const result = values[i];
- return result;
- }
- getSnapRecords(point) {
- const self = this;
- const coord = self.get('coord');
- const xScale = self.getXScale();
- const yScale = self.getYScale();
- const xfield = xScale.field;
- const dataArray = self.get('dataArray');
- if (!this.get('hasSorted')) {
- this._sort(dataArray);
- }
- let rst = [];
- const invertPoint = coord.invertPoint(point);
- let invertPointX = invertPoint.x;
- if (self.isInCircle() && !coord.transposed && invertPointX > (1 + xScale.rangeMax()) / 2) {
- invertPointX = xScale.rangeMin();
- }
- let xValue = xScale.invert(invertPointX);
- if (!xScale.isCategory) {
- xValue = self._getSnap(xScale, xValue);
- }
- const tmp = [];
- dataArray.forEach(function(data) {
- data.forEach(function(obj) {
- const originValue = Util.isNil(obj[FIELD_ORIGIN]) ? obj[xfield] : obj[FIELD_ORIGIN][xfield];
- if (self._isEqual(originValue, xValue, xScale)) {
- tmp.push(obj);
- }
- });
- });
- // special for pie chart
- if (this.hasAdjust('stack') && coord.isPolar && coord.transposed) {
- if (invertPointX >= 0 && invertPointX <= 1) {
- let yValue = yScale.invert(invertPoint.y);
- yValue = self._getSnap(yScale, yValue, tmp);
- tmp.forEach(obj => {
- if (Util.isArray(yValue) ? obj[FIELD_ORIGIN_Y].toString() === yValue.toString() : obj[FIELD_ORIGIN_Y] === yValue) {
- rst.push(obj);
- }
- });
- }
- } else {
- rst = tmp;
- }
- return rst;
- }
- _isEqual(originValue, value, scale) {
- if (scale.type === 'timeCat') {
- return scale._toTimeStamp(originValue) === value;
- }
- return value === originValue;
- }
- position(field) {
- this._setAttrOptions('position', {
- field
- });
- return this;
- }
- color(field, values) {
- this._createAttrOption('color', field, values, Global.colors);
- return this;
- }
- size(field, values) {
- this._createAttrOption('size', field, values, Global.sizes);
- return this;
- }
- shape(field, values) {
- const type = this.get('type');
- const shapes = Global.shapes[type] || [];
- this._createAttrOption('shape', field, values, shapes);
- return this;
- }
- style(field, cfg) {
- let styleOptions = this.get('styleOptions');
- if (!styleOptions) {
- styleOptions = {};
- this.set('styleOptions', styleOptions);
- }
- if (Util.isObject(field)) {
- cfg = field;
- field = null;
- }
- let fields;
- if (field) {
- fields = parseFields(field);
- }
- styleOptions.fields = fields;
- styleOptions.style = cfg;
- return this;
- }
- adjust(type) {
- if (Util.isString(type)) {
- type = { type };
- }
- this.set('adjust', type);
- return this;
- }
- animate(cfg) {
- this.set('animateCfg', cfg);
- return this;
- }
- changeData(data) {
- this.set('data', data);
- // 改变数据后,情况度量,因为需要重新实例化
- this.set('scales', {});
- this.init();
- }
- clearInner() {
- const container = this.get('container');
- if (container) {
- container.clear();
- // container.setMatrix([ 1, 0, 0, 1, 0, 0 ]);
- }
- }
- reset() {
- this.set('attrs', {});
- this.set('attrOptions', {});
- this.set('adjust', null);
- this.clearInner();
- }
- clear() {
- this.clearInner();
- }
- destroy() {
- this.clear();
- super.destroy();
- }
- _display(visible) {
- this.set('visible', visible);
- const container = this.get('container');
- const canvas = container.get('canvas');
- container.set('visible', visible);
- canvas.draw();
- }
- show() {
- this._display(true);
- }
- hide() {
- this._display(false);
- }
- }
- module.exports = Geom;
|