Skip to content

Commit

Permalink
chore(word-cloud): 主要完善词云图的测试用例 (#1745)
Browse files Browse the repository at this point in the history
* chore(word-cloud): 主要完善词云图的测试用例

* refactor(word-cloud/shapes): 删除冗余代码,提高测试覆盖率
  • Loading branch information
zhangzhonghe authored Oct 16, 2020
1 parent b5df7c9 commit f7febd6
Show file tree
Hide file tree
Showing 5 changed files with 168 additions and 83 deletions.
148 changes: 127 additions & 21 deletions __tests__/unit/utils/transform/word-cloud-spec.ts
Original file line number Diff line number Diff line change
@@ -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<DataItem, 'text' | 'value'>;

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 {
Expand All @@ -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();
Expand All @@ -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;
};
Expand All @@ -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]);
});
});
4 changes: 2 additions & 2 deletions src/plots/word-cloud/shapes/word-cloud.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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);
}

Expand All @@ -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',
};
}
15 changes: 9 additions & 6 deletions src/plots/word-cloud/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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;
/** 字体样式 */
Expand All @@ -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 */
Expand Down
8 changes: 0 additions & 8 deletions src/plots/word-cloud/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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);
}

Expand All @@ -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);
}
Loading

0 comments on commit f7febd6

Please sign in to comment.