diff --git a/plugins/plugin-chart-word-cloud/src/chart/WordCloud.tsx b/plugins/plugin-chart-word-cloud/src/chart/WordCloud.tsx index fc2d4523a7..8eb78df356 100644 --- a/plugins/plugin-chart-word-cloud/src/chart/WordCloud.tsx +++ b/plugins/plugin-chart-word-cloud/src/chart/WordCloud.tsx @@ -1,6 +1,6 @@ import React from 'react'; import cloudLayout, { Word } from 'd3-cloud'; -import { PlainObject, createEncoderFactory, DeriveEncoding } from 'encodable'; +import { PlainObject, createEncoderFactory, DeriveEncoding, Encoder } from 'encodable'; import { SupersetThemeProps, withTheme, seedRandom } from '@superset-ui/core'; export const ROTATION = { @@ -38,6 +38,7 @@ export interface WordCloudProps extends WordCloudVisualProps { export interface WordCloudState { words: Word[]; + scaleFactor: number; } const defaultProps: Required = { @@ -47,12 +48,23 @@ const defaultProps: Required = { type FullWordCloudProps = WordCloudProps & typeof defaultProps & SupersetThemeProps; +const SCALE_FACTOR_STEP = 0.5; +const MAX_SCALE_FACTOR = 3; +// Percentage of top results that will always be displayed. +// Needed to avoid clutter when shrinking a chart with many records. +const TOP_RESULTS_PERCENTAGE = 0.1; + class WordCloud extends React.PureComponent { static defaultProps = defaultProps; // Cannot name it isMounted because of conflict // with React's component function name - isComponentMounted = false; + isComponentMounted: boolean = false; + + state: WordCloudState = { + words: [], + scaleFactor: 1, + }; wordCloudEncoderFactory = createEncoderFactory({ channelTypes: { @@ -77,6 +89,7 @@ class WordCloud extends React.PureComponent super(props); this.state = { words: [], + scaleFactor: 1 }; this.setWords = this.setWords.bind(this); } @@ -111,13 +124,35 @@ class WordCloud extends React.PureComponent } update() { - const { data, width, height, rotation, encoding } = this.props; + const { data, encoding } = this.props; const encoder = this.createEncoder(encoding); encoder.setDomainFromDataset(data); + const sortedData = [...data].sort( + (a, b) => + encoder.channels.fontSize.encodeDatum(b, 0) - encoder.channels.fontSize.encodeDatum(a, 0), + ); + const topResultsCount = Math.max(sortedData.length * TOP_RESULTS_PERCENTAGE, 10); + const topResults = sortedData.slice(0, topResultsCount); + + // Ensure top results are always included in the final word cloud by scaling chart down if needed + this.generateCloud(encoder, 1, (words: Word[]) => + topResults.every((d: PlainObject) => + words.find(({ text }) => encoder.channels.text.getValueFromDatum(d) === text), + ), + ); + } + + generateCloud( + encoder: Encoder, + scaleFactor: number, + isValid: (word: Word[]) => boolean, + ) { + const { data, width, height, rotation } = this.props; + cloudLayout() - .size([width, height]) + .size([width * scaleFactor, height * scaleFactor]) // clone the data because cloudLayout mutates input .words(data.map(d => ({ ...d }))) .padding(5) @@ -128,20 +163,36 @@ class WordCloud extends React.PureComponent ) .fontWeight(d => encoder.channels.fontWeight.encodeDatum(d, 'normal')) .fontSize(d => encoder.channels.fontSize.encodeDatum(d, 0)) - .on('end', this.setWords) + .on('end', (words: Word[]) => { + if (isValid(words) || scaleFactor > MAX_SCALE_FACTOR) { + if (this.isComponentMounted) { + this.setState({ words, scaleFactor }); + } + } else { + this.generateCloud(encoder, scaleFactor + SCALE_FACTOR_STEP, isValid); + } + }) .start(); } render() { + const { scaleFactor } = this.state; const { width, height, encoding } = this.props; const { words } = this.state; const encoder = this.createEncoder(encoding); encoder.channels.color.setDomainFromDataset(words); + const viewBoxWidth = width * scaleFactor; + const viewBoxHeight = height * scaleFactor; + return ( - - + + {words.map(w => (