diff --git a/__tests__/unit/utils/transform/word-cloud-spec.ts b/__tests__/unit/utils/transform/word-cloud-spec.ts index a1f1a5ddf8..ee0955a2df 100644 --- a/__tests__/unit/utils/transform/word-cloud-spec.ts +++ b/__tests__/unit/utils/transform/word-cloud-spec.ts @@ -1,16 +1,22 @@ import DataSet from '@antv/data-set'; -import { DataItem, wordCloud } from '../../../../src/utils/transform/word-cloud'; +import { DataItem } from '../../../../src/plots/word-cloud/types'; +import { processImageMask } from '../../../../src/plots/word-cloud/utils'; +import { wordCloud, transform } from '../../../../src/utils/transform/word-cloud'; + +type Row = Pick; const { View } = DataSet; const options = { type: 'tag-cloud', fields: ['text', 'value'], + font: 'Impact', fontSize: 10, fontWeight: 'bold', + rotate: 90, + padding: 2, size: [800, 800], - padding: 0, + spiral: 'archimedean', timeInterval: 5000, - rotate: 90, }; const data = ['Hello', 'world', 'normally', 'you', 'want', 'more', 'words', 'than', 'this'].map((d) => { return { @@ -20,6 +26,22 @@ const data = ['Hello', 'world', 'normally', 'you', 'want', 'more', 'words', 'tha }; }); +function basicCommon(v: DataItem) { + expect(v.hasText).toBe(true); + expect(typeof v.text).toBe('string'); + expect(typeof v.value).toBe('number'); + expect(typeof v.font).toBe('string'); + expect(typeof v.style).toBe('string'); + expect(typeof v.weight === 'number' || typeof v.weight === 'string').toBe(true); + expect(typeof v.rotate).toBe('number'); + expect(typeof v.size).toBe('number'); + expect(typeof v.padding).toBe('number'); + expect(typeof v.width).toBe('number'); + expect(typeof v.height).toBe('number'); + expect(typeof v.x).toBe('number'); + expect(typeof v.y).toBe('number'); +} + describe('word-cloud', () => { it('with data-set', () => { const dv = new View(); @@ -41,38 +63,31 @@ describe('word-cloud', () => { it('default', () => { const result = wordCloud(data); - const firstRow = result[0]; - - expect(firstRow.hasText).toBe(true); - expect(typeof firstRow.x).toBe('number'); - expect(typeof firstRow.y).toBe('number'); - expect(typeof firstRow.text).toBe('string'); - expect(typeof firstRow.size).toBe('number'); - expect(typeof firstRow.font).toBe('string'); + basicCommon(result[0]); }); it('callback', () => { - const common = (row: DataItem) => { + const common = (row: Row) => { expect(typeof row.text).toBe('string'); expect(typeof row.value).toBe('number'); }; - const font = (row: DataItem) => { + const font = (row: Row) => { common(row); return 'font-test'; }; - const fontWeight = (row: DataItem): any => { + const fontWeight = (row: Row): any => { common(row); return 'fontWeight-test'; }; - const fontSize = (row: DataItem) => { + const fontSize = (row: Row) => { common(row); return 11; }; - const rotate = (row: DataItem) => { + const rotate = (row: Row) => { common(row); return 22; }; - const padding = (row: DataItem) => { + const padding = (row: Row) => { common(row); return 33; }; @@ -94,14 +109,105 @@ describe('word-cloud', () => { spiral, }); const firstRow = result[0]; - expect(firstRow.hasText).toBe(true); - expect(typeof firstRow.x).toBe('number'); - expect(typeof firstRow.y).toBe('number'); - expect(typeof firstRow.text).toBe('string'); + basicCommon(firstRow); expect(firstRow.font).toBe('font-test'); expect(firstRow.weight).toBe('fontWeight-test'); expect(firstRow.size).toBe(11); expect(firstRow.rotate).toBe(22); expect(firstRow.padding).toBe(33); }); + + it('data is empty', () => { + const result = wordCloud([]); + expect(result.length).toBe(2); + }); + + it('image mask', async () => { + const base64 = + ''; + + const img = await processImageMask(base64); + const result = wordCloud(data, { imageMask: img }); + basicCommon(result[0]); + }); + + it('spiral is rectangular', () => { + const result = wordCloud(data, { spiral: 'rectangular' }); + basicCommon(result[0]); + }); +}); + +describe('transform', () => { + it('default', () => { + const result = transform(data, { + fields: ['text', 'value'], + size: [800, 800], + }); + basicCommon(result[0]); + }); + + it('error of fields', () => { + expect(() => { + transform(data, { + fields: [null, null], + size: [800, 800], + }); + }).toThrow(); + }); + + it('fields is not exit', () => { + const result = transform(data, { + fields: ['a', 'b'], + size: [800, 800], + }); + + expect(result[0].text).toBe(undefined); + expect(result[0].value).toBe(undefined); + }); + + it('image mask with size is 0', async () => { + const base64 = + ''; + + const img = await processImageMask(base64); + const result = transform(data, { + fields: ['text', 'value'], + size: [0, 0], + imageMask: img, + }); + + // 画布大小为 0 时,返回的数组应为空数组,但该实现中, + // 最终会往返回的数组中 push 进两个元素,所以最终长度为 2。 + expect(result.length).toBe(2); + }); + + it('timeInterval is 0', () => { + const result = transform(data, { + fields: ['text', 'value'], + size: [800, 800], + timeInterval: 0, + }); + + // 时间间隔为 0 ,也就是说不给程序运行的时间, + // 所以最终返回的是空数据,只包含两个占位符数据。 + expect(result.length).toBe(2); + }); + + it('padding is 0', () => { + const result = transform(data, { + fields: ['text', 'value'], + size: [800, 800], + padding: 0, + }); + basicCommon(result[0]); + }); + + it('rotate is 0', () => { + const result = transform(data, { + fields: ['text', 'value'], + size: [800, 800], + rotate: 0, + }); + basicCommon(result[0]); + }); }); diff --git a/src/plots/word-cloud/shapes/word-cloud.ts b/src/plots/word-cloud/shapes/word-cloud.ts index 501ccfb22d..41ea50a74d 100644 --- a/src/plots/word-cloud/shapes/word-cloud.ts +++ b/src/plots/word-cloud/shapes/word-cloud.ts @@ -18,7 +18,7 @@ registerShape('point', 'word-cloud', { }, }); const rotate = cfg.data.rotate; - if (rotate) { + if (typeof rotate === 'number') { Util.rotate(shape, (rotate * Math.PI) / 180); } @@ -35,7 +35,7 @@ function getTextAttrs(cfg: Config): ShapeAttrs { textAlign: 'center', fontFamily: cfg.data.font, fontWeight: cfg.data.weight, - fill: cfg.color || cfg.defaultStyle?.stroke, + fill: cfg.color, textBaseline: 'alphabetic', }; } diff --git a/src/plots/word-cloud/types.ts b/src/plots/word-cloud/types.ts index aeb9214400..721a85a1ff 100644 --- a/src/plots/word-cloud/types.ts +++ b/src/plots/word-cloud/types.ts @@ -3,11 +3,14 @@ import { Options } from '../../types'; type FontWeight = ShapeAttrs['fontWeight']; -export interface DataItem { +interface Row { /** 文本内容 */ text: string; /** 该文本所占权重 */ value: number; +} + +export type DataItem = Row & { /** 字体 */ font?: string; /** 字体样式 */ @@ -30,21 +33,21 @@ export interface DataItem { x?: number; /** y 轴坐标 */ y?: number; -} +}; /** 词云字体样式 */ interface WordStyle { /** 词云的字体, 当为函数时,其参数是一个经过处理之后的数据元素的值 */ - readonly fontFamily?: string | ((row: DataItem) => string); + readonly fontFamily?: string | ((row: Row) => string); /** 设置字体的粗细, 当为函数时,其参数是一个经过处理之后的数据元素的值 */ - readonly fontWeight?: FontWeight | ((row: DataItem) => FontWeight); + readonly fontWeight?: FontWeight | ((row: Row) => FontWeight); /** * 每个单词所占的盒子的内边距,默认为 1。 越大单词之间的间隔越大。 * 当为函数时,其参数是一个经过处理之后的数据元素的值 */ - readonly padding?: number | ((row: DataItem) => number); + readonly padding?: number | ((row: Row) => number); /** 字体的大小范围,当为函数时,其参数是一个经过处理之后的数据元素的值 */ - readonly fontSize?: [number, number] | ((row: DataItem) => number); + readonly fontSize?: [number, number] | ((row: Row) => number); /** 旋转的最小角度和最大角度 默认 [0, 90] */ readonly rotation?: [number, number]; /** 旋转实际的步数,越大可能旋转角度越小, 默认是 2 */ diff --git a/src/plots/word-cloud/utils.ts b/src/plots/word-cloud/utils.ts index 28cdf1e8b5..ddd27fd0ef 100644 --- a/src/plots/word-cloud/utils.ts +++ b/src/plots/word-cloud/utils.ts @@ -208,10 +208,6 @@ function resolveRotate(options: WordCloudOptions) { * @param numbers */ function min(numbers: number[]) { - if (numbers.length === 0) { - throw new Error('min requires at least one data point'); - } - return Math.min(...numbers); } @@ -221,9 +217,5 @@ function min(numbers: number[]) { * @param numbers */ function max(numbers: number[]) { - if (numbers.length === 0) { - throw new Error('min requires at least one data point'); - } - return Math.max(...numbers); } diff --git a/src/utils/transform/word-cloud.ts b/src/utils/transform/word-cloud.ts index 970983cd48..e755c85147 100644 --- a/src/utils/transform/word-cloud.ts +++ b/src/utils/transform/word-cloud.ts @@ -1,9 +1,10 @@ import { deepMix, assign, isString } from '@antv/util'; +import { DataItem } from '../../plots/word-cloud/types'; import { Data } from '../../types'; type FontWeight = number | 'normal' | 'bold' | 'bolder' | 'lighter'; -export interface DataItem { +interface Row { /** 文本内容 */ text: string; /** 该文本所占权重 */ @@ -11,13 +12,13 @@ export interface DataItem { } export interface Options { - fields?: [string, string]; - font?: string | ((row: DataItem) => string); - fontSize?: number | ((row: DataItem) => number); - fontWeight?: FontWeight | ((row: DataItem) => FontWeight); - rotate?: number | ((row: DataItem) => number); - padding?: number | ((row: DataItem) => number); - size?: [number, number]; + fields: [string, string]; + size: [number, number]; + font?: string | ((row: Row) => string); + fontSize?: number | ((row: Row) => number); + fontWeight?: FontWeight | ((row: Row) => FontWeight); + rotate?: number | ((row: Row) => number); + padding?: number | ((row: Row) => number); spiral?: 'archimedean' | 'rectangular' | ((size: [number, number]) => (t: number) => number[]); timeInterval?: number; imageMask?: HTMLImageElement; @@ -40,19 +41,18 @@ const DEFAULT_OPTIONS: Options = { * 计算后的数据对象 * @param options */ -export function wordCloud(data: Data, options?: Options) { - return transform(data, options); +export function wordCloud(data: Data, options?: Partial) { + // 混入默认配置 + options = assign({} as Options, DEFAULT_OPTIONS, options); + return transform(data, options as Options); } -function transform(data: Data, options?: Options) { +export function transform(data: Data, options: Options) { // 深拷贝 data = deepMix([], data); - options = assign({} as Options, DEFAULT_OPTIONS, options); const layout = tagCloud(); ['font', 'fontSize', 'fontWeight', 'padding', 'rotate', 'size', 'spiral', 'timeInterval'].forEach((key) => { - // @ts-ignore - if (options[key]) { - // @ts-ignore + if (options[key] !== undefined && options[key] !== null) { layout[key](options[key]); } }); @@ -96,7 +96,7 @@ function transform(data: Data, options?: Options) { opacity: 0, }); - return tags; + return tags as DataItem[]; } /* @@ -326,24 +326,20 @@ const spirals = { function tagCloud() { let size = [256, 256], - text = cloudText, font = cloudFont, fontSize = cloudFontSize, - fontStyle = cloudFontNormal, fontWeight = cloudFontNormal, rotate = cloudRotate, padding = cloudPadding, spiral = archimedeanSpiral, words: any = [], - timeInterval = Infinity, - random = Math.random, - canvas = cloudCanvas; + timeInterval = Infinity; + const random = Math.random; + const text = cloudText; + const fontStyle = cloudFontNormal; + const canvas = cloudCanvas; const cloud: any = {}; - cloud.canvas = function (_) { - return arguments.length ? ((canvas = functor(_)), cloud) : canvas; - }; - cloud.start = function () { const [width, height] = size; const contextAndRatio = getContext(canvas()), @@ -502,51 +498,39 @@ function tagCloud() { }; cloud.timeInterval = function (_) { - return arguments.length ? ((timeInterval = _ == null ? Infinity : _), cloud) : timeInterval; + timeInterval = _ == null ? Infinity : _; }; cloud.words = function (_) { - return arguments.length ? ((words = _), cloud) : words; + words = _; }; cloud.size = function (_) { - return arguments.length ? ((size = [+_[0], +_[1]]), cloud) : size; + size = [+_[0], +_[1]]; }; cloud.font = function (_) { - return arguments.length ? ((font = functor(_)), cloud) : font; - }; - - cloud.fontStyle = function (_) { - return arguments.length ? ((fontStyle = functor(_)), cloud) : fontStyle; + font = functor(_); }; cloud.fontWeight = function (_) { - return arguments.length ? ((fontWeight = functor(_)), cloud) : fontWeight; + fontWeight = functor(_); }; cloud.rotate = function (_) { - return arguments.length ? ((rotate = functor(_)), cloud) : rotate; - }; - - cloud.text = function (_) { - return arguments.length ? ((text = functor(_)), cloud) : text; + rotate = functor(_); }; cloud.spiral = function (_) { - return arguments.length ? ((spiral = spirals[_] || _), cloud) : spiral; + spiral = spirals[_] || _; }; cloud.fontSize = function (_) { - return arguments.length ? ((fontSize = functor(_)), cloud) : fontSize; + fontSize = functor(_); }; cloud.padding = function (_) { - return arguments.length ? ((padding = functor(_)), cloud) : padding; - }; - - cloud.random = function (_) { - return arguments.length ? ((random = _), cloud) : random; + padding = functor(_); }; return cloud;