UNPKG

highcharts

Version:
1,579 lines (1,560 loc) 103 kB
// SPDX-License-Identifier: LicenseRef-Highcharts /** * @license Highcharts JS v12.6.0 (2026-04-13) * @module highcharts/modules/export-data * @requires highcharts * @requires highcharts/modules/exporting * * Export data module * * (c) 2010-2026 Highsoft AS * Author: Torstein Hønsi * * A commercial license may be required depending on use. * See www.highcharts.com/license */ (function webpackUniversalModuleDefinition(root, factory) { if(typeof exports === 'object' && typeof module === 'object') module.exports = factory(root["_Highcharts"], root["_Highcharts"]["AST"], root["_Highcharts"]["Chart"]); else if(typeof define === 'function' && define.amd) define("highcharts/modules/export-data", ["highcharts/highcharts"], function (amd1) {return factory(amd1,amd1["AST"],amd1["Chart"]);}); else if(typeof exports === 'object') exports["highcharts/modules/export-data"] = factory(root["_Highcharts"], root["_Highcharts"]["AST"], root["_Highcharts"]["Chart"]); else root["Highcharts"] = factory(root["Highcharts"], root["Highcharts"]["AST"], root["Highcharts"]["Chart"]); })(typeof window === 'undefined' ? this : window, (__WEBPACK_EXTERNAL_MODULE__944__, __WEBPACK_EXTERNAL_MODULE__660__, __WEBPACK_EXTERNAL_MODULE__960__) => { return /******/ (() => { // webpackBootstrap /******/ "use strict"; /******/ var __webpack_modules__ = ({ /***/ 660: /***/ ((module) => { module.exports = __WEBPACK_EXTERNAL_MODULE__660__; /***/ }), /***/ 944: /***/ ((module) => { module.exports = __WEBPACK_EXTERNAL_MODULE__944__; /***/ }), /***/ 960: /***/ ((module) => { module.exports = __WEBPACK_EXTERNAL_MODULE__960__; /***/ }) /******/ }); /************************************************************************/ /******/ // The module cache /******/ var __webpack_module_cache__ = {}; /******/ /******/ // The require function /******/ function __webpack_require__(moduleId) { /******/ // Check if module is in cache /******/ var cachedModule = __webpack_module_cache__[moduleId]; /******/ if (cachedModule !== undefined) { /******/ return cachedModule.exports; /******/ } /******/ // Create a new module (and put it into the cache) /******/ var module = __webpack_module_cache__[moduleId] = { /******/ // no module.id needed /******/ // no module.loaded needed /******/ exports: {} /******/ }; /******/ /******/ // Execute the module function /******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__); /******/ /******/ // Return the exports of the module /******/ return module.exports; /******/ } /******/ /************************************************************************/ /******/ /* webpack/runtime/compat get default export */ /******/ (() => { /******/ // getDefaultExport function for compatibility with non-harmony modules /******/ __webpack_require__.n = (module) => { /******/ var getter = module && module.__esModule ? /******/ () => (module['default']) : /******/ () => (module); /******/ __webpack_require__.d(getter, { a: getter }); /******/ return getter; /******/ }; /******/ })(); /******/ /******/ /* webpack/runtime/define property getters */ /******/ (() => { /******/ // define getter functions for harmony exports /******/ __webpack_require__.d = (exports, definition) => { /******/ for(var key in definition) { /******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) { /******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] }); /******/ } /******/ } /******/ }; /******/ })(); /******/ /******/ /* webpack/runtime/hasOwnProperty shorthand */ /******/ (() => { /******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop)) /******/ })(); /******/ /************************************************************************/ var __webpack_exports__ = {}; // EXPORTS __webpack_require__.d(__webpack_exports__, { "default": () => (/* binding */ export_data_src) }); // EXTERNAL MODULE: external {"amd":["highcharts/highcharts"],"commonjs":["highcharts"],"commonjs2":["highcharts"],"root":["Highcharts"]} var highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_ = __webpack_require__(944); var highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_default = /*#__PURE__*/__webpack_require__.n(highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_); ;// ./code/es-modules/Shared/DownloadURL.js /* * * * (c) 2015-2026 Highsoft AS * Author: Øystein Moseng * * A commercial license may be required depending on use. * See www.highcharts.com/license * * * Mixin for downloading content in the browser * * */ /* * * * Imports * * */ const { isSafari, win, win: { document: doc } } = (highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_default()); /* * * * Constants * * */ const domurl = win.URL || win.webkitURL || win; /* * * * Functions * * */ /** * Convert base64 dataURL to Blob if supported, otherwise returns undefined. * * @internal * @function Highcharts.dataURLtoBlob * * @param {string} dataURL * URL to convert. * * @return {string | undefined} * Blob. */ function dataURLtoBlob(dataURL) { const parts = dataURL .replace(/filename=.*;/, '') .match(/data:([^;]*)(;base64)?,([A-Z+\d\/]+)/i); if (parts && parts.length > 3 && (win.atob) && win.ArrayBuffer && win.Uint8Array && win.Blob && (domurl.createObjectURL)) { // Try to convert data URL to Blob const binStr = win.atob(parts[3]), buf = new win.ArrayBuffer(binStr.length), binary = new win.Uint8Array(buf); for (let i = 0; i < binary.length; ++i) { binary[i] = binStr.charCodeAt(i); } return domurl .createObjectURL(new win.Blob([binary], { 'type': parts[1] })); } } /** * Download a data URL in the browser. Can also take a blob as first param. * * @internal * @function Highcharts.downloadURL * * @param {string | global.URL} dataURL * The dataURL/Blob to download. * @param {string} filename * The name of the resulting file (w/extension). */ function downloadURL(dataURL, filename) { const nav = win.navigator, a = doc.createElement('a'); // IE specific blob implementation // Don't use for normal dataURLs if (typeof dataURL !== 'string' && !(dataURL instanceof String) && nav.msSaveOrOpenBlob) { nav.msSaveOrOpenBlob(dataURL, filename); return; } dataURL = '' + dataURL; if (nav.userAgent.length > 1000 /* RegexLimits.shortLimit */) { throw new Error('Input too long'); } const // Some browsers have limitations for data URL lengths. Try to convert // to Blob or fall back. Edge always needs that blob. isOldEdgeBrowser = /Edge\/\d+/.test(nav.userAgent), // Safari on iOS needs Blob in order to download PDF safariBlob = (isSafari && typeof dataURL === 'string' && dataURL.indexOf('data:application/pdf') === 0); if (safariBlob || isOldEdgeBrowser || dataURL.length > 2000000) { dataURL = dataURLtoBlob(dataURL) || ''; if (!dataURL) { throw new Error('Failed to convert to blob'); } } // Try HTML5 download attr if supported if (typeof a.download !== 'undefined') { a.href = dataURL; a.download = filename; // HTML5 download attribute doc.body.appendChild(a); a.click(); doc.body.removeChild(a); } else { // No download attr, just opening data URI try { if (!win.open(dataURL, 'chart')) { throw new Error('Failed to open window'); } } catch { // If window.open failed, try location.href win.location.href = dataURL; } } } /** * Asynchronously downloads a script from a provided location. * * @internal * @function Highcharts.getScript * * @param {string} scriptLocation * The location for the script to fetch. */ function getScript(scriptLocation) { return new Promise((resolve, reject) => { const head = doc.getElementsByTagName('head')[0], script = doc.createElement('script'); // Set type and location for the script script.type = 'text/javascript'; script.src = scriptLocation; // Resolve in case of a successful script fetching script.onload = () => { resolve(); }; // Reject in case of fail script.onerror = () => { const msg = `Error loading script ${scriptLocation}`; (0,highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_.error)(msg); reject(new Error(msg)); }; // Append the newly created script head.appendChild(script); }); } /** * Get a blob object from content, if blob is supported. * * @internal * @function Highcharts.getBlobFromContent * * @param {string} content * The content to create the blob from. * @param {string} type * The type of the content. * * @return {string | undefined} * The blob object, or undefined if not supported. * * @requires modules/exporting * @requires modules/export-data */ function getBlobFromContent(content, type) { const nav = win.navigator, domurl = win.URL || win.webkitURL || win; try { // MS specific if ((nav.msSaveOrOpenBlob) && win.MSBlobBuilder) { const blob = new win.MSBlobBuilder(); blob.append(content); return blob.getBlob('image/svg+xml'); } return domurl.createObjectURL(new win.Blob(['\uFEFF' + content], // #7084 { type: type })); // eslint-disable-next-line @typescript-eslint/no-unused-vars } catch (e) { // Ignore } } /* * * * Default Export * * */ /** @internal */ const DownloadURL = { dataURLtoBlob, downloadURL, getBlobFromContent, getScript }; /** @internal */ /* harmony default export */ const Shared_DownloadURL = (DownloadURL); // EXTERNAL MODULE: external {"amd":["highcharts/highcharts","AST"],"commonjs":["highcharts","AST"],"commonjs2":["highcharts","AST"],"root":["Highcharts","AST"]} var highcharts_AST_commonjs_highcharts_AST_commonjs2_highcharts_AST_root_Highcharts_AST_ = __webpack_require__(660); var highcharts_AST_commonjs_highcharts_AST_commonjs2_highcharts_AST_root_Highcharts_AST_default = /*#__PURE__*/__webpack_require__.n(highcharts_AST_commonjs_highcharts_AST_commonjs2_highcharts_AST_root_Highcharts_AST_); // EXTERNAL MODULE: external {"amd":["highcharts/highcharts","Chart"],"commonjs":["highcharts","Chart"],"commonjs2":["highcharts","Chart"],"root":["Highcharts","Chart"]} var highcharts_Chart_commonjs_highcharts_Chart_commonjs2_highcharts_Chart_root_Highcharts_Chart_ = __webpack_require__(960); var highcharts_Chart_commonjs_highcharts_Chart_commonjs2_highcharts_Chart_root_Highcharts_Chart_default = /*#__PURE__*/__webpack_require__.n(highcharts_Chart_commonjs_highcharts_Chart_commonjs2_highcharts_Chart_root_Highcharts_Chart_); ;// ./code/es-modules/Extensions/ExportData/ExportDataDefaults.js /* * * * Experimental data export module for Highcharts * * (c) 2010-2026 Highsoft AS * Author: Torstein Hønsi * * A commercial license may be required depending on use. * See www.highcharts.com/license * * * */ /* * * * Constants * * */ /** * @optionparent exporting * @internal */ const exporting = { /** * Caption for the data table. Same as chart title by default. Set to * `false` to disable. * * @sample highcharts/export-data/multilevel-table * Multiple table headers * * @type {boolean | string} * @since 6.0.4 * @requires modules/export-data * @apioption exporting.tableCaption */ /** * Options for exporting data to CSV or Excel, or displaying the data * in a HTML table or a JavaScript structure. * * This module adds data export options to the export menu and provides * functions like `Exporting.getCSV`, `Exporting.getTable`, * `Exporting.getDataRows` and `Exporting.viewData`. * * The XLS converter is limited and only creates a HTML string that is * passed for download, which works but creates a warning before * opening. The workaround for this is to use a third party XLSX * converter, as demonstrated in the sample below. * * @sample highcharts/export-data/categorized/ Categorized data * @sample highcharts/export-data/stock-timeaxis/ Highcharts Stock time axis * @sample highcharts/export-data/xlsx/ * Using a third party XLSX converter * * @since 6.0.0 * @requires modules/export-data */ csv: { /** * Options for annotations in the export-data table. * * @since 8.2.0 * @requires modules/export-data * @requires modules/annotations */ annotations: { /** * The way to mark the separator for annotations * combined in one export-data table cell. * * @since 8.2.0 * @requires modules/annotations */ itemDelimiter: '; ', /** * When several labels are assigned to a specific point, * they will be displayed in one field in the table. * * @sample highcharts/export-data/join-annotations/ * Concatenate point annotations with itemDelimiter set. * * @since 8.2.0 * @requires modules/annotations */ join: false }, /** * Formatter callback for the column headers. Parameters are: * - `item` - The series or axis object) * - `key` - The point key, for example y or z * - `keyLength` - The amount of value keys for this item, for * example a range series has the keys `low` and `high` so the * key length is 2. * * If [useMultiLevelHeaders](#exporting.useMultiLevelHeaders) is * true, columnHeaderFormatter by default returns an object with * columnTitle and topLevelColumnTitle for each key. Columns with * the same topLevelColumnTitle have their titles merged into a * single cell with colspan for table/Excel export. * * If `useMultiLevelHeaders` is false, or for CSV export, it returns * the series name, followed by the key if there is more than one * key. * * For the axis it returns the axis title or "Category" or * "DateTime" by default. * * Return `false` to use Highcharts' proposed header. * * @sample highcharts/export-data/multilevel-table * Multiple table headers * * @type {Function | null} */ columnHeaderFormatter: null, /** * Which date format to use for exported dates on a datetime X axis. * See `Highcharts.dateFormat`. */ dateFormat: '%Y-%m-%d %H:%M:%S', /** * Which decimal point to use for exported CSV. Defaults to the same * as the browser locale, typically `.` (English) or `,` (German, * French etc). * * @type {string | null} * @since 6.0.4 */ decimalPoint: null, /** * The item delimiter in the exported data. Use `;` for direct * exporting to Excel. Defaults to a best guess based on the browser * locale. If the locale _decimal point_ is `,`, the `itemDelimiter` * defaults to `;`, otherwise the `itemDelimiter` defaults to `,`. * * @type {string | null} */ itemDelimiter: null, /** * The line delimiter in the exported data, defaults to a newline. */ lineDelimiter: '\n' }, menuItemDefinitions: { /** * @requires modules/export-data */ downloadCSV: { /** * @see [lang.downloadCSV](#lang.downloadCSV) * @default downloadCSV */ textKey: 'downloadCSV', onclick: function () { this.exporting?.downloadCSV(); } }, /** * @requires modules/export-data */ downloadXLS: { /** * @see [lang.downloadXLS](#lang.downloadXLS) * @default downloadXLS */ textKey: 'downloadXLS', onclick: function () { this.exporting?.downloadXLS(); } }, /** * @requires modules/export-data */ viewData: { /** * @see [lang.viewData](#lang.viewData) * @default viewData */ textKey: 'viewData', onclick: function () { this.exporting?.wrapLoading(this.exporting.toggleDataTable); } } }, /** * Display a message when export is in progress. Uses * [Chart.showLoading()](/class-reference/Highcharts.Chart#showLoading). * * The message can be altered by changing * [lang.exportInProgress](#lang.exportInProgress). * * @since 11.3.0 * @requires modules/export-data */ showExportInProgress: true, /** * Show a HTML table below the chart with the chart's current data. * * @sample highcharts/export-data/showtable/ * Show the table * @sample highcharts/studies/exporting-table-html * Experiment with putting the table inside the subtitle to * allow exporting it. * * @since 6.0.0 * @requires modules/export-data */ showTable: false, /** * Use multi level headers in data table. If [csv.columnHeaderFormatter * ](#exporting.csv.columnHeaderFormatter) is defined, it has to return * objects in order for multi level headers to work. * * @sample highcharts/export-data/multilevel-table * Multiple table headers * * @since 6.0.4 * @requires modules/export-data */ useMultiLevelHeaders: true, /** * If using multi level table headers, use rowspans for headers that * have only one level. * * @sample highcharts/export-data/multilevel-table * Multiple table headers * * @since 6.0.4 * @requires modules/export-data */ useRowspanHeaders: true }; // TODO: no need to be a partial when Options are fully optional. /** * @optionparent lang * @internal */ const lang = { /** * The text for the menu item. * * @since 6.0.0 * @requires modules/export-data */ downloadCSV: 'Download CSV', /** * The text for the menu item. * * @since 6.0.0 * @requires modules/export-data */ downloadXLS: 'Download XLS', /** * The text for exported table. * * @since 8.1.0 * @requires modules/export-data */ exportData: { /** * The annotation column title. */ annotationHeader: 'Annotations', /** * The category column title. */ categoryHeader: 'Category', /** * The category column title when axis type set to "datetime". */ categoryDatetimeHeader: 'DateTime' }, /** * The text for the menu item. * * @since 6.0.0 * @requires modules/export-data */ viewData: 'View data table', /** * The text for the menu item. * * @since 8.2.0 * @requires modules/export-data */ hideData: 'Hide data table', /** * Text to show when export is in progress. * * @since 11.3.0 * @requires modules/export-data */ exportInProgress: 'Exporting...' }; /* * * * Default Export * * */ /** @internal */ const ExportDataDefaults = { exporting, lang }; /** @internal */ /* harmony default export */ const ExportData_ExportDataDefaults = (ExportDataDefaults); /* * * * API Options * * */ /** * Callback that fires while exporting data. This allows the modification of * data rows before processed into the final format. * * @type {Highcharts.ExportDataCallbackFunction} * @since 7.2.0 * @context Highcharts.Chart * @requires modules/exporting * @requires modules/export-data * @apioption chart.events.exportData */ /** * When set to `false` will prevent the series data from being included in * any form of data export. * * Since version 6.0.0 until 7.1.0 the option was existing undocumented * as `includeInCSVExport`. * * @type {boolean} * @since 7.1.0 * @requires modules/export-data * @apioption plotOptions.series.includeInDataExport */ (''); // Keep doclets above in JS file ;// ./code/es-modules/Shared/Utilities.js /* * * * (c) 2009-2026 Highsoft AS * * A commercial license may be required depending on use. * See www.highcharts.com/license * * * */ const { doc: Utilities_doc, win: Utilities_win } = (highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_default()); /** * Add an event listener. * * @function Highcharts.addEvent<T> * * @param {Highcharts.Class<T>|T} el * The element or object to add a listener to. It can be a * {@link HTMLDOMElement}, an {@link SVGElement} or any other object. * * @param {string} type * The event type. * * @param {Highcharts.EventCallbackFunction<T>|Function} fn * The function callback to execute when the event is fired. * * @param {Highcharts.EventOptionsObject} [options] * Options for adding the event. * * @sample highcharts/members/addevent * Use a general `render` event to draw shapes on a chart * * @return {Function} * A callback function to remove the added event. */ function addEvent(el, type, fn, options = {}) { // Add hcEvents to either the prototype (in case we're running addEvent on a // class) or the instance. If hasOwnProperty('hcEvents') is false, it is // inherited down the prototype chain, in which case we need to set the // property on this instance (which may itself be a prototype). const owner = typeof el === 'function' && el.prototype || el; if (!Object.hasOwnProperty.call(owner, 'hcEvents')) { owner.hcEvents = {}; } const events = owner.hcEvents; // Allow click events added to points, otherwise they will be prevented by // the TouchPointer.pinch function after a pinch zoom operation (#7091). if ((highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_default()).Point && // Without H a dependency loop occurs el instanceof (highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_default()).Point && el.series && el.series.chart) { el.series.chart.runTrackerClick = true; } // Handle DOM events // If the browser supports passive events, add it to improve performance // on touch events (#11353). const addEventListener = el.addEventListener; if (addEventListener) { addEventListener.call(el, type, fn, (highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_default()).supportsPassiveEvents ? { passive: options.passive === void 0 ? type.indexOf('touch') !== -1 : options.passive, capture: false } : false); } if (!events[type]) { events[type] = []; } const eventObject = { fn, order: typeof options.order === 'number' ? options.order : Infinity }; events[type].push(eventObject); // Order the calls events[type].sort((a, b) => a.order - b.order); // Return a function that can be called to remove this event. return function () { removeEvent(el, type, fn); }; } /** * Non-recursive method to find the lowest member of an array. `Math.min` raises * a maximum call stack size exceeded error in Chrome when trying to apply more * than 150.000 points. This method is slightly slower, but safe. * * @function Highcharts.arrayMin * * @param {Array<*>} data * An array of numbers. * * @return {number} * The lowest number. */ function arrayMin(data) { let i = data.length, min = data[0]; while (i--) { if (data[i] < min) { min = data[i]; } } return min; } /** * Non-recursive method to find the lowest member of an array. `Math.max` raises * a maximum call stack size exceeded error in Chrome when trying to apply more * than 150.000 points. This method is slightly slower, but safe. * * @function Highcharts.arrayMax * * @param {Array<*>} data * An array of numbers. * * @return {number} * The highest number. */ function arrayMax(data) { let i = data.length, max = data[0]; while (i--) { if (data[i] > max) { max = data[i]; } } return max; } /** * Set or get an attribute or an object of attributes. * * To use as a setter, pass a key and a value, or let the second argument be a * collection of keys and values. When using a collection, passing a value of * `null` or `undefined` will remove the attribute. * * To use as a getter, pass only a string as the second argument. * * @function Highcharts.attr * * @param {Highcharts.HTMLDOMElement|Highcharts.SVGDOMElement} elem * The DOM element to receive the attribute(s). * * @param {string|Highcharts.HTMLAttributes|Highcharts.SVGAttributes} [keyOrAttribs] * The property or an object of key-value pairs. * * @param {number|string} [value] * The value if a single property is set. * * @return {string|null|undefined} * When used as a getter, return the value. */ function attr(elem, keyOrAttribs, value) { const isGetter = isString(keyOrAttribs) && !defined(value); let ret; const attrSingle = (value, key) => { // Set the value if (defined(value)) { elem.setAttribute(key, value); // Get the value } else if (isGetter) { ret = elem.getAttribute(key); // IE7 and below cannot get class through getAttribute (#7850) if (!ret && key === 'class') { ret = elem.getAttribute(key + 'Name'); } // Remove the value } else { elem.removeAttribute(key); } }; // If keyOrAttribs is a string if (isString(keyOrAttribs)) { attrSingle(value, keyOrAttribs); // Else if keyOrAttribs is defined, it is a hash of key/value pairs } else { objectEach(keyOrAttribs, attrSingle); } return ret; } /** * Constrain a value to within a lower and upper threshold. * * @internal * @param {number} value The initial value * @param {number} min The lower threshold * @param {number} max The upper threshold * @return {number} Returns a number value within min and max. */ function clamp(value, min, max) { return value > min ? value < max ? value : max : min; } /** * Fix JS round off float errors. * * @function Highcharts.correctFloat * * @param {number} num * A float number to fix. * * @param {number} [prec=14] * The precision. * * @return {number} * The corrected float number. */ function correctFloat(num, prec) { // When the number is higher than 1e14 use the number (#16275) return num > 1e14 ? num : parseFloat(num.toPrecision(prec || 14)); } /** * Utility function to create an HTML element with attributes and styles. * * @function Highcharts.createElement * * @param {string} tag * The HTML tag. * * @param {Highcharts.HTMLAttributes} [attribs] * Attributes as an object of key-value pairs. * * @param {Highcharts.CSSObject} [styles] * Styles as an object of key-value pairs. * * @param {Highcharts.HTMLDOMElement} [parent] * The parent HTML object. * * @param {boolean} [nopad=false] * If true, remove all padding, border and margin. * * @return {Highcharts.HTMLDOMElement} * The created DOM element. */ function createElement(tag, attribs, styles, parent, nopad) { const el = Utilities_doc.createElement(tag); if (attribs) { extend(el, attribs); } if (nopad) { css(el, { padding: '0', border: 'none', margin: '0' }); } if (styles) { css(el, styles); } if (parent) { parent.appendChild(el); } return el; } /** * Utility for crisping a line position to the nearest full pixel depending on * the line width. * * @internal * @param {number} value The raw pixel position * @param {number} lineWidth The line width * @param {boolean} [inverted] Whether the containing group is inverted. * Crisping round numbers on the y-scale need to go * to the other side because the coordinate system * is flipped (scaleY is -1) * @return {number} The pixel position to use for a crisp display */ function crisp(value, lineWidth = 0, inverted) { const mod = lineWidth % 2 / 2, inverter = inverted ? -1 : 1; return (Math.round(value * inverter - mod) + mod) * inverter; } /** * Set CSS on a given element. * * @function Highcharts.css * * @param {Highcharts.HTMLDOMElement|Highcharts.SVGDOMElement} el * An HTML DOM element. * * @param {Highcharts.CSSObject} styles * Style object with camel case property names. * * @return {void} */ function css(el, styles) { extend(el.style, styles); } /** * Check if an object is null or undefined. * * @function Highcharts.defined * * @param {*} obj * The object to check. * * @return {boolean} * False if the object is null or undefined, otherwise true. */ function defined(obj) { return typeof obj !== 'undefined' && obj !== null; } /** * Utility method that destroys any SVGElement instances that are properties on * the given object. It loops all properties and invokes destroy if there is a * destroy method. The property is then delete. * * @function Highcharts.destroyObjectProperties * * @param {*} obj * The object to destroy properties on. * * @param {*} [except] * Exception, do not destroy this property, only delete it. */ function destroyObjectProperties(obj, except, destructablesOnly) { objectEach(obj, function (val, n) { // If the object is non-null and destroy is defined if (val !== except && val?.destroy) { // Invoke the destroy val.destroy(); } // Delete the property from the object if (val?.destroy || !destructablesOnly) { delete obj[n]; } }); } /** * Discard a HTML element * * @function Highcharts.discardElement * * @param {Highcharts.HTMLDOMElement} element * The HTML node to discard. */ function discardElement(element) { element?.parentElement?.removeChild(element); } // eslint-disable-next-line valid-jsdoc /** * Return the deep difference between two objects. It can either return the new * properties, or optionally return the old values of new properties. * @internal */ function diffObjects(newer, older, keepOlder, collectionsWithUpdate) { const ret = {}; /** * Recurse over a set of options and its current values, and store the * current values in the ret object. */ function diff(newer, older, ret, depth) { const keeper = keepOlder ? older : newer; objectEach(newer, function (newerVal, key) { if (!depth && collectionsWithUpdate && collectionsWithUpdate.indexOf(key) > -1 && older[key]) { newerVal = splat(newerVal); ret[key] = []; // Iterate over collections like series, xAxis or yAxis and map // the items by index. for (let i = 0; i < Math.max(newerVal.length, older[key].length); i++) { // Item exists in current data (#6347) if (older[key][i]) { // If the item is missing from the new data, we need to // save the whole config structure. Like when // responsively updating from a dual axis layout to a // single axis and back (#13544). if (newerVal[i] === void 0) { ret[key][i] = older[key][i]; // Otherwise, proceed } else { ret[key][i] = {}; diff(newerVal[i], older[key][i], ret[key][i], depth + 1); } } } } else if (isObject(newerVal, true) && !newerVal.nodeType // #10044 ) { ret[key] = isArray(newerVal) ? [] : {}; diff(newerVal, older[key] || {}, ret[key], depth + 1); // Delete empty nested objects if (Object.keys(ret[key]).length === 0 && // Except colorAxis which is a special case where the empty // object means it is enabled. Which is unfortunate and we // should try to find a better way. !(key === 'colorAxis' && depth === 0)) { delete ret[key]; } } else if (newer[key] !== older[key] || // If the newer key is explicitly undefined, keep it (#10525) (key in newer && !(key in older))) { if (key !== '__proto__' && key !== 'constructor') { ret[key] = keeper[key]; } } }); } diff(newer, older, ret, 0); return ret; } /** * Remove the last occurrence of an item from an array. * * @function Highcharts.erase * * @param {Array<*>} arr * The array. * * @param {*} item * The item to remove. * * @return {void} */ function erase(arr, item) { let i = arr.length; while (i--) { if (arr[i] === item) { arr.splice(i, 1); break; } } } /** * Utility function to extend an object with the members of another. * * @function Highcharts.extend<T> * * @param {T|undefined} a * The object to be extended. * * @param {Partial<T>} b * The object to add to the first one. * * @return {T} * Object a, the original object. */ function extend(a, b) { let n; if (!a) { a = {}; } for (n in b) { // eslint-disable-line guard-for-in a[n] = b[n]; } return a; } // eslint-disable-next-line valid-jsdoc /** * Extend a prototyped class by new members. * * @deprecated * @function Highcharts.extendClass<T> * * @param {Highcharts.Class<T>} parent * The parent prototype to inherit. * * @param {Highcharts.Dictionary<*>} members * A collection of prototype members to add or override compared to the * parent prototype. * * @return {Highcharts.Class<T>} * A new prototype. */ function extendClass(parent, members) { const obj = (function () { }); obj.prototype = new parent(); // eslint-disable-line new-cap extend(obj.prototype, members); return obj; } /** * Fire an event that was registered with {@link Highcharts#addEvent}. * * @function Highcharts.fireEvent<T> * * @param {T} el * The object to fire the event on. It can be a {@link HTMLDOMElement}, * an {@link SVGElement} or any other object. * * @param {string} type * The type of event. * * @param {Highcharts.Dictionary<*>|Event} [eventArguments] * Custom event arguments that are passed on as an argument to the event * handler. * * @param {Highcharts.EventCallbackFunction<T>|Function} [defaultFunction] * The default function to execute if the other listeners haven't * returned false. * * @return {void} */ function fireEvent(el, type, eventArguments, defaultFunction) { eventArguments = eventArguments || {}; if (Utilities_doc?.createEvent && (el.dispatchEvent || (el.fireEvent && // Enable firing events on Highcharts instance. el !== (highcharts_commonjs_highcharts_commonjs2_highcharts_root_Highcharts_default())))) { const e = Utilities_doc.createEvent('Events'); e.initEvent(type, true, true); eventArguments = extend(e, eventArguments); if (el.dispatchEvent) { el.dispatchEvent(eventArguments); } else { el.fireEvent(type, eventArguments); } } else if (el.hcEvents) { if (!eventArguments.target) { // We're running a custom event extend(eventArguments, { // Attach a simple preventDefault function to skip // default handler if called. The built-in // defaultPrevented property is not overwritable (#5112) preventDefault: function () { eventArguments.defaultPrevented = true; }, // Setting target to native events fails with clicking // the zoom-out button in Chrome. target: el, // If the type is not set, we're running a custom event // (#2297). If it is set, we're running a browser event. type: type }); } const events = []; let object = el; let multilevel = false; // Recurse up the inheritance chain and collect hcEvents set as own // objects on the prototypes. while (object.hcEvents) { if (Object.hasOwnProperty.call(object, 'hcEvents') && object.hcEvents[type]) { if (events.length) { multilevel = true; } events.unshift.apply(events, object.hcEvents[type]); } object = Object.getPrototypeOf(object); } // For performance reasons, only sort the event handlers in case we are // dealing with multiple levels in the prototype chain. Otherwise, the // events are already sorted in the addEvent function. if (multilevel) { // Order the calls events.sort((a, b) => a.order - b.order); } // Call the collected event handlers events.forEach((obj) => { // If the event handler returns false, prevent the default handler // from executing if (obj.fn.call(el, eventArguments, el) === false) { eventArguments.preventDefault(); } }); } // Run the default if not prevented if (defaultFunction && !eventArguments.defaultPrevented) { defaultFunction.call(el, eventArguments); } } /** * Convenience function to get the align factor, used several places for * computing positions * @internal */ const getAlignFactor = (align = '') => ({ center: 0.5, right: 1, middle: 0.5, bottom: 1 }[align] || 0); /** * Find the closest distance between two values of a two-dimensional array * @internal * @function Highcharts.getClosestDistance * * @param {Array<Array<number>>} arrays * An array of arrays of numbers * * @return {number | undefined} * The closest distance between values */ function getClosestDistance(arrays, onError) { const allowNegative = !onError; let closest, loopLength, distance, i; arrays.forEach((xData) => { if (xData.length > 1) { loopLength = xData.length - 1; for (i = loopLength; i > 0; i--) { distance = xData[i] - xData[i - 1]; if (distance < 0 && !allowNegative) { onError?.(); // Only one call onError = void 0; } else if (distance && (typeof closest === 'undefined' || distance < closest)) { closest = distance; } } } }); return closest; } /** * Get the magnitude of a number. * * @function Highcharts.getMagnitude * * @param {number} num * The number. * * @return {number} * The magnitude, where 1-9 are magnitude 1, 10-99 magnitude 2 etc. */ function getMagnitude(num) { return Math.pow(10, Math.floor(Math.log(num) / Math.LN10)); } /** * Returns the value of a property path on a given object. * * @internal * @function getNestedProperty * * @param {string} path * Path to the property, for example `custom.myValue`. * * @param {unknown} parent * Instance containing the property on the specific path. * * @return {unknown} * The unknown property value. */ function getNestedProperty(path, parent) { const pathElements = path.split('.'); while (pathElements.length && defined(parent)) { const pathElement = pathElements.shift(); // Filter on the key if (typeof pathElement === 'undefined' || pathElement === '__proto__') { return; // Undefined } if (pathElement === 'this') { let thisProp; if (isObject(parent)) { thisProp = parent['@this']; } return thisProp ?? parent; } const child = parent[pathElement.replace(/[\\'"]/g, '')]; // Filter on the child if (!defined(child) || typeof child === 'function' || typeof child.nodeType === 'number' || child === Utilities_win) { return; // Undefined } // Else, proceed parent = child; } return parent; } /** * Get the computed CSS value for given element and property, only for numerical * properties. For width and height, the dimension of the inner box (excluding * padding) is returned. Used for fitting the chart within the container. * * @function Highcharts.getStyle * * @param {Highcharts.HTMLDOMElement} el * An HTML element. * * @param {string} prop * The property name. * * @param {boolean} [toInt=true] * Parse to integer. * * @return {number|string|undefined} * The style value. */ function getStyle(el, prop, toInt) { let style; // For width and height, return the actual inner pixel size (#4913) if (prop === 'width') { let offsetWidth = Math.min(el.offsetWidth, el.scrollWidth); // In flex boxes, we need to use getBoundingClientRect and floor it, // because scrollWidth doesn't support subpixel precision (#6427) ... const boundingClientRectWidth = el.getBoundingClientRect?.().width; // ...unless if the containing div or its parents are transform-scaled // down, in which case the boundingClientRect can't be used as it is // also scaled down (#9871, #10498). if (boundingClientRectWidth < offsetWidth && boundingClientRectWidth >= offsetWidth - 1) { offsetWidth = Math.floor(boundingClientRectWidth); } return Math.max(0, // #8377 (offsetWidth - (getStyle(el, 'padding-left', true) || 0) - (getStyle(el, 'padding-right', true) || 0))); } if (prop === 'height') { return Math.max(0, // #8377 (Math.min(el.offsetHeight, el.scrollHeight) - (getStyle(el, 'padding-top', true) || 0) - (getStyle(el, 'padding-bottom', true) || 0))); } // Otherwise, get the computed style const css = Utilities_win.getComputedStyle(el, void 0); // eslint-disable-line no-undefined if (css) { style = css.getPropertyValue(prop); if (pick(toInt, prop !== 'opacity')) { style = pInt(style); } } return style; } /** * Return the value of the first element in the array that satisfies the * provided testing function. * * @function Highcharts.find<T> * * @param {Array<T>} arr * The array to test. * * @param {Function} callback * The callback function. The function receives the item as the first * argument. Return `true` if this item satisfies the condition. * * @return {T|undefined} * The value of the element. */ const find = Array.prototype.find ? function (arr, callback) { return arr.find(callback); } : // Legacy implementation. PhantomJS, IE <= 11 etc. #7223. function (arr, callback) { let i; const length = arr.length; for (i = 0; i < length; i++) { if (callback(arr[i], i)) { // eslint-disable-line node/callback-return return arr[i]; } } }; /** * Internal clear timeout. The function checks that the `id` was not removed * (e.g. by `chart.destroy()`). For the details see * [issue #7901](https://github.com/highcharts/highcharts/issues/7901). * * @internal * * @function Highcharts.clearTimeout * * @param {number|undefined} id * Id of a timeout. */ function internalClearTimeout(id) { if (defined(id)) { clearTimeout(id); } } /** * Utility function to check if an Object is a HTML Element. * * @function Highcharts.isDOMElement * * @param {*} obj * The item to check. * * @return {boolean} * True if the argument is a HTML Element. */ function isDOMElement(obj) { return isObject(obj) && typeof obj.nodeType === 'number'; } /** * Utility function to check if an Object is a class. * * @function Highcharts.isClass * * @param {object|undefined} obj * The item to check. * * @return {boolean} * True if the argument is a class. */ function isClass(obj) { const c = obj?.constructor; return !!(isObject(obj, true) && !isDOMElement(obj) && (c?.name && c.name !== 'Object')); } /** * Utility function to check if an item is a number and it is finite (not NaN, * Infinity or -Infinity). * * @function Highcharts.isNumber * * @param {*} n * The item to check. * * @return {boolean} * True if the item is a finite number */ function isNumber(n) { return typeof n === 'number' && !isNaN(n) && n < Infinity && n > -Infinity; } /** * Utility function to check for string type. * * @function Highcharts.isString * * @param {*} s * The item to check. * * @return {boolean} * True if the argument is a string. */ function isString(s) { return typeof s === 'string'; } /** * Utility function to check if an item is an array. * * @function Highcharts.isArray * * @param {*} obj * The item to check. * * @return {boolean} * True if the argument is an array. */ function isArray(obj) { const str = Object.prototype.toString.call(obj); return str === '[object Array]' || str === '[object Array Iterator]'; } /** * Utility function to check if object is a function. * * @function Highcharts.isFunction * * @param {*} obj * The item to check. * * @return {boolean} * True if the argument is a function. */ function isFunction(obj) { return typeof obj === 'function'; } /** * Utility function to check if an item is of type object. * * @function Highcharts.isObject * * @param {*} obj * The item to check. * * @param {boolean} [strict=false] * Also checks that the object is not an array. * * @return {boolean} * True if the argument is an object. */ function isObject(obj, strict) { return (!!obj && typeof obj === 'object' && (!strict || !isArray(obj))); // eslint-disable-line @typescript-eslint/no-explicit-any } /** * Utility function to deep merge two or more objects and return a third object. * If the first argument is true, the contents of the second object is copied * into the first object. The merge function can also be used with a single * object argument to create a deep copy of an object. * * @function Highcharts.merge<T> * * @param {true | T} extendOrSource * Whether to extend the left-side object, * or the first object to merge as a deep copy. * * @param {...Array<object|undefined>} [sources] * Object(s) to merge into the previous one. * * @return {T} * The merged object. If the first argument is true, the return is the * same as the second argument. */ function merge(extendOrSource, ...sources) { let i, args = [extendOrSource, ...sources], ret = {}; const doCopy = function (copy, original) { // An object is replacing a primitive if (typeof copy !== 'object') { copy = {}; } objectEach(original, function (value, key) { // Prototype pollution (#14883) if (key === '__proto__' || key === 'constructor') { return; } // Copy the contents of objects, but not arrays or DOM nodes if (isObject(value, true) && !isClass(value) && !isDOMElement(value)) { copy[key] = doCopy(copy[key] || {}, value); // Primitives and arrays are copied over directly } else { copy[key] = original[key]; } }); return copy; }; // If first argument is true, copy into the existing object. Used in // setOptions. if (e