12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103 |
- /* *
- * Experimental Highcharts module which enables visualization of a word cloud.
- *
- * (c) 2016-2019 Highsoft AS
- *
- * Authors: Jon Arild Nygard
- *
- * License: www.highcharts.com/license
- */
- 'use strict';
- import H from '../parts/Globals.js';
- import drawPoint from '../mixins/draw-point.js';
- import polygon from '../mixins/polygon.js';
- import '../parts/Series.js';
- var extend = H.extend,
- isArray = H.isArray,
- isNumber = H.isNumber,
- isObject = H.isObject,
- merge = H.merge,
- noop = H.noop,
- find = H.find,
- getBoundingBoxFromPolygon = polygon.getBoundingBoxFromPolygon,
- getPolygon = polygon.getPolygon,
- isPolygonsColliding = polygon.isPolygonsColliding,
- movePolygon = polygon.movePolygon,
- Series = H.Series;
- /**
- * Detects if there is a collision between two rectangles.
- *
- * @private
- * @function isRectanglesIntersecting
- *
- * @param {object} r1
- * First rectangle.
- *
- * @param {object} r2
- * Second rectangle.
- *
- * @return {boolean}
- * Returns true if the rectangles overlap.
- */
- function isRectanglesIntersecting(r1, r2) {
- return !(
- r2.left > r1.right ||
- r2.right < r1.left ||
- r2.top > r1.bottom ||
- r2.bottom < r1.top
- );
- }
- /**
- * Detects if a word collides with any previously placed words.
- *
- * @private
- * @function intersectsAnyWord
- *
- * @param {Highcharts.Point} point
- * Point which the word is connected to.
- *
- * @param {Array<Highcharts.Point>} points
- * Previously placed points to check against.
- *
- * @return {boolean}
- * Returns true if there is collision.
- */
- function intersectsAnyWord(point, points) {
- var intersects = false,
- rect = point.rect,
- polygon = point.polygon,
- lastCollidedWith = point.lastCollidedWith,
- isIntersecting = function (p) {
- var result = isRectanglesIntersecting(rect, p.rect);
- if (result && (point.rotation % 90 || p.roation % 90)) {
- result = isPolygonsColliding(
- polygon,
- p.polygon
- );
- }
- return result;
- };
- // If the point has already intersected a different point, chances are they
- // are still intersecting. So as an enhancement we check this first.
- if (lastCollidedWith) {
- intersects = isIntersecting(lastCollidedWith);
- // If they no longer intersects, remove the cache from the point.
- if (!intersects) {
- delete point.lastCollidedWith;
- }
- }
- // If not already found, then check if we can find a point that is
- // intersecting.
- if (!intersects) {
- intersects = !!find(points, function (p) {
- var result = isIntersecting(p);
- if (result) {
- point.lastCollidedWith = p;
- }
- return result;
- });
- }
- return intersects;
- }
- /**
- * Gives a set of cordinates for an Archimedian Spiral.
- *
- * @private
- * @function archimedeanSpiral
- *
- * @param {number} attempt
- * How far along the spiral we have traversed.
- *
- * @param {object} params
- * Additional parameters.
- *
- * @param {object} params.field
- * Size of field.
- *
- * @return {boolean|object}
- * Resulting coordinates, x and y. False if the word should be dropped
- * from the visualization.
- */
- function archimedeanSpiral(attempt, params) {
- var field = params.field,
- result = false,
- maxDelta = (field.width * field.width) + (field.height * field.height),
- t = attempt * 0.8; // 0.2 * 4 = 0.8. Enlarging the spiral.
- // Emergency brake. TODO make spiralling logic more foolproof.
- if (attempt <= 10000) {
- result = {
- x: t * Math.cos(t),
- y: t * Math.sin(t)
- };
- if (!(Math.min(Math.abs(result.x), Math.abs(result.y)) < maxDelta)) {
- result = false;
- }
- }
- return result;
- }
- /**
- * Gives a set of cordinates for an rectangular spiral.
- *
- * @private
- * @function squareSpiral
- *
- * @param {number} attempt
- * How far along the spiral we have traversed.
- *
- * @param {object} params
- * Additional parameters.
- *
- * @return {boolean|object}
- * Resulting coordinates, x and y. False if the word should be dropped
- * from the visualization.
- */
- function squareSpiral(attempt) {
- var a = attempt * 4,
- k = Math.ceil((Math.sqrt(a) - 1) / 2),
- t = 2 * k + 1,
- m = Math.pow(t, 2),
- isBoolean = function (x) {
- return typeof x === 'boolean';
- },
- result = false;
- t -= 1;
- if (attempt <= 10000) {
- if (isBoolean(result) && a >= m - t) {
- result = {
- x: k - (m - a),
- y: -k
- };
- }
- m -= t;
- if (isBoolean(result) && a >= m - t) {
- result = {
- x: -k,
- y: -k + (m - a)
- };
- }
- m -= t;
- if (isBoolean(result)) {
- if (a >= m - t) {
- result = {
- x: -k + (m - a),
- y: k
- };
- } else {
- result = {
- x: k,
- y: k - (m - a - t)
- };
- }
- }
- result.x *= 5;
- result.y *= 5;
- }
- return result;
- }
- /**
- * Gives a set of cordinates for an rectangular spiral.
- *
- * @private
- * @function rectangularSpiral
- *
- * @param {number} attempt
- * How far along the spiral we have traversed.
- *
- * @param {object} params
- * Additional parameters.
- *
- * @return {boolean|object}
- * Resulting coordinates, x and y. False if the word should be dropped
- * from the visualization.
- */
- function rectangularSpiral(attempt, params) {
- var result = squareSpiral(attempt, params),
- field = params.field;
- if (result) {
- result.x *= field.ratioX;
- result.y *= field.ratioY;
- }
- return result;
- }
- /**
- * @private
- * @function getRandomPosition
- *
- * @param {number} size
- *
- * @return {number}
- */
- function getRandomPosition(size) {
- return Math.round((size * (Math.random() + 0.5)) / 2);
- }
- /**
- * Calculates the proper scale to fit the cloud inside the plotting area.
- *
- * @private
- * @function getScale
- *
- * @param {number} targetWidth
- * Width of target area.
- *
- * @param {number} targetHeight
- * Height of target area.
- *
- * @param {object} field
- * The playing field.
- *
- * @param {Highcharts.Series} series
- * Series object.
- *
- * @return {number}
- * Returns the value to scale the playing field up to the size of the
- * target area.
- */
- function getScale(targetWidth, targetHeight, field) {
- var height = Math.max(Math.abs(field.top), Math.abs(field.bottom)) * 2,
- width = Math.max(Math.abs(field.left), Math.abs(field.right)) * 2,
- scaleX = width > 0 ? 1 / width * targetWidth : 1,
- scaleY = height > 0 ? 1 / height * targetHeight : 1;
- return Math.min(scaleX, scaleY);
- }
- /**
- * Calculates what is called the playing field. The field is the area which all
- * the words are allowed to be positioned within. The area is proportioned to
- * match the target aspect ratio.
- *
- * @private
- * @function getPlayingField
- *
- * @param {number} targetWidth
- * Width of the target area.
- *
- * @param {number} targetHeight
- * Height of the target area.
- *
- * @param {Array<Highcharts.Point>} data
- * Array of points.
- *
- * @param {object} data.dimensions
- * The height and width of the word.
- *
- * @return {object}
- * The width and height of the playing field.
- */
- function getPlayingField(
- targetWidth,
- targetHeight,
- data
- ) {
- var info = data.reduce(function (obj, point) {
- var dimensions = point.dimensions,
- x = Math.max(dimensions.width, dimensions.height);
- // Find largest height.
- obj.maxHeight = Math.max(obj.maxHeight, dimensions.height);
- // Find largest width.
- obj.maxWidth = Math.max(obj.maxWidth, dimensions.width);
- // Sum up the total maximum area of all the words.
- obj.area += x * x;
- return obj;
- }, {
- maxHeight: 0,
- maxWidth: 0,
- area: 0
- }),
- /**
- * Use largest width, largest height, or root of total area to give size
- * to the playing field.
- */
- x = Math.max(
- info.maxHeight, // Have enough space for the tallest word
- info.maxWidth, // Have enough space for the broadest word
- // Adjust 15% to account for close packing of words
- Math.sqrt(info.area) * 0.85
- ),
- ratioX = targetWidth > targetHeight ? targetWidth / targetHeight : 1,
- ratioY = targetHeight > targetWidth ? targetHeight / targetWidth : 1;
- return {
- width: x * ratioX,
- height: x * ratioY,
- ratioX: ratioX,
- ratioY: ratioY
- };
- }
- /**
- * Calculates a number of degrees to rotate, based upon a number of orientations
- * within a range from-to.
- *
- * @private
- * @function getRotation
- *
- * @param {number} orientations
- * Number of orientations.
- *
- * @param {number} index
- * Index of point, used to decide orientation.
- *
- * @param {number} from
- * The smallest degree of rotation.
- *
- * @param {number} to
- * The largest degree of rotation.
- *
- * @return {boolean|number}
- * Returns the resulting rotation for the word. Returns false if invalid
- * input parameters.
- */
- function getRotation(orientations, index, from, to) {
- var result = false, // Default to false
- range,
- intervals,
- orientation;
- // Check if we have valid input parameters.
- if (
- isNumber(orientations) &&
- isNumber(index) &&
- isNumber(from) &&
- isNumber(to) &&
- orientations > -1 &&
- index > -1 &&
- to > from
- ) {
- range = to - from;
- intervals = range / (orientations - 1);
- orientation = index % orientations;
- result = from + (orientation * intervals);
- }
- return result;
- }
- /**
- * Calculates the spiral positions and store them in scope for quick access.
- *
- * @private
- * @function getSpiral
- *
- * @param {Function} fn
- * The spiral function.
- *
- * @param {object} params
- * Additional parameters for the spiral.
- *
- * @return {Function}
- * Function with access to spiral positions.
- */
- function getSpiral(fn, params) {
- var length = 10000,
- i,
- arr = [];
- for (i = 1; i < length; i++) {
- arr.push(fn(i, params));
- }
- return function (attempt) {
- return attempt <= length ? arr[attempt - 1] : false;
- };
- }
- /**
- * Detects if a word is placed outside the playing field.
- *
- * @private
- * @function outsidePlayingField
- *
- * @param {Highcharts.Point} point
- * Point which the word is connected to.
- *
- * @param {object} field
- * The width and height of the playing field.
- *
- * @return {boolean}
- * Returns true if the word is placed outside the field.
- */
- function outsidePlayingField(rect, field) {
- var playingField = {
- left: -(field.width / 2),
- right: field.width / 2,
- top: -(field.height / 2),
- bottom: field.height / 2
- };
- return !(
- playingField.left < rect.left &&
- playingField.right > rect.right &&
- playingField.top < rect.top &&
- playingField.bottom > rect.bottom
- );
- }
- /**
- * Check if a point intersects with previously placed words, or if it goes
- * outside the field boundaries. If a collision, then try to adjusts the
- * position.
- *
- * @private
- * @function intersectionTesting
- *
- * @param {Highcharts.Point} point
- * Point to test for intersections.
- *
- * @param {object} options
- * Options object.
- *
- * @return {boolean|object}
- * Returns an object with how much to correct the positions. Returns
- * false if the word should not be placed at all.
- */
- function intersectionTesting(point, options) {
- var placed = options.placed,
- field = options.field,
- rectangle = options.rectangle,
- polygon = options.polygon,
- spiral = options.spiral,
- attempt = 1,
- delta = {
- x: 0,
- y: 0
- },
- // Make a copy to update values during intersection testing.
- rect = point.rect = extend({}, rectangle);
- point.polygon = polygon;
- point.rotation = options.rotation;
- /* while w intersects any previously placed words:
- do {
- move w a little bit along a spiral path
- } while any part of w is outside the playing field and
- the spiral radius is still smallish */
- while (
- delta !== false &&
- (
- intersectsAnyWord(point, placed) ||
- outsidePlayingField(rect, field)
- )
- ) {
- delta = spiral(attempt);
- if (isObject(delta)) {
- // Update the DOMRect with new positions.
- rect.left = rectangle.left + delta.x;
- rect.right = rectangle.right + delta.x;
- rect.top = rectangle.top + delta.y;
- rect.bottom = rectangle.bottom + delta.y;
- point.polygon = movePolygon(delta.x, delta.y, polygon);
- }
- attempt++;
- }
- return delta;
- }
- /**
- * Extends the playing field to have enough space to fit a given word.
- *
- * @private
- * @function extendPlayingField
- *
- * @param {object} field
- * The width, height and ratios of a playing field.
- *
- * @param {object} rectangle
- * The bounding box of the word to add space for.
- *
- * @return {object}
- * Returns the extended playing field with updated height and width.
- */
- function extendPlayingField(field, rectangle) {
- var height, width, ratioX, ratioY, x, extendWidth, extendHeight, result;
- if (isObject(field) && isObject(rectangle)) {
- height = (rectangle.bottom - rectangle.top);
- width = (rectangle.right - rectangle.left);
- ratioX = field.ratioX;
- ratioY = field.ratioY;
- // Use the same variable to extend both the height and width.
- x = ((width * ratioX) > (height * ratioY)) ? width : height;
- // Multiply variable with ratios to preserve aspect ratio.
- extendWidth = x * ratioX;
- extendHeight = x * ratioY;
- // Calculate the size of the new field after adding space for the word.
- result = merge(field, {
- // Add space on the left and right.
- width: field.width + (extendWidth * 2),
- // Add space on the top and bottom.
- height: field.height + (extendHeight * 2)
- });
- } else {
- result = field;
- }
- // Return the new extended field.
- return result;
- }
- /**
- * If a rectangle is outside a give field, then the boundaries of the field is
- * adjusted accordingly. Modifies the field object which is passed as the first
- * parameter.
- *
- * @private
- * @function updateFieldBoundaries
- *
- * @param {object} field
- * The bounding box of a playing field.
- *
- * @param {object} placement
- * The bounding box for a placed point.
- *
- * @return {object}
- * Returns a modified field object.
- */
- function updateFieldBoundaries(field, rectangle) {
- // TODO improve type checking.
- if (!isNumber(field.left) || field.left > rectangle.left) {
- field.left = rectangle.left;
- }
- if (!isNumber(field.right) || field.right < rectangle.right) {
- field.right = rectangle.right;
- }
- if (!isNumber(field.top) || field.top > rectangle.top) {
- field.top = rectangle.top;
- }
- if (!isNumber(field.bottom) || field.bottom < rectangle.bottom) {
- field.bottom = rectangle.bottom;
- }
- return field;
- }
- /**
- * A word cloud is a visualization of a set of words, where the size and
- * placement of a word is determined by how it is weighted.
- *
- * @sample highcharts/demo/wordcloud
- * Word Cloud chart
- *
- * @extends plotOptions.column
- * @excluding allAreas, boostThreshold, clip, colorAxis, compare,
- * compareBase, crisp, cropTreshold, dataGrouping, dataLabels,
- * depth, edgeColor, findNearestPointBy, getExtremesFromAll,
- * grouping, groupPadding, groupZPadding, joinBy, maxPointWidth,
- * minPointLength, navigatorOptions, negativeColor, pointInterval,
- * pointIntervalUnit, pointPadding, pointPlacement, pointRange,
- * pointStart, pointWidth, pointStart, pointWidth, shadow,
- * showCheckbox, showInNavigator, softThreshold, stacking,
- * threshold, zoneAxis, zones
- * @product highcharts
- * @since 6.0.0
- * @optionparent plotOptions.wordcloud
- */
- var wordCloudOptions = {
- /**
- * If there is no space for a word on the playing field, then this option
- * will allow the playing field to be extended to fit the word. If false
- * then the word will be dropped from the visualization.
- *
- * NB! This option is currently not decided to be published in the API, and
- * is therefore marked as private.
- *
- * @private
- */
- allowExtendPlayingField: true,
- animation: {
- duration: 500
- },
- borderWidth: 0,
- clip: false, // Something goes wrong with clip. // @todo fix this
- colorByPoint: true,
- /**
- * A threshold determining the minimum font size that can be applied to a
- * word.
- */
- minFontSize: 1,
- /**
- * The word with the largest weight will have a font size equal to this
- * value. The font size of a word is the ratio between its weight and the
- * largest occuring weight, multiplied with the value of maxFontSize.
- */
- maxFontSize: 25,
- /**
- * This option decides which algorithm is used for placement, and rotation
- * of a word. The choice of algorith is therefore a crucial part of the
- * resulting layout of the wordcloud. It is possible for users to add their
- * own custom placement strategies for use in word cloud. Read more about it
- * in our
- * [documentation](https://www.highcharts.com/docs/chart-and-series-types/word-cloud-series#custom-placement-strategies)
- *
- * @validvalue: ["center", "random"]
- */
- placementStrategy: 'center',
- /**
- * Rotation options for the words in the wordcloud.
- *
- * @sample highcharts/plotoptions/wordcloud-rotation
- * Word cloud with rotation
- */
- rotation: {
- /**
- * The smallest degree of rotation for a word.
- */
- from: 0,
- /**
- * The number of possible orientations for a word, within the range of
- * `rotation.from` and `rotation.to`.
- */
- orientations: 2,
- /**
- * The largest degree of rotation for a word.
- */
- to: 90
- },
- showInLegend: false,
- /**
- * Spiral used for placing a word after the initial position experienced a
- * collision with either another word or the borders.
- * It is possible for users to add their own custom spiralling algorithms
- * for use in word cloud. Read more about it in our
- * [documentation](https://www.highcharts.com/docs/chart-and-series-types/word-cloud-series#custom-spiralling-algorithm)
- *
- * @validvalue: ["archimedean", "rectangular", "square"]
- */
- spiral: 'rectangular',
- /**
- * CSS styles for the words.
- *
- * @type {Highcharts.CSSObject}
- * @default {"fontFamily":"sans-serif", "fontWeight": "900"}
- */
- style: {
- /** @ignore-option */
- fontFamily: 'sans-serif',
- /** @ignore-option */
- fontWeight: '900'
- },
- tooltip: {
- followPointer: true,
- pointFormat: '<span style="color:{point.color}">\u25CF</span> {series.name}: <b>{point.weight}</b><br/>'
- }
- };
- // Properties of the WordCloud series.
- var wordCloudSeries = {
- animate: Series.prototype.animate,
- animateDrilldown: noop,
- animateDrillupFrom: noop,
- bindAxes: function () {
- var wordcloudAxis = {
- endOnTick: false,
- gridLineWidth: 0,
- lineWidth: 0,
- maxPadding: 0,
- startOnTick: false,
- title: null,
- tickPositions: []
- };
- Series.prototype.bindAxes.call(this);
- extend(this.yAxis.options, wordcloudAxis);
- extend(this.xAxis.options, wordcloudAxis);
- },
- pointAttribs: function (point, state) {
- var attribs = H.seriesTypes.column.prototype
- .pointAttribs.call(this, point, state);
- delete attribs.stroke;
- delete attribs['stroke-width'];
- return attribs;
- },
- /**
- * Calculates the fontSize of a word based on its weight.
- *
- * @private
- * @function Highcharts.Series#deriveFontSize
- *
- * @param {number} [relativeWeight=0]
- * The weight of the word, on a scale 0-1.
- *
- * @param {number} [maxFontSize=1]
- * The maximum font size of a word.
- *
- * @param {number} [minFontSize=1]
- * The minimum font size of a word.
- *
- * @return {number}
- * Returns the resulting fontSize of a word. If minFontSize is
- * larger then maxFontSize the result will equal minFontSize.
- */
- deriveFontSize: function deriveFontSize(
- relativeWeight,
- maxFontSize,
- minFontSize
- ) {
- var weight = isNumber(relativeWeight) ? relativeWeight : 0,
- max = isNumber(maxFontSize) ? maxFontSize : 1,
- min = isNumber(minFontSize) ? minFontSize : 1;
- return Math.floor(Math.max(min, weight * max));
- },
- drawPoints: function () {
- var series = this,
- hasRendered = series.hasRendered,
- xAxis = series.xAxis,
- yAxis = series.yAxis,
- chart = series.chart,
- group = series.group,
- options = series.options,
- animation = options.animation,
- allowExtendPlayingField = options.allowExtendPlayingField,
- renderer = chart.renderer,
- testElement = renderer.text().add(group),
- placed = [],
- placementStrategy = series.placementStrategy[
- options.placementStrategy
- ],
- spiral,
- rotation = options.rotation,
- scale,
- weights = series.points
- .map(function (p) {
- return p.weight;
- }),
- maxWeight = Math.max.apply(null, weights),
- data = series.points
- .sort(function (a, b) {
- return b.weight - a.weight; // Sort descending
- }),
- field;
- // Get the dimensions for each word.
- // Used in calculating the playing field.
- data.forEach(function (point) {
- var relativeWeight = 1 / maxWeight * point.weight,
- fontSize = series.deriveFontSize(
- relativeWeight,
- options.maxFontSize,
- options.minFontSize
- ),
- css = extend({
- fontSize: fontSize + 'px'
- }, options.style),
- bBox;
- testElement.css(css).attr({
- x: 0,
- y: 0,
- text: point.name
- });
- bBox = testElement.getBBox(true);
- point.dimensions = {
- height: bBox.height,
- width: bBox.width
- };
- });
- // Calculate the playing field.
- field = getPlayingField(xAxis.len, yAxis.len, data);
- spiral = getSpiral(series.spirals[options.spiral], {
- field: field
- });
- // Draw all the points.
- data.forEach(function (point) {
- var relativeWeight = 1 / maxWeight * point.weight,
- fontSize = series.deriveFontSize(
- relativeWeight,
- options.maxFontSize,
- options.minFontSize
- ),
- css = extend({
- fontSize: fontSize + 'px'
- }, options.style),
- placement = placementStrategy(point, {
- data: data,
- field: field,
- placed: placed,
- rotation: rotation
- }),
- attr = extend(
- series.pointAttribs(point, point.selected && 'select'),
- {
- align: 'center',
- 'alignment-baseline': 'middle',
- x: placement.x,
- y: placement.y,
- text: point.name,
- rotation: placement.rotation
- }
- ),
- polygon = getPolygon(
- placement.x,
- placement.y,
- point.dimensions.width,
- point.dimensions.height,
- placement.rotation
- ),
- rectangle = getBoundingBoxFromPolygon(polygon),
- delta = intersectionTesting(point, {
- rectangle: rectangle,
- polygon: polygon,
- field: field,
- placed: placed,
- spiral: spiral,
- rotation: placement.rotation
- }),
- animate;
- // If there is no space for the word, extend the playing field.
- if (!delta && allowExtendPlayingField) {
- // Extend the playing field to fit the word.
- field = extendPlayingField(field, rectangle);
- // Run intersection testing one more time to place the word.
- delta = intersectionTesting(point, {
- rectangle: rectangle,
- polygon: polygon,
- field: field,
- placed: placed,
- spiral: spiral,
- rotation: placement.rotation
- });
- }
- // Check if point was placed, if so delete it, otherwise place it on
- // the correct positions.
- if (isObject(delta)) {
- attr.x += delta.x;
- attr.y += delta.y;
- rectangle.left += delta.x;
- rectangle.right += delta.x;
- rectangle.top += delta.y;
- rectangle.bottom += delta.y;
- field = updateFieldBoundaries(field, rectangle);
- placed.push(point);
- point.isNull = false;
- } else {
- point.isNull = true;
- }
- if (animation) {
- // Animate to new positions
- animate = {
- x: attr.x,
- y: attr.y
- };
- // Animate from center of chart
- if (!hasRendered) {
- attr.x = 0;
- attr.y = 0;
- // or animate from previous position
- } else {
- delete attr.x;
- delete attr.y;
- }
- }
- point.draw({
- animatableAttribs: animate,
- attribs: attr,
- css: css,
- group: group,
- renderer: renderer,
- shapeArgs: undefined,
- shapeType: 'text'
- });
- });
- // Destroy the element after use.
- testElement = testElement.destroy();
- // Scale the series group to fit within the plotArea.
- scale = getScale(xAxis.len, yAxis.len, field);
- series.group.attr({
- scaleX: scale,
- scaleY: scale
- });
- },
- hasData: function () {
- var series = this;
- return (
- isObject(series) &&
- series.visible === true &&
- isArray(series.points) &&
- series.points.length > 0
- );
- },
- // Strategies used for deciding rotation and initial position of a word. To
- // implement a custom strategy, have a look at the function random for
- // example.
- placementStrategy: {
- random: function (point, options) {
- var field = options.field,
- r = options.rotation;
- return {
- x: getRandomPosition(field.width) - (field.width / 2),
- y: getRandomPosition(field.height) - (field.height / 2),
- rotation: getRotation(r.orientations, point.index, r.from, r.to)
- };
- },
- center: function (point, options) {
- var r = options.rotation;
- return {
- x: 0,
- y: 0,
- rotation: getRotation(r.orientations, point.index, r.from, r.to)
- };
- }
- },
- pointArrayMap: ['weight'],
- // Spirals used for placing a word after the initial position experienced a
- // collision with either another word or the borders. To implement a custom
- // spiral, look at the function archimedeanSpiral for example.
- spirals: {
- 'archimedean': archimedeanSpiral,
- 'rectangular': rectangularSpiral,
- 'square': squareSpiral
- },
- utils: {
- extendPlayingField: extendPlayingField,
- getRotation: getRotation,
- isPolygonsColliding: isPolygonsColliding,
- rotate2DToOrigin: polygon.rotate2DToOrigin,
- rotate2DToPoint: polygon.rotate2DToPoint
- },
- getPlotBox: function () {
- var series = this,
- chart = series.chart,
- inverted = chart.inverted,
- // Swap axes for inverted (#2339)
- xAxis = series[(inverted ? 'yAxis' : 'xAxis')],
- yAxis = series[(inverted ? 'xAxis' : 'yAxis')],
- width = xAxis ? xAxis.len : chart.plotWidth,
- height = yAxis ? yAxis.len : chart.plotHeight,
- x = xAxis ? xAxis.left : chart.plotLeft,
- y = yAxis ? yAxis.top : chart.plotTop;
- return {
- translateX: x + (width / 2),
- translateY: y + (height / 2),
- scaleX: 1, // #1623
- scaleY: 1
- };
- }
- };
- // Properties of the Sunburst series.
- var wordCloudPoint = {
- draw: drawPoint,
- shouldDraw: function shouldDraw() {
- var point = this;
- return !point.isNull;
- },
- weight: 1
- };
- /**
- * A `wordcloud` series. If the [type](#series.wordcloud.type) option is not
- * specified, it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.wordcloud
- * @product highcharts
- * @apioption series.wordcloud
- */
- /**
- * An array of data points for the series. For the `wordcloud` series type,
- * points can be given in the following ways:
- *
- * 1. An array of arrays with 2 values. In this case, the values correspond to
- * `name,weight`.
- * ```js
- * data: [
- * ['Lorem', 4],
- * ['Ipsum', 1]
- * ]
- * ```
- *
- * 2. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.arearange.turboThreshold), this option is not
- * available.
- * ```js
- * data: [{
- * name: "Lorem",
- * weight: 4
- * }, {
- * name: "Ipsum",
- * weight: 1
- * }]
- * ```
- *
- * @type {Array<Array<string,number>|*>}
- * @extends series.line.data
- * @excluding drilldown, marker, x, y
- * @product highcharts
- * @apioption series.wordcloud.data
- */
- /**
- * The name decides the text for a word.
- *
- * @type {string}
- * @since 6.0.0
- * @product highcharts
- * @apioption series.sunburst.data.name
- */
- /**
- * The weighting of a word. The weight decides the relative size of a word
- * compared to the rest of the collection.
- *
- * @type {number}
- * @since 6.0.0
- * @product highcharts
- * @apioption series.sunburst.data.weight
- */
- /**
- * @private
- * @class
- * @name Highcharts.seriesTypes.wordcloud
- *
- * @augments Highcharts.Series
- */
- H.seriesType(
- 'wordcloud',
- 'column',
- wordCloudOptions,
- wordCloudSeries,
- wordCloudPoint
- );
|