diff --git a/notemyprogress/ajax.php b/notemyprogress/ajax.php index 1bd5ed4b377c9e9d30eb521c0dd39c70ad71774a..a2fb52b92b808a54f6ad7d8ab3c516d0fcb4dd41 100644 --- a/notemyprogress/ajax.php +++ b/notemyprogress/ajax.php @@ -73,6 +73,7 @@ $settings= optional_param('settings', false ,PARAM_TEXT); $url= optional_param('url', false ,PARAM_TEXT); //$courseid = optional_param('modulename', false ,PARAM_TEXT); $enable = optional_param('enable', false ,PARAM_BOOL); +$section = optional_param('section',false,PARAM_TEXT); /* */ @@ -193,17 +194,35 @@ if($action == 'saveconfigweek') {//Exemple: if the action passed is saveconfigwe array_push($params, $levels); array_push($params, $settings); array_push($params, $url); - array_push($params, $enable); + array_push($params, $section); if($courseid && $userid && $rules && $levels && $settings && $url) { $func = "local_notemyprogress_save_gamification_config"; } }elseif($action =='rankable'){ array_push($params, $courseid); array_push($params, $userid); + array_push($params, $url); if($courseid && $userid) { $func = "local_notemyprogress_set_rankable_player"; } -} +}elseif($action =='saveEnable'){ + array_push($params, $courseid); + array_push($params, $userid); + array_push($params,$enable); + array_push($params,$url); + if($courseid && $userid) { + $func = "local_notemyprogress_save_enable"; + } +}elseif($action =='studentGamificationViewed'){ + array_push($params, $courseid); + array_push($params, $userid); + array_push($params,$url); + if($courseid && $userid) { + $func = "local_notemyprogress_viewed_student_lvl"; + } +} + + if(isset($params) && isset($func)){ call_user_func_array($func, $params); @@ -337,7 +356,6 @@ function local_notemyprogress_get_student_sessions($weekcode, $courseid, $userid function local_notemyprogress_downloadMoodleLogs($beginDate, $lastDate, $courseid, $userid, $currentUrl) { $logs = new \local_notemyprogress\logs($courseid, $userid); - $logs->addLogsNMP("downloaded", "logfile", "LOGFILES", "moodle", $currentUrl, "File that contains all the activities performed on the Note My Progress plugin in a time interval"); $jsonData = $logs->searchLogsMoodle($beginDate, $lastDate); local_notemyprogress_ajax_response(array("jsonData"=>$jsonData)); @@ -348,31 +366,42 @@ function local_notemyprogress_addLogs($sectionname, $actiontype, $courseid, $use $logs->addLogsNMP($actiontype, $objectType, $sectionname, $objectName, $currentUrl, $objectDescription); local_notemyprogress_ajax_response(array("ok"=>"ok")); } -function local_notemyprogress_save_gamification_config($courseid, $userid, $rules, $levels, $settings, $url,$enable){ - // \local_notemyprogress\logs::create( - // "setgamification", - // "configweeks", - // "saved", - // "weeks_settings", - // $url, - // 4, - // $userid, - // $courseid - // ); - +function local_notemyprogress_save_gamification_config($courseid, $userid, $rules, $levels, $settings, $url,$section){ + $logs = new \local_notemyprogress\logs($courseid, $userid); + $logs->addLogsNMP("Saved", $section, "CONFIGURATION_GAMIFICATION", "configuration_gamification", $url, "GamificationSaved"); + $configLevels = new \local_notemyprogress\configgamification($courseid, $userid); $configLevels->save_levels($levels, $settings, $rules); - $configLevels->save_enable($enable); + //$configLevels->save_enable($enable); $message = get_string('fml_api_save_successful', 'local_notemyprogress'); local_notemyprogress_ajax_response($message); } -function local_notemyprogress_set_rankable_player($courseid, $userid){ +function local_notemyprogress_set_rankable_player($courseid, $userid,$url){ + $logs = new \local_notemyprogress\logs($courseid, $userid); + $logs->addLogsNMP("Saved", "section", "STUDENT_GAMIFICATION", "student_gamification", $url, "GamificationSaved"); GLobal $DB; - // $sql = "update {notemyprogress_xp} set rankable = ? where courseid = ? and userid = ?"; - // $DB->execute($sql, array(1,$courseid, $userid)); - $sql = "UPDATE {notemyprogress_xp} SET rankable = 1 WHERE courseid = 2 AND userid = 3"; - $DB->execute($sql); + $sql = "update {notemyprogress_xp} set rankable = ? where courseid = ? and userid = ?"; + $DB->execute($sql, array(1,$courseid, $userid)); + // $sql = "UPDATE {notemyprogress_xp} SET rankable = 1 WHERE courseid = 2 AND userid = 3"; + // $DB->execute($sql); $message = get_string('fml_api_save_successful', 'local_notemyprogress'); local_notemyprogress_ajax_response($message); -} \ No newline at end of file +} + +function local_notemyprogress_save_enable($courseid, $userid, $enable,$url){ + $logs = new \local_notemyprogress\logs($courseid, $userid); + $logs->addLogsNMP("Saved", "section", "CONFIGURATION_GAMIFICATION", "configuration_gamification", $url, "GamificationSaved"); + $configLevels = new \local_notemyprogress\configgamification($courseid, $userid); + $configLevels->save_enable($enable); + $message = get_string('fml_api_save_successful', 'local_notemyprogress'); + local_notemyprogress_ajax_response($message); + } + function local_notemyprogress_viewed_student_lvl($courseid, $userid,$url){ + $logs = new \local_notemyprogress\logs($courseid, $userid); + $logs->addLogsNMP("Viewed", "section", "STUDENT_GAMIFICATION", "student_gamification", $url, "GamificationSaved");; + // $configLevels = new \local_notemyprogress\configgamification($courseid, $userid); + // $configLevels->save_enable($enable); + // $message = get_string('fml_api_save_successful', 'local_notemyprogress'); + // local_notemyprogress_ajax_response($message); + } \ No newline at end of file diff --git a/notemyprogress/amd/build/gamification.js b/notemyprogress/amd/build/gamification.js index e83c2b5d5dcc9c72007e142993c3987a557cc6d6..5ba1ae7061ea5405269a932cd210be99e41a35be 100644 --- a/notemyprogress/amd/build/gamification.js +++ b/notemyprogress/amd/build/gamification.js @@ -4,13 +4,15 @@ define([ "local_notemyprogress/axios", "local_notemyprogress/alertify", "local_notemyprogress/pageheader", -], function (Vue, Vuetify, Axios, Alertify, PageHeader) { + "local_notemyprogress/chartdynamic", +], function (Vue, Vuetify, Axios, Alertify, PageHeader, ChartDynamic) { "use strict"; function init(content) { - //console.log(content); + console.log(content); Vue.use(Vuetify); Vue.component("pageheader", PageHeader); + Vue.component("chart", ChartDynamic); const app = new Vue({ delimiters: ["[[", "]]"], el: "#gamification", @@ -33,7 +35,13 @@ define([ setPointsOption: "calculated", pointsBase: 0, pointsBaseOri: 0, - swDisableEnable: "", + swDisableEnable: content.strings.swValue, + spreadData: [], + week_resources_categories: [], + week_resources_data: [], + week_resources_colors: "#FA4641", + indicators: content.indicators, + chartdata: content.chart_data, }, beforeMount() {}, @@ -113,10 +121,11 @@ define([ } return parseInt(x); }, - save_changes() { + save_changes(logParam) { this.notifications = ["Do you want to save the changes"]; Alertify.confirm(this.strings.save_warning_content, () => { - this.saveGamificationConfig(); + this.saveGamificationConfig(logParam); + console.log(logParam); }) // ON CONFIRM .set({ title: this.strings.save_warning_title }) .set({ @@ -126,7 +135,7 @@ define([ }, }); }, - saveGamificationConfig() { + saveGamificationConfig(logParam) { this.loading = true; let settings = { tit: this.settings.tit, @@ -142,10 +151,10 @@ define([ rules: JSON.stringify(this.rulesData), token: this.token, enable: this.enable, + section: logParam, }; let url = { url: window.location.href }; Axios({ - ///! try resolve that with a listener method: "post", url: M.cfg.wwwroot + @@ -162,8 +171,10 @@ define([ "&url=" + url.url + "&enable=" + - data.enable, - data: data, + data.enable + + "§ion=" + + data.section, + params: data, }) .then((response) => { // console.log(response); @@ -213,7 +224,204 @@ define([ disableEnable(swDisableEnable) { //traitement this.enable = swDisableEnable; + let data = { + courseid: this.courseid, + userid: this.userid, + action: "saveEnable", + enable: this.enable, + }; + Axios({ + method: "get", + url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", + params: data, + }) + .then((response) => { + if (response.status == 200 && response.data.ok) { + this.showNotifications(response.data.data, "success"); + } else { + let message = + response.data.error || this.strings.api_error_network; + this.showNotifications(message, "error"); //in this line "error" define the kind of message send + } + }) + .catch((e) => { + let message = e.response.data || this.strings.api_error_network; + this.showNotifications(message, "error"); + }) + .finally(() => { + this.loading = false; + }); + }, + //? ////////////////////////////////////////////////////////////////////// ?// + //? //////////////////////////// ChartPart /////////////////////////////// ?// + //? ////////////////////////////////////////////////////////////////////// ?// + chart_spread() { + let chart = new Object(); + chart.chart = { + type: "column", + backgroundColor: null, + }; + chart.title = { + text: this.strings.chartTitle, + }; + chart.colors = ["#118AB2"]; + (chart.xAxis = { + type: "category", + labels: { + rotation: -45, + style: { + fontSize: "13px", + fontFamily: "Verdana, sans-serif", + }, + }, + }), + (chart.yAxis = { + min: 0, + title: { + text: this.strings.chartYaxis, + }, + }); + chart.legend = { + enabled: false, + }; + (chart.series = [ + { + name: null, + data: this.chartdata, //[["level : 1", 1]], + dataLabels: { + enabled: true, + rotation: -90, + color: "#FFFFFF", + align: "right", + format: "{point.y:.1f}", // one decimal + y: 10, // 10 pixels down from the top + style: { + fontSize: "13px", + fontFamily: "Verdana, sans-serif", + }, + }, + }, + ]), + console.log("series: "); + console.log(chart.series); + return chart; }, + /* + build_inverted_time_chart() { + console.log("enter build_inverted_time_chart "); + //console.log(this.students_planification); + //console.log("this.data_report_meta_hours = "); + //console.log(this.data_report_meta_hours); + let chart = new Object(); + let meta = new Object(); + meta = this.chartdata_hours_week_dedication(); + console.log("meta = "); + console.log(meta); + let invest = [ + { + name: meta.labels[0], + y: meta.datasets[0].data[0], + }, + { + name: meta.labels[1], + y: meta.datasets[0].data[1], + }, + { + name: meta.labels[2], + y: meta.datasets[0].data[2], + }, + ]; + console.log("invest = "); + console.log(invest); + chart.chart = { + type: "bar", + backgroundColor: null, + style: { fontFamily: "poppins" }, + }; + chart.title = { text: null }; + chart.colors = this.inverted_time_colors; + chart.xAxis = { + type: "category", + crosshair: true, + }; + chart.yAxis = { + title: { + text: "Framboise", + }, + }; + chart.tooltip = { + shared: true, + useHTML: true, + formatter: function () { + let category_name = this.points[0].key; + let time = vue.convert_time(this.y); + return `<b>${category_name}: </b>${time}`; + }, + }; + chart.legend = { + enabled: false, + }; + chart.series = [ + { + colorByPoint: true, + data: invest, + }, + ]; + //console.log("this.inverted_time.data = "); + //console.log(this.inverted_time.data); + + // console.log("invest = "); + // console.log(invest); + return chart; + }, + chartdata_hours_week_dedication() { + var data = new Object(); + data.datasets = []; + + let inverted = + this.render_has == "teacher" + ? this.strings.inverted_time + : `${this.strings.myself} ${this.strings.inverted_time}`; + let planified = + this.render_has == "teacher" + ? this.strings.planified_time + : `${this.strings.myself} ${this.strings.planified_time}`; + data.labels = [inverted, planified]; + var dataset = new Object(); + dataset.label = "Horas"; + //console.log("data_report_meta_hours in chartdata_hours_week_dedication = "); + //console.log(this.data_report_meta_hours); + dataset.data = [ + parseFloat(this.data_report_meta_hours.horas_trabajadas), + parseInt(this.data_report_meta_hours.horas_planificadas), + ]; + dataset.backgroundColor = ["#ffa700", "#a0c2fa"]; + dataset.borderWidth = 0; + data.datasets.push(dataset); + + //if (this.render_has == "student" && this.compare_with_course) { + data.labels.splice(1, 0, this.strings.inverted_time_course); + //data.labels.splice(3, 0, this.strings.planified_time_course); + dataset.data.splice( + 1, + 0, + parseFloat(this.course_report_hours.horas_trabajadas) + ); + // dataset.data.splice( + // 3, + // 0, + // parseFloat(this.course_report_hours.horas_planificadas) + // ); + dataset.backgroundColor.splice(1, 0, "#ffa700"); + //dataset.backgroundColor.splice(3, 0, "#a0c2fa"); + //} + //console.log("data_report_meta_hours = "); + //console.log(this.data_report_meta_hours); + + //console.log("data = "); + //console.log(data); + return data; + },*/ }, }); } diff --git a/notemyprogress/amd/build/listener.min.js b/notemyprogress/amd/build/listener.min.js index f7412851521e015dd04ea00d52bd076ea76c2e2e..577b33cb6aedc1b3b10838f0ef40f60696806187 100644 --- a/notemyprogress/amd/build/listener.min.js +++ b/notemyprogress/amd/build/listener.min.js @@ -1 +1,11 @@ -define([],function(){return{init:function(){document.querySelector("#downloadButton").addEventListener("click",function(){console.log("clicked !")})}}}); \ No newline at end of file +define([], function () { + return { + init: function () { + document + .querySelector("#downloadButton") + .addEventListener("click", function () { + console.log("clicked !"); + }); + }, + }; +}); diff --git a/notemyprogress/amd/build/student_gamification.js b/notemyprogress/amd/build/student_gamification.js index a77d864883406dae90330837bdfab7cd94e991d2..d7fb47847300386cc9b36ae99c5382f37b04e6d9 100644 --- a/notemyprogress/amd/build/student_gamification.js +++ b/notemyprogress/amd/build/student_gamification.js @@ -31,7 +31,18 @@ define([ error_messages: [], save_successful: false, }, - beforeMount() {}, + beforeMount() { + let data = { + courseid: this.courseid, + userid: this.userid, + action: "studentGamificationViewed", + }; + Axios({ + method: "get", + url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", + params: data, + }); + }, computed: {}, methods: { get_help_content() { diff --git a/notemyprogress/amd/build/teacher.js b/notemyprogress/amd/build/teacher.js new file mode 100644 index 0000000000000000000000000000000000000000..59e8e40d2f52b57d58a94dc6da977a237f792f03 --- /dev/null +++ b/notemyprogress/amd/build/teacher.js @@ -0,0 +1,305 @@ +define([ + "local_notemyprogress/vue", + "local_notemyprogress/vuetify", + "local_notemyprogress/axios", + "local_notemyprogress/pagination", + "local_notemyprogress/chartstatic", + "local_notemyprogress/pageheader", + "local_notemyprogress/helpdialog", +], function ( + Vue, + Vuetify, + Axios, + Pagination, + ChartStatic, + PageHeader, + HelpDialog +) { + "use strict"; + + function init(content) { + // console.log(content); + Vue.use(Vuetify); + Vue.component("pagination", Pagination); + Vue.component("chart", ChartStatic); + Vue.component("pageheader", PageHeader); + Vue.component("helpdialog", HelpDialog); + let vue = new Vue({ + delimiters: ["[[", "]]"], + el: "#teacher", + vuetify: new Vuetify(), + data() { + return { + strings: content.strings, + groups: content.groups, + userid: content.userid, + courseid: content.courseid, + timezone: content.timezone, + render_has: content.profile_render, + + indicators: content.indicators, + week_resources_colors: content.week_resources_colors, + search: null, + week_resources_categories: [], + week_resources_data: [], + + help_dialog: false, + help_contents: [], + }; + }, + beforeMount() { + this.calculate_week_resources(); + }, + mounted() { + document.querySelector("#sessions-loader").style.display = "none"; + document.querySelector("#teacher").style.display = "block"; + }, + methods: { + get_help_content() { + let contents = []; + contents.push({ + title: this.strings.section_help_title, + description: this.strings.section_help_description, + }); + return contents; + }, + + get_course_grade() { + let grade = Number(this.indicators.course.grademax); + return this.isInt(grade) ? grade : grade.toFixed(2); + }, + + calculate_week_resources() { + let categories = [], + data = []; + let week_name; + this.indicators.weeks.forEach((week) => { + week_name = `${week.name} ${week.position + 1}`; + categories.push(week_name); + data.push(week.cms); + }); + let name = this.capitalizeFirstLetter( + this.strings.teacher_indicators_modules + ); + this.week_resources_categories = categories; + this.week_resources_data = [{ name, data }]; + }, + + build_week_resources_chart() { + let chart = new Object(); + chart.chart = { + type: "bar", + backgroundColor: null, + style: { fontFamily: "poppins" }, + }; + chart.title = { + text: null, + }; + chart.colors = this.week_resources_colors; + chart.xAxis = { + categories: this.week_resources_categories, + }; + chart.yAxis = { + min: 0, + title: { + text: this.strings.teacher_indicators_week_resources_yaxis_title, + }, + }; + chart.legend = { + enabled: false, + }; + chart.series = this.week_resources_data; + console.log(chart.series); + return chart; + }, + + build_weeks_sessions_chart() { + let chart = new Object(); + chart.chart = { + type: "heatmap", + backgroundColor: null, + style: { fontFamily: "poppins" }, + }; + chart.title = { + text: null, + }; + chart.xAxis = { + categories: this.strings.weeks, + }; + chart.yAxis = { + categories: this.indicators.sessions.categories, + title: null, + reversed: true, + }; + chart.colorAxis = { + min: 0, + minColor: "#E0E0E0", + maxColor: "#118AB2", + }; + chart.legend = { + layout: "horizontal", + verticalAlign: "bottom", + }; + chart.tooltip = { + formatter: function () { + let days = + vue.indicators.sessions.weeks[this.point.y][this.point.x] || ""; + let xCategoryName = vue.get_point_category_name(this.point, "x"); + let yCategoryName = vue.get_point_category_name(this.point, "y"); + let label = vue.strings.teacher_indicators_sessions; + if (this.point.value == 1) { + label = vue.strings.teacher_indicators_session; + } + return ( + "<b>" + + yCategoryName + + " " + + xCategoryName + + "</b>: " + + this.point.value + + " " + + label + + "<br/>" + + days + ); + }, + }; + chart.series = [ + { + borderWidth: 2, + borderColor: "#FAFAFA", + data: this.indicators.sessions.data, + }, + ]; + return chart; + }, + + table_headers() { + let headers = [ + { text: "", value: "id", align: "center", sortable: false }, + { text: this.strings.thead_name, value: "firstname" }, + { text: this.strings.thead_lastname, value: "lastname" }, + { text: this.strings.thead_email, value: "email" }, + { + text: this.strings.thead_progress, + value: "progress_percentage", + align: "center", + }, + { + text: this.strings.thead_sessions, + value: "sessions_number", + align: "center", + }, + { + text: this.strings.thead_time, + value: "inverted_time", + align: "center", + }, + ]; + return headers; + }, + + get_picture_url(userid) { + let url = `${M.cfg.wwwroot}/user/pix.php?file=/${userid}/f1.jpg`; + return url; + }, + + get_percentage_progress(value) { + return `${value} %`; + }, + + get_progress_tooltip(item) { + let module_label = this.strings.teacher_indicators_modules; + let finished_label = this.strings.teacher_indicators_finished; + if (item.cms.complete == 1) { + module_label = this.strings.teacher_indicators_module; + finished_label = this.strings.teacher_indicators_finalized; + } + return `${item.cms.complete} ${module_label} ${finished_label} ${this.strings.of_conector} ${item.cms.total}`; + }, + + get_point_category_name(point, dimension) { + let series = point.series, + isY = dimension === "y", + axis = series[isY ? "yAxis" : "xAxis"]; + return axis.categories[point[isY ? "y" : "x"]]; + }, + + capitalizeFirstLetter(string) { + return string.charAt(0).toUpperCase() + string.slice(1); + }, + + isInt(n) { + return n % 1 === 0; + }, + + open_chart_help(chart) { + let contents = []; + if (chart == "week_resources") { + contents.push({ + title: this.strings.week_resources_help_title, + description: this.strings.week_resources_help_description_p1, + }); + contents.push({ + description: this.strings.week_resources_help_description_p2, + }); + } else if (chart == "weeks_sessions") { + contents.push({ + title: this.strings.weeks_sessions_help_title, + description: this.strings.week_sessions_help_description_p1, + }); + contents.push({ + description: this.strings.week_sessions_help_description_p2, + }); + } else if (chart == "progress_table") { + contents.push({ + title: this.strings.progress_table_help_title, + description: this.strings.progress_table_help_description, + }); + } + this.help_contents = contents; + if (this.help_contents.length) { + this.help_dialog = true; + } + }, + + update_help_dialog(value) { + this.help_dialog = value; + }, + + get_timezone() { + let information = `${this.strings.ss_change_timezone} ${this.timezone}`; + return information; + }, + + addLogsIntoDB(action, objectName, objectType, objectDescription) { + let data = { + courseid: content.courseid, + userid: content.userid, + action: "addLogs", + sectionname: "TEACHER_GENERAL_INDICATORS", + actiontype: action, + objectType: objectType, + objectName: objectName, + currentUrl: document.location.href, + objectDescription: objectDescription, + }; + Axios({ + method: "get", + url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", + params: data, + }) + .then((response) => { + if (response.status == 200 && response.data.ok) { + } + }) + .catch((e) => {}); + }, + }, + }); + } + + return { + init: init, + }; +}); diff --git a/notemyprogress/amd/build/teacher.min.js b/notemyprogress/amd/build/teacher.min.js deleted file mode 100644 index 7b49e25001efbe51976e1ee2eb2274b2f88de329..0000000000000000000000000000000000000000 --- a/notemyprogress/amd/build/teacher.min.js +++ /dev/null @@ -1,2 +0,0 @@ -define(["local_notemyprogress/vue","local_notemyprogress/vuetify","local_notemyprogress/axios","local_notemyprogress/pagination","local_notemyprogress/chartstatic","local_notemyprogress/pageheader","local_notemyprogress/helpdialog"],function(e,t,s,i,r,o,n){"use strict";return{init:function(a){e.use(t),e.component("pagination",i),e.component("chart",r),e.component("pageheader",o),e.component("helpdialog",n);let l=new e({delimiters:["[[","]]"],el:"#teacher",vuetify:new t,data:()=>({strings:a.strings,groups:a.groups,userid:a.userid,courseid:a.courseid,timezone:a.timezone,render_has:a.profile_render,indicators:a.indicators,week_resources_colors:a.week_resources_colors,search:null,week_resources_categories:[],week_resources_data:[],help_dialog:!1,help_contents:[]}),beforeMount(){this.calculate_week_resources()},mounted(){document.querySelector("#sessions-loader").style.display="none",document.querySelector("#teacher").style.display="block"},methods:{get_help_content(){let e=[];return e.push({title:this.strings.section_help_title,description:this.strings.section_help_description}),e},get_course_grade(){let e=Number(this.indicators.course.grademax);return this.isInt(e)?e:e.toFixed(2)},calculate_week_resources(){let e,t=[],s=[];this.indicators.weeks.forEach(i=>{e=`${i.name} ${i.position+1}`,t.push(e),s.push(i.cms)});let i=this.capitalizeFirstLetter(this.strings.teacher_indicators_modules);this.week_resources_categories=t,this.week_resources_data=[{name:i,data:s}]},build_week_resources_chart(){let e=new Object;return e.chart={type:"bar",backgroundColor:null,style:{fontFamily:"poppins"}},e.title={text:null},e.colors=this.week_resources_colors,e.xAxis={categories:this.week_resources_categories},e.yAxis={min:0,title:{text:this.strings.teacher_indicators_week_resources_yaxis_title}},e.legend={enabled:!1},e.series=this.week_resources_data,e},build_weeks_sessions_chart(){let e=new Object;return e.chart={type:"heatmap",backgroundColor:null,style:{fontFamily:"poppins"}},e.title={text:null},e.xAxis={categories:this.strings.weeks},e.yAxis={categories:this.indicators.sessions.categories,title:null,reversed:!0},e.colorAxis={min:0,minColor:"#E0E0E0",maxColor:"#118AB2"},e.legend={layout:"horizontal",verticalAlign:"bottom"},e.tooltip={formatter:function(){let e=l.indicators.sessions.weeks[this.point.y][this.point.x]||"",t=l.get_point_category_name(this.point,"x"),s=l.get_point_category_name(this.point,"y"),i=l.strings.teacher_indicators_sessions;return 1==this.point.value&&(i=l.strings.teacher_indicators_session),"<b>"+s+" "+t+"</b>: "+this.point.value+" "+i+"<br/>"+e}},e.series=[{borderWidth:2,borderColor:"#FAFAFA",data:this.indicators.sessions.data}],e},table_headers(){return[{text:"",value:"id",align:"center",sortable:!1},{text:this.strings.thead_name,value:"firstname"},{text:this.strings.thead_lastname,value:"lastname"},{text:this.strings.thead_email,value:"email"},{text:this.strings.thead_progress,value:"progress_percentage",align:"center"},{text:this.strings.thead_sessions,value:"sessions_number",align:"center"},{text:this.strings.thead_time,value:"inverted_time",align:"center"}]},get_picture_url:e=>`${M.cfg.wwwroot}/user/pix.php?file=/${e}/f1.jpg`,get_percentage_progress:e=>`${e} %`,get_progress_tooltip(e){let t=this.strings.teacher_indicators_modules,s=this.strings.teacher_indicators_finished;return 1==e.cms.complete&&(t=this.strings.teacher_indicators_module,s=this.strings.teacher_indicators_finalized),`${e.cms.complete} ${t} ${s} ${this.strings.of_conector} ${e.cms.total}`},get_point_category_name(e,t){let s="y"===t;return e.series[s?"yAxis":"xAxis"].categories[e[s?"y":"x"]]},capitalizeFirstLetter:e=>e.charAt(0).toUpperCase()+e.slice(1),isInt:e=>e%1==0,open_chart_help(e){let t=[];"week_resources"==e?(t.push({title:this.strings.week_resources_help_title,description:this.strings.week_resources_help_description_p1}),t.push({description:this.strings.week_resources_help_description_p2})):"weeks_sessions"==e?(t.push({title:this.strings.weeks_sessions_help_title,description:this.strings.week_sessions_help_description_p1}),t.push({description:this.strings.week_sessions_help_description_p2})):"progress_table"==e&&t.push({title:this.strings.progress_table_help_title,description:this.strings.progress_table_help_description}),this.help_contents=t,this.help_contents.length&&(this.help_dialog=!0)},update_help_dialog(e){this.help_dialog=e},get_timezone(){return`${this.strings.ss_change_timezone} ${this.timezone}`},addLogsIntoDB(e,t,i,r){let o={courseid:a.courseid,userid:a.userid,action:"addLogs",sectionname:"TEACHER_GENERAL_INDICATORS",actiontype:e,objectType:i,objectName:t,currentUrl:document.location.href,objectDescription:r};s({method:"get",url:M.cfg.wwwroot+"/local/notemyprogress/ajax.php",params:o}).then(e=>{200==e.status&&e.data.ok}).catch(e=>{})}}})}}}); -//# sourceMappingURL=teacher.min.js.map diff --git a/notemyprogress/amd/build/teacher.min.js.map b/notemyprogress/amd/build/teacher.min.js.map deleted file mode 100644 index 6579b63acd185dff389ab267b839dbee24635552..0000000000000000000000000000000000000000 --- a/notemyprogress/amd/build/teacher.min.js.map +++ /dev/null @@ -1 +0,0 @@ -{"version":3,"sources":["../src/teacher.js"],"names":["define","Vue","Vuetify","Axios","Pagination","ChartStatic","PageHeader","HelpDialog","init","content","use","component","vue","delimiters","el","vuetify","data","strings","groups","userid","courseid","timezone","render_has","profile_render","indicators","week_resources_colors","search","week_resources_categories","week_resources_data","help_dialog","help_contents","beforeMount","calculate_week_resources","mounted","document","querySelector","style","display","methods","get_help_content","contents","title","section_help_title","description","section_help_description","get_course_grade","grade","course","grademax","isInt","toFixed","categories","week_name","weeks","forEach","week","name","position","push","cms","capitalizeFirstLetter","teacher_indicators_modules","build_week_resources_chart","chart","type","backgroundColor","fontFamily","text","colors","xAxis","yAxis","min","teacher_indicators_week_resources_yaxis_title","legend","enabled","series","build_weeks_sessions_chart","sessions","reversed","colorAxis","minColor","maxColor","layout","verticalAlign","tooltip","formatter","days","point","y","x","xCategoryName","get_point_category_name","yCategoryName","label","teacher_indicators_sessions","value","teacher_indicators_session","borderWidth","borderColor","table_headers","headers","align","sortable","thead_name","thead_lastname","thead_email","thead_progress","thead_sessions","thead_time","get_picture_url","url","M","cfg","wwwroot","get_percentage_progress","get_progress_tooltip","item","module_label","finished_label","teacher_indicators_finished","complete","teacher_indicators_module","teacher_indicators_finalized","of_conector","total","dimension","isY","axis","string","charAt","toUpperCase","slice","n","open_chart_help","week_resources_help_title","week_resources_help_description_p1","week_resources_help_description_p2","weeks_sessions_help_title","week_sessions_help_description_p1","week_sessions_help_description_p2","progress_table_help_title","progress_table_help_description","length","update_help_dialog","get_timezone","information","ss_change_timezone"],"mappings":"AAAAA,OAAM,8BAAC,CAAC,wBAAD,CACC,4BADD,CAEC,0BAFD,CAGC,+BAHD,CAIC,gCAJD,CAKC,+BALD,CAMC,+BAND,CAAD,CAQF,SAASC,CAAT,CAAcC,CAAd,CAAuBC,CAAvB,CAA8BC,CAA9B,CAA0CC,CAA1C,CAAuDC,CAAvD,CAAmEC,CAAnE,CAA+E,CAC3E,aAwOA,MAAO,CACHC,IAAI,CAvOR,SAAcC,CAAd,CAAuB,CAEnBR,CAAG,CAACS,GAAJ,CAAQR,CAAR,EACAD,CAAG,CAACU,SAAJ,CAAc,YAAd,CAA4BP,CAA5B,EACAH,CAAG,CAACU,SAAJ,CAAc,OAAd,CAAuBN,CAAvB,EACAJ,CAAG,CAACU,SAAJ,CAAc,YAAd,CAA4BL,CAA5B,EACAL,CAAG,CAACU,SAAJ,CAAc,YAAd,CAA4BJ,CAA5B,EACA,GAAIK,CAAAA,CAAG,CAAG,GAAIX,CAAAA,CAAJ,CAAQ,CACdY,UAAU,CAAE,CAAC,IAAD,CAAO,IAAP,CADE,CAEdC,EAAE,CAAE,UAFU,CAGdC,OAAO,CAAE,GAAIb,CAAAA,CAHC,CAIdc,IAJc,gBAIP,CACH,MAAO,CACHC,OAAO,CAAGR,CAAO,CAACQ,OADf,CAEHC,MAAM,CAAGT,CAAO,CAACS,MAFd,CAGHC,MAAM,CAAGV,CAAO,CAACU,MAHd,CAIHC,QAAQ,CAAGX,CAAO,CAACW,QAJhB,CAKHC,QAAQ,CAAGZ,CAAO,CAACY,QALhB,CAMHC,UAAU,CAAGb,CAAO,CAACc,cANlB,CAQHC,UAAU,CAAEf,CAAO,CAACe,UARjB,CASHC,qBAAqB,CAAEhB,CAAO,CAACgB,qBAT5B,CAUHC,MAAM,CAAE,IAVL,CAWHC,yBAAyB,CAAE,EAXxB,CAYHC,mBAAmB,CAAE,EAZlB,CAcHC,WAAW,GAdR,CAeHC,aAAa,CAAE,EAfZ,CAiBV,CAtBa,CAuBdC,WAvBc,uBAuBD,CACT,KAAKC,wBAAL,EACH,CAzBa,CA0BdC,OA1Bc,mBA0BL,CACLC,QAAQ,CAACC,aAAT,CAAuB,kBAAvB,EAA2CC,KAA3C,CAAiDC,OAAjD,CAA2D,MAA3D,CACAH,QAAQ,CAACC,aAAT,CAAuB,UAAvB,EAAmCC,KAAnC,CAAyCC,OAAzC,CAAmD,OACtD,CA7Ba,CA8BdC,OAAO,CAAG,CACNC,gBADM,4BACY,CACd,GAAIC,CAAAA,CAAQ,CAAG,CACD,CACVC,KAAK,CAAE,KAAKxB,OAAL,CAAayB,kBADV,CAEVC,WAAW,CAAE,KAAK1B,OAAL,CAAa2B,wBAFhB,CADC,CAAf,CAKA,MAAOJ,CAAAA,CACV,CARK,CAUNK,gBAVM,4BAUY,CACd,GAAIC,CAAAA,CAAK,EAAU,KAAKtB,UAAL,CAAgBuB,MAAhB,CAAuBC,QAA1C,CACA,MAAQ,MAAKC,KAAL,CAAWH,CAAX,CAAD,CAAsBA,CAAtB,CAA8BA,CAAK,CAACI,OAAN,CAAc,CAAd,CACxC,CAbK,CAeNlB,wBAfM,oCAeqB,IACnBmB,CAAAA,CAAU,CAAG,EADM,CACFnC,CAAI,CAAG,EADL,CAEnBoC,CAFmB,CAGvB,KAAK5B,UAAL,CAAgB6B,KAAhB,CAAsBC,OAAtB,CAA8B,SAAAC,CAAI,CAAI,CAClCH,CAAS,WAAMG,CAAI,CAACC,IAAX,aAAoBD,CAAI,CAACE,QAAL,CAAc,CAAlC,CAAT,CACAN,CAAU,CAACO,IAAX,CAAgBN,CAAhB,EACApC,CAAI,CAAC0C,IAAL,CAAUH,CAAI,CAACI,GAAf,CACH,CAJD,EAKA,GAAIH,CAAAA,CAAI,CAAG,KAAKI,qBAAL,CAA2B,KAAK3C,OAAL,CAAa4C,0BAAxC,CAAX,CACA,KAAKlC,yBAAL,CAAiCwB,CAAjC,CACA,KAAKvB,mBAAL,CAA2B,CAAC,CAAE4B,IAAI,CAAJA,CAAF,CAAQxC,IAAI,CAAJA,CAAR,CAAD,CAC9B,CA1BK,CA4BN8C,0BA5BM,sCA4BuB,CACzB,GAAIC,CAAAA,CAAK,GAAT,CACAA,CAAK,CAACA,KAAN,CAAc,CACVC,IAAI,CAAE,KADI,CAEVC,eAAe,CAAE,IAFP,CAGV7B,KAAK,CAAE,CAAC8B,UAAU,CAAE,SAAb,CAHG,CAAd,CAKAH,CAAK,CAACtB,KAAN,CAAc,CACV0B,IAAI,CAAE,IADI,CAAd,CAGAJ,CAAK,CAACK,MAAN,CAAe,KAAK3C,qBAApB,CACAsC,CAAK,CAACM,KAAN,CAAc,CACVlB,UAAU,CAAE,KAAKxB,yBADP,CAAd,CAGAoC,CAAK,CAACO,KAAN,CAAc,CACVC,GAAG,CAAE,CADK,CAEN9B,KAAK,CAAE,CACP0B,IAAI,CAAE,KAAKlD,OAAL,CAAauD,6CADZ,CAFD,CAAd,CAMAT,CAAK,CAACU,MAAN,CAAe,CACXC,OAAO,GADI,CAAf,CAGAX,CAAK,CAACY,MAAN,CAAe,KAAK/C,mBAApB,CACA,MAAOmC,CAAAA,CACV,CArDK,CAuDNa,0BAvDM,sCAuDuB,CACzB,GAAIb,CAAAA,CAAK,GAAT,CACAA,CAAK,CAACA,KAAN,CAAc,CACVC,IAAI,CAAE,SADI,CAEVC,eAAe,CAAE,IAFP,CAGV7B,KAAK,CAAE,CAAC8B,UAAU,CAAE,SAAb,CAHG,CAAd,CAKAH,CAAK,CAACtB,KAAN,CAAc,CACV0B,IAAI,CAAE,IADI,CAAd,CAGAJ,CAAK,CAACM,KAAN,CAAc,CACVlB,UAAU,CAAE,KAAKlC,OAAL,CAAaoC,KADf,CAAd,CAGAU,CAAK,CAACO,KAAN,CAAc,CACVnB,UAAU,CAAE,KAAK3B,UAAL,CAAgBqD,QAAhB,CAAyB1B,UAD3B,CAEVV,KAAK,CAAE,IAFG,CAGVqC,QAAQ,GAHE,CAAd,CAKAf,CAAK,CAACgB,SAAN,CAAkB,CACdR,GAAG,CAAE,CADS,CAEdS,QAAQ,CAAE,SAFI,CAGdC,QAAQ,CAAE,SAHI,CAAlB,CAKAlB,CAAK,CAACU,MAAN,CAAe,CACXS,MAAM,CAAE,YADG,CAEXC,aAAa,CAAE,QAFJ,CAAf,CAIApB,CAAK,CAACqB,OAAN,CAAgB,CACZC,SAAS,CAAE,oBAAY,IACfC,CAAAA,CAAI,CAAG1E,CAAG,CAACY,UAAJ,CAAeqD,QAAf,CAAwBxB,KAAxB,CAA8B,KAAKkC,KAAL,CAAWC,CAAzC,EAA4C,KAAKD,KAAL,CAAWE,CAAvD,GAA6D,EADrD,CAEfC,CAAa,CAAG9E,CAAG,CAAC+E,uBAAJ,CAA4B,KAAKJ,KAAjC,CAAwC,GAAxC,CAFD,CAGfK,CAAa,CAAGhF,CAAG,CAAC+E,uBAAJ,CAA4B,KAAKJ,KAAjC,CAAwC,GAAxC,CAHD,CAIfM,CAAK,CAAGjF,CAAG,CAACK,OAAJ,CAAY6E,2BAJL,CAKnB,GAAwB,CAApB,OAAKP,KAAL,CAAWQ,KAAf,CAA2B,CACvBF,CAAK,CAAGjF,CAAG,CAACK,OAAJ,CAAY+E,0BACvB,CACD,MAAO,MAAQJ,CAAR,CAAwB,GAAxB,CAA8BF,CAA9B,CAA8C,QAA9C,CACD,KAAKH,KAAL,CAAWQ,KADV,CACiB,GADjB,CACuBF,CADvB,CAC+B,OAD/B,CACyCP,CACnD,CAXW,CAAhB,CAaAvB,CAAK,CAACY,MAAN,CAAe,CAAC,CACZsB,WAAW,CAAE,CADD,CAEZC,WAAW,CAAE,SAFD,CAGZlF,IAAI,CAAE,KAAKQ,UAAL,CAAgBqD,QAAhB,CAAyB7D,IAHnB,CAAD,CAAf,CAKA,MAAO+C,CAAAA,CACV,CArGK,CAuGNoC,aAvGM,yBAuGS,CACX,GAAIC,CAAAA,CAAO,CAAG,CACV,CAAEjC,IAAI,CAAE,EAAR,CAAY4B,KAAK,CAAG,IAApB,CAA0BM,KAAK,CAAG,QAAlC,CAA4CC,QAAQ,GAApD,CADU,CAEV,CAAEnC,IAAI,CAAE,KAAKlD,OAAL,CAAasF,UAArB,CAAkCR,KAAK,CAAG,WAA1C,CAFU,CAGV,CAAE5B,IAAI,CAAE,KAAKlD,OAAL,CAAauF,cAArB,CAAsCT,KAAK,CAAG,UAA9C,CAHU,CAIV,CAAE5B,IAAI,CAAE,KAAKlD,OAAL,CAAawF,WAArB,CAAmCV,KAAK,CAAG,OAA3C,CAJU,CAKV,CAAE5B,IAAI,CAAE,KAAKlD,OAAL,CAAayF,cAArB,CAAsCX,KAAK,CAAG,qBAA9C,CAAsEM,KAAK,CAAG,QAA9E,CALU,CAMV,CAAElC,IAAI,CAAE,KAAKlD,OAAL,CAAa0F,cAArB,CAAsCZ,KAAK,CAAG,iBAA9C,CAAkEM,KAAK,CAAG,QAA1E,CANU,CAOV,CAAElC,IAAI,CAAE,KAAKlD,OAAL,CAAa2F,UAArB,CAAkCb,KAAK,CAAG,eAA1C,CAA2DM,KAAK,CAAG,QAAnE,CAPU,CAAd,CASA,MAAOD,CAAAA,CACV,CAlHK,CAoHNS,eApHM,0BAoHU1F,CApHV,CAoHiB,CACnB,GAAI2F,CAAAA,CAAG,WAAMC,CAAC,CAACC,GAAF,CAAMC,OAAZ,gCAA0C9F,CAA1C,WAAP,CACA,MAAO2F,CAAAA,CACV,CAvHK,CAyHNI,uBAzHM,kCAyHkBnB,CAzHlB,CAyHwB,CAC1B,gBAAUA,CAAV,MACH,CA3HK,CA6HNoB,oBA7HM,+BA6HeC,CA7Hf,CA6HoB,IAClBC,CAAAA,CAAY,CAAG,KAAKpG,OAAL,CAAa4C,0BADV,CAElByD,CAAc,CAAG,KAAKrG,OAAL,CAAasG,2BAFZ,CAGtB,GAAyB,CAArB,EAAAH,CAAI,CAACzD,GAAL,CAAS6D,QAAb,CAA4B,CACxBH,CAAY,CAAG,KAAKpG,OAAL,CAAawG,yBAA5B,CACAH,CAAc,CAAG,KAAKrG,OAAL,CAAayG,4BACjC,CACD,gBAAUN,CAAI,CAACzD,GAAL,CAAS6D,QAAnB,aAA+BH,CAA/B,aAA+CC,CAA/C,aAAiE,KAAKrG,OAAL,CAAa0G,WAA9E,aAA6FP,CAAI,CAACzD,GAAL,CAASiE,KAAtG,CACH,CArIK,CAuINjC,uBAvIM,kCAuIkBJ,CAvIlB,CAuIyBsC,CAvIzB,CAuIoC,CACtC,GAAIlD,CAAAA,CAAM,CAAGY,CAAK,CAACZ,MAAnB,CACImD,CAAG,CAAiB,GAAd,GAAAD,CADV,CAEIE,CAAI,CAAGpD,CAAM,CAACmD,CAAG,CAAG,OAAH,CAAa,OAAjB,CAFjB,CAGA,MAAOC,CAAAA,CAAI,CAAC5E,UAAL,CAAgBoC,CAAK,CAACuC,CAAG,CAAG,GAAH,CAAS,GAAb,CAArB,CACV,CA5IK,CA8INlE,qBA9IM,gCA8IgBoE,CA9IhB,CA8IwB,CAC1B,MAAOA,CAAAA,CAAM,CAACC,MAAP,CAAc,CAAd,EAAiBC,WAAjB,GAAiCF,CAAM,CAACG,KAAP,CAAa,CAAb,CAC3C,CAhJK,CAkJNlF,KAlJM,gBAkJAmF,CAlJA,CAkJG,CACL,MAAiB,EAAV,EAAAA,CAAC,CAAG,CACd,CApJK,CAsJNC,eAtJM,0BAsJUtE,CAtJV,CAsJiB,CACnB,GAAIvB,CAAAA,CAAQ,CAAG,EAAf,CACA,GAAa,gBAAT,EAAAuB,CAAJ,CAA+B,CAC3BvB,CAAQ,CAACkB,IAAT,CAAc,CACVjB,KAAK,CAAE,KAAKxB,OAAL,CAAaqH,yBADV,CAEV3F,WAAW,CAAE,KAAK1B,OAAL,CAAasH,kCAFhB,CAAd,EAIA/F,CAAQ,CAACkB,IAAT,CAAc,CACVf,WAAW,CAAE,KAAK1B,OAAL,CAAauH,kCADhB,CAAd,CAGH,CARD,IAQO,IAAa,gBAAT,EAAAzE,CAAJ,CAA+B,CAClCvB,CAAQ,CAACkB,IAAT,CAAc,CACVjB,KAAK,CAAE,KAAKxB,OAAL,CAAawH,yBADV,CAEV9F,WAAW,CAAE,KAAK1B,OAAL,CAAayH,iCAFhB,CAAd,EAIAlG,CAAQ,CAACkB,IAAT,CAAc,CACVf,WAAW,CAAE,KAAK1B,OAAL,CAAa0H,iCADhB,CAAd,CAGH,CARM,IAQA,IAAa,gBAAT,EAAA5E,CAAJ,CAA+B,CAClCvB,CAAQ,CAACkB,IAAT,CAAc,CACVjB,KAAK,CAAE,KAAKxB,OAAL,CAAa2H,yBADV,CAEVjG,WAAW,CAAE,KAAK1B,OAAL,CAAa4H,+BAFhB,CAAd,CAIH,CACD,KAAK/G,aAAL,CAAqBU,CAArB,CACA,GAAI,KAAKV,aAAL,CAAmBgH,MAAvB,CAA+B,CAC3B,KAAKjH,WAAL,GACH,CACJ,CAlLK,CAoLNkH,kBApLM,6BAoLchD,CApLd,CAoLqB,CACvB,KAAKlE,WAAL,CAAmBkE,CACtB,CAtLK,CAwLNiD,YAxLM,wBAwLQ,CACV,GAAIC,CAAAA,CAAW,WAAM,KAAKhI,OAAL,CAAaiI,kBAAnB,aAAyC,KAAK7H,QAA9C,CAAf,CACA,MAAO4H,CAAAA,CACV,CA3LK,CA9BI,CAAR,CA6Nb,CAEM,CAGV,CApPC,CAAN","sourcesContent":["define([\"local_notemyprogress/vue\",\r\n \"local_notemyprogress/vuetify\",\r\n \"local_notemyprogress/axios\",\r\n \"local_notemyprogress/pagination\",\r\n \"local_notemyprogress/chartstatic\",\r\n \"local_notemyprogress/pageheader\",\r\n \"local_notemyprogress/helpdialog\",\r\n ],\r\n function(Vue, Vuetify, Axios, Pagination, ChartStatic, PageHeader, HelpDialog) {\r\n \"use strict\";\r\n\r\n function init(content) {\r\n // console.log(content);\r\n Vue.use(Vuetify);\r\n Vue.component('pagination', Pagination);\r\n Vue.component('chart', ChartStatic);\r\n Vue.component('pageheader', PageHeader);\r\n Vue.component('helpdialog', HelpDialog);\r\n let vue = new Vue({\r\n delimiters: [\"[[\", \"]]\"],\r\n el: \"#teacher\",\r\n vuetify: new Vuetify(),\r\n data() {\r\n return {\r\n strings : content.strings,\r\n groups : content.groups,\r\n userid : content.userid,\r\n courseid : content.courseid,\r\n timezone : content.timezone,\r\n render_has : content.profile_render,\r\n\r\n indicators: content.indicators,\r\n week_resources_colors: content.week_resources_colors,\r\n search: null,\r\n week_resources_categories: [],\r\n week_resources_data: [],\r\n\r\n help_dialog: false,\r\n help_contents: [],\r\n }\r\n },\r\n beforeMount(){\r\n this.calculate_week_resources();\r\n },\r\n mounted(){\r\n document.querySelector(\"#sessions-loader\").style.display = \"none\";\r\n document.querySelector(\"#teacher\").style.display = \"block\";\r\n },\r\n methods : {\r\n get_help_content(){\r\n let contents = [];\r\n contents.push({\r\n title: this.strings.section_help_title,\r\n description: this.strings.section_help_description,\r\n });\r\n return contents;\r\n },\r\n\r\n get_course_grade(){\r\n let grade = Number(this.indicators.course.grademax);\r\n return (this.isInt(grade)) ? grade : grade.toFixed(2);\r\n },\r\n\r\n calculate_week_resources() {\r\n let categories = [], data = [];\r\n let week_name;\r\n this.indicators.weeks.forEach(week => {\r\n week_name = `${week.name} ${(week.position+1)}`;\r\n categories.push(week_name);\r\n data.push(week.cms);\r\n });\r\n let name = this.capitalizeFirstLetter(this.strings.teacher_indicators_modules);\r\n this.week_resources_categories = categories;\r\n this.week_resources_data = [{ name, data}];\r\n },\r\n\r\n build_week_resources_chart() {\r\n let chart = new Object();\r\n chart.chart = {\r\n type: 'bar',\r\n backgroundColor: null,\r\n style: {fontFamily: 'poppins'},\r\n };\r\n chart.title = {\r\n text: null,\r\n };\r\n chart.colors = this.week_resources_colors;\r\n chart.xAxis = {\r\n categories: this.week_resources_categories\r\n };\r\n chart.yAxis = {\r\n min: 0,\r\n title: {\r\n text: this.strings.teacher_indicators_week_resources_yaxis_title\r\n }\r\n };\r\n chart.legend = {\r\n enabled: false\r\n };\r\n chart.series = this.week_resources_data;\r\n return chart;\r\n },\r\n\r\n build_weeks_sessions_chart() {\r\n let chart = new Object();\r\n chart.chart = {\r\n type: 'heatmap',\r\n backgroundColor: null,\r\n style: {fontFamily: 'poppins'},\r\n };\r\n chart.title = {\r\n text: null,\r\n };\r\n chart.xAxis = {\r\n categories: this.strings.weeks,\r\n };\r\n chart.yAxis = {\r\n categories: this.indicators.sessions.categories,\r\n title: null,\r\n reversed: true,\r\n };\r\n chart.colorAxis = {\r\n min: 0,\r\n minColor: '#E0E0E0',\r\n maxColor: '#118AB2'\r\n };\r\n chart.legend = {\r\n layout: 'horizontal',\r\n verticalAlign: 'bottom',\r\n };\r\n chart.tooltip = {\r\n formatter: function () {\r\n let days = vue.indicators.sessions.weeks[this.point.y][this.point.x] || '';\r\n let xCategoryName = vue.get_point_category_name(this.point, 'x');\r\n let yCategoryName = vue.get_point_category_name(this.point, 'y');\r\n let label = vue.strings.teacher_indicators_sessions;\r\n if (this.point.value == 1) {\r\n label = vue.strings.teacher_indicators_session;\r\n }\r\n return '<b>' + yCategoryName + ' ' + xCategoryName + '</b>: '\r\n + this.point.value +' ' + label + '<br/>' + days;\r\n }\r\n };\r\n chart.series = [{\r\n borderWidth: 2,\r\n borderColor: '#FAFAFA',\r\n data: this.indicators.sessions.data,\r\n }];\r\n return chart;\r\n },\r\n\r\n table_headers(){\r\n let headers = [\r\n { text: '', value : 'id', align : 'center', sortable : false},\r\n { text: this.strings.thead_name , value : 'firstname'},\r\n { text: this.strings.thead_lastname , value : 'lastname'},\r\n { text: this.strings.thead_email , value : 'email'},\r\n { text: this.strings.thead_progress , value : 'progress_percentage', align : 'center'},\r\n { text: this.strings.thead_sessions , value : 'sessions_number', align : 'center'},\r\n { text: this.strings.thead_time , value : 'inverted_time', align : 'center'},\r\n ];\r\n return headers;\r\n },\r\n\r\n get_picture_url(userid){\r\n let url = `${M.cfg.wwwroot}/user/pix.php?file=/${userid}/f1.jpg`;\r\n return url;\r\n },\r\n\r\n get_percentage_progress(value){\r\n return `${value} %`;\r\n },\r\n\r\n get_progress_tooltip(item){\r\n let module_label = this.strings.teacher_indicators_modules;\r\n let finished_label = this.strings.teacher_indicators_finished;\r\n if (item.cms.complete == 1) {\r\n module_label = this.strings.teacher_indicators_module;\r\n finished_label = this.strings.teacher_indicators_finalized;\r\n }\r\n return `${item.cms.complete} ${module_label} ${finished_label} ${this.strings.of_conector} ${item.cms.total}`;\r\n },\r\n\r\n get_point_category_name(point, dimension) {\r\n let series = point.series,\r\n isY = dimension === 'y',\r\n axis = series[isY ? 'yAxis' : 'xAxis'];\r\n return axis.categories[point[isY ? 'y' : 'x']];\r\n },\r\n\r\n capitalizeFirstLetter(string) {\r\n return string.charAt(0).toUpperCase() + string.slice(1);\r\n },\r\n\r\n isInt(n) {\r\n return n % 1 === 0;\r\n },\r\n\r\n open_chart_help(chart) {\r\n let contents = [];\r\n if (chart == \"week_resources\") {\r\n contents.push({\r\n title: this.strings.week_resources_help_title,\r\n description: this.strings.week_resources_help_description_p1,\r\n });\r\n contents.push({\r\n description: this.strings.week_resources_help_description_p2,\r\n });\r\n } else if (chart == \"weeks_sessions\") {\r\n contents.push({\r\n title: this.strings.weeks_sessions_help_title,\r\n description: this.strings.week_sessions_help_description_p1,\r\n });\r\n contents.push({\r\n description: this.strings.week_sessions_help_description_p2,\r\n });\r\n } else if (chart == \"progress_table\") {\r\n contents.push({\r\n title: this.strings.progress_table_help_title,\r\n description: this.strings.progress_table_help_description,\r\n });\r\n }\r\n this.help_contents = contents;\r\n if (this.help_contents.length) {\r\n this.help_dialog = true;\r\n }\r\n },\r\n\r\n update_help_dialog (value) {\r\n this.help_dialog = value;\r\n },\r\n\r\n get_timezone(){\r\n let information = `${this.strings.ss_change_timezone} ${this.timezone}`\r\n return information;\r\n },\r\n\r\n }\r\n })\r\n }\r\n\r\n return {\r\n init : init\r\n };\r\n });"],"file":"teacher.min.js"} \ No newline at end of file diff --git a/notemyprogress/amd/src/logs.js b/notemyprogress/amd/src/logs.js index 11f3883b60293bf7ae8fb7f7e14bb168c131588f..edc81115e83e526f5c3d83be4a402cbbe5afa8ca 100644 --- a/notemyprogress/amd/src/logs.js +++ b/notemyprogress/amd/src/logs.js @@ -2,305 +2,379 @@ @author 2021 Éric Bart <bart.eric@hotmail.com> */ -define(["local_notemyprogress/vue", - "local_notemyprogress/vuetify", - "local_notemyprogress/axios", - "local_notemyprogress/moment", - "local_notemyprogress/pagination", - "local_notemyprogress/pageheader", - "local_notemyprogress/helpdialog", - "local_notemyprogress/alertify", - -], - function (Vue, Vuetify, Axios, Moment, Pagination, Pageheader, HelpDialog, Alertify) { - "use strict"; - - function init(content) { - const timeout = 60 * 120 * 1000 - Axios.defaults.timeout = timeout - Vue.use(Vuetify); - Vue.component('pagination', Pagination); - Vue.component('pageheader', Pageheader); - Vue.component('helpdialog', HelpDialog); - let vue = new Vue({ - delimiters: ["[[", "]]"], - el: "#logs", - vuetify: new Vuetify(), - data() { - return { - calendarData: {}, - strings: content.strings, - groups: content.groups, - userid: content.userid, - courseid: content.courseid, - timezone: content.timezone, - render_has: content.profile_render, - courseRole: content.courseRole, - loading: false, - errors: [], - pages: content.pages, - help_dialog: false, - help_contents: [], - dateRules: [ - v => !!v || this.strings.logs_invalid_date - ] +define([ + "local_notemyprogress/vue", + "local_notemyprogress/vuetify", + "local_notemyprogress/axios", + "local_notemyprogress/moment", + "local_notemyprogress/pagination", + "local_notemyprogress/pageheader", + "local_notemyprogress/helpdialog", + "local_notemyprogress/alertify", +], function ( + Vue, + Vuetify, + Axios, + Moment, + Pagination, + Pageheader, + HelpDialog, + Alertify +) { + "use strict"; + function init(content) { + const timeout = 60 * 120 * 1000; + Axios.defaults.timeout = timeout; + Vue.use(Vuetify); + Vue.component("pagination", Pagination); + Vue.component("pageheader", Pageheader); + Vue.component("helpdialog", HelpDialog); + let vue = new Vue({ + delimiters: ["[[", "]]"], + el: "#logs", + vuetify: new Vuetify(), + data() { + return { + calendarData: {}, + strings: content.strings, + groups: content.groups, + userid: content.userid, + courseid: content.courseid, + timezone: content.timezone, + render_has: content.profile_render, + courseRole: content.courseRole, + loading: false, + errors: [], + pages: content.pages, + help_dialog: false, + help_contents: [], + dateRules: [(v) => !!v || this.strings.logs_invalid_date], + }; + }, + beforeMount() { + document.querySelector("#downloadButtonMoodle").style.display = "none"; + document.querySelector("#downloadButtonNMP").style.display = "none"; + }, + mounted() { + document.querySelector(".v-application--wrap").style.minHeight = "60vh"; + document.querySelector("#sessions-loader").style.display = "none"; + document.querySelector("#helpMoodle").style.display = "block"; + document.querySelector("#helpNMP").style.display = "block"; + document.querySelector("#downloadButtonMoodle").style.display = "block"; + document.querySelector("#downloadButtonNMP").style.display = "block"; + }, + methods: { + get_Moodlefile() { + let lastDate = document.querySelector("#lastDateMoodle"); + let beginDate = document.querySelector("#beginDateMoodle"); + let timestampBeginDate = 0; + let timestampLastDate = 0; + let parsedBeginDate = []; + let parsedLastDate = []; + this.url = false; + this.loading = true; + var data = { + action: "downloadMOODLElogs", + courseid: this.courseid, + userid: this.userid, + beginDate: beginDate.value, + lastDate: lastDate.value, + currentUrl: window.location.href, + }; + if (beginDate.value != "" && lastDate.value != "") { + parsedBeginDate = beginDate.value.split("-"); + timestampBeginDate = new Date( + parsedBeginDate[0], + parsedBeginDate[1] - 1, + parsedBeginDate[2] + ); + parsedLastDate = lastDate.value.split("-"); + timestampLastDate = new Date( + parsedLastDate[0], + parsedLastDate[1] - 1, + parsedLastDate[2] + ); + if (timestampBeginDate.getTime() <= timestampLastDate.getTime()) { + if (timestampBeginDate.getTime() <= Date.now()) { + document.querySelector("#downloadButtonMoodle").innerHTML = + this.strings.logs_download_btn; + document.getElementById("downloadButtonMoodle").disabled = true; + Axios({ + method: "get", + url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", + timeout: timeout, + params: data, + }) + .then((response) => { + this.loading = false; + if (response.status == 200 && response.data.ok) { + let jsonData = response.data.data.jsonData; + jsonData = jsonData.map((row) => ({ + Role: row.user.role, + Email: row.user.email, + Fullname: row.user.firstname + " " + row.user.lastname, + Date: row.time.date, + Hour: row.time.hour, + ACTION_VERB: row.action.actionverb, + CourseID: row.course.courseid, + CourseName: row.course.coursename, + OBJECT_ID: row.action.objectid, + OBJECT_NAME: row.action.objectname, + OBJECT_TYPE: row.action.objecttype, + })); + let csvData = vue.objectToCSV(jsonData); + vue.downloadCSV(csvData, "Activity_Moodle"); + document.querySelector( + "#downloadButtonMoodle" + ).innerHTML = this.strings.logs_valid_Moodlebtn; + document.getElementById( + "downloadButtonMoodle" + ).disabled = false; + Alertify.success( + this.strings.logs_success_file_downloaded + ); + } else { + Alertify.error( + this.strings.logs_error_problem_encountered + ); + document.querySelector( + "#downloadButtonMoodle" + ).innerHTML = this.strings.logs_valid_Moodlebtn; + document.getElementById( + "downloadButtonMoodle" + ).disabled = false; } - }, - beforeMount() { - document.querySelector("#downloadButtonMoodle").style.display = "none"; - document.querySelector("#downloadButtonNMP").style.display = "none"; - }, - mounted() { - document.querySelector(".v-application--wrap").style.minHeight = "60vh"; - document.querySelector("#sessions-loader").style.display = "none"; - document.querySelector("#helpMoodle").style.display = "block"; - document.querySelector("#helpNMP").style.display = "block"; - document.querySelector("#downloadButtonMoodle").style.display = "block"; - document.querySelector("#downloadButtonNMP").style.display = "block"; - }, - methods: { - get_Moodlefile() { - let lastDate = document.querySelector("#lastDateMoodle"); - let beginDate = document.querySelector("#beginDateMoodle"); - let timestampBeginDate = 0; - let timestampLastDate = 0; - let parsedBeginDate = []; - let parsedLastDate = []; - this.url = false; - this.loading = true; - var data = { - action: "downloadMOODLElogs", - courseid: this.courseid, - userid: this.userid, - beginDate: beginDate.value, - lastDate: lastDate.value, - currentUrl: window.location.href, - } - if (beginDate.value != "" && lastDate.value != "") { - parsedBeginDate = beginDate.value.split("-"); - timestampBeginDate = new Date(parsedBeginDate[0], parsedBeginDate[1] - 1, parsedBeginDate[2]); - parsedLastDate = lastDate.value.split("-"); - timestampLastDate = new Date(parsedLastDate[0], parsedLastDate[1] - 1, parsedLastDate[2]); - if (timestampBeginDate.getTime() <= timestampLastDate.getTime()) { - if (timestampBeginDate.getTime() <= Date.now()) { - document.querySelector('#downloadButtonMoodle').innerHTML = this.strings.logs_download_btn; - document.getElementById('downloadButtonMoodle').disabled = true; - Axios({ - method: 'get', - url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", - timeout: timeout, - params: data, - }).then((response) => { - this.loading = false - if (response.status == 200 && response.data.ok) { - let jsonData = response.data.data.jsonData; - jsonData = jsonData.map(row => ({ - Role: row.user.role, - Email: row.user.email, - Fullname: row.user.firstname + ' ' + row.user.lastname, - Date: row.time.date, - Hour: row.time.hour, - ACTION_VERB: row.action.actionverb, - CourseID: row.course.courseid, - CourseName: row.course.coursename, - OBJECT_ID: row.action.objectid, - OBJECT_NAME: row.action.objectname, - OBJECT_TYPE: row.action.objecttype - })); - let csvData = vue.objectToCSV(jsonData); - vue.downloadCSV(csvData, "Activity_Moodle"); - document.querySelector('#downloadButtonMoodle').innerHTML = this.strings.logs_valid_Moodlebtn; - document.getElementById('downloadButtonMoodle').disabled = false; - Alertify.success(this.strings.logs_success_file_downloaded); - } else { - Alertify.error(this.strings.logs_error_problem_encountered); - document.querySelector('#downloadButtonMoodle').innerHTML = this.strings.logs_valid_Moodlebtn; - document.getElementById('downloadButtonMoodle').disabled = false; - } - }).catch((e) => { - Alertify.error(this.strings.logs_error_problem_encountered); - this.loading = false; - document.querySelector('#downloadButtonMoodle').innerHTML = this.strings.logs_valid_Moodlebtn; - document.getElementById('downloadButtonMoodle').disabled = false; - }).finally(() => { - this.loading = false; - document.querySelector('#downloadButtonMoodle').innerHTML = this.strings.logs_valid_Moodlebtn; - document.getElementById('downloadButtonMoodle').disabled = false; - }); - } else { - Alertify.error(this.strings.logs_error_begin_date_superior); - } - } else { - Alertify.error(this.strings.logs_error_begin_date_inferior); - } - } else { - Alertify.error(this.strings.logs_error_empty_dates); - } - }, - - objectToCSV(jsonData) { - let csvRows = []; + }) + .catch((e) => { + Alertify.error(this.strings.logs_error_problem_encountered); + this.loading = false; + document.querySelector("#downloadButtonMoodle").innerHTML = + this.strings.logs_valid_Moodlebtn; + document.getElementById( + "downloadButtonMoodle" + ).disabled = false; + }) + .finally(() => { + this.loading = false; + document.querySelector("#downloadButtonMoodle").innerHTML = + this.strings.logs_valid_Moodlebtn; + document.getElementById( + "downloadButtonMoodle" + ).disabled = false; + }); + } else { + Alertify.error(this.strings.logs_error_begin_date_superior); + } + } else { + Alertify.error(this.strings.logs_error_begin_date_inferior); + } + } else { + Alertify.error(this.strings.logs_error_empty_dates); + } + }, - let headers = Object.keys(jsonData[0]); - csvRows.push(headers.join(';')); + objectToCSV(jsonData) { + let csvRows = []; - for (const row of jsonData) { - const values = headers.map(header => { - const escaped = ('' + row[header]).replace(/"/g, '\\"'); - return `"${escaped}"`; - }) - csvRows.push(values.join(';')); - } + let headers = Object.keys(jsonData[0]); + csvRows.push(headers.join(";")); - return csvRows.join('\n'); - }, + for (const row of jsonData) { + const values = headers.map((header) => { + const escaped = ("" + row[header]).replace(/"/g, '\\"'); + return `"${escaped}"`; + }); + csvRows.push(values.join(";")); + } - downloadCSV(data, documentName) { - let blob = new Blob([data], { type: 'text/csv' }); - let url = window.URL.createObjectURL(blob); - const a = document.createElement('a'); - a.setAttribute('hidden', ''); - a.setAttribute('href', url); - a.setAttribute('download', documentName + '.csv'); - document.body.appendChild(a); - a.click(); - document.body.removeChild(a); - }, + return csvRows.join("\n"); + }, - getRapport() { - Alertify.confirm(this.strings.logs_download_details_description, - () => { - let path = M.cfg.wwwroot + "/local/notemyprogress/downloads/Details_Informations_LogsNMP.pdf"; - var link = document.createElement('a'); - link.href = path; - link.download = "Details_Informations_LogsNMP.pdf"; - link.click(); - Alertify.success(this.strings.logs_download_details_validation); - }).set({ title: this.strings.logs_download_details_title }) - .set({ labels: { cancel: this.strings.logs_download_details_cancel, ok: this.strings.logs_download_details_ok } }); - }, + downloadCSV(data, documentName) { + let blob = new Blob([data], { type: "text/csv" }); + let url = window.URL.createObjectURL(blob); + const a = document.createElement("a"); + a.setAttribute("hidden", ""); + a.setAttribute("href", url); + a.setAttribute("download", documentName + ".csv"); + document.body.appendChild(a); + a.click(); + document.body.removeChild(a); + }, - get_NMPfile() { - let lastDate = document.querySelector("#lastDateNMP"); - let beginDate = document.querySelector("#beginDateNMP"); - let timestampBeginDate = 0; - let timestampLastDate = 0; - let parsedBeginDate = []; - let parsedLastDate = []; - this.url = false; - this.loading = true; - var data = { - action: "downloadNMPlogs", - courseid: this.courseid, - userid: this.userid, - beginDate: beginDate.value, - lastDate: lastDate.value, - currentUrl: window.location.href, - } - if (beginDate.value != "" && lastDate.value != "") { - parsedBeginDate = beginDate.value.split("-"); - timestampBeginDate = new Date(parsedBeginDate[0], parsedBeginDate[1] - 1, parsedBeginDate[2]); - parsedLastDate = lastDate.value.split("-"); - timestampLastDate = new Date(parsedLastDate[0], parsedLastDate[1] - 1, parsedLastDate[2]); - if (timestampBeginDate.getTime() <= timestampLastDate.getTime()) { - if (timestampBeginDate.getTime() <= Date.now()) { - document.querySelector('#downloadButtonNMP').innerHTML = this.strings.logs_download_btn; - document.getElementById('downloadButtonNMP').disabled = true; - Axios({ - method: 'get', - url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", - timeout: timeout, - params: data, - }).then((response) => { - this.loading = false - if (response.status == 200 && response.data.ok) { - if (beginDate.value != "" || lastDate.value != "") { - let jsonData = response.data.data.jsonData; - jsonData = jsonData.map(row => ({ - Role: row.user.role, - Email: row.user.email, - Username: row.user.username, - Fullname: row.user.fullname, - Date: row.time.date, - Hour: row.time.hour, - CourseID: row.course.courseid, - SECTION_NAME: row.action.sectionname, - ACTION_TYPE: row.action.actiontype - })); - let csvData = vue.objectToCSV(jsonData); - vue.downloadCSV(csvData, "Activity_NoteMyProgress"); - document.querySelector('#downloadButtonNMP').innerHTML = this.strings.logs_valid_NMPbtn; - document.getElementById('downloadButtonNMP').disabled = false; - Alertify.success(this.strings.logs_success_file_downloaded); - } - } else { - Alertify.error(this.strings.logs_error_problem_encountered); - document.querySelector('#downloadButtonNMP').innerHTML = this.strings.logs_valid_NMPbtn; - document.getElementById('downloadButtonNMP').disabled = false; - } - }).catch((e) => { - Alertify.error(this.strings.logs_error_problem_encountered); - this.loading = false; - document.querySelector('#downloadButtonNMP').innerHTML = this.strings.logs_valid_NMPbtn; - document.getElementById('downloadButtonNMP').disabled = false; - }).finally(() => { - this.loading = false; - document.querySelector('#downloadButtonNMP').innerHTML = this.strings.logs_valid_NMPbtn; - document.getElementById('downloadButtonNMP').disabled = false; - }); - } else { //Si la date de début est supérieure à la date de fin - Alertify.error(this.strings.logs_error_begin_date_superior); - } - } else { //Si la date de début est inférieure à la date du jour - Alertify.error(this.strings.logs_error_begin_date_inferior); - } - } else { //Si les dates ne sont pas remplies - Alertify.error(this.strings.logs_error_empty_dates); - } - }, + getRapport() { + Alertify.confirm( + this.strings.logs_download_details_description, + () => { + let path = + M.cfg.wwwroot + + "/local/notemyprogress/downloads/Details_Informations_LogsNMP.pdf"; + var link = document.createElement("a"); + link.href = path; + link.download = "Details_Informations_LogsNMP.pdf"; + link.click(); + Alertify.success(this.strings.logs_download_details_validation); + } + ) + .set({ title: this.strings.logs_download_details_title }) + .set({ + labels: { + cancel: this.strings.logs_download_details_cancel, + ok: this.strings.logs_download_details_ok, + }, + }); + }, - get_help_content() { - var help_contents = []; - var help = new Object(); - help.title = this.strings.title; - help.description = this.strings.description; - help_contents.push(help); - return help_contents; - }, - - open_chart_help(chart) { - let contents = []; - if (chart == 'download_moodle') { - contents.push({ - title: this.strings.logs_download_moodle_help_title, - description: this.strings.logs_download_moodle_help_description, - }); - } else if (chart == "download_nmp") { - contents.push({ - title: this.strings.logs_download_nmp_help_title, - description: this.strings.logs_download_nmp_help_description, - }); - } - this.help_contents = contents; - if (this.help_contents.length) { - this.help_dialog = true; - } - }, + get_NMPfile() { + let lastDate = document.querySelector("#lastDateNMP"); + let beginDate = document.querySelector("#beginDateNMP"); + let timestampBeginDate = 0; + let timestampLastDate = 0; + let parsedBeginDate = []; + let parsedLastDate = []; + this.url = false; + this.loading = true; + var data = { + action: "downloadNMPlogs", + courseid: this.courseid, + userid: this.userid, + beginDate: beginDate.value, + lastDate: lastDate.value, + currentUrl: window.location.href, + }; + if (beginDate.value != "" && lastDate.value != "") { + parsedBeginDate = beginDate.value.split("-"); + timestampBeginDate = new Date( + parsedBeginDate[0], + parsedBeginDate[1] - 1, + parsedBeginDate[2] + ); + parsedLastDate = lastDate.value.split("-"); + timestampLastDate = new Date( + parsedLastDate[0], + parsedLastDate[1] - 1, + parsedLastDate[2] + ); + if (timestampBeginDate.getTime() <= timestampLastDate.getTime()) { + if (timestampBeginDate.getTime() <= Date.now()) { + document.querySelector("#downloadButtonNMP").innerHTML = + this.strings.logs_download_btn; + document.getElementById("downloadButtonNMP").disabled = true; + Axios({ + method: "get", + url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", + timeout: timeout, + params: data, + }) + .then((response) => { + this.loading = false; + if (response.status == 200 && response.data.ok) { + if (beginDate.value != "" || lastDate.value != "") { + let jsonData = response.data.data.jsonData; + jsonData = jsonData.map((row) => ({ + Role: row.user.role, + Email: row.user.email, + Username: row.user.username, + Fullname: row.user.fullname, + Date: row.time.date, + Hour: row.time.hour, + CourseID: row.course.courseid, + SECTION_NAME: row.action.sectionname, + ACTION_TYPE: row.action.actiontype, + })); + let csvData = vue.objectToCSV(jsonData); + vue.downloadCSV(csvData, "Activity_NoteMyProgress"); + document.querySelector("#downloadButtonNMP").innerHTML = + this.strings.logs_valid_NMPbtn; + document.getElementById( + "downloadButtonNMP" + ).disabled = false; + Alertify.success( + this.strings.logs_success_file_downloaded + ); + } + } else { + Alertify.error( + this.strings.logs_error_problem_encountered + ); + document.querySelector("#downloadButtonNMP").innerHTML = + this.strings.logs_valid_NMPbtn; + document.getElementById( + "downloadButtonNMP" + ).disabled = false; + } + }) + .catch((e) => { + Alertify.error(this.strings.logs_error_problem_encountered); + this.loading = false; + document.querySelector("#downloadButtonNMP").innerHTML = + this.strings.logs_valid_NMPbtn; + document.getElementById( + "downloadButtonNMP" + ).disabled = false; + }) + .finally(() => { + this.loading = false; + document.querySelector("#downloadButtonNMP").innerHTML = + this.strings.logs_valid_NMPbtn; + document.getElementById( + "downloadButtonNMP" + ).disabled = false; + }); + } else { + //Si la date de début est supérieure à la date de fin + Alertify.error(this.strings.logs_error_begin_date_superior); + } + } else { + //Si la date de début est inférieure à la date du jour + Alertify.error(this.strings.logs_error_begin_date_inferior); + } + } else { + //Si les dates ne sont pas remplies + Alertify.error(this.strings.logs_error_empty_dates); + } + }, - update_help_dialog(value) { - this.help_dialog = value; - }, + get_help_content() { + var help_contents = []; + var help = new Object(); + help.title = this.strings.title; + help.description = this.strings.description; + help_contents.push(help); + return help_contents; + }, - get_timezone() { - let information = `${this.strings.ss_change_timezone} ${this.timezone}` - return information; - }, - } - }) - } + open_chart_help(chart) { + let contents = []; + if (chart == "download_moodle") { + contents.push({ + title: this.strings.logs_download_moodle_help_title, + description: this.strings.logs_download_moodle_help_description, + }); + } else if (chart == "download_nmp") { + contents.push({ + title: this.strings.logs_download_nmp_help_title, + description: this.strings.logs_download_nmp_help_description, + }); + } + this.help_contents = contents; + if (this.help_contents.length) { + this.help_dialog = true; + } + }, - return { - init: init - }; - }); \ No newline at end of file + update_help_dialog(value) { + this.help_dialog = value; + }, + get_timezone() { + let information = `${this.strings.ss_change_timezone} ${this.timezone}`; + return information; + }, + }, + }); + } + return { + init: init, + }; +}); diff --git a/notemyprogress/amd/src/sessions.js b/notemyprogress/amd/src/sessions.js index 14d0d7990d3de6af94f026f4cfbbcb9dda6c3931..8d9df751b9905c8312ec27ab41c406ebc347749a 100644 --- a/notemyprogress/amd/src/sessions.js +++ b/notemyprogress/amd/src/sessions.js @@ -1,388 +1,438 @@ -define(["local_notemyprogress/vue", - "local_notemyprogress/vuetify", - "local_notemyprogress/axios", - "local_notemyprogress/moment", - "local_notemyprogress/pagination", - "local_notemyprogress/chartstatic", - "local_notemyprogress/pageheader", - "local_notemyprogress/helpdialog", - ], - function(Vue, Vuetify, Axios, Moment, Pagination, ChartStatic, PageHeader, HelpDialog) { - "use strict"; +define([ + "local_notemyprogress/vue", + "local_notemyprogress/vuetify", + "local_notemyprogress/axios", + "local_notemyprogress/moment", + "local_notemyprogress/pagination", + "local_notemyprogress/chartstatic", + "local_notemyprogress/pageheader", + "local_notemyprogress/helpdialog", +], function ( + Vue, + Vuetify, + Axios, + Moment, + Pagination, + ChartStatic, + PageHeader, + HelpDialog +) { + "use strict"; - function init(content) { - // console.log(content); - Vue.use(Vuetify); - Vue.component('pagination', Pagination); - Vue.component('chart', ChartStatic); - Vue.component('pageheader', PageHeader); - Vue.component('helpdialog', HelpDialog); - let vue = new Vue({ - delimiters: ["[[", "]]"], - el: "#work_sessions", - vuetify: new Vuetify(), - data() { - return { - strings : content.strings, - groups : content.groups, - userid : content.userid, - courseid : content.courseid, - timezone : content.timezone, - render_has : content.profile_render, - loading : false, - errors : [], + function init(content) { + // console.log(content); + Vue.use(Vuetify); + Vue.component("pagination", Pagination); + Vue.component("chart", ChartStatic); + Vue.component("pageheader", PageHeader); + Vue.component("helpdialog", HelpDialog); + let vue = new Vue({ + delimiters: ["[[", "]]"], + el: "#work_sessions", + vuetify: new Vuetify(), + data() { + return { + strings: content.strings, + groups: content.groups, + userid: content.userid, + courseid: content.courseid, + timezone: content.timezone, + render_has: content.profile_render, + loading: false, + errors: [], - pages : content.pages, - hours_sessions: content.indicators.sessions, - session_count: content.indicators.count, - inverted_time: content.indicators.time, - inverted_time_colors: content.inverted_time_colors, - sessions_count_colors: content.sessions_count_colors, + pages: content.pages, + hours_sessions: content.indicators.sessions, + session_count: content.indicators.count, + inverted_time: content.indicators.time, + inverted_time_colors: content.inverted_time_colors, + sessions_count_colors: content.sessions_count_colors, - search: null, + search: null, - help_dialog: false, - help_contents: [], - } - }, - mounted(){ - document.querySelector("#sessions-loader").style.display = "none"; - document.querySelector("#work_sessions").style.display = "block"; - setTimeout(function() { - vue.setGraphicsEventListeners(); - }, 500); - }, - methods : { - get_help_content(){ - let contents = []; - contents.push({ - title: this.strings.section_help_title, - description: this.strings.section_help_description, - }); - return contents; - }, + help_dialog: false, + help_contents: [], + }; + }, + mounted() { + document.querySelector("#sessions-loader").style.display = "none"; + document.querySelector("#work_sessions").style.display = "block"; + setTimeout(function () { + vue.setGraphicsEventListeners(); + }, 500); + }, + methods: { + get_help_content() { + let contents = []; + contents.push({ + title: this.strings.section_help_title, + description: this.strings.section_help_description, + }); + return contents; + }, - update_interactions(week){ - this.loading = true; - this.errors = []; - let data = { - action : "worksessions", - userid : this.userid, - courseid : this.courseid, - weekcode : week.weekcode, - profile : this.render_has, - } - Axios({ - method:'get', - url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", - params : data, - }).then((response) => { - if (response.status == 200 && response.data.ok) { - this.hours_sessions = response.data.data.indicators.sessions; - this.session_count = response.data.data.indicators.count; - this.inverted_time = response.data.data.indicators.time; - } else { - this.error_messages.push(this.strings.error_network); - } - }).catch((e) => { - this.errors.push(this.strings.api_error_network); - }).finally(() => { - this.loading = false; - //Ici, la page a fini de charger - vue.addLogsIntoDB("viewed", "week_"+week.weekcode, "week_section", "Week section that allows you to obtain information on a specific week"); - vue.setGraphicsEventListeners(); - }); - return this.data; - }, + update_interactions(week) { + this.loading = true; + this.errors = []; + let data = { + action: "worksessions", + userid: this.userid, + courseid: this.courseid, + weekcode: week.weekcode, + profile: this.render_has, + }; + Axios({ + method: "get", + url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", + params: data, + }) + .then((response) => { + if (response.status == 200 && response.data.ok) { + this.hours_sessions = response.data.data.indicators.sessions; + this.session_count = response.data.data.indicators.count; + this.inverted_time = response.data.data.indicators.time; + } else { + this.error_messages.push(this.strings.error_network); + } + }) + .catch((e) => { + this.errors.push(this.strings.api_error_network); + }) + .finally(() => { + this.loading = false; + //Ici, la page a fini de charger + vue.addLogsIntoDB( + "viewed", + "week_" + week.weekcode, + "week_section", + "Week section that allows you to obtain information on a specific week" + ); + vue.setGraphicsEventListeners(); + }); + return this.data; + }, - get_point_category_name(point, dimension) { - let series = point.series, - isY = dimension === 'y', - axis = series[isY ? 'yAxis' : 'xAxis']; - return axis.categories[point[isY ? 'y' : 'x']]; - }, + get_point_category_name(point, dimension) { + let series = point.series, + isY = dimension === "y", + axis = series[isY ? "yAxis" : "xAxis"]; + return axis.categories[point[isY ? "y" : "x"]]; + }, - build_hours_sessions_chart() { - let chart = new Object(); - chart.title = { - text: null, - }; - chart.chart = { - type: 'heatmap', - backgroundColor: null, - style: {fontFamily: 'poppins'}, - }; - chart.xAxis = { - categories: this.strings.days, - }; - chart.yAxis = { - categories: this.strings.hours, - title: null, - reversed: true, - }; - chart.colorAxis = { - min: 0, - minColor: '#E0E0E0', - maxColor: '#118AB2' - }; - chart.legend = { - layout: 'horizontal', - verticalAlign: 'bottom', - }; - chart.tooltip = { - formatter: function () { - let xCategoryName = vue.get_point_category_name(this.point, 'x'); - let yCategoryName = vue.get_point_category_name(this.point, 'y'); - let label = vue.strings.sessions_text; - if (this.point.value == 1) { - label = vue.strings.session_text; - } - return '<b>' + xCategoryName + ' ' + yCategoryName + '</b>: ' - + this.point.value +' ' + label; - } - }; - chart.series = [{ - borderWidth: 2, - borderColor: '#FAFAFA', - data: this.hours_sessions, - }]; - return chart; - }, + build_hours_sessions_chart() { + let chart = new Object(); + chart.title = { + text: null, + }; + chart.chart = { + type: "heatmap", + backgroundColor: null, + style: { fontFamily: "poppins" }, + }; + chart.xAxis = { + categories: this.strings.days, + }; + chart.yAxis = { + categories: this.strings.hours, + title: null, + reversed: true, + }; + chart.colorAxis = { + min: 0, + minColor: "#E0E0E0", + maxColor: "#118AB2", + }; + chart.legend = { + layout: "horizontal", + verticalAlign: "bottom", + }; + chart.tooltip = { + formatter: function () { + let xCategoryName = vue.get_point_category_name(this.point, "x"); + let yCategoryName = vue.get_point_category_name(this.point, "y"); + let label = vue.strings.sessions_text; + if (this.point.value == 1) { + label = vue.strings.session_text; + } + return ( + "<b>" + + xCategoryName + + " " + + yCategoryName + + "</b>: " + + this.point.value + + " " + + label + ); + }, + }; + chart.series = [ + { + borderWidth: 2, + borderColor: "#FAFAFA", + data: this.hours_sessions, + }, + ]; + return chart; + }, - build_inverted_time_chart() { - let chart = new Object(); - chart.chart = { - type: 'bar', - backgroundColor: null, - style: {fontFamily: 'poppins'}, - }; - chart.title = { - text: null, - }; - chart.colors = this.inverted_time_colors; - chart.xAxis = { - type: 'category', - crosshair: true, - }; - chart.yAxis = { - title: { - text: this.strings.time_inverted_x_axis, - } - }; - chart.tooltip = { - shared:true, - useHTML:true, - formatter: function () { - let category_name = this.points[0].key; - let time = vue.convert_time(this.y); - return `<b>${category_name}: </b>${time}`; - } - }; - chart.legend = { - enabled: false - }; - chart.series = [{ - colorByPoint: true, - data: this.inverted_time.data - }]; - return chart; - }, + build_inverted_time_chart() { + let chart = new Object(); + chart.chart = { + type: "bar", + backgroundColor: null, + style: { fontFamily: "poppins" }, + }; + chart.title = { + text: null, + }; + chart.colors = this.inverted_time_colors; + chart.xAxis = { + type: "category", + crosshair: true, + }; + chart.yAxis = { + title: { + text: this.strings.time_inverted_x_axis, + }, + }; + chart.tooltip = { + shared: true, + useHTML: true, + formatter: function () { + let category_name = this.points[0].key; + let time = vue.convert_time(this.y); + return `<b>${category_name}: </b>${time}`; + }, + }; + chart.legend = { + enabled: false, + }; + chart.series = [ + { + colorByPoint: true, + data: this.inverted_time.data, + }, + ]; + return chart; + }, - build_sessions_count_chart() { - let chart = new Object(); - chart.chart = { - backgroundColor: null, - style: {fontFamily: 'poppins'}, - }; - chart.title = { - text: null, - }; - chart.colors = this.sessions_count_colors; - chart.yAxis = { - title: { - text: this.strings.session_count_yaxis_title, - }, - allowDecimals: false - }; - chart.xAxis = { - categories: this.session_count.categories, - }; - chart.tooltip = { - valueSuffix: this.strings.session_count_tooltip_suffix, - }; - chart.legend = { - layout: 'horizontal', - verticalAlign: 'bottom', - }; - chart.series = this.session_count.data - return chart; - }, + build_sessions_count_chart() { + let chart = new Object(); + chart.chart = { + backgroundColor: null, + style: { fontFamily: "poppins" }, + }; + chart.title = { + text: null, + }; + chart.colors = this.sessions_count_colors; + chart.yAxis = { + title: { + text: this.strings.session_count_yaxis_title, + }, + allowDecimals: false, + }; + chart.xAxis = { + categories: this.session_count.categories, + }; + chart.tooltip = { + valueSuffix: this.strings.session_count_tooltip_suffix, + }; + chart.legend = { + layout: "horizontal", + verticalAlign: "bottom", + }; + chart.series = this.session_count.data; + return chart; + }, - convert_time(time) { - time *= 3600; // pasar las horas a segundos - let h = this.strings.hours_short; - let m = this.strings.minutes_short; - let s = this.strings.seconds_short; - let hours = Math.floor(time / 3600); - let minutes = Math.floor((time % 3600) / 60); - let seconds = Math.floor(time % 60); - let text; - if (hours >= 1) { - if (minutes >= 1) { - text = `${hours}${h} ${minutes}${m}`; - } else { - text = `${hours}${h}`; - } - } else if ((minutes >= 1)) { - if (seconds >= 1) { - text = `${minutes}${m} ${seconds}${s}`; - } else { - text = `${minutes}${m}`; - } - } else { - text = `${seconds}${s}`; - } - return text; - }, + convert_time(time) { + time *= 3600; // pasar las horas a segundos + let h = this.strings.hours_short; + let m = this.strings.minutes_short; + let s = this.strings.seconds_short; + let hours = Math.floor(time / 3600); + let minutes = Math.floor((time % 3600) / 60); + let seconds = Math.floor(time % 60); + let text; + if (hours >= 1) { + if (minutes >= 1) { + text = `${hours}${h} ${minutes}${m}`; + } else { + text = `${hours}${h}`; + } + } else if (minutes >= 1) { + if (seconds >= 1) { + text = `${minutes}${m} ${seconds}${s}`; + } else { + text = `${minutes}${m}`; + } + } else { + text = `${seconds}${s}`; + } + return text; + }, - open_chart_help(chart) { - let contents = []; - var action = ""; - var objectName = ""; - var objectType = ""; - var objectDescription = ""; - if (chart == "inverted_time") { - contents.push({ - title: this.strings.inverted_time_help_title, - description: this.strings.inverted_time_help_description_p1, - }); - contents.push({ - description: this.strings.inverted_time_help_description_p2, - }); - action = "viewed"; - objectType = "help"; - objectName = "invested_time"; - objectDescription = "Help section that provides information about the sessions per week chart"; - vue.addLogsIntoDB(action, objectName, objectType, objectDescription); - } else if (chart == "hours_sessions") { - contents.push({ - title: this.strings.hours_sessions_help_title, - description: this.strings.hours_sessions_help_description_p1, - }); - contents.push({ - description: this.strings.hours_sessions_help_description_p2, - }); - action = "viewed"; - objectType = "help"; - objectName = "hours_sessions"; - objectDescription = "Help section that provides information about the sessions per hour chart"; - vue.addLogsIntoDB(action, objectName, objectType, objectDescription); - } else if (chart == "sessions_count") { - contents.push({ - title: this.strings.sessions_count_help_title, - description: this.strings.sessions_count_help_description_p1, - }); - contents.push({ - description: this.strings.sessions_count_help_description_p2, - }); - action = "viewed"; - objectType = "help"; - objectName = "sessions_count"; - objectDescription = "Help section that provides information about the invested time chart"; - vue.addLogsIntoDB(action, objectName, objectType, objectDescription); - } - this.help_contents = contents; - if (this.help_contents.length) { - this.help_dialog = true; - } - }, + open_chart_help(chart) { + let contents = []; + var action = ""; + var objectName = ""; + var objectType = ""; + var objectDescription = ""; + if (chart == "inverted_time") { + contents.push({ + title: this.strings.inverted_time_help_title, + description: this.strings.inverted_time_help_description_p1, + }); + contents.push({ + description: this.strings.inverted_time_help_description_p2, + }); + action = "viewed"; + objectType = "help"; + objectName = "invested_time"; + objectDescription = + "Help section that provides information about the sessions per week chart"; + vue.addLogsIntoDB( + action, + objectName, + objectType, + objectDescription + ); + } else if (chart == "hours_sessions") { + contents.push({ + title: this.strings.hours_sessions_help_title, + description: this.strings.hours_sessions_help_description_p1, + }); + contents.push({ + description: this.strings.hours_sessions_help_description_p2, + }); + action = "viewed"; + objectType = "help"; + objectName = "hours_sessions"; + objectDescription = + "Help section that provides information about the sessions per hour chart"; + vue.addLogsIntoDB( + action, + objectName, + objectType, + objectDescription + ); + } else if (chart == "sessions_count") { + contents.push({ + title: this.strings.sessions_count_help_title, + description: this.strings.sessions_count_help_description_p1, + }); + contents.push({ + description: this.strings.sessions_count_help_description_p2, + }); + action = "viewed"; + objectType = "help"; + objectName = "sessions_count"; + objectDescription = + "Help section that provides information about the invested time chart"; + vue.addLogsIntoDB( + action, + objectName, + objectType, + objectDescription + ); + } + this.help_contents = contents; + if (this.help_contents.length) { + this.help_dialog = true; + } + }, - update_help_dialog (value) { - this.help_dialog = value; - }, + update_help_dialog(value) { + this.help_dialog = value; + }, - get_timezone(){ - let information = `${this.strings.ss_change_timezone} ${this.timezone}` - return information; - }, + get_timezone() { + let information = `${this.strings.ss_change_timezone} ${this.timezone}`; + return information; + }, - setGraphicsEventListeners() { - let graphics = document.querySelectorAll('.highcharts-container'); - if(graphics.length<1) { - setTimeout(vue.setGraphicsEventListeners,500); - } else { - graphics[0].id="investedTime"; - graphics[1].id="sessionsPerHour"; - graphics[2].id="sessionsPerWeek"; - graphics.forEach((graph) => { - graph.addEventListener('mouseenter', vue.addLogsViewGraphic); - }) - } - }, + setGraphicsEventListeners() { + let graphics = document.querySelectorAll(".highcharts-container"); + if (graphics.length < 1) { + setTimeout(vue.setGraphicsEventListeners, 500); + } else { + graphics[0].id = "investedTime"; + graphics[1].id = "sessionsPerHour"; + graphics[2].id = "sessionsPerWeek"; + graphics.forEach((graph) => { + graph.addEventListener("mouseenter", vue.addLogsViewGraphic); + }); + } + }, - addLogsViewGraphic(e) { - event.stopPropagation(); - var action = ""; - var objectName = ""; - var objectType = ""; - var objectDescription = ""; - switch(e.target.id) { - case "investedTime": - action = "viewed"; - objectName = "invested_time"; - objectType = "chart"; - objectDescription = "Bar chart that shows the average time invested by students as a function of the expected invested time"; - break; - case "sessionsPerHour": - action = "viewed"; - objectName = "hours_sessions"; - objectType = "chart"; - objectDescription = "Chart showing the number of sessions performed according to the time of day"; - break; - case "sessionsPerWeek": - action = "viewed"; - objectName = "sessions_count"; - objectType = "chart"; - objectDescription = "Chart showing the number of sessions performed per week"; - break; - default: - action = "viewed"; - objectName = ""; - objectType = "chart"; - objectDescription = "A chart"; - break; - } - vue.addLogsIntoDB(action, objectName, objectType, objectDescription); - }, + addLogsViewGraphic(e) { + event.stopPropagation(); + var action = ""; + var objectName = ""; + var objectType = ""; + var objectDescription = ""; + switch (e.target.id) { + case "investedTime": + action = "viewed"; + objectName = "invested_time"; + objectType = "chart"; + objectDescription = + "Bar chart that shows the average time invested by students as a function of the expected invested time"; + break; + case "sessionsPerHour": + action = "viewed"; + objectName = "hours_sessions"; + objectType = "chart"; + objectDescription = + "Chart showing the number of sessions performed according to the time of day"; + break; + case "sessionsPerWeek": + action = "viewed"; + objectName = "sessions_count"; + objectType = "chart"; + objectDescription = + "Chart showing the number of sessions performed per week"; + break; + default: + action = "viewed"; + objectName = ""; + objectType = "chart"; + objectDescription = "A chart"; + break; + } + vue.addLogsIntoDB(action, objectName, objectType, objectDescription); + }, - addLogsIntoDB(action, objectName, objectType, objectDescription) { - let data = { - courseid: content.courseid, - userid: content.userid, - action: "addLogs", - sectionname: "TEACHER_STUDY_SESSIONS", - actiontype: action, - objectType: objectType, - objectName: objectName, - currentUrl: document.location.href, - objectDescription: objectDescription, - }; - Axios({ - method:'get', - url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", - params : data, - }).then((response) => { - if (response.status == 200 && response.data.ok) { - } - }).catch((e) => { - }); - }, - } + addLogsIntoDB(action, objectName, objectType, objectDescription) { + let data = { + courseid: content.courseid, + userid: content.userid, + action: "addLogs", + sectionname: "TEACHER_STUDY_SESSIONS", + actiontype: action, + objectType: objectType, + objectName: objectName, + currentUrl: document.location.href, + objectDescription: objectDescription, + }; + Axios({ + method: "get", + url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", + params: data, + }) + .then((response) => { + if (response.status == 200 && response.data.ok) { + } }) + .catch((e) => {}); + }, + }, + }); + } - } - - return { - init : init - }; - }); \ No newline at end of file + return { + init: init, + }; +}); diff --git a/notemyprogress/amd/src/teacher.js b/notemyprogress/amd/src/teacher.js deleted file mode 100644 index 516f045979325e9d354104572559ed77df141938..0000000000000000000000000000000000000000 --- a/notemyprogress/amd/src/teacher.js +++ /dev/null @@ -1,268 +0,0 @@ -define(["local_notemyprogress/vue", - "local_notemyprogress/vuetify", - "local_notemyprogress/axios", - "local_notemyprogress/pagination", - "local_notemyprogress/chartstatic", - "local_notemyprogress/pageheader", - "local_notemyprogress/helpdialog", - ], - function(Vue, Vuetify, Axios, Pagination, ChartStatic, PageHeader, HelpDialog) { - "use strict"; - - function init(content) { - // console.log(content); - Vue.use(Vuetify); - Vue.component('pagination', Pagination); - Vue.component('chart', ChartStatic); - Vue.component('pageheader', PageHeader); - Vue.component('helpdialog', HelpDialog); - let vue = new Vue({ - delimiters: ["[[", "]]"], - el: "#teacher", - vuetify: new Vuetify(), - data() { - return { - strings : content.strings, - groups : content.groups, - userid : content.userid, - courseid : content.courseid, - timezone : content.timezone, - render_has : content.profile_render, - - indicators: content.indicators, - week_resources_colors: content.week_resources_colors, - search: null, - week_resources_categories: [], - week_resources_data: [], - - help_dialog: false, - help_contents: [], - } - }, - beforeMount(){ - this.calculate_week_resources(); - }, - mounted(){ - document.querySelector("#sessions-loader").style.display = "none"; - document.querySelector("#teacher").style.display = "block"; - }, - methods : { - get_help_content(){ - let contents = []; - contents.push({ - title: this.strings.section_help_title, - description: this.strings.section_help_description, - }); - return contents; - }, - - get_course_grade(){ - let grade = Number(this.indicators.course.grademax); - return (this.isInt(grade)) ? grade : grade.toFixed(2); - }, - - calculate_week_resources() { - let categories = [], data = []; - let week_name; - this.indicators.weeks.forEach(week => { - week_name = `${week.name} ${(week.position+1)}`; - categories.push(week_name); - data.push(week.cms); - }); - let name = this.capitalizeFirstLetter(this.strings.teacher_indicators_modules); - this.week_resources_categories = categories; - this.week_resources_data = [{ name, data}]; - }, - - build_week_resources_chart() { - let chart = new Object(); - chart.chart = { - type: 'bar', - backgroundColor: null, - style: {fontFamily: 'poppins'}, - }; - chart.title = { - text: null, - }; - chart.colors = this.week_resources_colors; - chart.xAxis = { - categories: this.week_resources_categories - }; - chart.yAxis = { - min: 0, - title: { - text: this.strings.teacher_indicators_week_resources_yaxis_title - } - }; - chart.legend = { - enabled: false - }; - chart.series = this.week_resources_data; - return chart; - }, - - build_weeks_sessions_chart() { - let chart = new Object(); - chart.chart = { - type: 'heatmap', - backgroundColor: null, - style: {fontFamily: 'poppins'}, - }; - chart.title = { - text: null, - }; - chart.xAxis = { - categories: this.strings.weeks, - }; - chart.yAxis = { - categories: this.indicators.sessions.categories, - title: null, - reversed: true, - }; - chart.colorAxis = { - min: 0, - minColor: '#E0E0E0', - maxColor: '#118AB2' - }; - chart.legend = { - layout: 'horizontal', - verticalAlign: 'bottom', - }; - chart.tooltip = { - formatter: function () { - let days = vue.indicators.sessions.weeks[this.point.y][this.point.x] || ''; - let xCategoryName = vue.get_point_category_name(this.point, 'x'); - let yCategoryName = vue.get_point_category_name(this.point, 'y'); - let label = vue.strings.teacher_indicators_sessions; - if (this.point.value == 1) { - label = vue.strings.teacher_indicators_session; - } - return '<b>' + yCategoryName + ' ' + xCategoryName + '</b>: ' - + this.point.value +' ' + label + '<br/>' + days; - } - }; - chart.series = [{ - borderWidth: 2, - borderColor: '#FAFAFA', - data: this.indicators.sessions.data, - }]; - return chart; - }, - - table_headers(){ - let headers = [ - { text: '', value : 'id', align : 'center', sortable : false}, - { text: this.strings.thead_name , value : 'firstname'}, - { text: this.strings.thead_lastname , value : 'lastname'}, - { text: this.strings.thead_email , value : 'email'}, - { text: this.strings.thead_progress , value : 'progress_percentage', align : 'center'}, - { text: this.strings.thead_sessions , value : 'sessions_number', align : 'center'}, - { text: this.strings.thead_time , value : 'inverted_time', align : 'center'}, - ]; - return headers; - }, - - get_picture_url(userid){ - let url = `${M.cfg.wwwroot}/user/pix.php?file=/${userid}/f1.jpg`; - return url; - }, - - get_percentage_progress(value){ - return `${value} %`; - }, - - get_progress_tooltip(item){ - let module_label = this.strings.teacher_indicators_modules; - let finished_label = this.strings.teacher_indicators_finished; - if (item.cms.complete == 1) { - module_label = this.strings.teacher_indicators_module; - finished_label = this.strings.teacher_indicators_finalized; - } - return `${item.cms.complete} ${module_label} ${finished_label} ${this.strings.of_conector} ${item.cms.total}`; - }, - - get_point_category_name(point, dimension) { - let series = point.series, - isY = dimension === 'y', - axis = series[isY ? 'yAxis' : 'xAxis']; - return axis.categories[point[isY ? 'y' : 'x']]; - }, - - capitalizeFirstLetter(string) { - return string.charAt(0).toUpperCase() + string.slice(1); - }, - - isInt(n) { - return n % 1 === 0; - }, - - open_chart_help(chart) { - let contents = []; - if (chart == "week_resources") { - contents.push({ - title: this.strings.week_resources_help_title, - description: this.strings.week_resources_help_description_p1, - }); - contents.push({ - description: this.strings.week_resources_help_description_p2, - }); - } else if (chart == "weeks_sessions") { - contents.push({ - title: this.strings.weeks_sessions_help_title, - description: this.strings.week_sessions_help_description_p1, - }); - contents.push({ - description: this.strings.week_sessions_help_description_p2, - }); - } else if (chart == "progress_table") { - contents.push({ - title: this.strings.progress_table_help_title, - description: this.strings.progress_table_help_description, - }); - } - this.help_contents = contents; - if (this.help_contents.length) { - this.help_dialog = true; - } - }, - - update_help_dialog (value) { - this.help_dialog = value; - }, - - get_timezone(){ - let information = `${this.strings.ss_change_timezone} ${this.timezone}` - return information; - }, - - addLogsIntoDB(action, objectName, objectType, objectDescription) { - let data = { - courseid: content.courseid, - userid: content.userid, - action: "addLogs", - sectionname: "TEACHER_GENERAL_INDICATORS", - actiontype: action, - objectType: objectType, - objectName: objectName, - currentUrl: document.location.href, - objectDescription: objectDescription, - }; - Axios({ - method:'get', - url: M.cfg.wwwroot + "/local/notemyprogress/ajax.php", - params : data, - }).then((response) => { - if (response.status == 200 && response.data.ok) { - } - }).catch((e) => { - }); - }, - - } - }) - } - - return { - init : init - }; - }); \ No newline at end of file diff --git a/notemyprogress/classes/configgamification.php b/notemyprogress/classes/configgamification.php index 0a4f947f7ea560db8d1b18b4322ca10f727f5ffc..3fb9d502f2dcd001c6e220e6eddce06380f743c5 100644 --- a/notemyprogress/classes/configgamification.php +++ b/notemyprogress/classes/configgamification.php @@ -115,7 +115,12 @@ class configgamification { $levelsData->levelsdata = json_decode($levelsData->levelsdata); $levelsData->settings = json_decode($levelsData->settings); $levelsData->rules = json_decode($levelsData->rules); - } + + + + } + $sql = "SELECT enablegamification from {notemyprogress_gamification} where courseid=? AND timecreated=? "; + $value = $DB->get_record_sql($sql, array("courseid="=>($courseid),"timecreated="=>$timecrated->maximum)); return $levelsData; } @@ -193,19 +198,6 @@ class configgamification { $list = []; $coreevents = $this->get_core_events(); -//! $list[] = [get_string('coresystem') => array_reduce(array_keys($coreevents), function($carry, $prefix) use ($coreevents) { -//! return array_merge($carry, array_reduce($coreevents[$prefix], function($carry, $eventclass) use ($prefix) { -//! $infos = self::get_event_infos($eventclass); -//! if ($infos) { -//! $carry[$infos['eventname']] = get_string('tg_colon', 'local_notemyprogress', (object) [ -//! 'a' => $prefix, -// ! 'b' => $infos['name'] -// ! ]); -// ! } -// ! return $carry; -// ! }, [])); -// ! }, [])]; - $coreevents = [get_string('coresystem') => array_reduce(array_keys($coreevents), function($carry, $prefix) use ($coreevents) { return array_merge($carry, array_reduce($coreevents[$prefix], function($carry, $eventclass) use ($prefix) { $infos = self::get_event_infos($eventclass); @@ -393,4 +385,25 @@ class configgamification { } return $eventclasses; } + /** + * Get the data needed to complete the graph. + * + * @param int $courseid the course id. + * @return array an array containing the apropriate data well formated + */ + public static function get_spread_chart($courseid){ + Global $DB; + $resultat = []; + $levelsData = "SELECT MAX(level) as lvl from {notemyprogress_xp} where courseid=?"; + $lvl = $DB->get_record_sql($levelsData, array("courseid="=>($courseid))); + $i = 1; + while($i<=$lvl->lvl){ + $levelCount = "SELECT COUNT(*) as cpt from {notemyprogress_xp} where courseid=? and level=?"; + $cpt = $DB->get_record_sql($levelCount, array("courseid="=>($courseid),"level="=>($i))); + $string = "level {$i} :"; + array_push( $resultat,[$string,intval($cpt->cpt)]); + $i = $i+1; + } + return $resultat; + } } \ No newline at end of file diff --git a/notemyprogress/classes/event_strategy.php b/notemyprogress/classes/event_strategy.php index d7ef122e8bac722b3ad5bce3cc7544881c679880..134b9aee5171eefbb2ab689e68ed484daf01eec2 100644 --- a/notemyprogress/classes/event_strategy.php +++ b/notemyprogress/classes/event_strategy.php @@ -358,11 +358,16 @@ class event_strategy { * * @return users */ - public function get_ranking() { + public function get_ranking($int) { global $DB; $users = array(); - $sql = "SELECT * FROM {{$this->table_xp}} WHERE courseid = {$this->course->id} and rankable = 1 ORDER BY points DESC"; + if ($int==1){ + $sql = "SELECT * FROM {{$this->table_xp}} WHERE courseid = {$this->course->id} and rankable = 1 ORDER BY points DESC"; + }else{ + $sql = "SELECT * FROM {{$this->table_xp}} WHERE courseid = {$this->course->id} ORDER BY points DESC"; + } + $rank = $DB->get_recordset_sql($sql); $index = 1; diff --git a/notemyprogress/gamification.php b/notemyprogress/gamification.php index 4a32d7751800de10ad1b1cceaafc055206835866..392dac98236677db5eaaea684b5bde729fa11283 100644 --- a/notemyprogress/gamification.php +++ b/notemyprogress/gamification.php @@ -23,7 +23,7 @@ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ require_once('locallib.php'); -global $COURSE, $USER; +global $COURSE, $USER , $DB; $courseid = required_param('courseid', PARAM_INT); $course = $DB->get_record('course', array('id' => $courseid), '*', MUST_EXIST); @@ -36,14 +36,27 @@ require_capability('local/notemyprogress:usepluggin', $context); require_capability('local/notemyprogress:teacher_gamification', $context); require_capability('local/notemyprogress:setweeks', $context); +$maxTimeSql = "SELECT MAX(timecreated) as maximum from {notemyprogress_gamification} where courseid=? "; +$timecrated = $DB->get_record_sql($maxTimeSql, array("courseid="=>($courseid))); + +$sql = "SELECT enablegamification from {notemyprogress_gamification} where courseid=? AND timecreated=? "; +$value = $DB->get_record_sql($sql, array("courseid="=>($courseid),"timecreated="=>$timecrated->maximum)); +if($value->enablegamification==1){ + $swValue = true; +}else{ + $swValue = false; +} + $actualLink = (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] === 'on' ? "https" : "http") . "://$_SERVER[HTTP_HOST]$_SERVER[REQUEST_URI]"; $logs = new \local_notemyprogress\logs($COURSE->id, $USER->id); -$logs->addLogsNMP("viewed", "section", "CONFIGURATION_GAMIFICATION", "configuration_gamification", $actualLink, "Section where teacher can configure gamification"); - +$logs->addLogsNMP("viewed", "section", "CONFIGURATION_GAMIFICATION", "configuration_gamification", $actualLink, "Section where teacher can configure gamification"); $configgamification = new \local_notemyprogress\configgamification($COURSE->id, $USER->id); $es = new \local_notemyprogress\event_strategy($COURSE, $USER); +$reports = new \local_notemyprogress\teacher($COURSE->id, $USER->id); + + $content = [ 'strings' =>[ @@ -87,15 +100,26 @@ $content = [ 'add_rule' => get_string("tg_section_settings_add_rule","local_notemyprogress"), 'earn' => get_string("tg_section_settings_earn","local_notemyprogress"), 'select_event' => get_string("tg_section_settings_select_event","local_notemyprogress"), - + 'enable'=>get_string("EnableGame","local_notemyprogress"), + 'swValue'=>$swValue, + 'chart' => $reports->get_chart_langs(), + 'pointError'=>get_string("game_point_error","local_notemyprogress"), + 'eventError'=>get_string("game_event_error","local_notemyprogress"), + 'nameError'=>get_string("game_name_error","local_notemyprogress"), + 'chartTitle'=>get_string("gm_Chart_Title","local_notemyprogress"), + 'chartYaxis'=>get_string("gm_Chart_Y","local_notemyprogress"), ], 'token' => local_notemyprogress_generate_token($COURSE->id, $USER->id, "teacher"), -// 'userid' => $USER->id, -// 'courseid' => $courseid, + 'userid' => $USER->id, + 'courseid' => $courseid, 'levels_data' => $configgamification->get_levels_data(), - 'ranking' => $es->get_ranking(), + 'ranking' => $es->get_ranking(0), 'events' => $configgamification->events_list(), - 'image' => $OUTPUT->image_url('badge', 'local_notemyprogress') + 'image' => $OUTPUT->image_url('badge', 'local_notemyprogress'), + 'indicators' => $reports->get_general_indicators(), + 'profile_render' => $reports->render_has(), + 'timezone' => $reports->timezone, + 'chart_data' => $configgamification->get_spread_chart($COURSE->id), ]; $templatecontext = [ diff --git a/notemyprogress/js/highcharts/highcharts.src.js b/notemyprogress/js/highcharts/highcharts.src.js index d8256d90570499e39c82e91ad80cb7294ee247c8..eab1ec4f806f46453de3965511828507d970445f 100644 --- a/notemyprogress/js/highcharts/highcharts.src.js +++ b/notemyprogress/js/highcharts/highcharts.src.js @@ -5,45166 +5,48241 @@ * * License: www.highcharts.com/license */ -'use strict'; +"use strict"; (function (root, factory) { - if (typeof module === 'object' && module.exports) { - factory['default'] = factory; - module.exports = root.document ? - factory(root) : - factory; - } else if (typeof define === 'function' && define.amd) { - define('highcharts/highcharts', function () { - return factory(root); + if (typeof module === "object" && module.exports) { + factory["default"] = factory; + module.exports = root.document ? factory(root) : factory; + } else if (typeof define === "function" && define.amd) { + define("highcharts/highcharts", function () { + return factory(root); + }); + } else { + if (root.Highcharts) { + root.Highcharts.error(16, true); + } + root.Highcharts = factory(root); + } +})(typeof window !== "undefined" ? window : this, function (win) { + var _modules = {}; + function _registerModule(obj, path, args, fn) { + if (!obj.hasOwnProperty(path)) { + obj[path] = fn.apply(null, args); + } + } + _registerModule(_modules, "Core/Globals.js", [], function () { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + /* globals Image, window */ + /** + * Reference to the global SVGElement class as a workaround for a name conflict + * in the Highcharts namespace. + * + * @global + * @typedef {global.SVGElement} GlobalSVGElement + * + * @see https://developer.mozilla.org/en-US/docs/Web/API/SVGElement + */ + // glob is a temporary fix to allow our es-modules to work. + var glob = // @todo UMD variable named `window`, and glob named `win` + typeof win !== "undefined" + ? win + : typeof window !== "undefined" + ? window + : {}, + doc = glob.document, + SVG_NS = "http://www.w3.org/2000/svg", + userAgent = (glob.navigator && glob.navigator.userAgent) || "", + svg = + doc && + doc.createElementNS && + !!doc.createElementNS(SVG_NS, "svg").createSVGRect, + isMS = /(edge|msie|trident)/i.test(userAgent) && !glob.opera, + isFirefox = userAgent.indexOf("Firefox") !== -1, + isChrome = userAgent.indexOf("Chrome") !== -1, + hasBidiBug = + isFirefox && parseInt(userAgent.split("Firefox/")[1], 10) < 4; // issue #38 + var H = { + product: "Highcharts", + version: "8.2.2", + deg2rad: (Math.PI * 2) / 360, + doc: doc, + hasBidiBug: hasBidiBug, + hasTouch: !!glob.TouchEvent, + isMS: isMS, + isWebKit: userAgent.indexOf("AppleWebKit") !== -1, + isFirefox: isFirefox, + isChrome: isChrome, + isSafari: !isChrome && userAgent.indexOf("Safari") !== -1, + isTouchDevice: /(Mobile|Android|Windows Phone)/.test(userAgent), + SVG_NS: SVG_NS, + chartCount: 0, + seriesTypes: {}, + symbolSizes: {}, + svg: svg, + win: glob, + marginNames: ["plotTop", "marginRight", "marginBottom", "plotLeft"], + noop: function () {}, + /** + * Theme options that should get applied to the chart. In module mode it + * might not be possible to change this property because of read-only + * restrictions, instead use {@link Highcharts.setOptions}. + * + * @name Highcharts.theme + * @type {Highcharts.Options} + */ + /** + * An array containing the current chart objects in the page. A chart's + * position in the array is preserved throughout the page's lifetime. When + * a chart is destroyed, the array item becomes `undefined`. + * + * @name Highcharts.charts + * @type {Array<Highcharts.Chart|undefined>} + */ + charts: [], + /** + * A hook for defining additional date format specifiers. New + * specifiers are defined as key-value pairs by using the + * specifier as key, and a function which takes the timestamp as + * value. This function returns the formatted portion of the + * date. + * + * @sample highcharts/global/dateformats/ + * Adding support for week number + * + * @name Highcharts.dateFormats + * @type {Highcharts.Dictionary<Highcharts.TimeFormatCallbackFunction>} + */ + dateFormats: {}, + }; + + return H; + }); + _registerModule( + _modules, + "Core/Utilities.js", + [_modules["Core/Globals.js"]], + function (H) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + /** + * An animation configuration. Animation configurations can also be defined as + * booleans, where `false` turns off animation and `true` defaults to a duration + * of 500ms and defer of 0ms. + * + * @interface Highcharts.AnimationOptionsObject + */ /** + * A callback function to exectute when the animation finishes. + * @name Highcharts.AnimationOptionsObject#complete + * @type {Function|undefined} + */ /** + * The animation defer in milliseconds. + * @name Highcharts.AnimationOptionsObject#defer + * @type {number|undefined} + */ /** + * The animation duration in milliseconds. + * @name Highcharts.AnimationOptionsObject#duration + * @type {number|undefined} + */ /** + * The name of an easing function as defined on the `Math` object. + * @name Highcharts.AnimationOptionsObject#easing + * @type {string|Function|undefined} + */ /** + * A callback function to execute on each step of each attribute or CSS property + * that's being animated. The first argument contains information about the + * animation and progress. + * @name Highcharts.AnimationOptionsObject#step + * @type {Function|undefined} + */ + /** + * Creates a frame for the animated SVG element. + * + * @callback Highcharts.AnimationStepCallbackFunction + * + * @param {Highcharts.SVGElement} this + * The SVG element to animate. + * + * @return {void} + */ + /** + * Interface description for a class. + * + * @interface Highcharts.Class<T> + * @extends Function + */ /** + * Class costructor. + * @function Highcharts.Class<T>#new + * @param {...Array<*>} args + * Constructor arguments. + * @return {T} + * Class instance. + */ + /** + * A style object with camel case property names to define visual appearance of + * a SVG element or HTML element. The properties can be whatever styles are + * supported on the given SVG or HTML element. + * + * @example + * { + * fontFamily: 'monospace', + * fontSize: '1.2em' + * } + * + * @interface Highcharts.CSSObject + */ /** + * @name Highcharts.CSSObject#[key:string] + * @type {boolean|number|string|undefined} + */ /** + * Background style for the element. + * @name Highcharts.CSSObject#background + * @type {string|undefined} + */ /** + * Background color of the element. + * @name Highcharts.CSSObject#backgroundColor + * @type {Highcharts.ColorString|undefined} + */ /** + * Border style for the element. + * @name Highcharts.CSSObject#border + * @type {string|undefined} + */ /** + * Radius of the element border. + * @name Highcharts.CSSObject#borderRadius + * @type {number|undefined} + */ /** + * Color used in the element. The 'contrast' option is a Highcharts custom + * property that results in black or white, depending on the background of the + * element. + * @name Highcharts.CSSObject#color + * @type {'contrast'|Highcharts.ColorString|undefined} + */ /** + * Style of the mouse cursor when resting over the element. + * @name Highcharts.CSSObject#cursor + * @type {Highcharts.CursorValue|undefined} + */ /** + * Font family of the element text. Multiple values have to be in decreasing + * preference order and separated by comma. + * @name Highcharts.CSSObject#fontFamily + * @type {string|undefined} + */ /** + * Font size of the element text. + * @name Highcharts.CSSObject#fontSize + * @type {string|undefined} + */ /** + * Font weight of the element text. + * @name Highcharts.CSSObject#fontWeight + * @type {string|undefined} + */ /** + * Height of the element. + * @name Highcharts.CSSObject#height + * @type {number|undefined} + */ /** + * Width of the element border. + * @name Highcharts.CSSObject#lineWidth + * @type {number|undefined} + */ /** + * Opacity of the element. + * @name Highcharts.CSSObject#opacity + * @type {number|undefined} + */ /** + * Space around the element content. + * @name Highcharts.CSSObject#padding + * @type {string|undefined} + */ /** + * Behaviour of the element when the mouse cursor rests over it. + * @name Highcharts.CSSObject#pointerEvents + * @type {string|undefined} + */ /** + * Positioning of the element. + * @name Highcharts.CSSObject#position + * @type {string|undefined} + */ /** + * Alignment of the element text. + * @name Highcharts.CSSObject#textAlign + * @type {string|undefined} + */ /** + * Additional decoration of the element text. + * @name Highcharts.CSSObject#textDecoration + * @type {string|undefined} + */ /** + * Outline style of the element text. + * @name Highcharts.CSSObject#textOutline + * @type {string|undefined} + */ /** + * Line break style of the element text. Highcharts SVG elements support + * `ellipsis` when a `width` is set. + * @name Highcharts.CSSObject#textOverflow + * @type {string|undefined} + */ /** + * Top spacing of the element relative to the parent element. + * @name Highcharts.CSSObject#top + * @type {string|undefined} + */ /** + * Animated transition of selected element properties. + * @name Highcharts.CSSObject#transition + * @type {string|undefined} + */ /** + * Line break style of the element text. + * @name Highcharts.CSSObject#whiteSpace + * @type {string|undefined} + */ /** + * Width of the element. + * @name Highcharts.CSSObject#width + * @type {number|undefined} + */ + /** + * All possible cursor styles. + * + * @typedef {'alias'|'all-scroll'|'auto'|'cell'|'col-resize'|'context-menu'|'copy'|'crosshair'|'default'|'e-resize'|'ew-resize'|'grab'|'grabbing'|'help'|'move'|'n-resize'|'ne-resize'|'nesw-resize'|'no-drop'|'none'|'not-allowed'|'ns-resize'|'nw-resize'|'nwse-resize'|'pointer'|'progress'|'row-resize'|'s-resize'|'se-resize'|'sw-resize'|'text'|'vertical-text'|'w-resize'|'wait'|'zoom-in'|'zoom-out'} Highcharts.CursorValue + */ + /** + * All possible dash styles. + * + * @typedef {'Dash'|'DashDot'|'Dot'|'LongDash'|'LongDashDot'|'LongDashDotDot'|'ShortDash'|'ShortDashDot'|'ShortDashDotDot'|'ShortDot'|'Solid'} Highcharts.DashStyleValue + */ + /** + * Generic dictionary in TypeScript notation. + * Use the native `Record<string, any>` instead. + * + * @deprecated + * @interface Highcharts.Dictionary<T> + */ /** + * @name Highcharts.Dictionary<T>#[key:string] + * @type {T} + */ + /** + * The function callback to execute when the event is fired. The `this` context + * contains the instance, that fired the event. + * + * @callback Highcharts.EventCallbackFunction<T> + * + * @param {T} this + * + * @param {Highcharts.Dictionary<*>|Event} [eventArguments] + * Event arguments. + * + * @return {boolean|void} + */ + /** + * The event options for adding function callback. + * + * @interface Highcharts.EventOptionsObject + */ /** + * The order the event handler should be called. This opens for having one + * handler be called before another, independent of in which order they were + * added. + * @name Highcharts.EventOptionsObject#order + * @type {number} + */ + /** + * Formats data as a string. Usually the data is accessible throught the `this` + * keyword. + * + * @callback Highcharts.FormatterCallbackFunction<T> + * + * @param {T} this + * Context to format + * + * @return {string} + * Formatted text + */ + /** + * An object of key-value pairs for HTML attributes. + * + * @typedef {Highcharts.Dictionary<boolean|number|string|Function>} Highcharts.HTMLAttributes + */ + /** + * An HTML DOM element. The type is a reference to the regular HTMLElement in + * the global scope. + * + * @typedef {global.HTMLElement} Highcharts.HTMLDOMElement + * + * @see https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement + */ + /** + * The iterator callback. + * + * @callback Highcharts.ObjectEachCallbackFunction<T> + * + * @param {T} this + * The context. + * + * @param {*} value + * The property value. + * + * @param {string} key + * The property key. + * + * @param {*} obj + * The object that objectEach is being applied to. + */ + /** + * An object containing `left` and `top` properties for the position in the + * page. + * + * @interface Highcharts.OffsetObject + */ /** + * Left distance to the page border. + * @name Highcharts.OffsetObject#left + * @type {number} + */ /** + * Top distance to the page border. + * @name Highcharts.OffsetObject#top + * @type {number} + */ + /** + * Describes a range. + * + * @interface Highcharts.RangeObject + */ /** + * Maximum number of the range. + * @name Highcharts.RangeObject#max + * @type {number} + */ /** + * Minimum number of the range. + * @name Highcharts.RangeObject#min + * @type {number} + */ + /** + * If a number is given, it defines the pixel length. If a percentage string is + * given, like for example `'50%'`, the setting defines a length relative to a + * base size, for example the size of a container. + * + * @typedef {number|string} Highcharts.RelativeSize + */ + /** + * Proceed function to call original (wrapped) function. + * + * @callback Highcharts.WrapProceedFunction + * + * @param {*} [arg1] + * Optional argument. Without any arguments defaults to first argument of + * the wrapping function. + * + * @param {*} [arg2] + * Optional argument. Without any arguments defaults to second argument + * of the wrapping function. + * + * @param {*} [arg3] + * Optional argument. Without any arguments defaults to third argument of + * the wrapping function. + * + * @return {*} + * Return value of the original function. + */ + /** + * The Highcharts object is the placeholder for all other members, and various + * utility functions. The most important member of the namespace would be the + * chart constructor. + * + * @example + * var chart = Highcharts.chart('container', { ... }); + * + * @namespace Highcharts + */ + H.timers = []; + var charts = H.charts, + doc = H.doc, + win = H.win; + /** + * Provide error messages for debugging, with links to online explanation. This + * function can be overridden to provide custom error handling. + * + * @sample highcharts/chart/highcharts-error/ + * Custom error handler + * + * @function Highcharts.error + * + * @param {number|string} code + * The error code. See + * [errors.xml](https://github.com/highcharts/highcharts/blob/master/errors/errors.xml) + * for available codes. If it is a string, the error message is printed + * directly in the console. + * + * @param {boolean} [stop=false] + * Whether to throw an error or just log a warning in the console. + * + * @param {Highcharts.Chart} [chart] + * Reference to the chart that causes the error. Used in 'debugger' + * module to display errors directly on the chart. + * Important note: This argument is undefined for errors that lack + * access to the Chart instance. + * + * @param {Highcharts.Dictionary<string>} [params] + * Additional parameters for the generated message. + * + * @return {void} + */ + function error(code, stop, chart, params) { + var severity = stop ? "Highcharts error" : "Highcharts warning"; + if (code === 32) { + code = severity + ": Deprecated member"; + } + var isCode = isNumber(code), + message = isCode + ? severity + + " #" + + code + + ": www.highcharts.com/errors/" + + code + + "/" + : code.toString(), + defaultHandler = function () { + if (stop) { + throw new Error(message); + } + // else ... + if ( + win.console && + error.messages.indexOf(message) === -1 // prevent console flooting + ) { + console.log(message); // eslint-disable-line no-console + } + }; + if (typeof params !== "undefined") { + var additionalMessages_1 = ""; + if (isCode) { + message += "?"; + } + objectEach(params, function (value, key) { + additionalMessages_1 += "\n - " + key + ": " + value; + if (isCode) { + message += encodeURI(key) + "=" + encodeURI(value); + } + }); + message += additionalMessages_1; + } + if (chart) { + fireEvent( + chart, + "displayError", + { code: code, message: message, params: params }, + defaultHandler + ); + } else { + defaultHandler(); + } + error.messages.push(message); + } + (function (error) { + error.messages = []; + })(error || (error = {})); + H.error = error; + /* eslint-disable valid-jsdoc */ + /** + * 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 {boolean} extend + * Whether to extend the left-side object (a) or return a whole new + * object. + * + * @param {T|undefined} a + * The first object to extend. When only this is given, the function + * returns a deep copy. + * + * @param {...Array<object|undefined>} [n] + * An object 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. + */ /** + * Utility function to deep merge two or more objects and return a third 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 {T|undefined} a + * The first object to extend. When only this is given, the function + * returns a deep copy. + * + * @param {...Array<object|undefined>} [n] + * An object 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() { + /* eslint-enable valid-jsdoc */ + var i, + args = arguments, + len, + ret = {}, + doCopy = function (copy, original) { + // An object is replacing a primitive + if (typeof copy !== "object") { + copy = {}; + } + objectEach(original, function (value, key) { + // 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 (args[0] === true) { + ret = args[1]; + args = Array.prototype.slice.call(args, 2); + } + // For each argument, extend the return + len = args.length; + for (i = 0; i < len; i++) { + ret = doCopy(ret, args[i]); + } + return ret; + } + H.merge = merge; + /** + * Constrain a value to within a lower and upper threshold. + * + * @private + * @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; + } + /** + * Shortcut for parseInt + * + * @private + * @function Highcharts.pInt + * + * @param {*} s + * any + * + * @param {number} [mag] + * Magnitude + * + * @return {number} + * number + */ + var pInt = (H.pInt = function pInt(s, mag) { + return parseInt(s, mag || 10); + }); + /** + * 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. + */ + var isString = (H.isString = 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. + */ + var isArray = (H.isArray = function isArray(obj) { + var str = Object.prototype.toString.call(obj); + return str === "[object Array]" || str === "[object Array Iterator]"; + }); + /** + * 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 + } + H.isObject = isObject; + /** + * 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. + */ + var isDOMElement = (H.isDOMElement = 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. + */ + var isClass = (H.isClass = function isClass(obj) { + var c = obj && obj.constructor; + return !!( + isObject(obj, true) && + !isDOMElement(obj) && + c && + 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 + */ + var isNumber = (H.isNumber = function isNumber(n) { + return ( + typeof n === "number" && !isNaN(n) && n < Infinity && n > -Infinity + ); + }); + /** + * Remove the last occurence of an item from an array. + * + * @function Highcharts.erase + * + * @param {Array<*>} arr + * The array. + * + * @param {*} item + * The item to remove. + * + * @return {void} + */ + var erase = (H.erase = function erase(arr, item) { + var i = arr.length; + while (i--) { + if (arr[i] === item) { + arr.splice(i, 1); + break; + } + } + }); + /** + * 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. + */ + var defined = (H.defined = function defined(obj) { + return typeof obj !== "undefined" && obj !== null; + }); + /** + * 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. 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} [prop] + * 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, prop, value) { + var ret; + // if the prop is a string + if (isString(prop)) { + // set the value + if (defined(value)) { + elem.setAttribute(prop, value); + // get the value + } else if (elem && elem.getAttribute) { + ret = elem.getAttribute(prop); + // IE7 and below cannot get class through getAttribute (#7850) + if (!ret && prop === "class") { + ret = elem.getAttribute(prop + "Name"); + } + } + // else if prop is defined, it is a hash of key/value pairs + } else { + objectEach(prop, function (val, key) { + elem.setAttribute(key, val); + }); + } + return ret; + } + H.attr = attr; + /** + * Check if an element is an array, and if not, make it into an array. + * + * @function Highcharts.splat + * + * @param {*} obj + * The object to splat. + * + * @return {Array} + * The produced or original array. + */ + var splat = (H.splat = function splat(obj) { + return isArray(obj) ? obj : [obj]; + }); + /** + * Set a timeout if the delay is given, otherwise perform the function + * synchronously. + * + * @function Highcharts.syncTimeout + * + * @param {Function} fn + * The function callback. + * + * @param {number} delay + * Delay in milliseconds. + * + * @param {*} [context] + * An optional context to send to the function callback. + * + * @return {number} + * An identifier for the timeout that can later be cleared with + * Highcharts.clearTimeout. Returns -1 if there is no timeout. + */ + var syncTimeout = (H.syncTimeout = function syncTimeout( + fn, + delay, + context + ) { + if (delay > 0) { + return setTimeout(fn, delay, context); + } + fn.call(0, context); + return -1; + }); + /** + * 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). + * + * @function Highcharts.clearTimeout + * + * @param {number} id + * Id of a timeout. + * + * @return {void} + */ + var internalClearTimeout = (H.clearTimeout = function (id) { + if (defined(id)) { + clearTimeout(id); + } + }); + /* eslint-disable valid-jsdoc */ + /** + * 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 {object} b + * The object to add to the first one. + * + * @return {T} + * Object a, the original object. + */ + var extend = (H.extend = function extend(a, b) { + /* eslint-enable valid-jsdoc */ + var n; + if (!a) { + a = {}; + } + for (n in b) { + // eslint-disable-line guard-for-in + a[n] = b[n]; + } + return a; + }); + /* eslint-disable valid-jsdoc */ + /** + * Return the first value that is not null or undefined. + * + * @function Highcharts.pick<T> + * + * @param {...Array<T|null|undefined>} items + * Variable number of arguments to inspect. + * + * @return {T} + * The value of the first argument that is not null or undefined. + */ + function pick() { + var args = arguments; + var length = args.length; + for (var i = 0; i < length; i++) { + var arg = args[i]; + if (typeof arg !== "undefined" && arg !== null) { + return arg; + } + } + } + H.pick = pick; + /** + * 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} + */ + var css = (H.css = function css(el, styles) { + if (H.isMS && !H.svg) { + // #2686 + if (styles && typeof styles.opacity !== "undefined") { + styles.filter = "alpha(opacity=" + styles.opacity * 100 + ")"; + } + } + extend(el.style, styles); + }); + /** + * 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. + */ + var createElement = (H.createElement = function createElement( + tag, + attribs, + styles, + parent, + nopad + ) { + var el = 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; + }); + // eslint-disable-next-line valid-jsdoc + /** + * Extend a prototyped class by new members. + * + * @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. + */ + var extendClass = (H.extendClass = function extendClass(parent, members) { + var obj = function () {}; + obj.prototype = new parent(); // eslint-disable-line new-cap + extend(obj.prototype, members); + return obj; + }); + /** + * Left-pad a string to a given length by adding a character repetetively. + * + * @function Highcharts.pad + * + * @param {number} number + * The input string or number. + * + * @param {number} [length] + * The desired string length. + * + * @param {string} [padder=0] + * The character to pad with. + * + * @return {string} + * The padded string. + */ + var pad = (H.pad = function pad(number, length, padder) { + return ( + new Array( + (length || 2) + 1 - String(number).replace("-", "").length + ).join(padder || "0") + number + ); + }); + /** + * Return a length based on either the integer value, or a percentage of a base. + * + * @function Highcharts.relativeLength + * + * @param {Highcharts.RelativeSize} value + * A percentage string or a number. + * + * @param {number} base + * The full length that represents 100%. + * + * @param {number} [offset=0] + * A pixel offset to apply for percentage values. Used internally in + * axis positioning. + * + * @return {number} + * The computed length. + */ + var relativeLength = (H.relativeLength = function relativeLength( + value, + base, + offset + ) { + return /%$/.test(value) + ? (base * parseFloat(value)) / 100 + (offset || 0) + : parseFloat(value); + }); + /** + * Wrap a method with extended functionality, preserving the original function. + * + * @function Highcharts.wrap + * + * @param {*} obj + * The context object that the method belongs to. In real cases, this is + * often a prototype. + * + * @param {string} method + * The name of the method to extend. + * + * @param {Highcharts.WrapProceedFunction} func + * A wrapper function callback. This function is called with the same + * arguments as the original function, except that the original function + * is unshifted and passed as the first argument. + */ + var wrap = (H.wrap = function wrap(obj, method, func) { + var proceed = obj[method]; + obj[method] = function () { + var args = Array.prototype.slice.call(arguments), + outerArgs = arguments, + ctx = this, + ret; + ctx.proceed = function () { + proceed.apply(ctx, arguments.length ? arguments : outerArgs); + }; + args.unshift(proceed); + ret = func.apply(this, args); + ctx.proceed = null; + return ret; + }; + }); + /** + * Format a string according to a subset of the rules of Python's String.format + * method. + * + * @example + * var s = Highcharts.format( + * 'The {color} fox was {len:.2f} feet long', + * { color: 'red', len: Math.PI } + * ); + * // => The red fox was 3.14 feet long + * + * @function Highcharts.format + * + * @param {string} str + * The string to format. + * + * @param {Record<string, *>} ctx + * The context, a collection of key-value pairs where each key is + * replaced by its value. + * + * @param {Highcharts.Chart} [chart] + * A `Chart` instance used to get numberFormatter and time. + * + * @return {string} + * The formatted string. + */ + var format = (H.format = function (str, ctx, chart) { + var splitter = "{", + isInside = false, + segment, + valueAndFormat, + ret = [], + val, + index; + var floatRegex = /f$/; + var decRegex = /\.([0-9])/; + var lang = H.defaultOptions.lang; + var time = (chart && chart.time) || H.time; + var numberFormatter = (chart && chart.numberFormatter) || numberFormat; + while (str) { + index = str.indexOf(splitter); + if (index === -1) { + break; + } + segment = str.slice(0, index); + if (isInside) { + // we're on the closing bracket looking back + valueAndFormat = segment.split(":"); + val = getNestedProperty(valueAndFormat.shift() || "", ctx); + // Format the replacement + if (valueAndFormat.length && typeof val === "number") { + segment = valueAndFormat.join(":"); + if (floatRegex.test(segment)) { + // float + var decimals = parseInt( + (segment.match(decRegex) || ["", "-1"])[1], + 10 + ); + if (val !== null) { + val = numberFormatter( + val, + decimals, + lang.decimalPoint, + segment.indexOf(",") > -1 ? lang.thousandsSep : "" + ); + } + } else { + val = time.dateFormat(segment, val); + } + } + // Push the result and advance the cursor + ret.push(val); + } else { + ret.push(segment); + } + str = str.slice(index + 1); // the rest + isInside = !isInside; // toggle + splitter = isInside ? "}" : "{"; // now look for next matching bracket + } + ret.push(str); + return ret.join(""); + }); + /** + * 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. + */ + var getMagnitude = (H.getMagnitude = function (num) { + return Math.pow(10, Math.floor(Math.log(num) / Math.LN10)); + }); + /** + * Take an interval and normalize it to multiples of round numbers. + * + * @deprecated + * @function Highcharts.normalizeTickInterval + * + * @param {number} interval + * The raw, un-rounded interval. + * + * @param {Array<*>} [multiples] + * Allowed multiples. + * + * @param {number} [magnitude] + * The magnitude of the number. + * + * @param {boolean} [allowDecimals] + * Whether to allow decimals. + * + * @param {boolean} [hasTickAmount] + * If it has tickAmount, avoid landing on tick intervals lower than + * original. + * + * @return {number} + * The normalized interval. + * + * @todo + * Move this function to the Axis prototype. It is here only for historical + * reasons. + */ + var normalizeTickInterval = (H.normalizeTickInterval = function ( + interval, + multiples, + magnitude, + allowDecimals, + hasTickAmount + ) { + var normalized, + i, + retInterval = interval; + // round to a tenfold of 1, 2, 2.5 or 5 + magnitude = pick(magnitude, 1); + normalized = interval / magnitude; + // multiples for a linear scale + if (!multiples) { + multiples = hasTickAmount + ? // Finer grained ticks when the tick amount is hard set, including + // when alignTicks is true on multiple axes (#4580). + [1, 1.2, 1.5, 2, 2.5, 3, 4, 5, 6, 8, 10] + : // Else, let ticks fall on rounder numbers + [1, 2, 2.5, 5, 10]; + // the allowDecimals option + if (allowDecimals === false) { + if (magnitude === 1) { + multiples = multiples.filter(function (num) { + return num % 1 === 0; + }); + } else if (magnitude <= 0.1) { + multiples = [1 / magnitude]; + } + } + } + // normalize the interval to the nearest multiple + for (i = 0; i < multiples.length; i++) { + retInterval = multiples[i]; + // only allow tick amounts smaller than natural + if ( + (hasTickAmount && retInterval * magnitude >= interval) || + (!hasTickAmount && + normalized <= + (multiples[i] + (multiples[i + 1] || multiples[i])) / 2) + ) { + break; + } + } + // Multiply back to the correct magnitude. Correct floats to appropriate + // precision (#6085). + retInterval = correctFloat( + retInterval * magnitude, + -Math.round(Math.log(0.001) / Math.LN10) + ); + return retInterval; + }); + /** + * Sort an object array and keep the order of equal items. The ECMAScript + * standard does not specify the behaviour when items are equal. + * + * @function Highcharts.stableSort + * + * @param {Array<*>} arr + * The array to sort. + * + * @param {Function} sortFunction + * The function to sort it with, like with regular Array.prototype.sort. + * + * @return {void} + */ + var stableSort = (H.stableSort = function stableSort(arr, sortFunction) { + // @todo It seems like Chrome since v70 sorts in a stable way internally, + // plus all other browsers do it, so over time we may be able to remove this + // function + var length = arr.length, + sortValue, + i; + // Add index to each item + for (i = 0; i < length; i++) { + arr[i].safeI = i; // stable sort index + } + arr.sort(function (a, b) { + sortValue = sortFunction(a, b); + return sortValue === 0 ? a.safeI - b.safeI : sortValue; }); - } else { - if (root.Highcharts) { - root.Highcharts.error(16, true); + // Remove index from items + for (i = 0; i < length; i++) { + delete arr[i].safeI; // stable sort index } - root.Highcharts = factory(root); - } -}(typeof window !== 'undefined' ? window : this, function (win) { - var _modules = {}; - function _registerModule(obj, path, args, fn) { - if (!obj.hasOwnProperty(path)) { - obj[path] = fn.apply(null, args); + }); + /** + * 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. + */ + var arrayMin = (H.arrayMin = function arrayMin(data) { + var i = data.length, + min = data[0]; + while (i--) { + if (data[i] < min) { + min = data[i]; + } } - } - _registerModule(_modules, 'Core/Globals.js', [], function () { - /* * + 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. + */ + var arrayMax = (H.arrayMax = function arrayMax(data) { + var i = data.length, + max = data[0]; + while (i--) { + if (data[i] > max) { + max = data[i]; + } + } + return max; + }); + /** + * 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. + */ + var destroyObjectProperties = (H.destroyObjectProperties = + function destroyObjectProperties(obj, except) { + objectEach(obj, function (val, n) { + // If the object is non-null and destroy is defined + if (val && val !== except && val.destroy) { + // Invoke the destroy + val.destroy(); + } + // Delete the property from the object. + delete obj[n]; + }); + }); + /** + * Discard a HTML element by moving it to the bin and delete. + * + * @function Highcharts.discardElement + * + * @param {Highcharts.HTMLDOMElement} element + * The HTML node to discard. + */ + var discardElement = (H.discardElement = function discardElement( + element + ) { + var garbageBin = H.garbageBin; + // create a garbage bin element, not part of the DOM + if (!garbageBin) { + garbageBin = createElement("div"); + } + // move the node and empty bin + if (element) { + garbageBin.appendChild(element); + } + garbageBin.innerHTML = ""; + }); + /** + * 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. + */ + var correctFloat = (H.correctFloat = function correctFloat(num, prec) { + return parseFloat(num.toPrecision(prec || 14)); + }); + /** + * The time unit lookup + * + * @ignore + */ + var timeUnits = (H.timeUnits = { + millisecond: 1, + second: 1000, + minute: 60000, + hour: 3600000, + day: 24 * 3600000, + week: 7 * 24 * 3600000, + month: 28 * 24 * 3600000, + year: 364 * 24 * 3600000, + }); + /** + * Format a number and return a string based on input settings. + * + * @sample highcharts/members/highcharts-numberformat/ + * Custom number format + * + * @function Highcharts.numberFormat + * + * @param {number} number + * The input number to format. + * + * @param {number} decimals + * The amount of decimals. A value of -1 preserves the amount in the + * input number. + * + * @param {string} [decimalPoint] + * The decimal point, defaults to the one given in the lang options, or + * a dot. + * + * @param {string} [thousandsSep] + * The thousands separator, defaults to the one given in the lang + * options, or a space character. + * + * @return {string} + * The formatted number. + */ + var numberFormat = (H.numberFormat = function numberFormat( + number, + decimals, + decimalPoint, + thousandsSep + ) { + number = +number || 0; + decimals = +decimals; + var lang = H.defaultOptions.lang, + origDec = (number.toString().split(".")[1] || "").split("e")[0] + .length, + strinteger, + thousands, + ret, + roundedNumber, + exponent = number.toString().split("e"), + fractionDigits; + if (decimals === -1) { + // Preserve decimals. Not huge numbers (#3793). + decimals = Math.min(origDec, 20); + } else if (!isNumber(decimals)) { + decimals = 2; + } else if (decimals && exponent[1] && exponent[1] < 0) { + // Expose decimals from exponential notation (#7042) + fractionDigits = decimals + +exponent[1]; + if (fractionDigits >= 0) { + // remove too small part of the number while keeping the notation + exponent[0] = (+exponent[0]) + .toExponential(fractionDigits) + .split("e")[0]; + decimals = fractionDigits; + } else { + // fractionDigits < 0 + exponent[0] = exponent[0].split(".")[0] || 0; + if (decimals < 20) { + // use number instead of exponential notation (#7405) + number = (exponent[0] * Math.pow(10, exponent[1])).toFixed( + decimals + ); + } else { + // or zero + number = 0; + } + exponent[1] = 0; + } + } + // Add another decimal to avoid rounding errors of float numbers. (#4573) + // Then use toFixed to handle rounding. + roundedNumber = ( + Math.abs(exponent[1] ? exponent[0] : number) + + Math.pow(10, -Math.max(decimals, origDec) - 1) + ).toFixed(decimals); + // A string containing the positive integer component of the number + strinteger = String(pInt(roundedNumber)); + // Leftover after grouping into thousands. Can be 0, 1 or 2. + thousands = strinteger.length > 3 ? strinteger.length % 3 : 0; + // Language + decimalPoint = pick(decimalPoint, lang.decimalPoint); + thousandsSep = pick(thousandsSep, lang.thousandsSep); + // Start building the return + ret = number < 0 ? "-" : ""; + // Add the leftover after grouping into thousands. For example, in the + // number 42 000 000, this line adds 42. + ret += thousands ? strinteger.substr(0, thousands) + thousandsSep : ""; + // Add the remaining thousands groups, joined by the thousands separator + ret += strinteger + .substr(thousands) + .replace(/(\d{3})(?=\d)/g, "$1" + thousandsSep); + // Add the decimal point and the decimal component + if (decimals) { + // Get the decimal component + ret += decimalPoint + roundedNumber.slice(-decimals); + } + if (exponent[1] && +ret !== 0) { + ret += "e" + exponent[1]; + } + return ret; + }); + /** + * Easing definition + * + * @private + * @function Math.easeInOutSine + * + * @param {number} pos + * Current position, ranging from 0 to 1. + * + * @return {number} + * Ease result + */ + Math.easeInOutSine = function (pos) { + return -0.5 * (Math.cos(Math.PI * pos) - 1); + }; + /** + * Returns the value of a property path on a given object. + * + * @private + * @function getNestedProperty + * + * @param {string} path + * Path to the property, for example `custom.myValue`. + * + * @param {unknown} obj + * Instance containing the property on the specific path. + * + * @return {unknown} + * The unknown property value. + */ + function getNestedProperty(path, obj) { + if (!path) { + return obj; + } + var pathElements = path.split(".").reverse(); + var subProperty = obj; + if (pathElements.length === 1) { + return subProperty[path]; + } + var pathElement = pathElements.pop(); + while ( + typeof pathElement !== "undefined" && + typeof subProperty !== "undefined" && + subProperty !== null + ) { + subProperty = subProperty[pathElement]; + pathElement = pathElements.pop(); + } + return subProperty; + } + /** + * 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} + * The numeric value. + */ + var getStyle = (H.getStyle = function (el, prop, toInt) { + var style; + // For width and height, return the actual inner pixel size (#4913) + if (prop === "width") { + var 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) ... + var boundingClientRectWidth = + el.getBoundingClientRect && 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 - + H.getStyle(el, "padding-left") - + H.getStyle(el, "padding-right") + ); + } + if (prop === "height") { + return Math.max( + 0, // #8377 + Math.min(el.offsetHeight, el.scrollHeight) - + H.getStyle(el, "padding-top") - + H.getStyle(el, "padding-bottom") + ); + } + if (!win.getComputedStyle) { + // SVG not supported, forgot to load oldie.js? + error(27, true); + } + // Otherwise, get the computed style + style = win.getComputedStyle(el, undefined); // eslint-disable-line no-undefined + if (style) { + style = style.getPropertyValue(prop); + if (pick(toInt, prop !== "opacity")) { + style = pInt(style); + } + } + return style; + }); + /** + * Search for an item in an array. + * + * @function Highcharts.inArray + * + * @deprecated + * + * @param {*} item + * The item to search for. + * + * @param {Array<*>} arr + * The array or node collection to search in. + * + * @param {number} [fromIndex=0] + * The index to start searching from. + * + * @return {number} + * The index within the array, or -1 if not found. + */ + var inArray = (H.inArray = function (item, arr, fromIndex) { + error(32, false, void 0, { "Highcharts.inArray": "use Array.indexOf" }); + return arr.indexOf(item, fromIndex); + }); + /* eslint-disable valid-jsdoc */ + /** + * 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. + */ + var find = (H.find = Array.prototype.find + ? /* eslint-enable valid-jsdoc */ + function (arr, callback) { + return arr.find(callback); + } + : // Legacy implementation. PhantomJS, IE <= 11 etc. #7223. + function (arr, callback) { + var i, + length = arr.length; + for (i = 0; i < length; i++) { + if (callback(arr[i], i)) { + // eslint-disable-line callback-return + return arr[i]; + } + } + }); + /** + * Returns an array of a given object's own properties. + * + * @function Highcharts.keys + * @deprecated + * + * @param {*} obj + * The object of which the properties are to be returned. + * + * @return {Array<string>} + * An array of strings that represents all the properties. + */ + H.keys = function (obj) { + error(32, false, void 0, { "Highcharts.keys": "use Object.keys" }); + return Object.keys(obj); + }; + /** + * Get the element's offset position, corrected for `overflow: auto`. + * + * @function Highcharts.offset + * + * @param {global.Element} el + * The DOM element. + * + * @return {Highcharts.OffsetObject} + * An object containing `left` and `top` properties for the position in + * the page. + */ + var offset = (H.offset = function offset(el) { + var docElem = doc.documentElement, + box = + el.parentElement || el.parentNode + ? el.getBoundingClientRect() + : { top: 0, left: 0 }; + return { + top: + box.top + + (win.pageYOffset || docElem.scrollTop) - + (docElem.clientTop || 0), + left: + box.left + + (win.pageXOffset || docElem.scrollLeft) - + (docElem.clientLeft || 0), + }; + }); + /* eslint-disable valid-jsdoc */ + /** + * Iterate over object key pairs in an object. + * + * @function Highcharts.objectEach<T> + * + * @param {*} obj + * The object to iterate over. + * + * @param {Highcharts.ObjectEachCallbackFunction<T>} fn + * The iterator callback. It passes three arguments: + * * value - The property value. + * * key - The property key. + * * obj - The object that objectEach is being applied to. + * + * @param {T} [ctx] + * The context. + * + * @return {void} + */ + var objectEach = (H.objectEach = function objectEach(obj, fn, ctx) { + /* eslint-enable valid-jsdoc */ + for (var key in obj) { + if (Object.hasOwnProperty.call(obj, key)) { + fn.call(ctx || obj[key], obj[key], key, obj); + } + } + }); + /** + * Iterate over an array. + * + * @deprecated + * @function Highcharts.each + * + * @param {Array<*>} arr + * The array to iterate over. + * + * @param {Function} fn + * The iterator callback. It passes three arguments: + * - `item`: The array item. + * - `index`: The item's index in the array. + * - `arr`: The array that each is being applied to. + * + * @param {*} [ctx] + * The context. + * + * @return {void} + */ + /** + * Filter an array by a callback. + * + * @deprecated + * @function Highcharts.grep + * + * @param {Array<*>} arr + * The array to filter. + * + * @param {Function} callback + * The callback function. The function receives the item as the first + * argument. Return `true` if the item is to be preserved. + * + * @return {Array<*>} + * A new, filtered array. + */ + /** + * Map an array by a callback. + * + * @deprecated + * @function Highcharts.map + * + * @param {Array<*>} arr + * The array to map. + * + * @param {Function} fn + * The callback function. Return the new value for the new array. + * + * @return {Array<*>} + * A new array item with modified items. + */ + /** + * Reduce an array to a single value. + * + * @deprecated + * @function Highcharts.reduce + * + * @param {Array<*>} arr + * The array to reduce. + * + * @param {Function} fn + * The callback function. Return the reduced value. Receives 4 + * arguments: Accumulated/reduced value, current value, current array + * index, and the array. + * + * @param {*} initialValue + * The initial value of the accumulator. + * + * @return {*} + * The reduced value. + */ + /** + * Test whether at least one element in the array passes the test implemented by + * the provided function. + * + * @deprecated + * @function Highcharts.some + * + * @param {Array<*>} arr + * The array to test + * + * @param {Function} fn + * The function to run on each item. Return truty to pass the test. + * Receives arguments `currentValue`, `index` and `array`. + * + * @param {*} ctx + * The context. + * + * @return {boolean} + */ + objectEach( + { + map: "map", + each: "forEach", + grep: "filter", + reduce: "reduce", + some: "some", + }, + function (val, key) { + H[key] = function (arr) { + var _a; + error( + 32, + false, + void 0, + ((_a = {}), (_a["Highcharts." + key] = "use Array." + val), _a) + ); + return Array.prototype[val].apply(arr, [].slice.call(arguments, 1)); + }; + } + ); + /* eslint-disable valid-jsdoc */ + /** + * 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. + * + * @return {Function} + * A callback function to remove the added event. + */ + var addEvent = (H.addEvent = function (el, type, fn, options) { + if (options === void 0) { + options = {}; + } + /* eslint-enable valid-jsdoc */ + var events, + addEventListener = el.addEventListener || H.addEventListenerPolyfill; + // If we're setting events directly on the constructor, use a separate + // collection, `protoEvents` to distinguish it from the item events in + // `hcEvents`. + if (typeof el === "function" && el.prototype) { + events = el.prototype.protoEvents = el.prototype.protoEvents || {}; + } else { + events = el.hcEvents = el.hcEvents || {}; + } + // Allow click events added to points, otherwise they will be prevented by + // the TouchPointer.pinch function after a pinch zoom operation (#7091). + if (H.Point && el instanceof H.Point && el.series && el.series.chart) { + el.series.chart.runTrackerClick = true; + } + // Handle DOM events + if (addEventListener) { + addEventListener.call(el, type, fn, false); + } + if (!events[type]) { + events[type] = []; + } + var eventObject = { + fn: fn, + order: typeof options.order === "number" ? options.order : Infinity, + }; + events[type].push(eventObject); + // Order the calls + events[type].sort(function (a, b) { + return a.order - b.order; + }); + // Return a function that can be called to remove this event. + return function () { + removeEvent(el, type, fn); + }; + }); + /* eslint-disable valid-jsdoc */ + /** + * Remove an event that was added with {@link Highcharts#addEvent}. + * + * @function Highcharts.removeEvent<T> + * + * @param {Highcharts.Class<T>|T} el + * The element to remove events on. + * + * @param {string} [type] + * The type of events to remove. If undefined, all events are removed + * from the element. + * + * @param {Highcharts.EventCallbackFunction<T>} [fn] + * The specific callback to remove. If undefined, all events that match + * the element and optionally the type are removed. + * + * @return {void} + */ + var removeEvent = (H.removeEvent = function removeEvent(el, type, fn) { + /* eslint-enable valid-jsdoc */ + var events; + /** + * @private + * @param {string} type - event type + * @param {Highcharts.EventCallbackFunction<T>} fn - callback + * @return {void} + */ + function removeOneEvent(type, fn) { + var removeEventListener = + el.removeEventListener || H.removeEventListenerPolyfill; + if (removeEventListener) { + removeEventListener.call(el, type, fn, false); + } + } + /** + * @private + * @param {any} eventCollection - collection + * @return {void} + */ + function removeAllEvents(eventCollection) { + var types, len; + if (!el.nodeName) { + return; // break on non-DOM events + } + if (type) { + types = {}; + types[type] = true; + } else { + types = eventCollection; + } + objectEach(types, function (_val, n) { + if (eventCollection[n]) { + len = eventCollection[n].length; + while (len--) { + removeOneEvent(n, eventCollection[n][len].fn); + } + } + }); + } + ["protoEvents", "hcEvents"].forEach(function (coll, i) { + var eventElem = i ? el : el.prototype; + var eventCollection = eventElem && eventElem[coll]; + if (eventCollection) { + if (type) { + events = eventCollection[type] || []; + if (fn) { + eventCollection[type] = events.filter(function (obj) { + return fn !== obj.fn; + }); + removeOneEvent(type, fn); + } else { + removeAllEvents(eventCollection); + eventCollection[type] = []; + } + } else { + removeAllEvents(eventCollection); + eventElem[coll] = {}; + } + } + }); + }); + /* eslint-disable valid-jsdoc */ + /** + * 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} + */ + var fireEvent = (H.fireEvent = function ( + el, + type, + eventArguments, + defaultFunction + ) { + /* eslint-enable valid-jsdoc */ + var e, i; + eventArguments = eventArguments || {}; + if (doc.createEvent && (el.dispatchEvent || el.fireEvent)) { + e = doc.createEvent("Events"); + e.initEvent(type, true, true); + extend(e, eventArguments); + if (el.dispatchEvent) { + el.dispatchEvent(e); + } else { + el.fireEvent(type, e); + } + } else { + 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, + // and setting it will cause en error in IE8 (#2465). + type: type, + }); + } + var fireInOrder = function (protoEvents, hcEvents) { + if (protoEvents === void 0) { + protoEvents = []; + } + if (hcEvents === void 0) { + hcEvents = []; + } + var iA = 0; + var iB = 0; + var length = protoEvents.length + hcEvents.length; + for (i = 0; i < length; i++) { + var obj = !protoEvents[iA] + ? hcEvents[iB++] + : !hcEvents[iB] + ? protoEvents[iA++] + : protoEvents[iA].order <= hcEvents[iB].order + ? protoEvents[iA++] + : hcEvents[iB++]; + // If the event handler return false, prevent the default + // handler from executing + if (obj.fn.call(el, eventArguments) === false) { + eventArguments.preventDefault(); + } + } + }; + fireInOrder( + el.protoEvents && el.protoEvents[type], + el.hcEvents && el.hcEvents[type] + ); + } + // Run the default if not prevented + if (defaultFunction && !eventArguments.defaultPrevented) { + defaultFunction.call(el, eventArguments); + } + }); + var serialMode; + /** + * Get a unique key for using in internal element id's and pointers. The key is + * composed of a random hash specific to this Highcharts instance, and a + * counter. + * + * @example + * var id = uniqueKey(); // => 'highcharts-x45f6hp-0' + * + * @function Highcharts.uniqueKey + * + * @return {string} + * A unique key. + */ + var uniqueKey = (H.uniqueKey = (function () { + var hash = Math.random().toString(36).substring(2, 9) + "-"; + var id = 0; + return function () { + return "highcharts-" + (serialMode ? "" : hash) + id++; + }; + })()); + /** + * Activates a serial mode for element IDs provided by + * {@link Highcharts.uniqueKey}. This mode can be used in automated tests, where + * a simple comparison of two rendered SVG graphics is needed. + * + * **Note:** This is only for testing purposes and will break functionality in + * webpages with multiple charts. + * + * @example + * if ( + * process && + * process.env.NODE_ENV === 'development' + * ) { + * Highcharts.useSerialIds(true); + * } + * + * @function Highcharts.useSerialIds + * + * @param {boolean} [mode] + * Changes the state of serial mode. + * + * @return {boolean|undefined} + * State of the serial mode. + */ + var useSerialIds = (H.useSerialIds = function (mode) { + return (serialMode = pick(mode, serialMode)); + }); + var isFunction = (H.isFunction = function (obj) { + return typeof obj === "function"; + }); + /** + * Get the updated default options. Until 3.0.7, merely exposing defaultOptions + * for outside modules wasn't enough because the setOptions method created a new + * object. + * + * @function Highcharts.getOptions + * + * @return {Highcharts.Options} + */ + var getOptions = (H.getOptions = function () { + return H.defaultOptions; + }); + /** + * Merge the default options with custom options and return the new options + * structure. Commonly used for defining reusable templates. + * + * @sample highcharts/global/useutc-false Setting a global option + * @sample highcharts/members/setoptions Applying a global theme + * + * @function Highcharts.setOptions + * + * @param {Highcharts.Options} options + * The new custom chart options. + * + * @return {Highcharts.Options} + * Updated options. + */ + var setOptions = (H.setOptions = function (options) { + // Copy in the default options + H.defaultOptions = merge(true, H.defaultOptions, options); + // Update the time object + if (options.time || options.global) { + H.time.update( + merge( + H.defaultOptions.global, + H.defaultOptions.time, + options.global, + options.time + ) + ); + } + return H.defaultOptions; + }); + // Register Highcharts as a plugin in jQuery + if (win.jQuery) { + /** + * Highcharts-extended JQuery. * - * (c) 2010-2020 Torstein Honsi + * @external JQuery + */ + /** + * Helper function to return the chart of the current JQuery selector + * element. * - * License: www.highcharts.com/license + * @function external:JQuery#highcharts * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @return {Highcharts.Chart} + * The chart that is linked to the JQuery selector element. + */ /** + * Factory function to create a chart in the current JQuery selector + * element. * - * */ - /* globals Image, window */ - /** - * Reference to the global SVGElement class as a workaround for a name conflict - * in the Highcharts namespace. - * - * @global - * @typedef {global.SVGElement} GlobalSVGElement - * - * @see https://developer.mozilla.org/en-US/docs/Web/API/SVGElement - */ - // glob is a temporary fix to allow our es-modules to work. - var glob = ( // @todo UMD variable named `window`, and glob named `win` - typeof win !== 'undefined' ? - win : - typeof window !== 'undefined' ? - window : - {}), doc = glob.document, SVG_NS = 'http://www.w3.org/2000/svg', userAgent = (glob.navigator && glob.navigator.userAgent) || '', svg = (doc && - doc.createElementNS && - !!doc.createElementNS(SVG_NS, 'svg').createSVGRect), isMS = /(edge|msie|trident)/i.test(userAgent) && !glob.opera, isFirefox = userAgent.indexOf('Firefox') !== -1, isChrome = userAgent.indexOf('Chrome') !== -1, hasBidiBug = (isFirefox && - parseInt(userAgent.split('Firefox/')[1], 10) < 4 // issue #38 - ); - var H = { - product: 'Highcharts', - version: '8.2.2', - deg2rad: Math.PI * 2 / 360, - doc: doc, - hasBidiBug: hasBidiBug, - hasTouch: !!glob.TouchEvent, - isMS: isMS, - isWebKit: userAgent.indexOf('AppleWebKit') !== -1, - isFirefox: isFirefox, - isChrome: isChrome, - isSafari: !isChrome && userAgent.indexOf('Safari') !== -1, - isTouchDevice: /(Mobile|Android|Windows Phone)/.test(userAgent), - SVG_NS: SVG_NS, - chartCount: 0, - seriesTypes: {}, - symbolSizes: {}, - svg: svg, - win: glob, - marginNames: ['plotTop', 'marginRight', 'marginBottom', 'plotLeft'], - noop: function () { }, - /** - * Theme options that should get applied to the chart. In module mode it - * might not be possible to change this property because of read-only - * restrictions, instead use {@link Highcharts.setOptions}. - * - * @name Highcharts.theme - * @type {Highcharts.Options} - */ - /** - * An array containing the current chart objects in the page. A chart's - * position in the array is preserved throughout the page's lifetime. When - * a chart is destroyed, the array item becomes `undefined`. - * - * @name Highcharts.charts - * @type {Array<Highcharts.Chart|undefined>} - */ - charts: [], - /** - * A hook for defining additional date format specifiers. New - * specifiers are defined as key-value pairs by using the - * specifier as key, and a function which takes the timestamp as - * value. This function returns the formatted portion of the - * date. - * - * @sample highcharts/global/dateformats/ - * Adding support for week number - * - * @name Highcharts.dateFormats - * @type {Highcharts.Dictionary<Highcharts.TimeFormatCallbackFunction>} - */ - dateFormats: {} - }; + * @function external:JQuery#highcharts + * + * @param {'Chart'|'Map'|'StockChart'|string} [className] + * Name of the factory class in the Highcharts namespace. + * + * @param {Highcharts.Options} [options] + * The chart options structure. + * + * @param {Highcharts.ChartCallbackFunction} [callback] + * Function to run when the chart has loaded and and all external + * images are loaded. Defining a + * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load) + * handler is equivalent. + * + * @return {JQuery} + * The current JQuery selector. + */ + win.jQuery.fn.highcharts = function () { + var args = [].slice.call(arguments); + if (this[0]) { + // this[0] is the renderTo div + // Create the chart + if (args[0]) { + new H[ // eslint-disable-line computed-property-spacing, no-new + // Constructor defaults to Chart + isString(args[0]) ? args.shift() : "Chart" + ](this[0], args[0], args[1]); + return this; + } + // When called without parameters or with the return argument, + // return an existing chart + return charts[attr(this[0], "data-highcharts-chart")]; + } + }; + } + // TODO use named exports when supported. + var utilitiesModule = { + addEvent: addEvent, + arrayMax: arrayMax, + arrayMin: arrayMin, + attr: attr, + clamp: clamp, + clearTimeout: internalClearTimeout, + correctFloat: correctFloat, + createElement: createElement, + css: css, + defined: defined, + destroyObjectProperties: destroyObjectProperties, + discardElement: discardElement, + erase: erase, + error: error, + extend: extend, + extendClass: extendClass, + find: find, + fireEvent: fireEvent, + format: format, + getMagnitude: getMagnitude, + getNestedProperty: getNestedProperty, + getOptions: getOptions, + getStyle: getStyle, + inArray: inArray, + isArray: isArray, + isClass: isClass, + isDOMElement: isDOMElement, + isFunction: isFunction, + isNumber: isNumber, + isObject: isObject, + isString: isString, + merge: merge, + normalizeTickInterval: normalizeTickInterval, + numberFormat: numberFormat, + objectEach: objectEach, + offset: offset, + pad: pad, + pick: pick, + pInt: pInt, + relativeLength: relativeLength, + removeEvent: removeEvent, + setOptions: setOptions, + splat: splat, + stableSort: stableSort, + syncTimeout: syncTimeout, + timeUnits: timeUnits, + uniqueKey: uniqueKey, + useSerialIds: useSerialIds, + wrap: wrap, + }; - return H; - }); - _registerModule(_modules, 'Core/Utilities.js', [_modules['Core/Globals.js']], function (H) { + return utilitiesModule; + } + ); + _registerModule( + _modules, + "Core/Color/Color.js", + [_modules["Core/Globals.js"], _modules["Core/Utilities.js"]], + function (H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var isNumber = U.isNumber, + merge = U.merge, + pInt = U.pInt; + /** + * A valid color to be parsed and handled by Highcharts. Highcharts internally + * supports hex colors like `#ffffff`, rgb colors like `rgb(255,255,255)` and + * rgba colors like `rgba(255,255,255,1)`. Other colors may be supported by the + * browsers and displayed correctly, but Highcharts is not able to process them + * and apply concepts like opacity and brightening. + * + * @typedef {string} Highcharts.ColorString + */ + /** + * A valid color type than can be parsed and handled by Highcharts. It can be a + * color string, a gradient object, or a pattern object. + * + * @typedef {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} Highcharts.ColorType + */ + /** + * Gradient options instead of a solid color. + * + * @example + * // Linear gradient used as a color option + * color: { + * linearGradient: { x1: 0, x2: 0, y1: 0, y2: 1 }, + * stops: [ + * [0, '#003399'], // start + * [0.5, '#ffffff'], // middle + * [1, '#3366AA'] // end + * ] + * } + * + * @interface Highcharts.GradientColorObject + */ /** + * Holds an object that defines the start position and the end position relative + * to the shape. + * @name Highcharts.GradientColorObject#linearGradient + * @type {Highcharts.LinearGradientColorObject|undefined} + */ /** + * Holds an object that defines the center position and the radius. + * @name Highcharts.GradientColorObject#radialGradient + * @type {Highcharts.RadialGradientColorObject|undefined} + */ /** + * The first item in each tuple is the position in the gradient, where 0 is the + * start of the gradient and 1 is the end of the gradient. Multiple stops can be + * applied. The second item is the color for each stop. This color can also be + * given in the rgba format. + * @name Highcharts.GradientColorObject#stops + * @type {Array<Highcharts.GradientColorStopObject>} + */ + /** + * Color stop tuple. + * + * @see Highcharts.GradientColorObject + * + * @interface Highcharts.GradientColorStopObject + */ /** + * @name Highcharts.GradientColorStopObject#0 + * @type {number} + */ /** + * @name Highcharts.GradientColorStopObject#1 + * @type {Highcharts.ColorString} + */ /** + * @name Highcharts.GradientColorStopObject#color + * @type {Highcharts.Color|undefined} + */ + /** + * Defines the start position and the end position for a gradient relative + * to the shape. Start position (x1, y1) and end position (x2, y2) are relative + * to the shape, where 0 means top/left and 1 is bottom/right. + * + * @interface Highcharts.LinearGradientColorObject + */ /** + * Start horizontal position of the gradient. Float ranges 0-1. + * @name Highcharts.LinearGradientColorObject#x1 + * @type {number} + */ /** + * End horizontal position of the gradient. Float ranges 0-1. + * @name Highcharts.LinearGradientColorObject#x2 + * @type {number} + */ /** + * Start vertical position of the gradient. Float ranges 0-1. + * @name Highcharts.LinearGradientColorObject#y1 + * @type {number} + */ /** + * End vertical position of the gradient. Float ranges 0-1. + * @name Highcharts.LinearGradientColorObject#y2 + * @type {number} + */ + /** + * Defines the center position and the radius for a gradient. + * + * @interface Highcharts.RadialGradientColorObject + */ /** + * Center horizontal position relative to the shape. Float ranges 0-1. + * @name Highcharts.RadialGradientColorObject#cx + * @type {number} + */ /** + * Center vertical position relative to the shape. Float ranges 0-1. + * @name Highcharts.RadialGradientColorObject#cy + * @type {number} + */ /** + * Radius relative to the shape. Float ranges 0-1. + * @name Highcharts.RadialGradientColorObject#r + * @type {number} + */ + (""); // detach doclets above + /* * + * + * Class + * + * */ + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * Handle color operations. Some object methods are chainable. + * + * @class + * @name Highcharts.Color + * + * @param {Highcharts.ColorType} input + * The input color in either rbga or hex format + */ + var Color = /** @class */ (function () { /* * * - * (c) 2010-2020 Torstein Honsi + * Constructors * - * License: www.highcharts.com/license + * */ + function Color(input) { + // Collection of parsers. This can be extended from the outside by pushing + // parsers to Highcharts.Color.prototype.parsers. + this.parsers = [ + { + // RGBA color + // eslint-disable-next-line max-len + regex: + /rgba\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]?(?:\.[0-9]+)?)\s*\)/, + parse: function (result) { + return [ + pInt(result[1]), + pInt(result[2]), + pInt(result[3]), + parseFloat(result[4], 10), + ]; + }, + }, + { + // RGB color + regex: + /rgb\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*\)/, + parse: function (result) { + return [pInt(result[1]), pInt(result[2]), pInt(result[3]), 1]; + }, + }, + ]; + this.rgba = []; + // Backwards compatibility, allow class overwrite + if (H.Color !== Color) { + return new H.Color(input); + } + // Backwards compatibility, allow instanciation without new (#13053) + if (!(this instanceof Color)) { + return new Color(input); + } + this.init(input); + } + /* * * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * Static Functions * * */ /** - * An animation configuration. Animation configurations can also be defined as - * booleans, where `false` turns off animation and `true` defaults to a duration - * of 500ms and defer of 0ms. + * Creates a color instance out of a color string or object. * - * @interface Highcharts.AnimationOptionsObject - */ /** - * A callback function to exectute when the animation finishes. - * @name Highcharts.AnimationOptionsObject#complete - * @type {Function|undefined} - */ /** - * The animation defer in milliseconds. - * @name Highcharts.AnimationOptionsObject#defer - * @type {number|undefined} - */ /** - * The animation duration in milliseconds. - * @name Highcharts.AnimationOptionsObject#duration - * @type {number|undefined} - */ /** - * The name of an easing function as defined on the `Math` object. - * @name Highcharts.AnimationOptionsObject#easing - * @type {string|Function|undefined} - */ /** - * A callback function to execute on each step of each attribute or CSS property - * that's being animated. The first argument contains information about the - * animation and progress. - * @name Highcharts.AnimationOptionsObject#step - * @type {Function|undefined} - */ - /** - * Creates a frame for the animated SVG element. - * - * @callback Highcharts.AnimationStepCallbackFunction - * - * @param {Highcharts.SVGElement} this - * The SVG element to animate. - * - * @return {void} - */ - /** - * Interface description for a class. + * @function Highcharts.Color.parse * - * @interface Highcharts.Class<T> - * @extends Function - */ /** - * Class costructor. - * @function Highcharts.Class<T>#new - * @param {...Array<*>} args - * Constructor arguments. - * @return {T} - * Class instance. - */ - /** - * A style object with camel case property names to define visual appearance of - * a SVG element or HTML element. The properties can be whatever styles are - * supported on the given SVG or HTML element. + * @param {Highcharts.ColorType} input + * The input color in either rbga or hex format. * - * @example - * { - * fontFamily: 'monospace', - * fontSize: '1.2em' - * } + * @return {Highcharts.Color} + * Color instance. + */ + Color.parse = function (input) { + return new Color(input); + }; + /* * * - * @interface Highcharts.CSSObject - */ /** - * @name Highcharts.CSSObject#[key:string] - * @type {boolean|number|string|undefined} - */ /** - * Background style for the element. - * @name Highcharts.CSSObject#background - * @type {string|undefined} - */ /** - * Background color of the element. - * @name Highcharts.CSSObject#backgroundColor - * @type {Highcharts.ColorString|undefined} - */ /** - * Border style for the element. - * @name Highcharts.CSSObject#border - * @type {string|undefined} - */ /** - * Radius of the element border. - * @name Highcharts.CSSObject#borderRadius - * @type {number|undefined} - */ /** - * Color used in the element. The 'contrast' option is a Highcharts custom - * property that results in black or white, depending on the background of the - * element. - * @name Highcharts.CSSObject#color - * @type {'contrast'|Highcharts.ColorString|undefined} - */ /** - * Style of the mouse cursor when resting over the element. - * @name Highcharts.CSSObject#cursor - * @type {Highcharts.CursorValue|undefined} - */ /** - * Font family of the element text. Multiple values have to be in decreasing - * preference order and separated by comma. - * @name Highcharts.CSSObject#fontFamily - * @type {string|undefined} - */ /** - * Font size of the element text. - * @name Highcharts.CSSObject#fontSize - * @type {string|undefined} - */ /** - * Font weight of the element text. - * @name Highcharts.CSSObject#fontWeight - * @type {string|undefined} - */ /** - * Height of the element. - * @name Highcharts.CSSObject#height - * @type {number|undefined} - */ /** - * Width of the element border. - * @name Highcharts.CSSObject#lineWidth - * @type {number|undefined} - */ /** - * Opacity of the element. - * @name Highcharts.CSSObject#opacity - * @type {number|undefined} - */ /** - * Space around the element content. - * @name Highcharts.CSSObject#padding - * @type {string|undefined} - */ /** - * Behaviour of the element when the mouse cursor rests over it. - * @name Highcharts.CSSObject#pointerEvents - * @type {string|undefined} - */ /** - * Positioning of the element. - * @name Highcharts.CSSObject#position - * @type {string|undefined} - */ /** - * Alignment of the element text. - * @name Highcharts.CSSObject#textAlign - * @type {string|undefined} - */ /** - * Additional decoration of the element text. - * @name Highcharts.CSSObject#textDecoration - * @type {string|undefined} - */ /** - * Outline style of the element text. - * @name Highcharts.CSSObject#textOutline - * @type {string|undefined} - */ /** - * Line break style of the element text. Highcharts SVG elements support - * `ellipsis` when a `width` is set. - * @name Highcharts.CSSObject#textOverflow - * @type {string|undefined} - */ /** - * Top spacing of the element relative to the parent element. - * @name Highcharts.CSSObject#top - * @type {string|undefined} - */ /** - * Animated transition of selected element properties. - * @name Highcharts.CSSObject#transition - * @type {string|undefined} - */ /** - * Line break style of the element text. - * @name Highcharts.CSSObject#whiteSpace - * @type {string|undefined} - */ /** - * Width of the element. - * @name Highcharts.CSSObject#width - * @type {number|undefined} - */ - /** - * All possible cursor styles. - * - * @typedef {'alias'|'all-scroll'|'auto'|'cell'|'col-resize'|'context-menu'|'copy'|'crosshair'|'default'|'e-resize'|'ew-resize'|'grab'|'grabbing'|'help'|'move'|'n-resize'|'ne-resize'|'nesw-resize'|'no-drop'|'none'|'not-allowed'|'ns-resize'|'nw-resize'|'nwse-resize'|'pointer'|'progress'|'row-resize'|'s-resize'|'se-resize'|'sw-resize'|'text'|'vertical-text'|'w-resize'|'wait'|'zoom-in'|'zoom-out'} Highcharts.CursorValue - */ - /** - * All possible dash styles. - * - * @typedef {'Dash'|'DashDot'|'Dot'|'LongDash'|'LongDashDot'|'LongDashDotDot'|'ShortDash'|'ShortDashDot'|'ShortDashDotDot'|'ShortDot'|'Solid'} Highcharts.DashStyleValue - */ - /** - * Generic dictionary in TypeScript notation. - * Use the native `Record<string, any>` instead. + * Functions * - * @deprecated - * @interface Highcharts.Dictionary<T> - */ /** - * @name Highcharts.Dictionary<T>#[key:string] - * @type {T} - */ + * */ /** - * The function callback to execute when the event is fired. The `this` context - * contains the instance, that fired the event. + * Parse the input color to rgba array * - * @callback Highcharts.EventCallbackFunction<T> - * - * @param {T} this + * @private + * @function Highcharts.Color#init * - * @param {Highcharts.Dictionary<*>|Event} [eventArguments] - * Event arguments. + * @param {Highcharts.ColorType} input + * The input color in either rbga or hex format * - * @return {boolean|void} + * @return {void} */ + Color.prototype.init = function (input) { + var result, rgba, i, parser, len; + this.input = input = + Color.names[ + input && input.toLowerCase ? input.toLowerCase() : "" + ] || input; + // Gradients + if (input && input.stops) { + this.stops = input.stops.map(function (stop) { + return new Color(stop[1]); + }); + // Solid colors + } else { + // Bitmasking as input[0] is not working for legacy IE. + if (input && input.charAt && input.charAt() === "#") { + len = input.length; + input = parseInt(input.substr(1), 16); + // Handle long-form, e.g. #AABBCC + if (len === 7) { + rgba = [ + (input & 0xff0000) >> 16, + (input & 0xff00) >> 8, + input & 0xff, + 1, + ]; + // Handle short-form, e.g. #ABC + // In short form, the value is assumed to be the same + // for both nibbles for each component. e.g. #ABC = #AABBCC + } else if (len === 4) { + rgba = [ + ((input & 0xf00) >> 4) | ((input & 0xf00) >> 8), + ((input & 0xf0) >> 4) | (input & 0xf0), + ((input & 0xf) << 4) | (input & 0xf), + 1, + ]; + } + } + // Otherwise, check regex parsers + if (!rgba) { + i = this.parsers.length; + while (i-- && !rgba) { + parser = this.parsers[i]; + result = parser.regex.exec(input); + if (result) { + rgba = parser.parse(result); + } + } + } + } + this.rgba = rgba || []; + }; /** - * The event options for adding function callback. - * - * @interface Highcharts.EventOptionsObject - */ /** - * The order the event handler should be called. This opens for having one - * handler be called before another, independent of in which order they were - * added. - * @name Highcharts.EventOptionsObject#order - * @type {number} - */ - /** - * Formats data as a string. Usually the data is accessible throught the `this` - * keyword. + * Return the color or gradient stops in the specified format * - * @callback Highcharts.FormatterCallbackFunction<T> + * @function Highcharts.Color#get * - * @param {T} this - * Context to format + * @param {string} [format] + * Possible values are 'a', 'rgb', 'rgba' (default). * - * @return {string} - * Formatted text + * @return {Highcharts.ColorType} + * This color as a string or gradient stops. */ + Color.prototype.get = function (format) { + var input = this.input, + rgba = this.rgba, + ret; + if (typeof this.stops !== "undefined") { + ret = merge(input); + ret.stops = [].concat(ret.stops); + this.stops.forEach(function (stop, i) { + ret.stops[i] = [ret.stops[i][0], stop.get(format)]; + }); + // it's NaN if gradient colors on a column chart + } else if (rgba && isNumber(rgba[0])) { + if (format === "rgb" || (!format && rgba[3] === 1)) { + ret = "rgb(" + rgba[0] + "," + rgba[1] + "," + rgba[2] + ")"; + } else if (format === "a") { + ret = rgba[3]; + } else { + ret = "rgba(" + rgba.join(",") + ")"; + } + } else { + ret = input; + } + return ret; + }; /** - * An object of key-value pairs for HTML attributes. + * Brighten the color instance. * - * @typedef {Highcharts.Dictionary<boolean|number|string|Function>} Highcharts.HTMLAttributes - */ + * @function Highcharts.Color#brighten + * + * @param {number} alpha + * The alpha value. + * + * @return {Highcharts.Color} + * This color with modifications. + */ + Color.prototype.brighten = function (alpha) { + var i, + rgba = this.rgba; + if (this.stops) { + this.stops.forEach(function (stop) { + stop.brighten(alpha); + }); + } else if (isNumber(alpha) && alpha !== 0) { + for (i = 0; i < 3; i++) { + rgba[i] += pInt(alpha * 255); + if (rgba[i] < 0) { + rgba[i] = 0; + } + if (rgba[i] > 255) { + rgba[i] = 255; + } + } + } + return this; + }; /** - * An HTML DOM element. The type is a reference to the regular HTMLElement in - * the global scope. + * Set the color's opacity to a given alpha value. * - * @typedef {global.HTMLElement} Highcharts.HTMLDOMElement + * @function Highcharts.Color#setOpacity * - * @see https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement + * @param {number} alpha + * Opacity between 0 and 1. + * + * @return {Highcharts.Color} + * Color with modifications. */ + Color.prototype.setOpacity = function (alpha) { + this.rgba[3] = alpha; + return this; + }; /** - * The iterator callback. + * Return an intermediate color between two colors. * - * @callback Highcharts.ObjectEachCallbackFunction<T> + * @function Highcharts.Color#tweenTo * - * @param {T} this - * The context. + * @param {Highcharts.Color} to + * The color object to tween to. * - * @param {*} value - * The property value. + * @param {number} pos + * The intermediate position, where 0 is the from color (current + * color item), and 1 is the `to` color. + * + * @return {Highcharts.ColorString} + * The intermediate color in rgba notation. + */ + Color.prototype.tweenTo = function (to, pos) { + // Check for has alpha, because rgba colors perform worse due to lack of + // support in WebKit. + var fromRgba = this.rgba, + toRgba = to.rgba, + hasAlpha, + ret; + // Unsupported color, return to-color (#3920, #7034) + if (!toRgba.length || !fromRgba || !fromRgba.length) { + ret = to.input || "none"; + // Interpolate + } else { + hasAlpha = toRgba[3] !== 1 || fromRgba[3] !== 1; + ret = + (hasAlpha ? "rgba(" : "rgb(") + + Math.round(toRgba[0] + (fromRgba[0] - toRgba[0]) * (1 - pos)) + + "," + + Math.round(toRgba[1] + (fromRgba[1] - toRgba[1]) * (1 - pos)) + + "," + + Math.round(toRgba[2] + (fromRgba[2] - toRgba[2]) * (1 - pos)) + + (hasAlpha + ? "," + (toRgba[3] + (fromRgba[3] - toRgba[3]) * (1 - pos)) + : "") + + ")"; + } + return ret; + }; + /* * * - * @param {string} key - * The property key. + * Static Properties * - * @param {*} obj - * The object that objectEach is being applied to. - */ - /** - * An object containing `left` and `top` properties for the position in the - * page. + * */ + // Collection of named colors. Can be extended from the outside by adding + // colors to Highcharts.Color.names. + Color.names = { + white: "#ffffff", + black: "#000000", + }; + return Color; + })(); + H.Color = Color; + /** + * Creates a color instance out of a color string. + * + * @function Highcharts.color + * + * @param {Highcharts.ColorType} input + * The input color in either rbga or hex format + * + * @return {Highcharts.Color} + * Color instance + */ + H.color = Color.parse; + /* * + * + * Export + * + * */ + + return Color; + } + ); + _registerModule( + _modules, + "Core/Animation/Fx.js", + [_modules["Core/Globals.js"], _modules["Core/Utilities.js"]], + function (H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var win = H.win; + var isNumber = U.isNumber, + objectEach = U.objectEach; + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * An animator object used internally. One instance applies to one property + * (attribute or style prop) on one element. Animation is always initiated + * through {@link SVGElement#animate}. + * + * @example + * var rect = renderer.rect(0, 0, 10, 10).add(); + * rect.animate({ width: 100 }); + * + * @private + * @class + * @name Highcharts.Fx + */ + var Fx = /** @class */ (function () { + /* * * - * @interface Highcharts.OffsetObject - */ /** - * Left distance to the page border. - * @name Highcharts.OffsetObject#left - * @type {number} - */ /** - * Top distance to the page border. - * @name Highcharts.OffsetObject#top - * @type {number} - */ - /** - * Describes a range. - * - * @interface Highcharts.RangeObject - */ /** - * Maximum number of the range. - * @name Highcharts.RangeObject#max - * @type {number} - */ /** - * Minimum number of the range. - * @name Highcharts.RangeObject#min - * @type {number} - */ - /** - * If a number is given, it defines the pixel length. If a percentage string is - * given, like for example `'50%'`, the setting defines a length relative to a - * base size, for example the size of a container. + * Constructors * - * @typedef {number|string} Highcharts.RelativeSize - */ + * */ /** - * Proceed function to call original (wrapped) function. * - * @callback Highcharts.WrapProceedFunction + * @param {Highcharts.HTMLDOMElement|Highcharts.SVGElement} elem + * The element to animate. * - * @param {*} [arg1] - * Optional argument. Without any arguments defaults to first argument of - * the wrapping function. + * @param {Partial<Highcharts.AnimationOptionsObject>} options + * Animation options. * - * @param {*} [arg2] - * Optional argument. Without any arguments defaults to second argument - * of the wrapping function. + * @param {string} prop + * The single attribute or CSS property to animate. + */ + function Fx(elem, options, prop) { + this.pos = NaN; + this.options = options; + this.elem = elem; + this.prop = prop; + } + /* * * - * @param {*} [arg3] - * Optional argument. Without any arguments defaults to third argument of - * the wrapping function. + * Functions * - * @return {*} - * Return value of the original function. - */ + * */ /** - * The Highcharts object is the placeholder for all other members, and various - * utility functions. The most important member of the namespace would be the - * chart constructor. + * Set the current step of a path definition on SVGElement. * - * @example - * var chart = Highcharts.chart('container', { ... }); + * @function Highcharts.Fx#dSetter * - * @namespace Highcharts + * @return {void} */ - H.timers = []; - var charts = H.charts, - doc = H.doc, - win = H.win; + Fx.prototype.dSetter = function () { + var paths = this.paths, + start = paths && paths[0], + end = paths && paths[1], + path = [], + now = this.now || 0; + // Land on the final path without adjustment points appended in the ends + if (now === 1 || !start || !end) { + path = this.toD || []; + } else if (start.length === end.length && now < 1) { + for (var i = 0; i < end.length; i++) { + // Tween between the start segment and the end segment. Start + // with a copy of the end segment and tween the appropriate + // numerics + var startSeg = start[i]; + var endSeg = end[i]; + var tweenSeg = []; + for (var j = 0; j < endSeg.length; j++) { + var startItem = startSeg[j]; + var endItem = endSeg[j]; + // Tween numbers + if ( + typeof startItem === "number" && + typeof endItem === "number" && + // Arc boolean flags + !(endSeg[0] === "A" && (j === 4 || j === 5)) + ) { + tweenSeg[j] = startItem + now * (endItem - startItem); + // Strings, take directly from the end segment + } else { + tweenSeg[j] = endItem; + } + } + path.push(tweenSeg); + } + // If animation is finished or length not matching, land on right value + } else { + path = end; + } + this.elem.attr("d", path, void 0, true); + }; /** - * Provide error messages for debugging, with links to online explanation. This - * function can be overridden to provide custom error handling. + * Update the element with the current animation step. * - * @sample highcharts/chart/highcharts-error/ - * Custom error handler + * @function Highcharts.Fx#update * - * @function Highcharts.error + * @return {void} + */ + Fx.prototype.update = function () { + var elem = this.elem, + prop = this.prop, // if destroyed, it is null + now = this.now, + step = this.options.step; + // Animation setter defined from outside + if (this[prop + "Setter"]) { + this[prop + "Setter"](); + // Other animations on SVGElement + } else if (elem.attr) { + if (elem.element) { + elem.attr(prop, now, null, true); + } + // HTML styles, raw HTML content like container size + } else { + elem.style[prop] = now + this.unit; + } + if (step) { + step.call(elem, now, this); + } + }; + /** + * Run an animation. * - * @param {number|string} code - * The error code. See - * [errors.xml](https://github.com/highcharts/highcharts/blob/master/errors/errors.xml) - * for available codes. If it is a string, the error message is printed - * directly in the console. + * @function Highcharts.Fx#run * - * @param {boolean} [stop=false] - * Whether to throw an error or just log a warning in the console. + * @param {number} from + * The current value, value to start from. * - * @param {Highcharts.Chart} [chart] - * Reference to the chart that causes the error. Used in 'debugger' - * module to display errors directly on the chart. - * Important note: This argument is undefined for errors that lack - * access to the Chart instance. + * @param {number} to + * The end value, value to land on. * - * @param {Highcharts.Dictionary<string>} [params] - * Additional parameters for the generated message. + * @param {string} unit + * The property unit, for example `px`. * * @return {void} */ - function error(code, stop, chart, params) { - var severity = stop ? 'Highcharts error' : 'Highcharts warning'; - if (code === 32) { - code = severity + ": Deprecated member"; - } - var isCode = isNumber(code), - message = isCode ? - severity + " #" + code + ": www.highcharts.com/errors/" + code + "/" : - code.toString(), - defaultHandler = function () { - if (stop) { - throw new Error(message); - } - // else ... - if (win.console && - error.messages.indexOf(message) === -1 // prevent console flooting - ) { - console.log(message); // eslint-disable-line no-console - } - }; - if (typeof params !== 'undefined') { - var additionalMessages_1 = ''; - if (isCode) { - message += '?'; - } - objectEach(params, function (value, key) { - additionalMessages_1 += "\n - " + key + ": " + value; - if (isCode) { - message += encodeURI(key) + '=' + encodeURI(value); - } - }); - message += additionalMessages_1; - } - if (chart) { - fireEvent(chart, 'displayError', { code: code, message: message, params: params }, defaultHandler); + Fx.prototype.run = function (from, to, unit) { + var self = this, + options = self.options, + timer = function (gotoEnd) { + return timer.stopped ? false : self.step(gotoEnd); + }, + requestAnimationFrame = + win.requestAnimationFrame || + function (step) { + setTimeout(step, 13); + }, + step = function () { + for (var i = 0; i < H.timers.length; i++) { + if (!H.timers[i]()) { + H.timers.splice(i--, 1); + } + } + if (H.timers.length) { + requestAnimationFrame(step); + } + }; + if (from === to && !this.elem["forceAnimate:" + this.prop]) { + delete options.curAnim[this.prop]; + if (options.complete && Object.keys(options.curAnim).length === 0) { + options.complete.call(this.elem); } - else { - defaultHandler(); + } else { + // #7166 + this.startTime = +new Date(); + this.start = from; + this.end = to; + this.unit = unit; + this.now = this.start; + this.pos = 0; + timer.elem = this.elem; + timer.prop = this.prop; + if (timer() && H.timers.push(timer) === 1) { + requestAnimationFrame(step); } - error.messages.push(message); - } - (function (error) { - error.messages = []; - })(error || (error = {})); - H.error = error; - /* eslint-disable valid-jsdoc */ + } + }; /** - * 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. + * Run a single step in the animation. * - * @function Highcharts.merge<T> + * @function Highcharts.Fx#step * - * @param {boolean} extend - * Whether to extend the left-side object (a) or return a whole new - * object. + * @param {boolean} [gotoEnd] + * Whether to go to the endpoint of the animation after abort. * - * @param {T|undefined} a - * The first object to extend. When only this is given, the function - * returns a deep copy. - * - * @param {...Array<object|undefined>} [n] - * An object 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. - */ /** - * Utility function to deep merge two or more objects and return a third 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 {T|undefined} a - * The first object to extend. When only this is given, the function - * returns a deep copy. - * - * @param {...Array<object|undefined>} [n] - * An object 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() { - /* eslint-enable valid-jsdoc */ - var i, - args = arguments, - len, - ret = {}, - doCopy = function (copy, - original) { - // An object is replacing a primitive - if (typeof copy !== 'object') { - copy = {}; - } - objectEach(original, function (value, key) { - // 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 (args[0] === true) { - ret = args[1]; - args = Array.prototype.slice.call(args, 2); - } - // For each argument, extend the return - len = args.length; - for (i = 0; i < len; i++) { - ret = doCopy(ret, args[i]); + * @return {boolean} + * Returns `true` if animation continues. + */ + Fx.prototype.step = function (gotoEnd) { + var t = +new Date(), + ret, + done, + options = this.options, + elem = this.elem, + complete = options.complete, + duration = options.duration, + curAnim = options.curAnim; + if (elem.attr && !elem.element) { + // #2616, element is destroyed + ret = false; + } else if (gotoEnd || t >= duration + this.startTime) { + this.now = this.end; + this.pos = 1; + this.update(); + curAnim[this.prop] = true; + done = true; + objectEach(curAnim, function (val) { + if (val !== true) { + done = false; + } + }); + if (done && complete) { + complete.call(elem); } - return ret; - } - H.merge = merge; + ret = false; + } else { + this.pos = options.easing((t - this.startTime) / duration); + this.now = this.start + (this.end - this.start) * this.pos; + this.update(); + ret = true; + } + return ret; + }; /** - * Constrain a value to within a lower and upper threshold. + * Prepare start and end values so that the path can be animated one to one. * - * @private - * @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; - } - /** - * Shortcut for parseInt + * @function Highcharts.Fx#initPath * - * @private - * @function Highcharts.pInt + * @param {Highcharts.SVGElement} elem + * The SVGElement item. * - * @param {*} s - * any + * @param {Highcharts.SVGPathArray|undefined} fromD + * Starting path definition. * - * @param {number} [mag] - * Magnitude + * @param {Highcharts.SVGPathArray} toD + * Ending path definition. * - * @return {number} - * number + * @return {Array<Highcharts.SVGPathArray,Highcharts.SVGPathArray>} + * An array containing start and end paths in array form so that + * they can be animated in parallel. */ - var pInt = H.pInt = function pInt(s, - mag) { - return parseInt(s, - mag || 10); + Fx.prototype.initPath = function (elem, fromD, toD) { + var shift, + startX = elem.startX, + endX = elem.endX, + fullLength, + i, + start = fromD && fromD.slice(), // copy + end = toD.slice(), // copy + isArea = elem.isArea, + positionFactor = isArea ? 2 : 1, + reverse; + if (!start) { + return [end, end]; + } + /** + * If shifting points, prepend a dummy point to the end path. + * @private + * @param {Highcharts.SVGPathArray} arr - array + * @param {Highcharts.SVGPathArray} other - array + * @return {void} + */ + function prepend(arr, other) { + while (arr.length < fullLength) { + // Move to, line to or curve to? + var moveSegment = arr[0], + otherSegment = other[fullLength - arr.length]; + if (otherSegment && moveSegment[0] === "M") { + if (otherSegment[0] === "C") { + arr[0] = [ + "C", + moveSegment[1], + moveSegment[2], + moveSegment[1], + moveSegment[2], + moveSegment[1], + moveSegment[2], + ]; + } else { + arr[0] = ["L", moveSegment[1], moveSegment[2]]; + } + } + // Prepend a copy of the first point + arr.unshift(moveSegment); + // For areas, the bottom path goes back again to the left, so we + // need to append a copy of the last point. + if (isArea) { + arr.push(arr[arr.length - 1]); + } + } + } + /** + * Copy and append last point until the length matches the end length. + * @private + * @param {Highcharts.SVGPathArray} arr - array + * @param {Highcharts.SVGPathArray} other - array + * @return {void} + */ + function append(arr, other) { + while (arr.length < fullLength) { + // Pull out the slice that is going to be appended or inserted. + // In a line graph, the positionFactor is 1, and the last point + // is sliced out. In an area graph, the positionFactor is 2, + // causing the middle two points to be sliced out, since an area + // path starts at left, follows the upper path then turns and + // follows the bottom back. + var segmentToAdd = arr[arr.length / positionFactor - 1].slice(); + // Disable the first control point of curve segments + if (segmentToAdd[0] === "C") { + segmentToAdd[1] = segmentToAdd[5]; + segmentToAdd[2] = segmentToAdd[6]; + } + if (!isArea) { + arr.push(segmentToAdd); + } else { + var lowerSegmentToAdd = + arr[arr.length / positionFactor].slice(); + arr.splice(arr.length / 2, 0, segmentToAdd, lowerSegmentToAdd); + } + } + } + // For sideways animation, find out how much we need to shift to get the + // start path Xs to match the end path Xs. + if (startX && endX) { + for (i = 0; i < startX.length; i++) { + // Moving left, new points coming in on right + if (startX[i] === endX[0]) { + shift = i; + break; + // Moving right + } else if (startX[0] === endX[endX.length - startX.length + i]) { + shift = i; + reverse = true; + break; + // Fixed from the right side, "scaling" left + } else if ( + startX[startX.length - 1] === + endX[endX.length - startX.length + i] + ) { + shift = startX.length - i; + break; + } + } + if (typeof shift === "undefined") { + start = []; + } + } + if (start.length && isNumber(shift)) { + // The common target length for the start and end array, where both + // arrays are padded in opposite ends + fullLength = end.length + shift * positionFactor; + if (!reverse) { + prepend(end, start); + append(start, end); + } else { + prepend(start, end); + append(end, start); + } + } + return [start, end]; }; /** - * Utility function to check for string type. + * Handle animation of the color attributes directly. * - * @function Highcharts.isString + * @function Highcharts.Fx#fillSetter * - * @param {*} s - * The item to check. - * - * @return {boolean} - * True if the argument is a string. + * @return {void} */ - var isString = H.isString = function isString(s) { - return typeof s === 'string'; + Fx.prototype.fillSetter = function () { + Fx.prototype.strokeSetter.apply(this, arguments); }; /** - * Utility function to check if an item is an array. - * - * @function Highcharts.isArray + * Handle animation of the color attributes directly. * - * @param {*} obj - * The item to check. + * @function Highcharts.Fx#strokeSetter * - * @return {boolean} - * True if the argument is an array. + * @return {void} */ - var isArray = H.isArray = function isArray(obj) { - var str = Object.prototype.toString.call(obj); - return str === '[object Array]' || str === '[object Array Iterator]'; + Fx.prototype.strokeSetter = function () { + this.elem.attr( + this.prop, + H.color(this.start).tweenTo(H.color(this.end), this.pos), + null, + true + ); }; - /** - * Utility function to check if an item is of type object. - * - * @function Highcharts.isObject - * - * @param {*} obj - * The item to check. + return Fx; + })(); + H.Fx = Fx; + + return Fx; + } + ); + _registerModule( + _modules, + "Core/Animation/AnimationUtilities.js", + [ + _modules["Core/Animation/Fx.js"], + _modules["Core/Globals.js"], + _modules["Core/Utilities.js"], + ], + function (Fx, H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var defined = U.defined, + getStyle = U.getStyle, + isArray = U.isArray, + isNumber = U.isNumber, + isObject = U.isObject, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick; + /** + * Set the global animation to either a given value, or fall back to the given + * chart's animation option. + * + * @function Highcharts.setAnimation + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>|undefined} animation + * The animation object. + * + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @todo + * This function always relates to a chart, and sets a property on the renderer, + * so it should be moved to the SVGRenderer. + */ + var setAnimation = (H.setAnimation = function setAnimation( + animation, + chart + ) { + chart.renderer.globalAnimation = pick( + animation, + chart.options.chart.animation, + true + ); + }); + /** + * Get the animation in object form, where a disabled animation is always + * returned as `{ duration: 0 }`. + * + * @function Highcharts.animObject + * + * @param {boolean|Highcharts.AnimationOptionsObject} [animation=0] + * An animation setting. Can be an object with duration, complete and + * easing properties, or a boolean to enable or disable. + * + * @return {Highcharts.AnimationOptionsObject} + * An object with at least a duration property. + */ + var animObject = (H.animObject = function animObject(animation) { + return isObject(animation) + ? H.merge({ duration: 500, defer: 0 }, animation) + : { duration: animation ? 500 : 0, defer: 0 }; + }); + /** + * Get the defer as a number value from series animation options. + * + * @function Highcharts.getDeferredAnimation + * + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @param {boolean|Highcharts.AnimationOptionsObject} animation + * An animation setting. Can be an object with duration, complete and + * easing properties, or a boolean to enable or disable. + * + * @param {Highcharts.Series} [series] + * Series to defer animation. + * + * @return {number} + * The numeric value. + */ + var getDeferredAnimation = (H.getDeferredAnimation = function ( + chart, + animation, + series + ) { + var labelAnimation = animObject(animation); + var s = series ? [series] : chart.series; + var defer = 0; + var duration = 0; + s.forEach(function (series) { + var seriesAnim = animObject(series.options.animation); + defer = + animation && defined(animation.defer) + ? labelAnimation.defer + : Math.max(defer, seriesAnim.duration + seriesAnim.defer); + duration = Math.min(labelAnimation.duration, seriesAnim.duration); + }); + // Disable defer for exporting + if (chart.renderer.forExport) { + defer = 0; + } + var anim = { + defer: Math.max(0, defer - duration), + duration: Math.min(defer, duration), + }; + return anim; + }); + /** + * The global animate method, which uses Fx to create individual animators. + * + * @function Highcharts.animate + * + * @param {Highcharts.HTMLDOMElement|Highcharts.SVGElement} el + * The element to animate. + * + * @param {Highcharts.CSSObject|Highcharts.SVGAttributes} params + * An object containing key-value pairs of the properties to animate. + * Supports numeric as pixel-based CSS properties for HTML objects and + * attributes for SVGElements. + * + * @param {Partial<Highcharts.AnimationOptionsObject>} [opt] + * Animation options. + * + * @return {void} + */ + var animate = (H.animate = function (el, params, opt) { + var start, + unit = "", + end, + fx, + args; + if (!isObject(opt)) { + // Number or undefined/null + args = arguments; + opt = { + duration: args[2], + easing: args[3], + complete: args[4], + }; + } + if (!isNumber(opt.duration)) { + opt.duration = 400; + } + opt.easing = + typeof opt.easing === "function" + ? opt.easing + : Math[opt.easing] || Math.easeInOutSine; + opt.curAnim = merge(params); + objectEach(params, function (val, prop) { + // Stop current running animation of this property + stop(el, prop); + fx = new Fx(el, opt, prop); + end = null; + if (prop === "d" && isArray(params.d)) { + fx.paths = fx.initPath(el, el.pathArray, params.d); + fx.toD = params.d; + start = 0; + end = 1; + } else if (el.attr) { + start = el.attr(prop); + } else { + start = parseFloat(getStyle(el, prop)) || 0; + if (prop !== "opacity") { + unit = "px"; + } + } + if (!end) { + end = val; + } + if (end && end.match && end.match("px")) { + end = end.replace(/px/g, ""); // #4351 + } + fx.run(start, end, unit); + }); + }); + /** + * Stop running animation. + * + * @function Highcharts.stop + * + * @param {Highcharts.SVGElement} el + * The SVGElement to stop animation on. + * + * @param {string} [prop] + * The property to stop animating. If given, the stop method will stop a + * single property from animating, while others continue. + * + * @return {void} + * + * @todo + * A possible extension to this would be to stop a single property, when + * we want to continue animating others. Then assign the prop to the timer + * in the Fx.run method, and check for the prop here. This would be an + * improvement in all cases where we stop the animation from .attr. Instead of + * stopping everything, we can just stop the actual attributes we're setting. + */ + var stop = (H.stop = function (el, prop) { + var i = H.timers.length; + // Remove timers related to this element (#4519) + while (i--) { + if (H.timers[i].elem === el && (!prop || prop === H.timers[i].prop)) { + H.timers[i].stopped = true; // #4667 + } + } + }); + var animationExports = { + animate: animate, + animObject: animObject, + getDeferredAnimation: getDeferredAnimation, + setAnimation: setAnimation, + stop: stop, + }; + + return animationExports; + } + ); + _registerModule( + _modules, + "Core/Renderer/SVG/SVGElement.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Color/Color.js"], + _modules["Core/Globals.js"], + _modules["Core/Utilities.js"], + ], + function (A, Color, H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animate = A.animate, + animObject = A.animObject, + stop = A.stop; + var deg2rad = H.deg2rad, + doc = H.doc, + hasTouch = H.hasTouch, + isFirefox = H.isFirefox, + noop = H.noop, + svg = H.svg, + SVG_NS = H.SVG_NS, + win = H.win; + var attr = U.attr, + createElement = U.createElement, + css = U.css, + defined = U.defined, + erase = U.erase, + extend = U.extend, + fireEvent = U.fireEvent, + isArray = U.isArray, + isFunction = U.isFunction, + isNumber = U.isNumber, + isString = U.isString, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick, + pInt = U.pInt, + syncTimeout = U.syncTimeout, + uniqueKey = U.uniqueKey; + /** + * The horizontal alignment of an element. + * + * @typedef {"center"|"left"|"right"} Highcharts.AlignValue + */ + /** + * Options to align the element relative to the chart or another box. + * + * @interface Highcharts.AlignObject + */ /** + * Horizontal alignment. Can be one of `left`, `center` and `right`. + * + * @name Highcharts.AlignObject#align + * @type {Highcharts.AlignValue|undefined} + * + * @default left + */ /** + * Vertical alignment. Can be one of `top`, `middle` and `bottom`. + * + * @name Highcharts.AlignObject#verticalAlign + * @type {Highcharts.VerticalAlignValue|undefined} + * + * @default top + */ /** + * Horizontal pixel offset from alignment. + * + * @name Highcharts.AlignObject#x + * @type {number|undefined} + * + * @default 0 + */ /** + * Vertical pixel offset from alignment. + * + * @name Highcharts.AlignObject#y + * @type {number|undefined} + * + * @default 0 + */ /** + * Use the `transform` attribute with translateX and translateY custom + * attributes to align this elements rather than `x` and `y` attributes. + * + * @name Highcharts.AlignObject#alignByTranslate + * @type {boolean|undefined} + * + * @default false + */ + /** + * Bounding box of an element. + * + * @interface Highcharts.BBoxObject + * @extends Highcharts.PositionObject + */ /** + * Height of the bounding box. + * + * @name Highcharts.BBoxObject#height + * @type {number} + */ /** + * Width of the bounding box. + * + * @name Highcharts.BBoxObject#width + * @type {number} + */ /** + * Horizontal position of the bounding box. + * + * @name Highcharts.BBoxObject#x + * @type {number} + */ /** + * Vertical position of the bounding box. + * + * @name Highcharts.BBoxObject#y + * @type {number} + */ + /** + * An object of key-value pairs for SVG attributes. Attributes in Highcharts + * elements for the most parts correspond to SVG, but some are specific to + * Highcharts, like `zIndex`, `rotation`, `rotationOriginX`, + * `rotationOriginY`, `translateX`, `translateY`, `scaleX` and `scaleY`. SVG + * attributes containing a hyphen are _not_ camel-cased, they should be + * quoted to preserve the hyphen. + * + * @example + * { + * 'stroke': '#ff0000', // basic + * 'stroke-width': 2, // hyphenated + * 'rotation': 45 // custom + * 'd': ['M', 10, 10, 'L', 30, 30, 'z'] // path definition, note format + * } + * + * @interface Highcharts.SVGAttributes + */ /** + * @name Highcharts.SVGAttributes#[key:string] + * @type {*} + */ /** + * @name Highcharts.SVGAttributes#d + * @type {string|Highcharts.SVGPathArray|undefined} + */ /** + * @name Highcharts.SVGAttributes#fill + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} + */ /** + * @name Highcharts.SVGAttributes#inverted + * @type {boolean|undefined} + */ /** + * @name Highcharts.SVGAttributes#matrix + * @type {Array<number>|undefined} + */ /** + * @name Highcharts.SVGAttributes#rotation + * @type {number|undefined} + */ /** + * @name Highcharts.SVGAttributes#rotationOriginX + * @type {number|undefined} + */ /** + * @name Highcharts.SVGAttributes#rotationOriginY + * @type {number|undefined} + */ /** + * @name Highcharts.SVGAttributes#scaleX + * @type {number|undefined} + */ /** + * @name Highcharts.SVGAttributes#scaleY + * @type {number|undefined} + */ /** + * @name Highcharts.SVGAttributes#stroke + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} + */ /** + * @name Highcharts.SVGAttributes#style + * @type {string|Highcharts.CSSObject|undefined} + */ /** + * @name Highcharts.SVGAttributes#translateX + * @type {number|undefined} + */ /** + * @name Highcharts.SVGAttributes#translateY + * @type {number|undefined} + */ /** + * @name Highcharts.SVGAttributes#zIndex + * @type {number|undefined} + */ + /** + * An SVG DOM element. The type is a reference to the regular SVGElement in the + * global scope. + * + * @typedef {globals.GlobalSVGElement} Highcharts.SVGDOMElement + * + * @see https://developer.mozilla.org/en-US/docs/Web/API/SVGElement + */ + /** + * The vertical alignment of an element. + * + * @typedef {"bottom"|"middle"|"top"} Highcharts.VerticalAlignValue + */ + (""); // detach doclets above + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The SVGElement prototype is a JavaScript wrapper for SVG elements used in the + * rendering layer of Highcharts. Combined with the + * {@link Highcharts.SVGRenderer} + * object, these prototypes allow freeform annotation in the charts or even in + * HTML pages without instanciating a chart. The SVGElement can also wrap HTML + * labels, when `text` or `label` elements are created with the `useHTML` + * parameter. + * + * The SVGElement instances are created through factory functions on the + * {@link Highcharts.SVGRenderer} + * object, like + * {@link Highcharts.SVGRenderer#rect|rect}, + * {@link Highcharts.SVGRenderer#path|path}, + * {@link Highcharts.SVGRenderer#text|text}, + * {@link Highcharts.SVGRenderer#label|label}, + * {@link Highcharts.SVGRenderer#g|g} + * and more. + * + * @class + * @name Highcharts.SVGElement + */ + var SVGElement = /** @class */ (function () { + function SVGElement() { + /* * + * + * Properties + * + * */ + this.element = void 0; + this.height = void 0; + this.opacity = 1; // Default base for animation + this.renderer = void 0; + this.SVG_NS = SVG_NS; + // Custom attributes used for symbols, these should be filtered out when + // setting SVGElement attributes (#9375). + this.symbolCustomAttribs = [ + "x", + "y", + "width", + "height", + "r", + "start", + "end", + "innerR", + "anchorX", + "anchorY", + "rounded", + ]; + this.width = void 0; + } + /* * * - * @param {boolean} [strict=false] - * Also checks that the object is not an array. + * Functions * - * @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 - } - H.isObject = isObject; + * */ /** - * Utility function to check if an Object is a HTML Element. + * Get the current value of an attribute or pseudo attribute, + * used mainly for animation. Called internally from + * the {@link Highcharts.SVGRenderer#attr} function. * - * @function Highcharts.isDOMElement + * @private + * @function Highcharts.SVGElement#_defaultGetter * - * @param {*} obj - * The item to check. + * @param {string} key + * Property key. * - * @return {boolean} - * True if the argument is a HTML Element. - */ - var isDOMElement = H.isDOMElement = function isDOMElement(obj) { - return isObject(obj) && typeof obj.nodeType === 'number'; + * @return {number|string} + * Property value. + */ + SVGElement.prototype._defaultGetter = function (key) { + var ret = pick( + this[key + "Value"], // align getter + this[key], + this.element ? this.element.getAttribute(key) : null, + 0 + ); + if (/^[\-0-9\.]+$/.test(ret)) { + // is numerical + ret = parseFloat(ret); + } + return ret; }; /** - * Utility function to check if an Object is a class. + * @private + * @function Highcharts.SVGElement#_defaultSetter + * + * @param {string} value * - * @function Highcharts.isClass + * @param {string} key * - * @param {object|undefined} obj - * The item to check. + * @param {Highcharts.SVGDOMElement} element * - * @return {boolean} - * True if the argument is a class. + * @return {void} */ - var isClass = H.isClass = function isClass(obj) { - var c = obj && obj.constructor; - return !!(isObject(obj, true) && - !isDOMElement(obj) && - (c && c.name && c.name !== 'Object')); + SVGElement.prototype._defaultSetter = function (value, key, element) { + element.setAttribute(key, value); + }; + /** + * Add the element to the DOM. All elements must be added this way. + * + * @sample highcharts/members/renderer-g + * Elements added to a group + * + * @function Highcharts.SVGElement#add + * + * @param {Highcharts.SVGElement} [parent] + * The parent item to add it to. If undefined, the element is added + * to the {@link Highcharts.SVGRenderer.box}. + * + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. + */ + SVGElement.prototype.add = function (parent) { + var renderer = this.renderer, + element = this.element, + inserted; + if (parent) { + this.parentGroup = parent; + } + // Mark as inverted + this.parentInverted = parent && parent.inverted; + // Build formatted text + if ( + typeof this.textStr !== "undefined" && + this.element.nodeName === "text" // Not for SVGLabel instances + ) { + renderer.buildText(this); + } + // Mark as added + this.added = true; + // If we're adding to renderer root, or other elements in the group + // have a z index, we need to handle it + if (!parent || parent.handleZ || this.zIndex) { + inserted = this.zIndexSetter(); + } + // If zIndex is not handled, append at the end + if (!inserted) { + (parent ? parent.element : renderer.box).appendChild(element); + } + // fire an event for internal hooks + if (this.onAdd) { + this.onAdd(); + } + return this; + }; + /** + * Add a class name to an element. + * + * @function Highcharts.SVGElement#addClass + * + * @param {string} className + * The new class name to add. + * + * @param {boolean} [replace=false] + * When true, the existing class name(s) will be overwritten with the new + * one. When false, the new one is added. + * + * @return {Highcharts.SVGElement} + * Return the SVG element for chainability. + */ + SVGElement.prototype.addClass = function (className, replace) { + var currentClassName = replace ? "" : this.attr("class") || ""; + // Trim the string and remove duplicates + className = (className || "") + .split(/ /g) + .reduce( + function (newClassName, name) { + if (currentClassName.indexOf(name) === -1) { + newClassName.push(name); + } + return newClassName; + }, + currentClassName ? [currentClassName] : [] + ) + .join(" "); + if (className !== currentClassName) { + this.attr("class", className); + } + return this; }; /** - * Utility function to check if an item is a number and it is finite (not NaN, - * Infinity or -Infinity). + * This method is executed in the end of `attr()`, after setting all + * attributes in the hash. In can be used to efficiently consolidate + * multiple attributes in one SVG property -- e.g., translate, rotate and + * scale are merged in one "transform" attribute in the SVG node. * - * @function Highcharts.isNumber + * @private + * @function Highcharts.SVGElement#afterSetters + */ + SVGElement.prototype.afterSetters = function () { + // Update transform. Do this outside the loop to prevent redundant + // updating for batch setting of attributes. + if (this.doTransform) { + this.updateTransform(); + this.doTransform = false; + } + }; + /** + * Align the element relative to the chart or another box. * - * @param {*} n - * The item to check. + * @function Highcharts.SVGElement#align * - * @return {boolean} - * True if the item is a finite number - */ - var isNumber = H.isNumber = function isNumber(n) { - return typeof n === 'number' && !isNaN(n) && n < Infinity && n > -Infinity; + * @param {Highcharts.AlignObject} [alignOptions] + * The alignment options. The function can be called without this + * parameter in order to re-align an element after the box has been + * updated. + * + * @param {boolean} [alignByTranslate] + * Align element by translation. + * + * @param {string|Highcharts.BBoxObject} [box] + * The box to align to, needs a width and height. When the box is a + * string, it refers to an object in the Renderer. For example, when + * box is `spacingBox`, it refers to `Renderer.spacingBox` which + * holds `width`, `height`, `x` and `y` properties. + * + * @return {Highcharts.SVGElement} Returns the SVGElement for chaining. + */ + SVGElement.prototype.align = function ( + alignOptions, + alignByTranslate, + box + ) { + var align, + vAlign, + x, + y, + attribs = {}, + alignTo, + renderer = this.renderer, + alignedObjects = renderer.alignedObjects, + alignFactor, + vAlignFactor; + // First call on instanciate + if (alignOptions) { + this.alignOptions = alignOptions; + this.alignByTranslate = alignByTranslate; + if (!box || isString(box)) { + this.alignTo = alignTo = box || "renderer"; + // prevent duplicates, like legendGroup after resize + erase(alignedObjects, this); + alignedObjects.push(this); + box = void 0; // reassign it below + } + // When called on resize, no arguments are supplied + } else { + alignOptions = this.alignOptions; + alignByTranslate = this.alignByTranslate; + alignTo = this.alignTo; + } + box = pick(box, renderer[alignTo], renderer); + // Assign variables + align = alignOptions.align; + vAlign = alignOptions.verticalAlign; + // default: left align + x = (box.x || 0) + (alignOptions.x || 0); + // default: top align + y = (box.y || 0) + (alignOptions.y || 0); + // Align + if (align === "right") { + alignFactor = 1; + } else if (align === "center") { + alignFactor = 2; + } + if (alignFactor) { + x += (box.width - (alignOptions.width || 0)) / alignFactor; + } + attribs[alignByTranslate ? "translateX" : "x"] = Math.round(x); + // Vertical align + if (vAlign === "bottom") { + vAlignFactor = 1; + } else if (vAlign === "middle") { + vAlignFactor = 2; + } + if (vAlignFactor) { + y += (box.height - (alignOptions.height || 0)) / vAlignFactor; + } + attribs[alignByTranslate ? "translateY" : "y"] = Math.round(y); + // Animate only if already placed + this[this.placed ? "animate" : "attr"](attribs); + this.placed = true; + this.alignAttr = attribs; + return this; + }; + /** + * @private + * @function Highcharts.SVGElement#alignSetter + * @param {"left"|"center"|"right"} value + */ + SVGElement.prototype.alignSetter = function (value) { + var convert = { + left: "start", + center: "middle", + right: "end", + }; + if (convert[value]) { + this.alignValue = value; + this.element.setAttribute("text-anchor", convert[value]); + } }; /** - * Remove the last occurence of an item from an array. + * Animate to given attributes or CSS properties. * - * @function Highcharts.erase + * @sample highcharts/members/element-on/ + * Setting some attributes by animation * - * @param {Array<*>} arr - * The array. + * @function Highcharts.SVGElement#animate * - * @param {*} item - * The item to remove. + * @param {Highcharts.SVGAttributes} params + * SVG attributes or CSS to animate. * - * @return {void} - */ - var erase = H.erase = function erase(arr, - item) { - var i = arr.length; - while (i--) { - if (arr[i] === item) { - arr.splice(i, 1); - break; - } + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [options] + * Animation options. + * + * @param {Function} [complete] + * Function to perform at the end of animation. + * + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. + */ + SVGElement.prototype.animate = function (params, options, complete) { + var _this = this; + var animOptions = animObject( + pick(options, this.renderer.globalAnimation, true) + ), + deferTime = animOptions.defer; + // When the page is hidden save resources in the background by not + // running animation at all (#9749). + if (pick(doc.hidden, doc.msHidden, doc.webkitHidden, false)) { + animOptions.duration = 0; + } + if (animOptions.duration !== 0) { + // allows using a callback with the global animation without + // overwriting it + if (complete) { + animOptions.complete = complete; } + // If defer option is defined delay the animation #12901 + syncTimeout(function () { + if (_this.element) { + animate(_this, params, animOptions); + } + }, deferTime); + } else { + this.attr(params, void 0, complete); + // Call the end step synchronously + objectEach( + params, + function (val, prop) { + if (animOptions.step) { + animOptions.step.call(this, val, { prop: prop, pos: 1 }); + } + }, + this + ); + } + return this; }; /** - * Check if an object is null or undefined. - * - * @function Highcharts.defined + * Apply a text outline through a custom CSS property, by copying the text + * element and apply stroke to the copy. Used internally. Contrast checks at + * [example](https://jsfiddle.net/highcharts/43soe9m1/2/). * - * @param {*} obj - * The object to check. + * @example + * // Specific color + * text.css({ + * textOutline: '1px black' + * }); + * // Automatic contrast + * text.css({ + * color: '#000000', // black text + * textOutline: '1px contrast' // => white outline + * }); * - * @return {boolean} - * False if the object is null or undefined, otherwise true. - */ - var defined = H.defined = function defined(obj) { - return typeof obj !== 'undefined' && obj !== null; + * @private + * @function Highcharts.SVGElement#applyTextOutline + * + * @param {string} textOutline + * A custom CSS `text-outline` setting, defined by `width color`. + */ + SVGElement.prototype.applyTextOutline = function (textOutline) { + var elem = this.element, + tspans, + hasContrast = textOutline.indexOf("contrast") !== -1, + styles = {}, + color, + strokeWidth, + firstRealChild; + // When the text shadow is set to contrast, use dark stroke for light + // text and vice versa. + if (hasContrast) { + styles.textOutline = textOutline = textOutline.replace( + /contrast/g, + this.renderer.getContrast(elem.style.fill) + ); + } + // Extract the stroke width and color + textOutline = textOutline.split(" "); + color = textOutline[textOutline.length - 1]; + strokeWidth = textOutline[0]; + if (strokeWidth && strokeWidth !== "none" && H.svg) { + this.fakeTS = true; // Fake text shadow + tspans = [].slice.call(elem.getElementsByTagName("tspan")); + // In order to get the right y position of the clone, + // copy over the y setter + this.ySetter = this.xSetter; + // Since the stroke is applied on center of the actual outline, we + // need to double it to get the correct stroke-width outside the + // glyphs. + strokeWidth = strokeWidth.replace( + /(^[\d\.]+)(.*?)$/g, + function (match, digit, unit) { + return 2 * digit + unit; + } + ); + // Remove shadows from previous runs. + this.removeTextOutline(tspans); + // Check if the element contains RTL characters. + // Comparing against Hebrew and Arabic characters, + // excluding Arabic digits. Source: + // https://www.unicode.org/Public/UNIDATA/extracted/DerivedBidiClass.txt + var isRTL_1 = elem.textContent + ? /^[\u0591-\u065F\u066A-\u07FF\uFB1D-\uFDFD\uFE70-\uFEFC]/.test( + elem.textContent + ) + : false; + // For each of the tspans, create a stroked copy behind it. + firstRealChild = elem.firstChild; + tspans.forEach(function (tspan, y) { + var clone; + // Let the first line start at the correct X position + if (y === 0) { + tspan.setAttribute("x", elem.getAttribute("x")); + y = elem.getAttribute("y"); + tspan.setAttribute("y", y || 0); + if (y === null) { + elem.setAttribute("y", 0); + } + } + // Create the clone and apply outline properties. + // For RTL elements apply outline properties for orginal element + // to prevent outline from overlapping the text. + // For RTL in Firefox keep the orginal order (#10162). + clone = tspan.cloneNode(true); + attr(isRTL_1 && !isFirefox ? tspan : clone, { + class: "highcharts-text-outline", + fill: color, + stroke: color, + "stroke-width": strokeWidth, + "stroke-linejoin": "round", + }); + elem.insertBefore(clone, firstRealChild); + }); + // Create a whitespace between tspan and clone, + // to fix the display of Arabic characters in Firefox. + if (isRTL_1 && isFirefox && tspans[0]) { + var whitespace = tspans[0].cloneNode(true); + whitespace.textContent = " "; + elem.insertBefore(whitespace, firstRealChild); + } + } }; /** - * 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. To use as a getter, pass only a string as the second argument. + * @function Highcharts.SVGElement#attr + * @param {string} key + * @return {number|string} + */ /** + * Apply native and custom attributes to the SVG elements. * - * @function Highcharts.attr + * In order to set the rotation center for rotation, set x and y to 0 and + * use `translateX` and `translateY` attributes to position the element + * instead. * - * @param {Highcharts.HTMLDOMElement|Highcharts.SVGDOMElement} elem - * The DOM element to receive the attribute(s). + * Attributes frequently used in Highcharts are `fill`, `stroke`, + * `stroke-width`. * - * @param {string|Highcharts.HTMLAttributes|Highcharts.SVGAttributes} [prop] - * The property or an object of key-value pairs. + * @sample highcharts/members/renderer-rect/ + * Setting some attributes * - * @param {number|string} [value] - * The value if a single property is set. + * @example + * // Set multiple attributes + * element.attr({ + * stroke: 'red', + * fill: 'blue', + * x: 10, + * y: 10 + * }); * - * @return {string|null|undefined} - * When used as a getter, return the value. - */ - function attr(elem, prop, value) { - var ret; - // if the prop is a string - if (isString(prop)) { - // set the value - if (defined(value)) { - elem.setAttribute(prop, value); - // get the value - } - else if (elem && elem.getAttribute) { - ret = elem.getAttribute(prop); - // IE7 and below cannot get class through getAttribute (#7850) - if (!ret && prop === 'class') { - ret = elem.getAttribute(prop + 'Name'); - } + * // Set a single attribute + * element.attr('stroke', 'red'); + * + * // Get an attribute + * element.attr('stroke'); // => 'red' + * + * @function Highcharts.SVGElement#attr + * + * @param {string|Highcharts.SVGAttributes} [hash] + * The native and custom SVG attributes. + * + * @param {number|string|Highcharts.SVGPathArray} [val] + * If the type of the first argument is `string`, the second can be a + * value, which will serve as a single attribute setter. If the first + * argument is a string and the second is undefined, the function + * serves as a getter and the current value of the property is + * returned. + * + * @param {Function} [complete] + * A callback function to execute after setting the attributes. This + * makes the function compliant and interchangeable with the + * {@link SVGElement#animate} function. + * + * @param {boolean} [continueAnimation=true] + * Used internally when `.attr` is called as part of an animation + * step. Otherwise, calling `.attr` for an attribute will stop + * animation for that attribute. + * + * @return {Highcharts.SVGElement} + * If used as a setter, it returns the current + * {@link Highcharts.SVGElement} so the calls can be chained. If + * used as a getter, the current value of the attribute is returned. + */ + SVGElement.prototype.attr = function ( + hash, + val, + complete, + continueAnimation + ) { + var key, + element = this.element, + hasSetSymbolSize, + ret = this, + skipAttr, + setter, + symbolCustomAttribs = this.symbolCustomAttribs; + // single key-value pair + if (typeof hash === "string" && typeof val !== "undefined") { + key = hash; + hash = {}; + hash[key] = val; + } + // used as a getter: first argument is a string, second is undefined + if (typeof hash === "string") { + ret = (this[hash + "Getter"] || this._defaultGetter).call( + this, + hash, + element + ); + // setter + } else { + objectEach( + hash, + function eachAttribute(val, key) { + skipAttr = false; + // Unless .attr is from the animator update, stop current + // running animation of this property + if (!continueAnimation) { + stop(this, key); + } + // Special handling of symbol attributes + if ( + this.symbolName && + symbolCustomAttribs.indexOf(key) !== -1 + ) { + if (!hasSetSymbolSize) { + this.symbolAttr(hash); + hasSetSymbolSize = true; + } + skipAttr = true; + } + if (this.rotation && (key === "x" || key === "y")) { + this.doTransform = true; + } + if (!skipAttr) { + setter = this[key + "Setter"] || this._defaultSetter; + setter.call(this, val, key, element); + // Let the shadow follow the main element + if ( + !this.styledMode && + this.shadows && + /^(width|height|visibility|x|y|d|transform|cx|cy|r)$/.test( + key + ) + ) { + this.updateShadows(key, val, setter); + } } - // else if prop is defined, it is a hash of key/value pairs - } - else { - objectEach(prop, function (val, key) { - elem.setAttribute(key, val); - }); - } - return ret; - } - H.attr = attr; + }, + this + ); + this.afterSetters(); + } + // In accordance with animate, run a complete callback + if (complete) { + complete.call(this); + } + return ret; + }; /** - * Check if an element is an array, and if not, make it into an array. + * Apply a clipping rectangle to this element. * - * @function Highcharts.splat + * @function Highcharts.SVGElement#clip * - * @param {*} obj - * The object to splat. + * @param {Highcharts.ClipRectElement} [clipRect] + * The clipping rectangle. If skipped, the current clip is removed. * - * @return {Array} - * The produced or original array. + * @return {Highcharts.SVGElement} + * Returns the SVG element to allow chaining. */ - var splat = H.splat = function splat(obj) { - return isArray(obj) ? obj : [obj]; + SVGElement.prototype.clip = function (clipRect) { + return this.attr( + "clip-path", + clipRect + ? "url(" + this.renderer.url + "#" + clipRect.id + ")" + : "none" + ); }; /** - * Set a timeout if the delay is given, otherwise perform the function - * synchronously. - * - * @function Highcharts.syncTimeout - * - * @param {Function} fn - * The function callback. - * - * @param {number} delay - * Delay in milliseconds. - * - * @param {*} [context] - * An optional context to send to the function callback. - * - * @return {number} - * An identifier for the timeout that can later be cleared with - * Highcharts.clearTimeout. Returns -1 if there is no timeout. - */ - var syncTimeout = H.syncTimeout = function syncTimeout(fn, - delay, - context) { - if (delay > 0) { - return setTimeout(fn, - delay, - context); - } - fn.call(0, context); - return -1; + * Calculate the coordinates needed for drawing a rectangle crisply and + * return the calculated attributes. + * + * @function Highcharts.SVGElement#crisp + * + * @param {Highcharts.RectangleObject} rect + * Rectangle to crisp. + * + * @param {number} [strokeWidth] + * The stroke width to consider when computing crisp positioning. It can + * also be set directly on the rect parameter. + * + * @return {Highcharts.RectangleObject} + * The modified rectangle arguments. + */ + SVGElement.prototype.crisp = function (rect, strokeWidth) { + var wrapper = this, + normalizer; + strokeWidth = strokeWidth || rect.strokeWidth || 0; + // Math.round because strokeWidth can sometimes have roundoff errors + normalizer = (Math.round(strokeWidth) % 2) / 2; + // normalize for crisp edges + rect.x = Math.floor(rect.x || wrapper.x || 0) + normalizer; + rect.y = Math.floor(rect.y || wrapper.y || 0) + normalizer; + rect.width = Math.floor( + (rect.width || wrapper.width || 0) - 2 * normalizer + ); + rect.height = Math.floor( + (rect.height || wrapper.height || 0) - 2 * normalizer + ); + if (defined(rect.strokeWidth)) { + rect.strokeWidth = strokeWidth; + } + return rect; }; /** - * 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). + * Build and apply an SVG gradient out of a common JavaScript configuration + * object. This function is called from the attribute setters. An event + * hook is added for supporting other complex color types. * - * @function Highcharts.clearTimeout + * @private + * @function Highcharts.SVGElement#complexColor * - * @param {number} id - * Id of a timeout. + * @param {Highcharts.GradientColorObject|Highcharts.PatternObject} colorOptions + * The gradient or pattern options structure. * - * @return {void} - */ - var internalClearTimeout = H.clearTimeout = function (id) { - if (defined(id)) { - clearTimeout(id); + * @param {string} prop + * The property to apply, can either be `fill` or `stroke`. + * + * @param {Highcharts.SVGDOMElement} elem + * SVG element to apply the gradient on. + */ + SVGElement.prototype.complexColor = function ( + colorOptions, + prop, + elem + ) { + var renderer = this.renderer, + colorObject, + gradName, + gradAttr, + radAttr, + gradients, + stops, + stopColor, + stopOpacity, + radialReference, + id, + key = [], + value; + fireEvent( + this.renderer, + "complexColor", + { + args: arguments, + }, + function () { + // Apply linear or radial gradients + if (colorOptions.radialGradient) { + gradName = "radialGradient"; + } else if (colorOptions.linearGradient) { + gradName = "linearGradient"; + } + if (gradName) { + gradAttr = colorOptions[gradName]; + gradients = renderer.gradients; + stops = colorOptions.stops; + radialReference = elem.radialReference; + // Keep < 2.2 kompatibility + if (isArray(gradAttr)) { + colorOptions[gradName] = gradAttr = { + x1: gradAttr[0], + y1: gradAttr[1], + x2: gradAttr[2], + y2: gradAttr[3], + gradientUnits: "userSpaceOnUse", + }; + } + // Correct the radial gradient for the radial reference system + if ( + gradName === "radialGradient" && + radialReference && + !defined(gradAttr.gradientUnits) + ) { + // Save the radial attributes for updating + radAttr = gradAttr; + gradAttr = merge( + gradAttr, + renderer.getRadialAttr(radialReference, radAttr), + { gradientUnits: "userSpaceOnUse" } + ); + } + // Build the unique key to detect whether we need to create a + // new element (#1282) + objectEach(gradAttr, function (val, n) { + if (n !== "id") { + key.push(n, val); + } + }); + objectEach(stops, function (val) { + key.push(val); + }); + key = key.join(","); + // Check if a gradient object with the same config object is + // created within this renderer + if (gradients[key]) { + id = gradients[key].attr("id"); + } else { + // Set the id and create the element + gradAttr.id = id = uniqueKey(); + var gradientObject_1 = (gradients[key] = renderer + .createElement(gradName) + .attr(gradAttr) + .add(renderer.defs)); + gradientObject_1.radAttr = radAttr; + // The gradient needs to keep a list of stops to be able to + // destroy them + gradientObject_1.stops = []; + stops.forEach(function (stop) { + var stopObject; + if (stop[1].indexOf("rgba") === 0) { + colorObject = Color.parse(stop[1]); + stopColor = colorObject.get("rgb"); + stopOpacity = colorObject.get("a"); + } else { + stopColor = stop[1]; + stopOpacity = 1; + } + stopObject = renderer + .createElement("stop") + .attr({ + offset: stop[0], + "stop-color": stopColor, + "stop-opacity": stopOpacity, + }) + .add(gradientObject_1); + // Add the stop element to the gradient + gradientObject_1.stops.push(stopObject); + }); + } + // Set the reference to the gradient object + value = "url(" + renderer.url + "#" + id + ")"; + elem.setAttribute(prop, value); + elem.gradient = key; + // Allow the color to be concatenated into tooltips formatters + // etc. (#2995) + colorOptions.toString = function () { + return value; + }; + } } + ); }; - /* eslint-disable valid-jsdoc */ /** - * Utility function to extend an object with the members of another. + * Set styles for the element. In addition to CSS styles supported by + * native SVG and HTML elements, there are also some custom made for + * Highcharts, like `width`, `ellipsis` and `textOverflow` for SVG text + * elements. * - * @function Highcharts.extend<T> + * @sample highcharts/members/renderer-text-on-chart/ + * Styled text * - * @param {T|undefined} a - * The object to be extended. + * @function Highcharts.SVGElement#css * - * @param {object} b - * The object to add to the first one. - * - * @return {T} - * Object a, the original object. - */ - var extend = H.extend = function extend(a, - b) { - /* eslint-enable valid-jsdoc */ - var n; - if (!a) { - a = {}; + * @param {Highcharts.CSSObject} styles + * The new CSS styles. + * + * @return {Highcharts.SVGElement} + * Return the SVG element for chaining. + */ + SVGElement.prototype.css = function (styles) { + var oldStyles = this.styles, + newStyles = {}, + elem = this.element, + textWidth, + serializedCss = "", + hyphenate, + hasNew = !oldStyles, + // These CSS properties are interpreted internally by the SVG + // renderer, but are not supported by SVG and should not be added to + // the DOM. In styled mode, no CSS should find its way to the DOM + // whatsoever (#6173, #6474). + svgPseudoProps = ["textOutline", "textOverflow", "width"]; + // convert legacy + if (styles && styles.color) { + styles.fill = styles.color; + } + // Filter out existing styles to increase performance (#2640) + if (oldStyles) { + objectEach(styles, function (style, n) { + if (oldStyles && oldStyles[n] !== style) { + newStyles[n] = style; + hasNew = true; + } + }); + } + if (hasNew) { + // Merge the new styles with the old ones + if (oldStyles) { + styles = extend(oldStyles, newStyles); + } + // Get the text width from style + if (styles) { + // Previously set, unset it (#8234) + if (styles.width === null || styles.width === "auto") { + delete this.textWidth; + // Apply new + } else if ( + elem.nodeName.toLowerCase() === "text" && + styles.width + ) { + textWidth = this.textWidth = pInt(styles.width); + } + } + // store object + this.styles = styles; + if (textWidth && !svg && this.renderer.forExport) { + delete styles.width; } - for (n in b) { // eslint-disable-line guard-for-in - a[n] = b[n]; + // Serialize and set style attribute + if (elem.namespaceURI === this.SVG_NS) { + // #7633 + hyphenate = function (a, b) { + return "-" + b.toLowerCase(); + }; + objectEach(styles, function (style, n) { + if (svgPseudoProps.indexOf(n) === -1) { + serializedCss += + n.replace(/([A-Z])/g, hyphenate) + ":" + style + ";"; + } + }); + if (serializedCss) { + attr(elem, "style", serializedCss); // #1881 + } + } else { + css(elem, styles); } - return a; + if (this.added) { + // Rebuild text after added. Cache mechanisms in the buildText + // will prevent building if there are no significant changes. + if (this.element.nodeName === "text") { + this.renderer.buildText(this); + } + // Apply text outline after added + if (styles && styles.textOutline) { + this.applyTextOutline(styles.textOutline); + } + } + } + return this; }; - /* eslint-disable valid-jsdoc */ /** - * Return the first value that is not null or undefined. - * - * @function Highcharts.pick<T> - * - * @param {...Array<T|null|undefined>} items - * Variable number of arguments to inspect. - * - * @return {T} - * The value of the first argument that is not null or undefined. - */ - function pick() { - var args = arguments; - var length = args.length; - for (var i = 0; i < length; i++) { - var arg = args[i]; - if (typeof arg !== 'undefined' && arg !== null) { - return arg; - } + * @private + * @function Highcharts.SVGElement#dashstyleSetter + * @param {string} value + */ + SVGElement.prototype.dashstyleSetter = function (value) { + var i, + strokeWidth = this["stroke-width"]; + // If "inherit", like maps in IE, assume 1 (#4981). With HC5 and the new + // strokeWidth function, we should be able to use that instead. + if (strokeWidth === "inherit") { + strokeWidth = 1; + } + value = value && value.toLowerCase(); + if (value) { + var v = value + .replace("shortdashdotdot", "3,1,1,1,1,1,") + .replace("shortdashdot", "3,1,1,1") + .replace("shortdot", "1,1,") + .replace("shortdash", "3,1,") + .replace("longdash", "8,3,") + .replace(/dot/g, "1,3,") + .replace("dash", "4,3,") + .replace(/,$/, "") + .split(","); // ending comma + i = v.length; + while (i--) { + v[i] = "" + pInt(v[i]) * pick(strokeWidth, NaN); } - } - H.pick = pick; + value = v.join(",").replace(/NaN/g, "none"); // #3226 + this.element.setAttribute("stroke-dasharray", value); + } + }; /** - * Set CSS on a given element. - * - * @function Highcharts.css - * - * @param {Highcharts.HTMLDOMElement|Highcharts.SVGDOMElement} el - * An HTML DOM element. + * Destroy the element and element wrapper and clear up the DOM and event + * hooks. + * + * @function Highcharts.SVGElement#destroy + */ + SVGElement.prototype.destroy = function () { + var wrapper = this, + element = wrapper.element || {}, + renderer = wrapper.renderer, + parentToClean = + (renderer.isSVG && + element.nodeName === "SPAN" && + wrapper.parentGroup) || + void 0, + grandParent, + ownerSVGElement = element.ownerSVGElement, + i; + // remove events + element.onclick = + element.onmouseout = + element.onmouseover = + element.onmousemove = + element.point = + null; + stop(wrapper); // stop running animations + if (wrapper.clipPath && ownerSVGElement) { + var clipPath_1 = wrapper.clipPath; + // Look for existing references to this clipPath and remove them + // before destroying the element (#6196). + // The upper case version is for Edge + [].forEach.call( + ownerSVGElement.querySelectorAll("[clip-path],[CLIP-PATH]"), + function (el) { + var clipPathAttr = el.getAttribute("clip-path"); + if (clipPathAttr.indexOf(clipPath_1.element.id) > -1) { + el.removeAttribute("clip-path"); + } + } + ); + wrapper.clipPath = clipPath_1.destroy(); + } + // Destroy stops in case this is a gradient object @todo old code? + if (wrapper.stops) { + for (i = 0; i < wrapper.stops.length; i++) { + wrapper.stops[i].destroy(); + } + wrapper.stops.length = 0; + wrapper.stops = void 0; + } + // remove element + wrapper.safeRemoveChild(element); + if (!renderer.styledMode) { + wrapper.destroyShadows(); + } + // In case of useHTML, clean up empty containers emulating SVG groups + // (#1960, #2393, #2697). + while ( + parentToClean && + parentToClean.div && + parentToClean.div.childNodes.length === 0 + ) { + grandParent = parentToClean.parentGroup; + wrapper.safeRemoveChild(parentToClean.div); + delete parentToClean.div; + parentToClean = grandParent; + } + // remove from alignObjects + if (wrapper.alignTo) { + erase(renderer.alignedObjects, wrapper); + } + objectEach(wrapper, function (val, key) { + // Destroy child elements of a group + if ( + wrapper[key] && + wrapper[key].parentGroup === wrapper && + wrapper[key].destroy + ) { + wrapper[key].destroy(); + } + // Delete all properties + delete wrapper[key]; + }); + return; + }; + /** + * Destroy shadows on the element. * - * @param {Highcharts.CSSObject} styles - * Style object with camel case property names. + * @private + * @function Highcharts.SVGElement#destroyShadows * * @return {void} */ - var css = H.css = function css(el, - styles) { - if (H.isMS && !H.svg) { // #2686 - if (styles && typeof styles.opacity !== 'undefined') { - styles.filter = - 'alpha(opacity=' + (styles.opacity * 100) + ')'; - } + SVGElement.prototype.destroyShadows = function () { + (this.shadows || []).forEach(function (shadow) { + this.safeRemoveChild(shadow); + }, this); + this.shadows = void 0; + }; + /** + * @private + */ + SVGElement.prototype.destroyTextPath = function (elem, path) { + var textElement = elem.getElementsByTagName("text")[0]; + var tspans; + if (textElement) { + // Remove textPath attributes + textElement.removeAttribute("dx"); + textElement.removeAttribute("dy"); + // Remove ID's: + path.element.setAttribute("id", ""); + // Check if textElement includes textPath, + if ( + this.textPathWrapper && + textElement.getElementsByTagName("textPath").length + ) { + // Move nodes to <text> + tspans = this.textPathWrapper.element.childNodes; + // Now move all <tspan>'s to the <textPath> node + while (tspans.length) { + textElement.appendChild(tspans[0]); + } + // Remove <textPath> from the DOM + textElement.removeChild(this.textPathWrapper.element); } - extend(el.style, styles); + } else if (elem.getAttribute("dx") || elem.getAttribute("dy")) { + // Remove textPath attributes from elem + // to get correct text-outline position + elem.removeAttribute("dx"); + elem.removeAttribute("dy"); + } + if (this.textPathWrapper) { + // Set textPathWrapper to undefined and destroy it + this.textPathWrapper = this.textPathWrapper.destroy(); + } }; /** - * 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. + * @private + * @function Highcharts.SVGElement#dSettter + * @param {number|string|Highcharts.SVGPathArray} value + * @param {string} key + * @param {Highcharts.SVGDOMElement} element + */ + SVGElement.prototype.dSetter = function (value, key, element) { + if (isArray(value)) { + // Backwards compatibility, convert one-dimensional array into an + // array of segments + if (typeof value[0] === "string") { + value = this.renderer.pathToSegments(value); + } + this.pathArray = value; + value = value.reduce(function (acc, seg, i) { + if (!seg || !seg.join) { + return (seg || "").toString(); + } + return (i ? acc + " " : "") + seg.join(" "); + }, ""); + } + if (/(NaN| {2}|^$)/.test(value)) { + value = "M 0 0"; + } + // Check for cache before resetting. Resetting causes disturbance in the + // DOM, causing flickering in some cases in Edge/IE (#6747). Also + // possible performance gain. + if (this[key] !== value) { + element.setAttribute(key, value); + this[key] = value; + } + }; + /** + * Fade out an element by animating its opacity down to 0, and hide it on + * complete. Used internally for the tooltip. * - * @param {boolean} [nopad=false] - * If true, remove all padding, border and margin. + * @function Highcharts.SVGElement#fadeOut * - * @return {Highcharts.HTMLDOMElement} - * The created DOM element. + * @param {number} [duration=150] + * The fade duration in milliseconds. */ - var createElement = H.createElement = function createElement(tag, - attribs, - styles, - parent, - nopad) { - var el = doc.createElement(tag); - if (attribs) { - extend(el, attribs); + SVGElement.prototype.fadeOut = function (duration) { + var elemWrapper = this; + elemWrapper.animate( + { + opacity: 0, + }, + { + duration: pick(duration, 150), + complete: function () { + // #3088, assuming we're only using this for tooltips + elemWrapper.attr({ y: -9999 }).hide(); + }, } - if (nopad) { - css(el, { padding: '0', border: 'none', margin: '0' }); + ); + }; + /** + * @private + * @function Highcharts.SVGElement#fillSetter + * @param {Highcharts.ColorType} value + * @param {string} key + * @param {Highcharts.SVGDOMElement} element + */ + SVGElement.prototype.fillSetter = function (value, key, element) { + if (typeof value === "string") { + element.setAttribute(key, value); + } else if (value) { + this.complexColor(value, key, element); + } + }; + /** + * Get the bounding box (width, height, x and y) for the element. Generally + * used to get rendered text size. Since this is called a lot in charts, + * the results are cached based on text properties, in order to save DOM + * traffic. The returned bounding box includes the rotation, so for example + * a single text line of rotation 90 will report a greater height, and a + * width corresponding to the line-height. + * + * @sample highcharts/members/renderer-on-chart/ + * Draw a rectangle based on a text's bounding box + * + * @function Highcharts.SVGElement#getBBox + * + * @param {boolean} [reload] + * Skip the cache and get the updated DOM bouding box. + * + * @param {number} [rot] + * Override the element's rotation. This is internally used on axis + * labels with a value of 0 to find out what the bounding box would + * be have been if it were not rotated. + * + * @return {Highcharts.BBoxObject} + * The bounding box with `x`, `y`, `width` and `height` properties. + */ + SVGElement.prototype.getBBox = function (reload, rot) { + var wrapper = this, + bBox, // = wrapper.bBox, + renderer = wrapper.renderer, + width, + height, + element = wrapper.element, + styles = wrapper.styles, + fontSize, + textStr = wrapper.textStr, + toggleTextShadowShim, + cache = renderer.cache, + cacheKeys = renderer.cacheKeys, + isSVG = element.namespaceURI === wrapper.SVG_NS, + cacheKey; + var rotation = pick(rot, wrapper.rotation, 0); + fontSize = renderer.styledMode + ? element && + SVGElement.prototype.getStyle.call(element, "font-size") + : styles && styles.fontSize; + // Avoid undefined and null (#7316) + if (defined(textStr)) { + cacheKey = textStr.toString(); + // Since numbers are monospaced, and numerical labels appear a lot + // in a chart, we assume that a label of n characters has the same + // bounding box as others of the same length. Unless there is inner + // HTML in the label. In that case, leave the numbers as is (#5899). + if (cacheKey.indexOf("<") === -1) { + cacheKey = cacheKey.replace(/[0-9]/g, "0"); } - if (styles) { - css(el, styles); + // Properties that affect bounding box + cacheKey += [ + "", + rotation, + fontSize, + wrapper.textWidth, + styles && styles.textOverflow, + styles && styles.fontWeight, // #12163 + ].join(","); + } + if (cacheKey && !reload) { + bBox = cache[cacheKey]; + } + // No cache found + if (!bBox) { + // SVG elements + if (isSVG || renderer.forExport) { + try { + // Fails in Firefox if the container has display: none. + // When the text shadow shim is used, we need to hide the + // fake shadows to get the correct bounding box (#3872) + toggleTextShadowShim = + this.fakeTS && + function (display) { + [].forEach.call( + element.querySelectorAll(".highcharts-text-outline"), + function (tspan) { + tspan.style.display = display; + } + ); + }; + // Workaround for #3842, Firefox reporting wrong bounding + // box for shadows + if (isFunction(toggleTextShadowShim)) { + toggleTextShadowShim("none"); + } + bBox = element.getBBox + ? // SVG: use extend because IE9 is not allowed to change + // width and height in case of rotation (below) + extend({}, element.getBBox()) + : { + // Legacy IE in export mode + width: element.offsetWidth, + height: element.offsetHeight, + }; + // #3842 + if (isFunction(toggleTextShadowShim)) { + toggleTextShadowShim(""); + } + } catch (e) { + (""); + } + // If the bBox is not set, the try-catch block above failed. The + // other condition is for Opera that returns a width of + // -Infinity on hidden elements. + if (!bBox || bBox.width < 0) { + bBox = { width: 0, height: 0 }; + } + // VML Renderer or useHTML within SVG + } else { + bBox = wrapper.htmlGetBBox(); + } + // True SVG elements as well as HTML elements in modern browsers + // using the .useHTML option need to compensated for rotation + if (renderer.isSVG) { + width = bBox.width; + height = bBox.height; + // Workaround for wrong bounding box in IE, Edge and Chrome on + // Windows. With Highcharts' default font, IE and Edge report + // a box height of 16.899 and Chrome rounds it to 17. If this + // stands uncorrected, it results in more padding added below + // the text than above when adding a label border or background. + // Also vertical positioning is affected. + // https://jsfiddle.net/highcharts/em37nvuj/ + // (#1101, #1505, #1669, #2568, #6213). + if (isSVG) { + bBox.height = height = + { + "11px,17": 14, + "13px,20": 16, + }[styles && styles.fontSize + "," + Math.round(height)] || + height; + } + // Adjust for rotated text + if (rotation) { + var rad = rotation * deg2rad; + bBox.width = + Math.abs(height * Math.sin(rad)) + + Math.abs(width * Math.cos(rad)); + bBox.height = + Math.abs(height * Math.cos(rad)) + + Math.abs(width * Math.sin(rad)); + } } - if (parent) { - parent.appendChild(el); + // Cache it. When loading a chart in a hidden iframe in Firefox and + // IE/Edge, the bounding box height is 0, so don't cache it (#5620). + if (cacheKey && bBox.height > 0) { + // Rotate (#4681) + while (cacheKeys.length > 250) { + delete cache[cacheKeys.shift()]; + } + if (!cache[cacheKey]) { + cacheKeys.push(cacheKey); + } + cache[cacheKey] = bBox; } - return el; + } + return bBox; }; - // eslint-disable-next-line valid-jsdoc /** - * Extend a prototyped class by new members. + * Get the computed style. Only in styled mode. * - * @function Highcharts.extendClass<T> + * @example + * chart.series[0].points[0].graphic.getStyle('stroke-width'); // => '1px' * - * @param {Highcharts.Class<T>} parent - * The parent prototype to inherit. + * @function Highcharts.SVGElement#getStyle * - * @param {Highcharts.Dictionary<*>} members - * A collection of prototype members to add or override compared to the - * parent prototype. + * @param {string} prop + * The property name to check for. * - * @return {Highcharts.Class<T>} - * A new prototype. + * @return {string} + * The current computed value. */ - var extendClass = H.extendClass = function extendClass(parent, - members) { - var obj = (function () { }); - obj.prototype = new parent(); // eslint-disable-line new-cap - extend(obj.prototype, members); - return obj; + SVGElement.prototype.getStyle = function (prop) { + return win + .getComputedStyle(this.element || this, "") + .getPropertyValue(prop); }; /** - * Left-pad a string to a given length by adding a character repetetively. - * - * @function Highcharts.pad + * Check if an element has the given class name. * - * @param {number} number - * The input string or number. + * @function Highcharts.SVGElement#hasClass * - * @param {number} [length] - * The desired string length. + * @param {string} className + * The class name to check for. * - * @param {string} [padder=0] - * The character to pad with. - * - * @return {string} - * The padded string. + * @return {boolean} + * Whether the class name is found. */ - var pad = H.pad = function pad(number, length, padder) { - return new Array((length || 2) + - 1 - - String(number) - .replace('-', '') - .length).join(padder || '0') + number; + SVGElement.prototype.hasClass = function (className) { + return ("" + this.attr("class")).split(" ").indexOf(className) !== -1; }; /** - * Return a length based on either the integer value, or a percentage of a base. - * - * @function Highcharts.relativeLength + * Hide the element, similar to setting the `visibility` attribute to + * `hidden`. * - * @param {Highcharts.RelativeSize} value - * A percentage string or a number. + * @function Highcharts.SVGElement#hide * - * @param {number} base - * The full length that represents 100%. + * @param {boolean} [hideByTranslation=false] + * The flag to determine if element should be hidden by moving out + * of the viewport. Used for example for dataLabels. * - * @param {number} [offset=0] - * A pixel offset to apply for percentage values. Used internally in - * axis positioning. - * - * @return {number} - * The computed length. - */ - var relativeLength = H.relativeLength = function relativeLength(value, - base, - offset) { - return (/%$/).test(value) ? - (base * parseFloat(value) / 100) + (offset || 0) : - parseFloat(value); - }; - /** - * Wrap a method with extended functionality, preserving the original function. - * - * @function Highcharts.wrap - * - * @param {*} obj - * The context object that the method belongs to. In real cases, this is - * often a prototype. - * - * @param {string} method - * The name of the method to extend. - * - * @param {Highcharts.WrapProceedFunction} func - * A wrapper function callback. This function is called with the same - * arguments as the original function, except that the original function - * is unshifted and passed as the first argument. - */ - var wrap = H.wrap = function wrap(obj, - method, - func) { - var proceed = obj[method]; - obj[method] = function () { - var args = Array.prototype.slice.call(arguments), - outerArgs = arguments, - ctx = this, - ret; - ctx.proceed = function () { - proceed.apply(ctx, arguments.length ? arguments : outerArgs); - }; - args.unshift(proceed); - ret = func.apply(this, args); - ctx.proceed = null; - return ret; - }; + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. + */ + SVGElement.prototype.hide = function (hideByTranslation) { + if (hideByTranslation) { + this.attr({ y: -9999 }); + } else { + this.attr({ visibility: "hidden" }); + } + return this; }; /** - * Format a string according to a subset of the rules of Python's String.format - * method. - * - * @example - * var s = Highcharts.format( - * 'The {color} fox was {len:.2f} feet long', - * { color: 'red', len: Math.PI } - * ); - * // => The red fox was 3.14 feet long - * - * @function Highcharts.format - * - * @param {string} str - * The string to format. - * - * @param {Record<string, *>} ctx - * The context, a collection of key-value pairs where each key is - * replaced by its value. - * - * @param {Highcharts.Chart} [chart] - * A `Chart` instance used to get numberFormatter and time. - * - * @return {string} - * The formatted string. - */ - var format = H.format = function (str, - ctx, - chart) { - var splitter = '{', - isInside = false, - segment, - valueAndFormat, - ret = [], - val, - index; - var floatRegex = /f$/; - var decRegex = /\.([0-9])/; - var lang = H.defaultOptions.lang; - var time = chart && chart.time || H.time; - var numberFormatter = chart && chart.numberFormatter || numberFormat; - while (str) { - index = str.indexOf(splitter); - if (index === -1) { - break; - } - segment = str.slice(0, index); - if (isInside) { // we're on the closing bracket looking back - valueAndFormat = segment.split(':'); - val = getNestedProperty(valueAndFormat.shift() || '', ctx); - // Format the replacement - if (valueAndFormat.length && typeof val === 'number') { - segment = valueAndFormat.join(':'); - if (floatRegex.test(segment)) { // float - var decimals = parseInt((segment.match(decRegex) || ['', '-1'])[1], 10); - if (val !== null) { - val = numberFormatter(val, decimals, lang.decimalPoint, segment.indexOf(',') > -1 ? lang.thousandsSep : ''); - } - } - else { - val = time.dateFormat(segment, val); - } - } - // Push the result and advance the cursor - ret.push(val); - } - else { - ret.push(segment); - } - str = str.slice(index + 1); // the rest - isInside = !isInside; // toggle - splitter = isInside ? '}' : '{'; // now look for next matching bracket - } - ret.push(str); - return ret.join(''); + * @private + */ + SVGElement.prototype.htmlGetBBox = function () { + return { height: 0, width: 0, x: 0, y: 0 }; }; /** - * Get the magnitude of a number. + * Initialize the SVG element. This function only exists to make the + * initialization process overridable. It should not be called directly. + * + * @function Highcharts.SVGElement#init + * + * @param {Highcharts.SVGRenderer} renderer + * The SVGRenderer instance to initialize to. + * + * @param {string} nodeName + * The SVG node name. + */ + SVGElement.prototype.init = function (renderer, nodeName) { + /** + * The primary DOM node. Each `SVGElement` instance wraps a main DOM + * node, but may also represent more nodes. + * + * @name Highcharts.SVGElement#element + * @type {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} + */ + this.element = + nodeName === "span" + ? createElement(nodeName) + : doc.createElementNS(this.SVG_NS, nodeName); + /** + * The renderer that the SVGElement belongs to. + * + * @name Highcharts.SVGElement#renderer + * @type {Highcharts.SVGRenderer} + */ + this.renderer = renderer; + fireEvent(this, "afterInit"); + }; + /** + * Invert a group, rotate and flip. This is used internally on inverted + * charts, where the points and graphs are drawn as if not inverted, then + * the series group elements are inverted. * - * @function Highcharts.getMagnitude + * @function Highcharts.SVGElement#invert * - * @param {number} num - * The number. + * @param {boolean} inverted + * Whether to invert or not. An inverted shape can be un-inverted by + * setting it to false. * - * @return {number} - * The magnitude, where 1-9 are magnitude 1, 10-99 magnitude 2 etc. + * @return {Highcharts.SVGElement} + * Return the SVGElement for chaining. */ - var getMagnitude = H.getMagnitude = function (num) { - return Math.pow(10, - Math.floor(Math.log(num) / Math.LN10)); + SVGElement.prototype.invert = function (inverted) { + var wrapper = this; + wrapper.inverted = inverted; + wrapper.updateTransform(); + return wrapper; }; /** - * Take an interval and normalize it to multiples of round numbers. - * - * @deprecated - * @function Highcharts.normalizeTickInterval - * - * @param {number} interval - * The raw, un-rounded interval. - * - * @param {Array<*>} [multiples] - * Allowed multiples. - * - * @param {number} [magnitude] - * The magnitude of the number. + * Add an event listener. This is a simple setter that replaces all other + * events of the same type, opposed to the {@link Highcharts#addEvent} + * function. * - * @param {boolean} [allowDecimals] - * Whether to allow decimals. + * @sample highcharts/members/element-on/ + * A clickable rectangle + * + * @function Highcharts.SVGElement#on + * + * @param {string} eventType + * The event type. If the type is `click`, Highcharts will internally + * translate it to a `touchstart` event on touch devices, to prevent the + * browser from waiting for a click event from firing. + * + * @param {Function} handler + * The handler callback. + * + * @return {Highcharts.SVGElement} + * The SVGElement for chaining. + */ + SVGElement.prototype.on = function (eventType, handler) { + var svgElement = this, + element = svgElement.element, + touchStartPos, + touchEventFired; + // touch + if (hasTouch && eventType === "click") { + element.ontouchstart = function (e) { + // save touch position for later calculation + touchStartPos = { + clientX: e.touches[0].clientX, + clientY: e.touches[0].clientY, + }; + }; + // Instead of ontouchstart, event handlers should be called + // on touchend - similar to how current mouseup events are called + element.ontouchend = function (e) { + // hasMoved is a boolean variable containing logic if page + // was scrolled, so if touch position changed more than + // ~4px (value borrowed from general touch handler) + var hasMoved = touchStartPos.clientX + ? Math.sqrt( + Math.pow( + touchStartPos.clientX - e.changedTouches[0].clientX, + 2 + ) + + Math.pow( + touchStartPos.clientY - e.changedTouches[0].clientY, + 2 + ) + ) >= 4 + : false; + if (!hasMoved) { + // only call handlers if page was not scrolled + handler.call(element, e); + } + touchEventFired = true; + if (e.cancelable !== false) { + // prevent other events from being fired. #9682 + e.preventDefault(); + } + }; + element.onclick = function (e) { + // Do not call onclick handler if touch event was fired already. + if (!touchEventFired) { + handler.call(element, e); + } + }; + } else { + // simplest possible event model for internal use + element["on" + eventType] = handler; + } + return this; + }; + /** + * @private + * @function Highcharts.SVGElement#opacitySetter + * @param {string} value + * @param {string} key + * @param {Highcharts.SVGDOMElement} element + */ + SVGElement.prototype.opacitySetter = function (value, key, element) { + // Round off to avoid float errors, like tests where opacity lands on + // 9.86957e-06 instead of 0 + var opacity = Number(Number(value).toFixed(3)); + this.opacity = opacity; + element.setAttribute(key, opacity); + }; + /** + * Remove a class name from the element. * - * @param {boolean} [hasTickAmount] - * If it has tickAmount, avoid landing on tick intervals lower than - * original. + * @function Highcharts.SVGElement#removeClass * - * @return {number} - * The normalized interval. + * @param {string|RegExp} className + * The class name to remove. * - * @todo - * Move this function to the Axis prototype. It is here only for historical - * reasons. + * @return {Highcharts.SVGElement} Returns the SVG element for chainability. */ - var normalizeTickInterval = H.normalizeTickInterval = function (interval, - multiples, - magnitude, - allowDecimals, - hasTickAmount) { - var normalized, - i, - retInterval = interval; - // round to a tenfold of 1, 2, 2.5 or 5 - magnitude = pick(magnitude, 1); - normalized = interval / magnitude; - // multiples for a linear scale - if (!multiples) { - multiples = hasTickAmount ? - // Finer grained ticks when the tick amount is hard set, including - // when alignTicks is true on multiple axes (#4580). - [1, 1.2, 1.5, 2, 2.5, 3, 4, 5, 6, 8, 10] : - // Else, let ticks fall on rounder numbers - [1, 2, 2.5, 5, 10]; - // the allowDecimals option - if (allowDecimals === false) { - if (magnitude === 1) { - multiples = multiples.filter(function (num) { - return num % 1 === 0; - }); - } - else if (magnitude <= 0.1) { - multiples = [1 / magnitude]; - } - } - } - // normalize the interval to the nearest multiple - for (i = 0; i < multiples.length; i++) { - retInterval = multiples[i]; - // only allow tick amounts smaller than natural - if ((hasTickAmount && - retInterval * magnitude >= interval) || - (!hasTickAmount && - (normalized <= - (multiples[i] + - (multiples[i + 1] || multiples[i])) / 2))) { - break; - } + SVGElement.prototype.removeClass = function (className) { + return this.attr( + "class", + ("" + this.attr("class")) + .replace( + isString(className) + ? new RegExp("(^| )" + className + "( |$)") // #12064, #13590 + : className, + " " + ) + .replace(/ +/g, " ") + .trim() + ); + }; + /** + * @private + * @param {Array<Highcharts.SVGDOMElement>} tspans + * Text spans. + */ + SVGElement.prototype.removeTextOutline = function (tspans) { + // Iterate from the end to + // support removing items inside the cycle (#6472). + var i = tspans.length, + tspan; + while (i--) { + tspan = tspans[i]; + if (tspan.getAttribute("class") === "highcharts-text-outline") { + // Remove then erase + erase(tspans, this.element.removeChild(tspan)); } - // Multiply back to the correct magnitude. Correct floats to appropriate - // precision (#6085). - retInterval = correctFloat(retInterval * magnitude, -Math.round(Math.log(0.001) / Math.LN10)); - return retInterval; + } }; /** - * Sort an object array and keep the order of equal items. The ECMAScript - * standard does not specify the behaviour when items are equal. - * - * @function Highcharts.stableSort + * Removes an element from the DOM. * - * @param {Array<*>} arr - * The array to sort. - * - * @param {Function} sortFunction - * The function to sort it with, like with regular Array.prototype.sort. + * @private + * @function Highcharts.SVGElement#safeRemoveChild * - * @return {void} + * @param {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} element + * The DOM node to remove. */ - var stableSort = H.stableSort = function stableSort(arr, - sortFunction) { - // @todo It seems like Chrome since v70 sorts in a stable way internally, - // plus all other browsers do it, so over time we may be able to remove this - // function - var length = arr.length, - sortValue, - i; - // Add index to each item - for (i = 0; i < length; i++) { - arr[i].safeI = i; // stable sort index + SVGElement.prototype.safeRemoveChild = function (element) { + var parentNode = element.parentNode; + if (parentNode) { + parentNode.removeChild(element); + } + }; + /** + * Set the coordinates needed to draw a consistent radial gradient across + * a shape regardless of positioning inside the chart. Used on pie slices + * to make all the slices have the same radial reference point. + * + * @function Highcharts.SVGElement#setRadialReference + * + * @param {Array<number>} coordinates + * The center reference. The format is `[centerX, centerY, diameter]` in + * pixels. + * + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. + */ + SVGElement.prototype.setRadialReference = function (coordinates) { + var existingGradient = + this.element.gradient && + this.renderer.gradients[this.element.gradient]; + this.element.radialReference = coordinates; + // On redrawing objects with an existing gradient, the gradient needs + // to be repositioned (#3801) + if (existingGradient && existingGradient.radAttr) { + existingGradient.animate( + this.renderer.getRadialAttr(coordinates, existingGradient.radAttr) + ); + } + return this; + }; + /** + * @private + * @function Highcharts.SVGElement#setTextPath + * @param {Highcharts.SVGElement} path + * Path to follow. + * @param {Highcharts.DataLabelsTextPathOptionsObject} textPathOptions + * Options. + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. + */ + SVGElement.prototype.setTextPath = function (path, textPathOptions) { + var elem = this.element, + attribsMap = { + textAnchor: "text-anchor", + }, + attrs, + adder = false, + textPathElement, + textPathId, + textPathWrapper = this.textPathWrapper, + tspans, + firstTime = !textPathWrapper; + // Defaults + textPathOptions = merge( + true, + { + enabled: true, + attributes: { + dy: -5, + startOffset: "50%", + textAnchor: "middle", + }, + }, + textPathOptions + ); + attrs = textPathOptions.attributes; + if (path && textPathOptions && textPathOptions.enabled) { + // In case of fixed width for a text, string is rebuilt + // (e.g. ellipsis is applied), so we need to rebuild textPath too + if ( + textPathWrapper && + textPathWrapper.element.parentNode === null + ) { + // When buildText functionality was triggered again + // and deletes textPathWrapper parentNode + firstTime = true; + textPathWrapper = textPathWrapper.destroy(); + } else if (textPathWrapper) { + // Case after drillup when spans were added into + // the DOM outside the textPathWrapper parentGroup + this.removeTextOutline.call( + textPathWrapper.parentGroup, + [].slice.call(elem.getElementsByTagName("tspan")) + ); + } + // label() has padding, text() doesn't + if (this.options && this.options.padding) { + attrs.dx = -this.options.padding; + } + if (!textPathWrapper) { + // Create <textPath>, defer the DOM adder + this.textPathWrapper = textPathWrapper = + this.renderer.createElement("textPath"); + adder = true; + } + textPathElement = textPathWrapper.element; + // Set ID for the path + textPathId = path.element.getAttribute("id"); + if (!textPathId) { + path.element.setAttribute("id", (textPathId = uniqueKey())); + } + // Change DOM structure, by placing <textPath> tag in <text> + if (firstTime) { + tspans = elem.getElementsByTagName("tspan"); + // Now move all <tspan>'s to the <textPath> node + while (tspans.length) { + // Remove "y" from tspans, as Firefox translates them + tspans[0].setAttribute("y", 0); + // Remove "x" from tspans + if (isNumber(attrs.dx)) { + tspans[0].setAttribute("x", -attrs.dx); + } + textPathElement.appendChild(tspans[0]); + } + } + // Add <textPath> to the DOM + if (adder && textPathWrapper) { + textPathWrapper.add({ + // label() is placed in a group, text() is standalone + element: this.text ? this.text.element : elem, + }); + } + // Set basic options: + // Use `setAttributeNS` because Safari needs this.. + textPathElement.setAttributeNS( + "http://www.w3.org/1999/xlink", + "href", + this.renderer.url + "#" + textPathId + ); + // Presentation attributes: + // dx/dy options must by set on <text> (parent), + // the rest should be set on <textPath> + if (defined(attrs.dy)) { + textPathElement.parentNode.setAttribute("dy", attrs.dy); + delete attrs.dy; } - arr.sort(function (a, b) { - sortValue = sortFunction(a, b); - return sortValue === 0 ? a.safeI - b.safeI : sortValue; + if (defined(attrs.dx)) { + textPathElement.parentNode.setAttribute("dx", attrs.dx); + delete attrs.dx; + } + // Additional attributes + objectEach(attrs, function (val, key) { + textPathElement.setAttribute(attribsMap[key] || key, val); }); - // Remove index from items - for (i = 0; i < length; i++) { - delete arr[i].safeI; // stable sort index + // Remove translation, text that follows path does not need that + elem.removeAttribute("transform"); + // Remove shadows and text outlines + this.removeTextOutline.call( + textPathWrapper, + [].slice.call(elem.getElementsByTagName("tspan")) + ); + // Remove background and border for label(), see #10545 + // Alternatively, we can disable setting background rects in + // series.drawDataLabels() + if (this.text && !this.renderer.styledMode) { + this.attr({ + fill: "none", + "stroke-width": 0, + }); + } + // Disable some functions + this.updateTransform = noop; + this.applyTextOutline = noop; + } else if (textPathWrapper) { + // Reset to prototype + delete this.updateTransform; + delete this.applyTextOutline; + // Restore DOM structure: + this.destroyTextPath(elem, path); + // Bring attributes back + this.updateTransform(); + // Set textOutline back for text() + if (this.options && this.options.rotation) { + this.applyTextOutline(this.options.style.textOutline); } + } + return this; }; /** - * 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. + * Add a shadow to the element. Must be called after the element is added to + * the DOM. In styled mode, this method is not used, instead use `defs` and + * filters. * - * @function Highcharts.arrayMin + * @example + * renderer.rect(10, 100, 100, 100) + * .attr({ fill: 'red' }) + * .shadow(true); * - * @param {Array<*>} data - * An array of numbers. + * @function Highcharts.SVGElement#shadow * - * @return {number} - * The lowest number. - */ - var arrayMin = H.arrayMin = function arrayMin(data) { - var 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. + * @param {boolean|Highcharts.ShadowOptionsObject} [shadowOptions] + * The shadow options. If `true`, the default options are applied. If + * `false`, the current shadow will be removed. * - * @function Highcharts.arrayMax + * @param {Highcharts.SVGElement} [group] + * The SVG group element where the shadows will be applied. The + * default is to add it to the same parent as the current element. + * Internally, this is ised for pie slices, where all the shadows are + * added to an element behind all the slices. * - * @param {Array<*>} data - * An array of numbers. + * @param {boolean} [cutOff] + * Used internally for column shadows. * - * @return {number} - * The highest number. + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. */ - var arrayMax = H.arrayMax = function arrayMax(data) { - var i = data.length, - max = data[0]; - while (i--) { - if (data[i] > max) { - max = data[i]; - } + SVGElement.prototype.shadow = function (shadowOptions, group, cutOff) { + var shadows = [], + i, + shadow, + element = this.element, + strokeWidth, + shadowElementOpacity, + update = false, + oldShadowOptions = this.oldShadowOptions, + // compensate for inverted plot area + transform; + var defaultShadowOptions = { + color: "#000000", + offsetX: 1, + offsetY: 1, + opacity: 0.15, + width: 3, + }; + var options; + if (shadowOptions === true) { + options = defaultShadowOptions; + } else if (typeof shadowOptions === "object") { + options = extend(defaultShadowOptions, shadowOptions); + } + // Update shadow when options change (#12091). + if (options) { + // Go over each key to look for change + if (options && oldShadowOptions) { + objectEach(options, function (value, key) { + if (value !== oldShadowOptions[key]) { + update = true; + } + }); + } + if (update) { + this.destroyShadows(); + } + this.oldShadowOptions = options; + } + if (!options) { + this.destroyShadows(); + } else if (!this.shadows) { + shadowElementOpacity = options.opacity / options.width; + transform = this.parentInverted + ? "translate(-1,-1)" + : "translate(" + options.offsetX + ", " + options.offsetY + ")"; + for (i = 1; i <= options.width; i++) { + shadow = element.cloneNode(false); + strokeWidth = options.width * 2 + 1 - 2 * i; + attr(shadow, { + stroke: shadowOptions.color || "#000000", + "stroke-opacity": shadowElementOpacity * i, + "stroke-width": strokeWidth, + transform: transform, + fill: "none", + }); + shadow.setAttribute( + "class", + (shadow.getAttribute("class") || "") + " highcharts-shadow" + ); + if (cutOff) { + attr( + shadow, + "height", + Math.max(attr(shadow, "height") - strokeWidth, 0) + ); + shadow.cutHeight = strokeWidth; + } + if (group) { + group.element.appendChild(shadow); + } else if (element.parentNode) { + element.parentNode.insertBefore(shadow, element); + } + shadows.push(shadow); } - return max; + this.shadows = shadows; + } + return this; }; /** - * 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. + * Show the element after it has been hidden. * - * @function Highcharts.destroyObjectProperties + * @function Highcharts.SVGElement#show * - * @param {*} obj - * The object to destroy properties on. + * @param {boolean} [inherit=false] + * Set the visibility attribute to `inherit` rather than `visible`. + * The difference is that an element with `visibility="visible"` + * will be visible even if the parent is hidden. * - * @param {*} [except] - * Exception, do not destroy this property, only delete it. + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. */ - var destroyObjectProperties = H.destroyObjectProperties = - function destroyObjectProperties(obj, - except) { - objectEach(obj, - function (val, - n) { - // If the object is non-null and destroy is defined - if (val && val !== except && val.destroy) { - // Invoke the destroy - val.destroy(); - } - // Delete the property from the object. - delete obj[n]; - }); - }; + SVGElement.prototype.show = function (inherit) { + return this.attr({ visibility: inherit ? "inherit" : "visible" }); + }; /** - * Discard a HTML element by moving it to the bin and delete. + * WebKit and Batik have problems with a stroke-width of zero, so in this + * case we remove the stroke attribute altogether. #1270, #1369, #3065, + * #3072. * - * @function Highcharts.discardElement - * - * @param {Highcharts.HTMLDOMElement} element - * The HTML node to discard. - */ - var discardElement = H.discardElement = function discardElement(element) { - var garbageBin = H.garbageBin; - // create a garbage bin element, not part of the DOM - if (!garbageBin) { - garbageBin = createElement('div'); - } - // move the node and empty bin - if (element) { - garbageBin.appendChild(element); - } - garbageBin.innerHTML = ''; + * @private + * @function Highcharts.SVGElement#strokeSetter + * @param {number|string} value + * @param {string} key + * @param {Highcharts.SVGDOMElement} element + */ + SVGElement.prototype.strokeSetter = function (value, key, element) { + this[key] = value; + // Only apply the stroke attribute if the stroke width is defined and + // larger than 0 + if (this.stroke && this["stroke-width"]) { + // Use prototype as instance may be overridden + SVGElement.prototype.fillSetter.call( + this, + this.stroke, + "stroke", + element + ); + element.setAttribute("stroke-width", this["stroke-width"]); + this.hasStroke = true; + } else if (key === "stroke-width" && value === 0 && this.hasStroke) { + element.removeAttribute("stroke"); + this.hasStroke = false; + } else if (this.renderer.styledMode && this["stroke-width"]) { + element.setAttribute("stroke-width", this["stroke-width"]); + this.hasStroke = true; + } }; /** - * Fix JS round off float errors. + * Get the computed stroke width in pixel values. This is used extensively + * when drawing shapes to ensure the shapes are rendered crisp and + * positioned correctly relative to each other. Using + * `shape-rendering: crispEdges` leaves us less control over positioning, + * for example when we want to stack columns next to each other, or position + * things pixel-perfectly within the plot box. * - * @function Highcharts.correctFloat + * The common pattern when placing a shape is: + * - Create the SVGElement and add it to the DOM. In styled mode, it will + * now receive a stroke width from the style sheet. In classic mode we + * will add the `stroke-width` attribute. + * - Read the computed `elem.strokeWidth()`. + * - Place it based on the stroke width. * - * @param {number} num - * A float number to fix. - * - * @param {number} [prec=14] - * The precision. + * @function Highcharts.SVGElement#strokeWidth * * @return {number} - * The corrected float number. - */ - var correctFloat = H.correctFloat = function correctFloat(num, - prec) { - return parseFloat(num.toPrecision(prec || 14)); + * The stroke width in pixels. Even if the given stroke widtch (in CSS or by + * attributes) is based on `em` or other units, the pixel size is returned. + */ + SVGElement.prototype.strokeWidth = function () { + // In non-styled mode, read the stroke width as set by .attr + if (!this.renderer.styledMode) { + return this["stroke-width"] || 0; + } + // In styled mode, read computed stroke width + var val = this.getStyle("stroke-width"), + ret = 0, + dummy; + // Read pixel values directly + if (val.indexOf("px") === val.length - 2) { + ret = pInt(val); + // Other values like em, pt etc need to be measured + } else if (val !== "") { + dummy = doc.createElementNS(SVG_NS, "rect"); + attr(dummy, { + width: val, + "stroke-width": 0, + }); + this.element.parentNode.appendChild(dummy); + ret = dummy.getBBox().width; + dummy.parentNode.removeChild(dummy); + } + return ret; }; /** - * The time unit lookup + * If one of the symbol size affecting parameters are changed, + * check all the others only once for each call to an element's + * .attr() method * - * @ignore - */ - var timeUnits = H.timeUnits = { - millisecond: 1, - second: 1000, - minute: 60000, - hour: 3600000, - day: 24 * 3600000, - week: 7 * 24 * 3600000, - month: 28 * 24 * 3600000, - year: 364 * 24 * 3600000 - }; + * @private + * @function Highcharts.SVGElement#symbolAttr + * + * @param {Highcharts.SVGAttributes} hash + * The attributes to set. + */ + SVGElement.prototype.symbolAttr = function (hash) { + var wrapper = this; + [ + "x", + "y", + "r", + "start", + "end", + "width", + "height", + "innerR", + "anchorX", + "anchorY", + "clockwise", + ].forEach(function (key) { + wrapper[key] = pick(hash[key], wrapper[key]); + }); + wrapper.attr({ + d: wrapper.renderer.symbols[wrapper.symbolName]( + wrapper.x, + wrapper.y, + wrapper.width, + wrapper.height, + wrapper + ), + }); + }; /** - * Format a number and return a string based on input settings. - * - * @sample highcharts/members/highcharts-numberformat/ - * Custom number format - * - * @function Highcharts.numberFormat - * - * @param {number} number - * The input number to format. - * - * @param {number} decimals - * The amount of decimals. A value of -1 preserves the amount in the - * input number. - * - * @param {string} [decimalPoint] - * The decimal point, defaults to the one given in the lang options, or - * a dot. - * - * @param {string} [thousandsSep] - * The thousands separator, defaults to the one given in the lang - * options, or a space character. - * - * @return {string} - * The formatted number. - */ - var numberFormat = H.numberFormat = function numberFormat(number, - decimals, - decimalPoint, - thousandsSep) { - number = +number || 0; - decimals = +decimals; - var lang = H.defaultOptions.lang, origDec = (number.toString().split('.')[1] || '').split('e')[0].length, strinteger, thousands, ret, roundedNumber, exponent = number.toString().split('e'), fractionDigits; - if (decimals === -1) { - // Preserve decimals. Not huge numbers (#3793). - decimals = Math.min(origDec, 20); - } - else if (!isNumber(decimals)) { - decimals = 2; - } - else if (decimals && exponent[1] && exponent[1] < 0) { - // Expose decimals from exponential notation (#7042) - fractionDigits = decimals + +exponent[1]; - if (fractionDigits >= 0) { - // remove too small part of the number while keeping the notation - exponent[0] = (+exponent[0]).toExponential(fractionDigits) - .split('e')[0]; - decimals = fractionDigits; - } - else { - // fractionDigits < 0 - exponent[0] = exponent[0].split('.')[0] || 0; - if (decimals < 20) { - // use number instead of exponential notation (#7405) - number = (exponent[0] * Math.pow(10, exponent[1])) - .toFixed(decimals); - } - else { - // or zero - number = 0; - } - exponent[1] = 0; - } - } - // Add another decimal to avoid rounding errors of float numbers. (#4573) - // Then use toFixed to handle rounding. - roundedNumber = (Math.abs(exponent[1] ? exponent[0] : number) + - Math.pow(10, -Math.max(decimals, origDec) - 1)).toFixed(decimals); - // A string containing the positive integer component of the number - strinteger = String(pInt(roundedNumber)); - // Leftover after grouping into thousands. Can be 0, 1 or 2. - thousands = strinteger.length > 3 ? strinteger.length % 3 : 0; - // Language - decimalPoint = pick(decimalPoint, lang.decimalPoint); - thousandsSep = pick(thousandsSep, lang.thousandsSep); - // Start building the return - ret = number < 0 ? '-' : ''; - // Add the leftover after grouping into thousands. For example, in the - // number 42 000 000, this line adds 42. - ret += thousands ? strinteger.substr(0, thousands) + thousandsSep : ''; - // Add the remaining thousands groups, joined by the thousands separator - ret += strinteger - .substr(thousands) - .replace(/(\d{3})(?=\d)/g, '$1' + thousandsSep); - // Add the decimal point and the decimal component - if (decimals) { - // Get the decimal component - ret += decimalPoint + roundedNumber.slice(-decimals); - } - if (exponent[1] && +ret !== 0) { - ret += 'e' + exponent[1]; + * @private + * @function Highcharts.SVGElement#textSetter + * @param {string} value + */ + SVGElement.prototype.textSetter = function (value) { + if (value !== this.textStr) { + // Delete size caches when the text changes + // delete this.bBox; // old code in series-label + delete this.textPxLength; + this.textStr = value; + if (this.added) { + this.renderer.buildText(this); } - return ret; + } }; /** - * Easing definition - * * @private - * @function Math.easeInOutSine + * @function Highcharts.SVGElement#titleSetter + * @param {string} value + */ + SVGElement.prototype.titleSetter = function (value) { + var titleNode = this.element.getElementsByTagName("title")[0]; + if (!titleNode) { + titleNode = doc.createElementNS(this.SVG_NS, "title"); + this.element.appendChild(titleNode); + } + // Remove text content if it exists + if (titleNode.firstChild) { + titleNode.removeChild(titleNode.firstChild); + } + titleNode.appendChild( + doc.createTextNode( + // #3276, #3895 + String(pick(value, "")) + .replace(/<[^>]*>/g, "") + .replace(/</g, "<") + .replace(/>/g, ">") + ) + ); + }; + /** + * Bring the element to the front. Alternatively, a new zIndex can be set. * - * @param {number} pos - * Current position, ranging from 0 to 1. + * @sample highcharts/members/element-tofront/ + * Click an element to bring it to front * - * @return {number} - * Ease result + * @function Highcharts.SVGElement#toFront + * + * @return {Highcharts.SVGElement} + * Returns the SVGElement for chaining. */ - Math.easeInOutSine = function (pos) { - return -0.5 * (Math.cos(Math.PI * pos) - 1); + SVGElement.prototype.toFront = function () { + var element = this.element; + element.parentNode.appendChild(element); + return this; }; /** - * Returns the value of a property path on a given object. + * Move an object and its children by x and y values. * - * @private - * @function getNestedProperty + * @function Highcharts.SVGElement#translate * - * @param {string} path - * Path to the property, for example `custom.myValue`. + * @param {number} x + * The x value. * - * @param {unknown} obj - * Instance containing the property on the specific path. + * @param {number} y + * The y value. * - * @return {unknown} - * The unknown property value. + * @return {Highcharts.SVGElement} */ - function getNestedProperty(path, obj) { - if (!path) { - return obj; - } - var pathElements = path.split('.').reverse(); - var subProperty = obj; - if (pathElements.length === 1) { - return subProperty[path]; - } - var pathElement = pathElements.pop(); - while (typeof pathElement !== 'undefined' && - typeof subProperty !== 'undefined' && - subProperty !== null) { - subProperty = subProperty[pathElement]; - pathElement = pathElements.pop(); - } - return subProperty; - } + SVGElement.prototype.translate = function (x, y) { + return this.attr({ + translateX: x, + translateY: y, + }); + }; /** - * 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. + * Update the shadow elements with new attributes. * - * @function Highcharts.getStyle - * - * @param {Highcharts.HTMLDOMElement} el - * An HTML element. + * @private + * @function Highcharts.SVGElement#updateShadows * - * @param {string} prop - * The property name. + * @param {string} key + * The attribute name. * - * @param {boolean} [toInt=true] - * Parse to integer. + * @param {number} value + * The value of the attribute. * - * @return {number|string} - * The numeric value. + * @param {Function} setter + * The setter function, inherited from the parent wrapper. */ - var getStyle = H.getStyle = function (el, - prop, - toInt) { - var style; - // For width and height, return the actual inner pixel size (#4913) - if (prop === 'width') { - var 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) ... - var boundingClientRectWidth = el.getBoundingClientRect && - 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 - - H.getStyle(el, 'padding-left') - - H.getStyle(el, 'padding-right'))); - } - if (prop === 'height') { - return Math.max(0, // #8377 - Math.min(el.offsetHeight, el.scrollHeight) - - H.getStyle(el, 'padding-top') - - H.getStyle(el, 'padding-bottom')); - } - if (!win.getComputedStyle) { - // SVG not supported, forgot to load oldie.js? - error(27, true); - } - // Otherwise, get the computed style - style = win.getComputedStyle(el, undefined); // eslint-disable-line no-undefined - if (style) { - style = style.getPropertyValue(prop); - if (pick(toInt, prop !== 'opacity')) { - style = pInt(style); - } + SVGElement.prototype.updateShadows = function (key, value, setter) { + var shadows = this.shadows; + if (shadows) { + var i = shadows.length; + while (i--) { + setter.call( + shadows[i], + key === "height" + ? Math.max(value - (shadows[i].cutHeight || 0), 0) + : key === "d" + ? this.d + : value, + key, + shadows[i] + ); } - return style; + } }; /** - * Search for an item in an array. + * Update the transform attribute based on internal properties. Deals with + * the custom `translateX`, `translateY`, `rotation`, `scaleX` and `scaleY` + * attributes and updates the SVG `transform` attribute. * - * @function Highcharts.inArray - * - * @deprecated + * @private + * @function Highcharts.SVGElement#updateTransform + */ + SVGElement.prototype.updateTransform = function () { + var wrapper = this, + translateX = wrapper.translateX || 0, + translateY = wrapper.translateY || 0, + scaleX = wrapper.scaleX, + scaleY = wrapper.scaleY, + inverted = wrapper.inverted, + rotation = wrapper.rotation, + matrix = wrapper.matrix, + element = wrapper.element, + transform; + // Flipping affects translate as adjustment for flipping around the + // group's axis + if (inverted) { + translateX += wrapper.width; + translateY += wrapper.height; + } + // Apply translate. Nearly all transformed elements have translation, + // so instead of checking for translate = 0, do it always (#1767, + // #1846). + transform = ["translate(" + translateX + "," + translateY + ")"]; + // apply matrix + if (defined(matrix)) { + transform.push("matrix(" + matrix.join(",") + ")"); + } + // apply rotation + if (inverted) { + transform.push("rotate(90) scale(-1,1)"); + } else if (rotation) { + // text rotation + transform.push( + "rotate(" + + rotation + + " " + + pick(this.rotationOriginX, element.getAttribute("x"), 0) + + " " + + pick(this.rotationOriginY, element.getAttribute("y") || 0) + + ")" + ); + } + // apply scale + if (defined(scaleX) || defined(scaleY)) { + transform.push( + "scale(" + pick(scaleX, 1) + " " + pick(scaleY, 1) + ")" + ); + } + if (transform.length) { + element.setAttribute("transform", transform.join(" ")); + } + }; + /** + * @private + * @function Highcharts.SVGElement#visibilitySetter * - * @param {*} item - * The item to search for. + * @param {string} value * - * @param {Array<*>} arr - * The array or node collection to search in. + * @param {string} key * - * @param {number} [fromIndex=0] - * The index to start searching from. + * @param {Highcharts.SVGDOMElement} element * - * @return {number} - * The index within the array, or -1 if not found. + * @return {void} */ - var inArray = H.inArray = function (item, - arr, - fromIndex) { - error(32, - false, - void 0, { 'Highcharts.inArray': 'use Array.indexOf' }); - return arr.indexOf(item, fromIndex); + SVGElement.prototype.visibilitySetter = function (value, key, element) { + // IE9-11 doesn't handle visibilty:inherit well, so we remove the + // attribute instead (#2881, #3909) + if (value === "inherit") { + element.removeAttribute(key); + } else if (this[key] !== value) { + // #6747 + element.setAttribute(key, value); + } + this[key] = value; }; - /* eslint-disable valid-jsdoc */ - /** - * 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. - */ - var find = H.find = Array.prototype.find ? - /* eslint-enable valid-jsdoc */ - function (arr, - callback) { - return arr.find(callback); - } : - // Legacy implementation. PhantomJS, IE <= 11 etc. #7223. - function (arr, callback) { - var i, - length = arr.length; - for (i = 0; i < length; i++) { - if (callback(arr[i], i)) { // eslint-disable-line callback-return - return arr[i]; - } - } - }; /** - * Returns an array of a given object's own properties. - * - * @function Highcharts.keys - * @deprecated + * @private + * @function Highcharts.SVGElement#xGetter * - * @param {*} obj - * The object of which the properties are to be returned. + * @param {string} key * - * @return {Array<string>} - * An array of strings that represents all the properties. + * @return {number|string|null} */ - H.keys = function (obj) { - error(32, false, void 0, { 'Highcharts.keys': 'use Object.keys' }); - return Object.keys(obj); + SVGElement.prototype.xGetter = function (key) { + if (this.element.nodeName === "circle") { + if (key === "x") { + key = "cx"; + } else if (key === "y") { + key = "cy"; + } + } + return this._defaultGetter(key); }; /** - * Get the element's offset position, corrected for `overflow: auto`. - * - * @function Highcharts.offset - * - * @param {global.Element} el - * The DOM element. - * - * @return {Highcharts.OffsetObject} - * An object containing `left` and `top` properties for the position in - * the page. - */ - var offset = H.offset = function offset(el) { - var docElem = doc.documentElement, - box = (el.parentElement || el.parentNode) ? - el.getBoundingClientRect() : - { top: 0, - left: 0 }; - return { - top: box.top + (win.pageYOffset || docElem.scrollTop) - - (docElem.clientTop || 0), - left: box.left + (win.pageXOffset || docElem.scrollLeft) - - (docElem.clientLeft || 0) - }; + * @private + * @function Highcharts.SVGElement#zIndexSetter + * @param {number} [value] + * @param {string} [key] + * @return {boolean} + */ + SVGElement.prototype.zIndexSetter = function (value, key) { + var renderer = this.renderer, + parentGroup = this.parentGroup, + parentWrapper = parentGroup || renderer, + parentNode = parentWrapper.element || renderer.box, + childNodes, + otherElement, + otherZIndex, + element = this.element, + inserted = false, + undefinedOtherZIndex, + svgParent = parentNode === renderer.box, + run = this.added, + i; + if (defined(value)) { + // So we can read it for other elements in the group + element.setAttribute("data-z-index", value); + value = +value; + if (this[key] === value) { + // Only update when needed (#3865) + run = false; + } + } else if (defined(this[key])) { + element.removeAttribute("data-z-index"); + } + this[key] = value; + // Insert according to this and other elements' zIndex. Before .add() is + // called, nothing is done. Then on add, or by later calls to + // zIndexSetter, the node is placed on the right place in the DOM. + if (run) { + value = this.zIndex; + if (value && parentGroup) { + parentGroup.handleZ = true; + } + childNodes = parentNode.childNodes; + for (i = childNodes.length - 1; i >= 0 && !inserted; i--) { + otherElement = childNodes[i]; + otherZIndex = otherElement.getAttribute("data-z-index"); + undefinedOtherZIndex = !defined(otherZIndex); + if (otherElement !== element) { + if ( + // Negative zIndex versus no zIndex: + // On all levels except the highest. If the parent is + // <svg>, then we don't want to put items before <desc> + // or <defs> + value < 0 && + undefinedOtherZIndex && + !svgParent && + !i + ) { + parentNode.insertBefore(element, childNodes[i]); + inserted = true; + } else if ( + // Insert after the first element with a lower zIndex + pInt(otherZIndex) <= value || + // If negative zIndex, add this before first undefined + // zIndex element + (undefinedOtherZIndex && (!defined(value) || value >= 0)) + ) { + parentNode.insertBefore( + element, + childNodes[i + 1] || null // null for oldIE export + ); + inserted = true; + } + } + } + if (!inserted) { + parentNode.insertBefore( + element, + childNodes[svgParent ? 3 : 0] || null // null for oldIE + ); + inserted = true; + } + } + return inserted; }; - /* eslint-disable valid-jsdoc */ - /** - * Iterate over object key pairs in an object. - * - * @function Highcharts.objectEach<T> + return SVGElement; + })(); + // Some shared setters and getters + SVGElement.prototype["stroke-widthSetter"] = + SVGElement.prototype.strokeSetter; + SVGElement.prototype.yGetter = SVGElement.prototype.xGetter; + SVGElement.prototype.matrixSetter = + SVGElement.prototype.rotationOriginXSetter = + SVGElement.prototype.rotationOriginYSetter = + SVGElement.prototype.rotationSetter = + SVGElement.prototype.scaleXSetter = + SVGElement.prototype.scaleYSetter = + SVGElement.prototype.translateXSetter = + SVGElement.prototype.translateYSetter = + SVGElement.prototype.verticalAlignSetter = + function (value, key) { + this[key] = value; + this.doTransform = true; + }; + H.SVGElement = SVGElement; + + return H.SVGElement; + } + ); + _registerModule( + _modules, + "Core/Renderer/SVG/SVGLabel.js", + [ + _modules["Core/Renderer/SVG/SVGElement.js"], + _modules["Core/Utilities.js"], + ], + function (SVGElement, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var __extends = + (this && this.__extends) || + (function () { + var extendStatics = function (d, b) { + extendStatics = + Object.setPrototypeOf || + ({ __proto__: [] } instanceof Array && + function (d, b) { + d.__proto__ = b; + }) || + function (d, b) { + for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; + }; + return extendStatics(d, b); + }; + return function (d, b) { + extendStatics(d, b); + function __() { + this.constructor = d; + } + d.prototype = + b === null + ? Object.create(b) + : ((__.prototype = b.prototype), new __()); + }; + })(); + var defined = U.defined, + extend = U.extend, + isNumber = U.isNumber, + merge = U.merge, + removeEvent = U.removeEvent; + /** + * SVG label to render text. + * @private + * @class + * @name Highcharts.SVGLabel + * @augments Highcharts.SVGElement + */ + var SVGLabel = /** @class */ (function (_super) { + __extends(SVGLabel, _super); + /* * * - * @param {*} obj - * The object to iterate over. + * Constructors * - * @param {Highcharts.ObjectEachCallbackFunction<T>} fn - * The iterator callback. It passes three arguments: - * * value - The property value. - * * key - The property key. - * * obj - The object that objectEach is being applied to. + * */ + function SVGLabel( + renderer, + str, + x, + y, + shape, + anchorX, + anchorY, + useHTML, + baseline, + className + ) { + var _this = _super.call(this) || this; + _this.init(renderer, "g"); + _this.textStr = str; + _this.x = x; + _this.y = y; + _this.anchorX = anchorX; + _this.anchorY = anchorY; + _this.baseline = baseline; + _this.className = className; + if (className !== "button") { + _this.addClass("highcharts-label"); + } + if (className) { + _this.addClass("highcharts-" + className); + } + _this.text = renderer.text("", 0, 0, useHTML).attr({ + zIndex: 1, + }); + // Validate the shape argument + var hasBGImage; + if (typeof shape === "string") { + hasBGImage = /^url\((.*?)\)$/.test(shape); + if (_this.renderer.symbols[shape] || hasBGImage) { + _this.symbolKey = shape; + } + } + _this.bBox = SVGLabel.emptyBBox; + _this.padding = 3; + _this.paddingLeft = 0; + _this.baselineOffset = 0; + _this.needsBox = renderer.styledMode || hasBGImage; + _this.deferredAttr = {}; + _this.alignFactor = 0; + return _this; + } + /* * * - * @param {T} [ctx] - * The context. + * Functions * - * @return {void} - */ - var objectEach = H.objectEach = function objectEach(obj, - fn, - ctx) { - /* eslint-enable valid-jsdoc */ - for (var key in obj) { - if (Object.hasOwnProperty.call(obj, - key)) { - fn.call(ctx || obj[key], - obj[key], + * */ + SVGLabel.prototype.alignSetter = function (value) { + var alignFactor = { + left: 0, + center: 0.5, + right: 1, + }[value]; + if (alignFactor !== this.alignFactor) { + this.alignFactor = alignFactor; + // Bounding box exists, means we're dynamically changing + if (this.bBox && isNumber(this.xSetting)) { + this.attr({ x: this.xSetting }); // #5134 + } + } + }; + SVGLabel.prototype.anchorXSetter = function (value, key) { + this.anchorX = value; + this.boxAttr( key, - obj); - } + Math.round(value) - this.getCrispAdjust() - this.xSetting + ); + }; + SVGLabel.prototype.anchorYSetter = function (value, key) { + this.anchorY = value; + this.boxAttr(key, value - this.ySetting); + }; + /* + * Set a box attribute, or defer it if the box is not yet created + */ + SVGLabel.prototype.boxAttr = function (key, value) { + if (this.box) { + this.box.attr(key, value); + } else { + this.deferredAttr[key] = value; + } + }; + /* + * Pick up some properties and apply them to the text instead of the + * wrapper. + */ + SVGLabel.prototype.css = function (styles) { + if (styles) { + var textStyles = {}, + isWidth, + isFontStyle; + // Create a copy to avoid altering the original object + // (#537) + styles = merge(styles); + SVGLabel.textProps.forEach(function (prop) { + if (typeof styles[prop] !== "undefined") { + textStyles[prop] = styles[prop]; + delete styles[prop]; + } + }); + this.text.css(textStyles); + isWidth = "width" in textStyles; + isFontStyle = + "fontSize" in textStyles || "fontWeight" in textStyles; + // Update existing text, box (#9400, #12163) + if (isWidth || isFontStyle) { + this.updateBoxSize(); + // Keep updated (#9400, #12163) + if (isFontStyle) { + this.updateTextPadding(); + } } + } + return SVGElement.prototype.css.call(this, styles); }; - /** - * Iterate over an array. - * - * @deprecated - * @function Highcharts.each + /* + * Destroy and release memory. + */ + SVGLabel.prototype.destroy = function () { + // Added by button implementation + removeEvent(this.element, "mouseenter"); + removeEvent(this.element, "mouseleave"); + if (this.text) { + this.text.destroy(); + } + if (this.box) { + this.box = this.box.destroy(); + } + // Call base implementation to destroy the rest + SVGElement.prototype.destroy.call(this); + return void 0; + }; + SVGLabel.prototype.fillSetter = function (value, key) { + if (value) { + this.needsBox = true; + } + // for animation getter (#6776) + this.fill = value; + this.boxAttr(key, value); + }; + /* + * Return the bounding box of the box, not the group. + */ + SVGLabel.prototype.getBBox = function () { + var bBox = this.bBox; + var padding = this.padding; + return { + width: bBox.width + 2 * padding, + height: bBox.height + 2 * padding, + x: bBox.x - padding, + y: bBox.y - padding, + }; + }; + SVGLabel.prototype.getCrispAdjust = function () { + return this.renderer.styledMode && this.box + ? (this.box.strokeWidth() % 2) / 2 + : ((this["stroke-width"] ? parseInt(this["stroke-width"], 10) : 0) % + 2) / + 2; + }; + SVGLabel.prototype.heightSetter = function (value) { + this.heightSetting = value; + }; + // Event handling. In case of useHTML, we need to make sure that events + // are captured on the span as well, and that mouseenter/mouseleave + // between the SVG group and the HTML span are not treated as real + // enter/leave events. #13310. + SVGLabel.prototype.on = function (eventType, handler) { + var label = this; + var text = label.text; + var span = text && text.element.tagName === "SPAN" ? text : void 0; + var selectiveHandler; + if (span) { + selectiveHandler = function (e) { + if ( + (eventType === "mouseenter" || eventType === "mouseleave") && + e.relatedTarget instanceof Element && + (label.element.contains(e.relatedTarget) || + span.element.contains(e.relatedTarget)) + ) { + return; + } + handler.call(label.element, e); + }; + span.on(eventType, selectiveHandler); + } + SVGElement.prototype.on.call( + label, + eventType, + selectiveHandler || handler + ); + return label; + }; + /* + * After the text element is added, get the desired size of the border + * box and add it before the text in the DOM. + */ + SVGLabel.prototype.onAdd = function () { + var str = this.textStr; + this.text.add(this); + this.attr({ + // Alignment is available now (#3295, 0 not rendered if given + // as a value) + text: defined(str) ? str : "", + x: this.x, + y: this.y, + }); + if (this.box && defined(this.anchorX)) { + this.attr({ + anchorX: this.anchorX, + anchorY: this.anchorY, + }); + } + }; + SVGLabel.prototype.paddingSetter = function (value) { + if (defined(value) && value !== this.padding) { + this.padding = value; + this.updateTextPadding(); + } + }; + SVGLabel.prototype.paddingLeftSetter = function (value) { + if (defined(value) && value !== this.paddingLeft) { + this.paddingLeft = value; + this.updateTextPadding(); + } + }; + SVGLabel.prototype.rSetter = function (value, key) { + this.boxAttr(key, value); + }; + SVGLabel.prototype.shadow = function (b) { + if (b && !this.renderer.styledMode) { + this.updateBoxSize(); + if (this.box) { + this.box.shadow(b); + } + } + return this; + }; + SVGLabel.prototype.strokeSetter = function (value, key) { + // for animation getter (#6776) + this.stroke = value; + this.boxAttr(key, value); + }; + SVGLabel.prototype["stroke-widthSetter"] = function (value, key) { + if (value) { + this.needsBox = true; + } + this["stroke-width"] = value; + this.boxAttr(key, value); + }; + SVGLabel.prototype["text-alignSetter"] = function (value) { + this.textAlign = value; + }; + SVGLabel.prototype.textSetter = function (text) { + if (typeof text !== "undefined") { + // Must use .attr to ensure transforms are done (#10009) + this.text.attr({ text: text }); + } + this.updateBoxSize(); + this.updateTextPadding(); + }; + /* + * This function runs after the label is added to the DOM (when the bounding + * box is available), and after the text of the label is updated to detect + * the new bounding box and reflect it in the border box. + */ + SVGLabel.prototype.updateBoxSize = function () { + var style = this.text.element.style, + crispAdjust, + attribs = {}; + var padding = this.padding; + var paddingLeft = this.paddingLeft; + // #12165 error when width is null (auto) + // #12163 when fontweight: bold, recalculate bBox withot cache + // #3295 && 3514 box failure when string equals 0 + var bBox = + (!isNumber(this.widthSetting) || + !isNumber(this.heightSetting) || + this.textAlign) && + defined(this.text.textStr) + ? this.text.getBBox() + : SVGLabel.emptyBBox; + this.width = + (this.widthSetting || bBox.width || 0) + 2 * padding + paddingLeft; + this.height = (this.heightSetting || bBox.height || 0) + 2 * padding; + // Update the label-scoped y offset. Math.min because of inline + // style (#9400) + this.baselineOffset = + padding + + Math.min( + this.renderer.fontMetrics(style && style.fontSize, this.text).b, + // When the height is 0, there is no bBox, so go with the font + // metrics. Highmaps CSS demos. + bBox.height || Infinity + ); + if (this.needsBox) { + // Create the border box if it is not already present + if (!this.box) { + // Symbol definition exists (#5324) + var box = (this.box = this.symbolKey + ? this.renderer.symbol(this.symbolKey) + : this.renderer.rect()); + box.addClass( + // Don't use label className for buttons + (this.className === "button" ? "" : "highcharts-label-box") + + (this.className + ? " highcharts-" + this.className + "-box" + : "") + ); + box.add(this); + crispAdjust = this.getCrispAdjust(); + attribs.x = crispAdjust; + attribs.y = + (this.baseline ? -this.baselineOffset : 0) + crispAdjust; + } + // Apply the box attributes + attribs.width = Math.round(this.width); + attribs.height = Math.round(this.height); + this.box.attr(extend(attribs, this.deferredAttr)); + this.deferredAttr = {}; + } + this.bBox = bBox; + }; + /* + * This function runs after setting text or padding, but only if padding + * is changed. + */ + SVGLabel.prototype.updateTextPadding = function () { + var text = this.text; + // Determine y based on the baseline + var textY = this.baseline ? 0 : this.baselineOffset; + var textX = this.paddingLeft + this.padding; + // compensate for alignment + if ( + defined(this.widthSetting) && + this.bBox && + (this.textAlign === "center" || this.textAlign === "right") + ) { + textX += + { center: 0.5, right: 1 }[this.textAlign] * + (this.widthSetting - this.bBox.width); + } + // update if anything changed + if (textX !== text.x || textY !== text.y) { + text.attr("x", textX); + // #8159 - prevent misplaced data labels in treemap + // (useHTML: true) + if (text.hasBoxWidthChanged) { + this.bBox = text.getBBox(true); + this.updateBoxSize(); + } + if (typeof textY !== "undefined") { + text.attr("y", textY); + } + } + // record current values + text.x = textX; + text.y = textY; + }; + SVGLabel.prototype.widthSetter = function (value) { + // width:auto => null + this.widthSetting = isNumber(value) ? value : void 0; + }; + SVGLabel.prototype.xSetter = function (value) { + this.x = value; // for animation getter + if (this.alignFactor) { + value -= + this.alignFactor * + ((this.widthSetting || this.bBox.width) + 2 * this.padding); + // Force animation even when setting to the same value (#7898) + this["forceAnimate:x"] = true; + } + this.xSetting = Math.round(value); + this.attr("translateX", this.xSetting); + }; + SVGLabel.prototype.ySetter = function (value) { + this.ySetting = this.y = Math.round(value); + this.attr("translateY", this.ySetting); + }; + /* * * - * @param {Array<*>} arr - * The array to iterate over. + * Static Properties * - * @param {Function} fn - * The iterator callback. It passes three arguments: - * - `item`: The array item. - * - `index`: The item's index in the array. - * - `arr`: The array that each is being applied to. + * */ + SVGLabel.emptyBBox = { width: 0, height: 0, x: 0, y: 0 }; + /* * * - * @param {*} [ctx] - * The context. + * Properties * - * @return {void} - */ + * */ /** - * Filter an array by a callback. + * For labels, these CSS properties are applied to the `text` node directly. * - * @deprecated - * @function Highcharts.grep + * @private + * @name Highcharts.SVGLabel#textProps + * @type {Array<string>} + */ + SVGLabel.textProps = [ + "color", + "cursor", + "direction", + "fontFamily", + "fontSize", + "fontStyle", + "fontWeight", + "lineHeight", + "textAlign", + "textDecoration", + "textOutline", + "textOverflow", + "width", + ]; + return SVGLabel; + })(SVGElement); + + return SVGLabel; + } + ); + _registerModule( + _modules, + "Core/Renderer/SVG/SVGRenderer.js", + [ + _modules["Core/Color/Color.js"], + _modules["Core/Globals.js"], + _modules["Core/Renderer/SVG/SVGElement.js"], + _modules["Core/Renderer/SVG/SVGLabel.js"], + _modules["Core/Utilities.js"], + ], + function (Color, H, SVGElement, SVGLabel, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var addEvent = U.addEvent, + attr = U.attr, + createElement = U.createElement, + css = U.css, + defined = U.defined, + destroyObjectProperties = U.destroyObjectProperties, + extend = U.extend, + isArray = U.isArray, + isNumber = U.isNumber, + isObject = U.isObject, + isString = U.isString, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick, + pInt = U.pInt, + splat = U.splat, + uniqueKey = U.uniqueKey; + /** + * A clipping rectangle that can be applied to one or more {@link SVGElement} + * instances. It is instanciated with the {@link SVGRenderer#clipRect} function + * and applied with the {@link SVGElement#clip} function. + * + * @example + * var circle = renderer.circle(100, 100, 100) + * .attr({ fill: 'red' }) + * .add(); + * var clipRect = renderer.clipRect(100, 100, 100, 100); + * + * // Leave only the lower right quarter visible + * circle.clip(clipRect); + * + * @typedef {Highcharts.SVGElement} Highcharts.ClipRectElement + */ + /** + * The font metrics. + * + * @interface Highcharts.FontMetricsObject + */ /** + * The baseline relative to the top of the box. + * + * @name Highcharts.FontMetricsObject#b + * @type {number} + */ /** + * The font size. + * + * @name Highcharts.FontMetricsObject#f + * @type {number} + */ /** + * The line height. + * + * @name Highcharts.FontMetricsObject#h + * @type {number} + */ + /** + * An object containing `x` and `y` properties for the position of an element. + * + * @interface Highcharts.PositionObject + */ /** + * X position of the element. + * @name Highcharts.PositionObject#x + * @type {number} + */ /** + * Y position of the element. + * @name Highcharts.PositionObject#y + * @type {number} + */ + /** + * A rectangle. + * + * @interface Highcharts.RectangleObject + */ /** + * Height of the rectangle. + * @name Highcharts.RectangleObject#height + * @type {number} + */ /** + * Width of the rectangle. + * @name Highcharts.RectangleObject#width + * @type {number} + */ /** + * Horizontal position of the rectangle. + * @name Highcharts.RectangleObject#x + * @type {number} + */ /** + * Vertical position of the rectangle. + * @name Highcharts.RectangleObject#y + * @type {number} + */ + /** + * The shadow options. + * + * @interface Highcharts.ShadowOptionsObject + */ /** + * The shadow color. + * @name Highcharts.ShadowOptionsObject#color + * @type {Highcharts.ColorString|undefined} + * @default #000000 + */ /** + * The horizontal offset from the element. + * + * @name Highcharts.ShadowOptionsObject#offsetX + * @type {number|undefined} + * @default 1 + */ /** + * The vertical offset from the element. + * @name Highcharts.ShadowOptionsObject#offsetY + * @type {number|undefined} + * @default 1 + */ /** + * The shadow opacity. + * + * @name Highcharts.ShadowOptionsObject#opacity + * @type {number|undefined} + * @default 0.15 + */ /** + * The shadow width or distance from the element. + * @name Highcharts.ShadowOptionsObject#width + * @type {number|undefined} + * @default 3 + */ + /** + * @interface Highcharts.SizeObject + */ /** + * @name Highcharts.SizeObject#height + * @type {number} + */ /** + * @name Highcharts.SizeObject#width + * @type {number} + */ + /** + * Serialized form of an SVG definition, including children. Some key + * property names are reserved: tagName, textContent, and children. + * + * @interface Highcharts.SVGDefinitionObject + */ /** + * @name Highcharts.SVGDefinitionObject#[key:string] + * @type {boolean|number|string|Array<Highcharts.SVGDefinitionObject>|undefined} + */ /** + * @name Highcharts.SVGDefinitionObject#children + * @type {Array<Highcharts.SVGDefinitionObject>|undefined} + */ /** + * @name Highcharts.SVGDefinitionObject#tagName + * @type {string|undefined} + */ /** + * @name Highcharts.SVGDefinitionObject#textContent + * @type {string|undefined} + */ + /** + * Array of path commands, that will go into the `d` attribute of an SVG + * element. + * + * @typedef {Array<(Array<Highcharts.SVGPathCommand>|Array<Highcharts.SVGPathCommand,number>|Array<Highcharts.SVGPathCommand,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number,number,number,number>)>} Highcharts.SVGPathArray + */ + /** + * Possible path commands in an SVG path array. Valid values are `A`, `C`, `H`, + * `L`, `M`, `Q`, `S`, `T`, `V`, `Z`. + * + * @typedef {string} Highcharts.SVGPathCommand + * @validvalue ["a","c","h","l","m","q","s","t","v","z","A","C","H","L","M","Q","S","T","V","Z"] + */ + /** + * An extendable collection of functions for defining symbol paths. Symbols are + * used internally for point markers, button and label borders and backgrounds, + * or custom shapes. Extendable by adding to {@link SVGRenderer#symbols}. + * + * @interface Highcharts.SymbolDictionary + */ /** + * @name Highcharts.SymbolDictionary#[key:string] + * @type {Function|undefined} + */ /** + * @name Highcharts.SymbolDictionary#arc + * @type {Function|undefined} + */ /** + * @name Highcharts.SymbolDictionary#callout + * @type {Function|undefined} + */ /** + * @name Highcharts.SymbolDictionary#circle + * @type {Function|undefined} + */ /** + * @name Highcharts.SymbolDictionary#diamond + * @type {Function|undefined} + */ /** + * @name Highcharts.SymbolDictionary#square + * @type {Function|undefined} + */ /** + * @name Highcharts.SymbolDictionary#triangle + * @type {Function|undefined} + */ + /** + * Can be one of `arc`, `callout`, `circle`, `diamond`, `square`, `triangle`, + * and `triangle-down`. Symbols are used internally for point markers, button + * and label borders and backgrounds, or custom shapes. Extendable by adding to + * {@link SVGRenderer#symbols}. + * + * @typedef {"arc"|"callout"|"circle"|"diamond"|"square"|"triangle"|"triangle-down"} Highcharts.SymbolKeyValue + */ + /** + * Additional options, depending on the actual symbol drawn. + * + * @interface Highcharts.SymbolOptionsObject + */ /** + * The anchor X position for the `callout` symbol. This is where the chevron + * points to. + * + * @name Highcharts.SymbolOptionsObject#anchorX + * @type {number|undefined} + */ /** + * The anchor Y position for the `callout` symbol. This is where the chevron + * points to. + * + * @name Highcharts.SymbolOptionsObject#anchorY + * @type {number|undefined} + */ /** + * The end angle of an `arc` symbol. + * + * @name Highcharts.SymbolOptionsObject#end + * @type {number|undefined} + */ /** + * Whether to draw `arc` symbol open or closed. + * + * @name Highcharts.SymbolOptionsObject#open + * @type {boolean|undefined} + */ /** + * The radius of an `arc` symbol, or the border radius for the `callout` symbol. + * + * @name Highcharts.SymbolOptionsObject#r + * @type {number|undefined} + */ /** + * The start angle of an `arc` symbol. + * + * @name Highcharts.SymbolOptionsObject#start + * @type {number|undefined} + */ + /* eslint-disable no-invalid-this, valid-jsdoc */ + var charts = H.charts, + deg2rad = H.deg2rad, + doc = H.doc, + isFirefox = H.isFirefox, + isMS = H.isMS, + isWebKit = H.isWebKit, + noop = H.noop, + svg = H.svg, + SVG_NS = H.SVG_NS, + symbolSizes = H.symbolSizes, + win = H.win; + /** + * Allows direct access to the Highcharts rendering layer in order to draw + * primitive shapes like circles, rectangles, paths or text directly on a chart, + * or independent from any chart. The SVGRenderer represents a wrapper object + * for SVG in modern browsers. Through the VMLRenderer, part of the `oldie.js` + * module, it also brings vector graphics to IE <= 8. + * + * An existing chart's renderer can be accessed through {@link Chart.renderer}. + * The renderer can also be used completely decoupled from a chart. + * + * @sample highcharts/members/renderer-on-chart + * Annotating a chart programmatically. + * @sample highcharts/members/renderer-basic + * Independent SVG drawing. + * + * @example + * // Use directly without a chart object. + * var renderer = new Highcharts.Renderer(parentNode, 600, 400); + * + * @class + * @name Highcharts.SVGRenderer + * + * @param {Highcharts.HTMLDOMElement} container + * Where to put the SVG in the web page. + * + * @param {number} width + * The width of the SVG. + * + * @param {number} height + * The height of the SVG. + * + * @param {Highcharts.CSSObject} [style] + * The box style, if not in styleMode + * + * @param {boolean} [forExport=false] + * Whether the rendered content is intended for export. + * + * @param {boolean} [allowHTML=true] + * Whether the renderer is allowed to include HTML text, which will be + * projected on top of the SVG. + * + * @param {boolean} [styledMode=false] + * Whether the renderer belongs to a chart that is in styled mode. + * If it does, it will avoid setting presentational attributes in + * some cases, but not when set explicitly through `.attr` and `.css` + * etc. + */ + var SVGRenderer = /** @class */ (function () { + /* * * - * @param {Array<*>} arr - * The array to filter. + * Constructors + * + * */ + function SVGRenderer( + container, + width, + height, + style, + forExport, + allowHTML, + styledMode + ) { + /* * + * + * Properties + * + * */ + this.alignedObjects = void 0; + /** + * The root `svg` node of the renderer. + * + * @name Highcharts.SVGRenderer#box + * @type {Highcharts.SVGDOMElement} + */ + this.box = void 0; + /** + * The wrapper for the root `svg` node of the renderer. + * + * @name Highcharts.SVGRenderer#boxWrapper + * @type {Highcharts.SVGElement} + */ + this.boxWrapper = void 0; + this.cache = void 0; + this.cacheKeys = void 0; + this.chartIndex = void 0; + /** + * A pointer to the `defs` node of the root SVG. + * + * @name Highcharts.SVGRenderer#defs + * @type {Highcharts.SVGElement} + */ + this.defs = void 0; + this.globalAnimation = void 0; + this.gradients = void 0; + this.height = void 0; + this.imgCount = void 0; + this.isSVG = void 0; + this.style = void 0; + /** + * Page url used for internal references. + * + * @private + * @name Highcharts.SVGRenderer#url + * @type {string} + */ + this.url = void 0; + this.width = void 0; + this.init( + container, + width, + height, + style, + forExport, + allowHTML, + styledMode + ); + } + /* * * - * @param {Function} callback - * The callback function. The function receives the item as the first - * argument. Return `true` if the item is to be preserved. + * Functions * - * @return {Array<*>} - * A new, filtered array. - */ + * */ /** - * Map an array by a callback. + * Initialize the SVGRenderer. Overridable initializer function that takes + * the same parameters as the constructor. * - * @deprecated - * @function Highcharts.map - * - * @param {Array<*>} arr - * The array to map. + * @function Highcharts.SVGRenderer#init * - * @param {Function} fn - * The callback function. Return the new value for the new array. + * @param {Highcharts.HTMLDOMElement} container + * Where to put the SVG in the web page. * - * @return {Array<*>} - * A new array item with modified items. - */ - /** - * Reduce an array to a single value. + * @param {number} width + * The width of the SVG. * - * @deprecated - * @function Highcharts.reduce + * @param {number} height + * The height of the SVG. * - * @param {Array<*>} arr - * The array to reduce. + * @param {Highcharts.CSSObject} [style] + * The box style, if not in styleMode * - * @param {Function} fn - * The callback function. Return the reduced value. Receives 4 - * arguments: Accumulated/reduced value, current value, current array - * index, and the array. + * @param {boolean} [forExport=false] + * Whether the rendered content is intended for export. * - * @param {*} initialValue - * The initial value of the accumulator. + * @param {boolean} [allowHTML=true] + * Whether the renderer is allowed to include HTML text, which will be + * projected on top of the SVG. * - * @return {*} - * The reduced value. - */ + * @param {boolean} [styledMode=false] + * Whether the renderer belongs to a chart that is in styled mode. If it + * does, it will avoid setting presentational attributes in some cases, but + * not when set explicitly through `.attr` and `.css` etc. + */ + SVGRenderer.prototype.init = function ( + container, + width, + height, + style, + forExport, + allowHTML, + styledMode + ) { + var renderer = this, + boxWrapper, + element, + desc; + boxWrapper = renderer.createElement("svg").attr({ + version: "1.1", + class: "highcharts-root", + }); + if (!styledMode) { + boxWrapper.css(this.getStyle(style)); + } + element = boxWrapper.element; + container.appendChild(element); + // Always use ltr on the container, otherwise text-anchor will be + // flipped and text appear outside labels, buttons, tooltip etc (#3482) + attr(container, "dir", "ltr"); + // For browsers other than IE, add the namespace attribute (#1978) + if (container.innerHTML.indexOf("xmlns") === -1) { + attr(element, "xmlns", this.SVG_NS); + } + // object properties + renderer.isSVG = true; + this.box = element; + this.boxWrapper = boxWrapper; + renderer.alignedObjects = []; + // #24, #672, #1070 + this.url = + (isFirefox || isWebKit) && doc.getElementsByTagName("base").length + ? win.location.href + .split("#")[0] // remove the hash + .replace(/<[^>]*>/g, "") // wing cut HTML + // escape parantheses and quotes + .replace(/([\('\)])/g, "\\$1") + // replace spaces (needed for Safari only) + .replace(/ /g, "%20") + : ""; + // Add description + desc = this.createElement("desc").add(); + desc.element.appendChild( + doc.createTextNode("Created with Highcharts 8.2.2") + ); + renderer.defs = this.createElement("defs").add(); + renderer.allowHTML = allowHTML; + renderer.forExport = forExport; + renderer.styledMode = styledMode; + renderer.gradients = {}; // Object where gradient SvgElements are stored + renderer.cache = {}; // Cache for numerical bounding boxes + renderer.cacheKeys = []; + renderer.imgCount = 0; + renderer.setSize(width, height, false); + // Issue 110 workaround: + // In Firefox, if a div is positioned by percentage, its pixel position + // may land between pixels. The container itself doesn't display this, + // but an SVG element inside this container will be drawn at subpixel + // precision. In order to draw sharp lines, this must be compensated + // for. This doesn't seem to work inside iframes though (like in + // jsFiddle). + var subPixelFix, rect; + if (isFirefox && container.getBoundingClientRect) { + subPixelFix = function () { + css(container, { left: 0, top: 0 }); + rect = container.getBoundingClientRect(); + css(container, { + left: Math.ceil(rect.left) - rect.left + "px", + top: Math.ceil(rect.top) - rect.top + "px", + }); + }; + // run the fix now + subPixelFix(); + // run it on resize + renderer.unSubPixelFix = addEvent(win, "resize", subPixelFix); + } + }; + /** + * General method for adding a definition to the SVG `defs` tag. Can be used + * for gradients, fills, filters etc. Styled mode only. A hook for adding + * general definitions to the SVG's defs tag. Definitions can be referenced + * from the CSS by its `id`. Read more in + * [gradients, shadows and patterns](https://www.highcharts.com/docs/chart-design-and-style/gradients-shadows-and-patterns). + * Styled mode only. + * + * @function Highcharts.SVGRenderer#definition + * + * @param {Highcharts.SVGDefinitionObject} def + * A serialized form of an SVG definition, including children. + * + * @return {Highcharts.SVGElement} + * The inserted node. + */ + SVGRenderer.prototype.definition = function (def) { + var ren = this; + /** + * @private + * @param {Highcharts.SVGDefinitionObject} config - SVG definition + * @param {Highcharts.SVGElement} [parent] - parent node + */ + function recurse(config, parent) { + var ret; + splat(config).forEach(function (item) { + var node = ren.createElement(item.tagName), + attr = {}; + // Set attributes + objectEach(item, function (val, key) { + if ( + key !== "tagName" && + key !== "children" && + key !== "textContent" + ) { + attr[key] = val; + } + }); + node.attr(attr); + // Add to the tree + node.add(parent || ren.defs); + // Add text content + if (item.textContent) { + node.element.appendChild(doc.createTextNode(item.textContent)); + } + // Recurse + recurse(item.children || [], node); + ret = node; + }); + // Return last node added (on top level it's the only one) + return ret; + } + return recurse(def); + }; /** - * Test whether at least one element in the array passes the test implemented by - * the provided function. + * Get the global style setting for the renderer. * - * @deprecated - * @function Highcharts.some + * @private + * @function Highcharts.SVGRenderer#getStyle * - * @param {Array<*>} arr - * The array to test + * @param {Highcharts.CSSObject} style + * Style settings. * - * @param {Function} fn - * The function to run on each item. Return truty to pass the test. - * Receives arguments `currentValue`, `index` and `array`. + * @return {Highcharts.CSSObject} + * The style settings mixed with defaults. + */ + SVGRenderer.prototype.getStyle = function (style) { + this.style = extend( + { + fontFamily: + '"Lucida Grande", "Lucida Sans Unicode", ' + + "Arial, Helvetica, sans-serif", + fontSize: "12px", + }, + style + ); + return this.style; + }; + /** + * Apply the global style on the renderer, mixed with the default styles. * - * @param {*} ctx - * The context. + * @function Highcharts.SVGRenderer#setStyle * - * @return {boolean} + * @param {Highcharts.CSSObject} style + * CSS to apply. */ - objectEach({ - map: 'map', - each: 'forEach', - grep: 'filter', - reduce: 'reduce', - some: 'some' - }, function (val, key) { - H[key] = function (arr) { - var _a; - error(32, false, void 0, (_a = {}, _a["Highcharts." + key] = "use Array." + val, _a)); - return Array.prototype[val].apply(arr, [].slice.call(arguments, 1)); - }; - }); - /* eslint-disable valid-jsdoc */ + SVGRenderer.prototype.setStyle = function (style) { + this.boxWrapper.css(this.getStyle(style)); + }; /** - * Add an event listener. + * Detect whether the renderer is hidden. This happens when one of the + * parent elements has `display: none`. Used internally to detect when we + * needto render preliminarily in another div to get the text bounding boxes + * right. * - * @function Highcharts.addEvent<T> + * @function Highcharts.SVGRenderer#isHidden * - * @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. + * @return {boolean} + * True if it is hidden. + */ + SVGRenderer.prototype.isHidden = function () { + return !this.boxWrapper.getBBox().width; + }; + /** + * Destroys the renderer and its allocated members. + * + * @function Highcharts.SVGRenderer#destroy + * + * @return {null} + */ + SVGRenderer.prototype.destroy = function () { + var renderer = this, + rendererDefs = renderer.defs; + renderer.box = null; + renderer.boxWrapper = renderer.boxWrapper.destroy(); + // Call destroy on all gradient elements + destroyObjectProperties(renderer.gradients || {}); + renderer.gradients = null; + // Defs are null in VMLRenderer + // Otherwise, destroy them here. + if (rendererDefs) { + renderer.defs = rendererDefs.destroy(); + } + // Remove sub pixel fix handler (#982) + if (renderer.unSubPixelFix) { + renderer.unSubPixelFix(); + } + renderer.alignedObjects = null; + return null; + }; + /** + * Create a wrapper for an SVG element. Serves as a factory for + * {@link SVGElement}, but this function is itself mostly called from + * primitive factories like {@link SVGRenderer#path}, {@link + * SVGRenderer#rect} or {@link SVGRenderer#text}. * - * @param {Highcharts.EventCallbackFunction<T>|Function} fn - * The function callback to execute when the event is fired. + * @function Highcharts.SVGRenderer#createElement * - * @param {Highcharts.EventOptionsObject} [options] - * Options for adding the event. + * @param {string} nodeName + * The node name, for example `rect`, `g` etc. * - * @return {Function} - * A callback function to remove the added event. + * @return {Highcharts.SVGElement} + * The generated SVGElement. */ - var addEvent = H.addEvent = function (el, - type, - fn, - options) { - if (options === void 0) { options = {}; } - /* eslint-enable valid-jsdoc */ - var events, - addEventListener = (el.addEventListener || H.addEventListenerPolyfill); - // If we're setting events directly on the constructor, use a separate - // collection, `protoEvents` to distinguish it from the item events in - // `hcEvents`. - if (typeof el === 'function' && el.prototype) { - events = el.prototype.protoEvents = el.prototype.protoEvents || {}; - } - else { - events = el.hcEvents = el.hcEvents || {}; - } - // Allow click events added to points, otherwise they will be prevented by - // the TouchPointer.pinch function after a pinch zoom operation (#7091). - if (H.Point && - el instanceof H.Point && - el.series && - el.series.chart) { - el.series.chart.runTrackerClick = true; - } - // Handle DOM events - if (addEventListener) { - addEventListener.call(el, type, fn, false); - } - if (!events[type]) { - events[type] = []; - } - var eventObject = { - fn: fn, - order: typeof options.order === 'number' ? options.order : Infinity - }; - events[type].push(eventObject); - // Order the calls - events[type].sort(function (a, b) { - return a.order - b.order; - }); - // Return a function that can be called to remove this event. - return function () { - removeEvent(el, type, fn); - }; + SVGRenderer.prototype.createElement = function (nodeName) { + var wrapper = new this.Element(); + wrapper.init(this, nodeName); + return wrapper; }; - /* eslint-disable valid-jsdoc */ /** - * Remove an event that was added with {@link Highcharts#addEvent}. - * - * @function Highcharts.removeEvent<T> + * Get converted radial gradient attributes according to the radial + * reference. Used internally from the {@link SVGElement#colorGradient} + * function. * - * @param {Highcharts.Class<T>|T} el - * The element to remove events on. + * @private + * @function Highcharts.SVGRenderer#getRadialAttr + */ + SVGRenderer.prototype.getRadialAttr = function ( + radialReference, + gradAttr + ) { + return { + cx: + radialReference[0] - + radialReference[2] / 2 + + gradAttr.cx * radialReference[2], + cy: + radialReference[1] - + radialReference[2] / 2 + + gradAttr.cy * radialReference[2], + r: gradAttr.r * radialReference[2], + }; + }; + /** + * Truncate the text node contents to a given length. Used when the css + * width is set. If the `textOverflow` is `ellipsis`, the text is truncated + * character by character to the given length. If not, the text is + * word-wrapped line by line. * - * @param {string} [type] - * The type of events to remove. If undefined, all events are removed - * from the element. + * @private + * @function Highcharts.SVGRenderer#truncate * - * @param {Highcharts.EventCallbackFunction<T>} [fn] - * The specific callback to remove. If undefined, all events that match - * the element and optionally the type are removed. + * @return {boolean} + * True if tspan is too long. + */ + SVGRenderer.prototype.truncate = function ( + wrapper, + tspan, + text, + words, + startAt, + width, + getString + ) { + var renderer = this, + rotation = wrapper.rotation, + str, + // Word wrap can not be truncated to shorter than one word, ellipsis + // text can be completely blank. + minIndex = words ? 1 : 0, + maxIndex = (text || words).length, + currentIndex = maxIndex, + // Cache the lengths to avoid checking the same twice + lengths = [], + updateTSpan = function (s) { + if (tspan.firstChild) { + tspan.removeChild(tspan.firstChild); + } + if (s) { + tspan.appendChild(doc.createTextNode(s)); + } + }, + getSubStringLength = function (charEnd, concatenatedEnd) { + // charEnd is useed when finding the character-by-character + // break for ellipsis, concatenatedEnd is used for word-by-word + // break for word wrapping. + var end = concatenatedEnd || charEnd; + if (typeof lengths[end] === "undefined") { + // Modern browsers + if (tspan.getSubStringLength) { + // Fails with DOM exception on unit-tests/legend/members + // of unknown reason. Desired width is 0, text content + // is "5" and end is 1. + try { + lengths[end] = + startAt + + tspan.getSubStringLength(0, words ? end + 1 : end); + } catch (e) { + (""); + } + // Legacy + } else if (renderer.getSpanWidth) { + // #9058 jsdom + updateTSpan(getString(text || words, charEnd)); + lengths[end] = + startAt + renderer.getSpanWidth(wrapper, tspan); + } + } + return lengths[end]; + }, + actualWidth, + truncated; + wrapper.rotation = 0; // discard rotation when computing box + actualWidth = getSubStringLength(tspan.textContent.length); + truncated = startAt + actualWidth > width; + if (truncated) { + // Do a binary search for the index where to truncate the text + while (minIndex <= maxIndex) { + currentIndex = Math.ceil((minIndex + maxIndex) / 2); + // When checking words for word-wrap, we need to build the + // string and measure the subStringLength at the concatenated + // word length. + if (words) { + str = getString(words, currentIndex); + } + actualWidth = getSubStringLength( + currentIndex, + str && str.length - 1 + ); + if (minIndex === maxIndex) { + // Complete + minIndex = maxIndex + 1; + } else if (actualWidth > width) { + // Too large. Set max index to current. + maxIndex = currentIndex - 1; + } else { + // Within width. Set min index to current. + minIndex = currentIndex; + } + } + // If max index was 0 it means the shortest possible text was also + // too large. For ellipsis that means only the ellipsis, while for + // word wrap it means the whole first word. + if (maxIndex === 0) { + // Remove ellipsis + updateTSpan(""); + // If the new text length is one less than the original, we don't + // need the ellipsis + } else if (!(text && maxIndex === text.length - 1)) { + updateTSpan(str || getString(text || words, currentIndex)); + } + } + // When doing line wrapping, prepare for the next line by removing the + // items from this line. + if (words) { + words.splice(0, currentIndex); + } + wrapper.actualWidth = actualWidth; + wrapper.rotation = rotation; // Apply rotation again. + return truncated; + }; + /** + * Parse a simple HTML string into SVG tspans. Called internally when text + * is set on an SVGElement. The function supports a subset of HTML tags, CSS + * text features like `width`, `text-overflow`, `white-space`, and also + * attributes like `href` and `style`. * - * @return {void} - */ - var removeEvent = H.removeEvent = function removeEvent(el, - type, - fn) { - /* eslint-enable valid-jsdoc */ - var events; - /** - * @private - * @param {string} type - event type - * @param {Highcharts.EventCallbackFunction<T>} fn - callback - * @return {void} - */ - function removeOneEvent(type, fn) { - var removeEventListener = (el.removeEventListener || H.removeEventListenerPolyfill); - if (removeEventListener) { - removeEventListener.call(el, type, fn, false); - } + * @private + * @function Highcharts.SVGRenderer#buildText + * + * @param {Highcharts.SVGElement} wrapper + * The parent SVGElement. + */ + SVGRenderer.prototype.buildText = function (wrapper) { + var textNode = wrapper.element, + renderer = this, + forExport = renderer.forExport, + textStr = pick(wrapper.textStr, "").toString(), + hasMarkup = textStr.indexOf("<") !== -1, + lines, + childNodes = textNode.childNodes, + truncated, + parentX = attr(textNode, "x"), + textStyles = wrapper.styles, + width = wrapper.textWidth, + textLineHeight = textStyles && textStyles.lineHeight, + textOutline = textStyles && textStyles.textOutline, + ellipsis = textStyles && textStyles.textOverflow === "ellipsis", + noWrap = textStyles && textStyles.whiteSpace === "nowrap", + fontSize = textStyles && textStyles.fontSize, + textCache, + isSubsequentLine, + i = childNodes.length, + tempParent = width && !wrapper.added && this.box, + getLineHeight = function (tspan) { + var fontSizeStyle; + if (!renderer.styledMode) { + fontSizeStyle = /(px|em)$/.test(tspan && tspan.style.fontSize) + ? tspan.style.fontSize + : fontSize || renderer.style.fontSize || 12; + } + return textLineHeight + ? pInt(textLineHeight) + : renderer.fontMetrics( + fontSizeStyle, + // Get the computed size from parent if not explicit + tspan.getAttribute("style") ? tspan : textNode + ).h; + }, + unescapeEntities = function (inputStr, except) { + objectEach(renderer.escapes, function (value, key) { + if (!except || except.indexOf(value) === -1) { + inputStr = inputStr + .toString() + .replace(new RegExp(value, "g"), key); + } + }); + return inputStr; + }, + parseAttribute = function (s, attr) { + var start, delimiter; + start = s.indexOf("<"); + s = s.substring(start, s.indexOf(">") - start); + start = s.indexOf(attr + "="); + if (start !== -1) { + start = start + attr.length + 1; + delimiter = s.charAt(start); + if (delimiter === '"' || delimiter === "'") { + // eslint-disable-line quotes + s = s.substring(start + 1); + return s.substring(0, s.indexOf(delimiter)); + } + } + }; + var regexMatchBreaks = /<br.*?>/g; + // The buildText code is quite heavy, so if we're not changing something + // that affects the text, skip it (#6113). + textCache = [ + textStr, + ellipsis, + noWrap, + textLineHeight, + textOutline, + fontSize, + width, + ].join(","); + if (textCache === wrapper.textCache) { + return; + } + wrapper.textCache = textCache; + // Remove old text + while (i--) { + textNode.removeChild(childNodes[i]); + } + // Skip tspans, add text directly to text node. The forceTSpan is a hook + // used in text outline hack. + if ( + !hasMarkup && + !textOutline && + !ellipsis && + !width && + (textStr.indexOf(" ") === -1 || + (noWrap && !regexMatchBreaks.test(textStr))) + ) { + textNode.appendChild(doc.createTextNode(unescapeEntities(textStr))); + // Complex strings, add more logic + } else { + if (tempParent) { + // attach it to the DOM to read offset width + tempParent.appendChild(textNode); } - /** - * @private - * @param {any} eventCollection - collection - * @return {void} - */ - function removeAllEvents(eventCollection) { - var types, - len; - if (!el.nodeName) { - return; // break on non-DOM events - } - if (type) { - types = {}; - types[type] = true; - } - else { - types = eventCollection; - } - objectEach(types, function (_val, n) { - if (eventCollection[n]) { - len = eventCollection[n].length; - while (len--) { - removeOneEvent(n, eventCollection[n][len].fn); - } - } - }); + if (hasMarkup) { + lines = renderer.styledMode + ? textStr + .replace( + /<(b|strong)>/g, + '<span class="highcharts-strong">' + ) + .replace( + /<(i|em)>/g, + '<span class="highcharts-emphasized">' + ) + : textStr + .replace(/<(b|strong)>/g, '<span style="font-weight:bold">') + .replace(/<(i|em)>/g, '<span style="font-style:italic">'); + lines = lines + .replace(/<a/g, "<span") + .replace(/<\/(b|strong|i|em|a)>/g, "</span>") + .split(regexMatchBreaks); + } else { + lines = [textStr]; } - ['protoEvents', 'hcEvents'].forEach(function (coll, i) { - var eventElem = i ? el : el.prototype; - var eventCollection = eventElem && eventElem[coll]; - if (eventCollection) { - if (type) { - events = (eventCollection[type] || []); - if (fn) { - eventCollection[type] = events.filter(function (obj) { - return fn !== obj.fn; + // Trim empty lines (#5261) + lines = lines.filter(function (line) { + return line !== ""; + }); + // build the lines + lines.forEach(function (line, lineNo) { + var spans, + spanNo = 0, + lineLength = 0; + line = line + // Trim to prevent useless/costly process on the spaces + // (#5258) + .replace(/^\s+|\s+$/g, "") + .replace(/<span/g, "|||<span") + .replace(/<\/span>/g, "</span>|||"); + spans = line.split("|||"); + spans.forEach(function buildTextSpans(span) { + if (span !== "" || spans.length === 1) { + var attributes = {}, + tspan = doc.createElementNS(renderer.SVG_NS, "tspan"), + a, + classAttribute, + styleAttribute, // #390 + hrefAttribute; + classAttribute = parseAttribute(span, "class"); + if (classAttribute) { + attr(tspan, "class", classAttribute); + } + styleAttribute = parseAttribute(span, "style"); + if (styleAttribute) { + styleAttribute = styleAttribute.replace( + /(;| |^)color([ :])/, + "$1fill$2" + ); + attr(tspan, "style", styleAttribute); + } + // For anchors, wrap the tspan in an <a> tag and apply + // the href attribute as is (#13559). Not for export + // (#1529) + hrefAttribute = parseAttribute(span, "href"); + if (hrefAttribute && !forExport) { + if ( + // Stop JavaScript links, vulnerable to XSS + hrefAttribute + .split(":")[0] + .toLowerCase() + .indexOf("javascript") === -1 + ) { + a = doc.createElementNS(renderer.SVG_NS, "a"); + attr(a, "href", hrefAttribute); + attr(tspan, "class", "highcharts-anchor"); + a.appendChild(tspan); + if (!renderer.styledMode) { + css(tspan, { cursor: "pointer" }); + } + } + } + // Strip away unsupported HTML tags (#7126) + span = unescapeEntities( + span.replace(/<[a-zA-Z\/](.|\n)*?>/g, "") || " " + ); + // Nested tags aren't supported, and cause crash in + // Safari (#1596) + if (span !== " ") { + // add the text node + tspan.appendChild(doc.createTextNode(span)); + // First span in a line, align it to the left + if (!spanNo) { + if (lineNo && parentX !== null) { + attributes.x = parentX; + } + } else { + attributes.dx = 0; // #16 + } + // add attributes + attr(tspan, attributes); + // Append it + textNode.appendChild(a || tspan); + // first span on subsequent line, add the line + // height + if (!spanNo && isSubsequentLine) { + // allow getting the right offset height in + // exporting in IE + if (!svg && forExport) { + css(tspan, { display: "block" }); + } + // Set the line height based on the font size of + // either the text element or the tspan element + attr(tspan, "dy", getLineHeight(tspan)); + } + // Check width and apply soft breaks or ellipsis + if (width) { + var words = span.replace(/([^\^])-/g, "$1- ").split(" "), // #1273 + hasWhiteSpace = + !noWrap && + (spans.length > 1 || lineNo || words.length > 1), + wrapLineNo = 0, + dy = getLineHeight(tspan); + if (ellipsis) { + truncated = renderer.truncate( + wrapper, + tspan, + span, + void 0, + 0, + // Target width + Math.max( + 0, + // Substract the font face to make + // room for the ellipsis itself + width - parseInt(fontSize || 12, 10) + ), + // Build the text to test for + function (text, currentIndex) { + return text.substring(0, currentIndex) + "\u2026"; + } + ); + } else if (hasWhiteSpace) { + while (words.length) { + // For subsequent lines, create tspans + // with the same style attributes as the + // parent text node. + if (words.length && !noWrap && wrapLineNo > 0) { + tspan = doc.createElementNS(SVG_NS, "tspan"); + attr(tspan, { + dy: dy, + x: parentX, }); - removeOneEvent(type, fn); - } - else { - removeAllEvents(eventCollection); - eventCollection[type] = []; + if (styleAttribute) { + // #390 + attr(tspan, "style", styleAttribute); + } + // Start by appending the full + // remaining text + tspan.appendChild( + doc.createTextNode( + words.join(" ").replace(/- /g, "-") + ) + ); + textNode.appendChild(tspan); + } + // For each line, truncate the remaining + // words into the line length. + renderer.truncate( + wrapper, + tspan, + null, + words, + wrapLineNo === 0 ? lineLength : 0, + width, + // Build the text to test for + function (text, currentIndex) { + return words + .slice(0, currentIndex) + .join(" ") + .replace(/- /g, "-"); + } + ); + lineLength = wrapper.actualWidth; + wrapLineNo++; } + } } - else { - removeAllEvents(eventCollection); - eventElem[coll] = {}; - } + spanNo++; + } } + }); + // To avoid beginning lines that doesn't add to the textNode + // (#6144) + isSubsequentLine = isSubsequentLine || textNode.childNodes.length; }); + if (ellipsis && truncated) { + wrapper.attr( + "title", + unescapeEntities(wrapper.textStr || "", ["<", ">"]) // #7179 + ); + } + if (tempParent) { + tempParent.removeChild(textNode); + } + // Apply the text outline + if (isString(textOutline) && wrapper.applyTextOutline) { + wrapper.applyTextOutline(textOutline); + } + } }; - /* eslint-disable valid-jsdoc */ /** - * Fire an event that was registered with {@link Highcharts#addEvent}. + * Returns white for dark colors and black for bright colors. * - * @function Highcharts.fireEvent<T> + * @function Highcharts.SVGRenderer#getContrast * - * @param {T} el - * The object to fire the event on. It can be a {@link HTMLDOMElement}, - * an {@link SVGElement} or any other object. + * @param {Highcharts.ColorString} rgba + * The color to get the contrast for. * - * @param {string} type - * The type of event. + * @return {Highcharts.ColorString} + * The contrast color, either `#000000` or `#FFFFFF`. + */ + SVGRenderer.prototype.getContrast = function (rgba) { + rgba = Color.parse(rgba).rgba; + // The threshold may be discussed. Here's a proposal for adding + // different weight to the color channels (#6216) + rgba[0] *= 1; // red + rgba[1] *= 1.2; // green + rgba[2] *= 0.5; // blue + return rgba[0] + rgba[1] + rgba[2] > 1.8 * 255 + ? "#000000" + : "#FFFFFF"; + }; + /** + * Create a button with preset states. * - * @param {Highcharts.Dictionary<*>|Event} [eventArguments] - * Custom event arguments that are passed on as an argument to the event - * handler. + * @function Highcharts.SVGRenderer#button * - * @param {Highcharts.EventCallbackFunction<T>|Function} [defaultFunction] - * The default function to execute if the other listeners haven't - * returned false. + * @param {string} text + * The text or HTML to draw. * - * @return {void} - */ - var fireEvent = H.fireEvent = function (el, - type, - eventArguments, - defaultFunction) { - /* eslint-enable valid-jsdoc */ - var e, - i; - eventArguments = eventArguments || {}; - if (doc.createEvent && - (el.dispatchEvent || el.fireEvent)) { - e = doc.createEvent('Events'); - e.initEvent(type, true, true); - extend(e, eventArguments); - if (el.dispatchEvent) { - el.dispatchEvent(e); - } - else { - el.fireEvent(type, e); - } + * @param {number} x + * The x position of the button's left side. + * + * @param {number} y + * The y position of the button's top side. + * + * @param {Highcharts.EventCallbackFunction<Highcharts.SVGElement>} callback + * The function to execute on button click or touch. + * + * @param {Highcharts.SVGAttributes} [normalState] + * SVG attributes for the normal state. + * + * @param {Highcharts.SVGAttributes} [hoverState] + * SVG attributes for the hover state. + * + * @param {Highcharts.SVGAttributes} [pressedState] + * SVG attributes for the pressed state. + * + * @param {Highcharts.SVGAttributes} [disabledState] + * SVG attributes for the disabled state. + * + * @param {Highcharts.SymbolKeyValue} [shape=rect] + * The shape type. + * + * @param {boolean} [useHTML=false] + * Wether to use HTML to render the label. + * + * @return {Highcharts.SVGElement} + * The button element. + */ + SVGRenderer.prototype.button = function ( + text, + x, + y, + callback, + normalState, + hoverState, + pressedState, + disabledState, + shape, + useHTML + ) { + var label = this.label( + text, + x, + y, + shape, + void 0, + void 0, + useHTML, + void 0, + "button" + ), + curState = 0, + styledMode = this.styledMode, + // Make a copy of normalState (#13798) + // (reference to options.rangeSelector.buttonTheme) + normalState = normalState ? merge(normalState) : normalState, + userNormalStyle = (normalState && normalState.style) || {}; + // Remove stylable attributes + if (normalState && normalState.style) { + delete normalState.style; + } + // Default, non-stylable attributes + label.attr(merge({ padding: 8, r: 2 }, normalState)); + if (!styledMode) { + // Presentational + var normalStyle, hoverStyle, pressedStyle, disabledStyle; + // Normal state - prepare the attributes + normalState = merge( + { + fill: "#f7f7f7", + stroke: "#cccccc", + "stroke-width": 1, + style: { + color: "#333333", + cursor: "pointer", + fontWeight: "normal", + }, + }, + { + style: userNormalStyle, + }, + normalState + ); + normalStyle = normalState.style; + delete normalState.style; + // Hover state + hoverState = merge( + normalState, + { + fill: "#e6e6e6", + }, + hoverState + ); + hoverStyle = hoverState.style; + delete hoverState.style; + // Pressed state + pressedState = merge( + normalState, + { + fill: "#e6ebf5", + style: { + color: "#000000", + fontWeight: "bold", + }, + }, + pressedState + ); + pressedStyle = pressedState.style; + delete pressedState.style; + // Disabled state + disabledState = merge( + normalState, + { + style: { + color: "#cccccc", + }, + }, + disabledState + ); + disabledStyle = disabledState.style; + delete disabledState.style; + } + // Add the events. IE9 and IE10 need mouseover and mouseout to funciton + // (#667). + addEvent( + label.element, + isMS ? "mouseover" : "mouseenter", + function () { + if (curState !== 3) { + label.setState(1); + } } - else { - 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, - // and setting it will cause en error in IE8 (#2465). - type: type - }); - } - var fireInOrder = function (protoEvents, - hcEvents) { - if (protoEvents === void 0) { protoEvents = []; } - if (hcEvents === void 0) { hcEvents = []; } - var iA = 0; - var iB = 0; - var length = protoEvents.length + hcEvents.length; - for (i = 0; i < length; i++) { - var obj = (!protoEvents[iA] ? - hcEvents[iB++] : - !hcEvents[iB] ? - protoEvents[iA++] : - protoEvents[iA].order <= hcEvents[iB].order ? - protoEvents[iA++] : - hcEvents[iB++]); - // If the event handler return false, prevent the default - // handler from executing - if (obj.fn.call(el, eventArguments) === false) { - eventArguments.preventDefault(); - } - } - }; - fireInOrder(el.protoEvents && el.protoEvents[type], el.hcEvents && el.hcEvents[type]); + ); + addEvent( + label.element, + isMS ? "mouseout" : "mouseleave", + function () { + if (curState !== 3) { + label.setState(curState); + } + } + ); + label.setState = function (state) { + // Hover state is temporary, don't record it + if (state !== 1) { + label.state = curState = state; + } + // Update visuals + label + .removeClass(/highcharts-button-(normal|hover|pressed|disabled)/) + .addClass( + "highcharts-button-" + + ["normal", "hover", "pressed", "disabled"][state || 0] + ); + if (!styledMode) { + label + .attr( + [normalState, hoverState, pressedState, disabledState][ + state || 0 + ] + ) + .css( + [normalStyle, hoverStyle, pressedStyle, disabledStyle][ + state || 0 + ] + ); } - // Run the default if not prevented - if (defaultFunction && !eventArguments.defaultPrevented) { - defaultFunction.call(el, eventArguments); + }; + // Presentational attributes + if (!styledMode) { + label + .attr(normalState) + .css(extend({ cursor: "default" }, normalStyle)); + } + return label.on("click", function (e) { + if (curState !== 3) { + callback.call(label, e); } + }); }; - var serialMode; /** - * Get a unique key for using in internal element id's and pointers. The key is - * composed of a random hash specific to this Highcharts instance, and a - * counter. + * Make a straight line crisper by not spilling out to neighbour pixels. * - * @example - * var id = uniqueKey(); // => 'highcharts-x45f6hp-0' + * @function Highcharts.SVGRenderer#crispLine * - * @function Highcharts.uniqueKey + * @param {Highcharts.SVGPathArray} points + * The original points on the format `[['M', 0, 0], ['L', 100, 0]]`. * - * @return {string} - * A unique key. - */ - var uniqueKey = H.uniqueKey = (function () { - var hash = Math.random().toString(36).substring(2, 9) + '-'; - var id = 0; - return function () { - return 'highcharts-' + (serialMode ? '' : hash) + id++; - }; - }()); + * @param {number} width + * The width of the line. + * + * @param {string} roundingFunction + * The rounding function name on the `Math` object, can be one of + * `round`, `floor` or `ceil`. + * + * @return {Highcharts.SVGPathArray} + * The original points array, but modified to render crisply. + */ + SVGRenderer.prototype.crispLine = function ( + points, + width, + roundingFunction + ) { + if (roundingFunction === void 0) { + roundingFunction = "round"; + } + var start = points[0]; + var end = points[1]; + // Normalize to a crisp line + if (start[1] === end[1]) { + // Substract due to #1129. Now bottom and left axis gridlines behave + // the same. + start[1] = end[1] = + Math[roundingFunction](start[1]) - (width % 2) / 2; + } + if (start[2] === end[2]) { + start[2] = end[2] = + Math[roundingFunction](start[2]) + (width % 2) / 2; + } + return points; + }; /** - * Activates a serial mode for element IDs provided by - * {@link Highcharts.uniqueKey}. This mode can be used in automated tests, where - * a simple comparison of two rendered SVG graphics is needed. + * Draw a path, wraps the SVG `path` element. * - * **Note:** This is only for testing purposes and will break functionality in - * webpages with multiple charts. + * @sample highcharts/members/renderer-path-on-chart/ + * Draw a path in a chart + * @sample highcharts/members/renderer-path/ + * Draw a path independent from a chart * * @example - * if ( - * process && - * process.env.NODE_ENV === 'development' - * ) { - * Highcharts.useSerialIds(true); - * } + * var path = renderer.path(['M', 10, 10, 'L', 30, 30, 'z']) + * .attr({ stroke: '#ff00ff' }) + * .add(); * - * @function Highcharts.useSerialIds + * @function Highcharts.SVGRenderer#path * - * @param {boolean} [mode] - * Changes the state of serial mode. + * @param {Highcharts.SVGPathArray} [path] + * An SVG path definition in array form. * - * @return {boolean|undefined} - * State of the serial mode. - */ - var useSerialIds = H.useSerialIds = function (mode) { - return (serialMode = pick(mode, - serialMode)); - }; - var isFunction = H.isFunction = function (obj) { - return typeof obj === 'function'; + * @return {Highcharts.SVGElement} + * The generated wrapper element. + * + */ /** + * Draw a path, wraps the SVG `path` element. + * + * @function Highcharts.SVGRenderer#path + * + * @param {Highcharts.SVGAttributes} [attribs] + * The initial attributes. + * + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ + SVGRenderer.prototype.path = function (path) { + var attribs = this.styledMode + ? {} + : { + fill: "none", + }; + if (isArray(path)) { + attribs.d = path; + } else if (isObject(path)) { + // attributes + extend(attribs, path); + } + return this.createElement("path").attr(attribs); }; /** - * Get the updated default options. Until 3.0.7, merely exposing defaultOptions - * for outside modules wasn't enough because the setOptions method created a new - * object. + * Draw a circle, wraps the SVG `circle` element. * - * @function Highcharts.getOptions + * @sample highcharts/members/renderer-circle/ + * Drawing a circle * - * @return {Highcharts.Options} - */ - var getOptions = H.getOptions = function () { - return H.defaultOptions; - }; - /** - * Merge the default options with custom options and return the new options - * structure. Commonly used for defining reusable templates. + * @function Highcharts.SVGRenderer#circle * - * @sample highcharts/global/useutc-false Setting a global option - * @sample highcharts/members/setoptions Applying a global theme + * @param {number} [x] + * The center x position. * - * @function Highcharts.setOptions + * @param {number} [y] + * The center y position. * - * @param {Highcharts.Options} options - * The new custom chart options. + * @param {number} [r] + * The radius. * - * @return {Highcharts.Options} - * Updated options. - */ - var setOptions = H.setOptions = function (options) { - // Copy in the default options - H.defaultOptions = merge(true, - H.defaultOptions, - options); - // Update the time object - if (options.time || options.global) { - H.time.update(merge(H.defaultOptions.global, H.defaultOptions.time, options.global, options.time)); - } - return H.defaultOptions; + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ /** + * Draw a circle, wraps the SVG `circle` element. + * + * @function Highcharts.SVGRenderer#circle + * + * @param {Highcharts.SVGAttributes} [attribs] + * The initial attributes. + * + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ + SVGRenderer.prototype.circle = function (x, y, r) { + var attribs = isObject(x) + ? x + : typeof x === "undefined" + ? {} + : { x: x, y: y, r: r }, + wrapper = this.createElement("circle"); + // Setting x or y translates to cx and cy + wrapper.xSetter = wrapper.ySetter = function (value, key, element) { + element.setAttribute("c" + key, value); + }; + return wrapper.attr(attribs); }; - // Register Highcharts as a plugin in jQuery - if (win.jQuery) { - /** - * Highcharts-extended JQuery. - * - * @external JQuery - */ - /** - * Helper function to return the chart of the current JQuery selector - * element. - * - * @function external:JQuery#highcharts - * - * @return {Highcharts.Chart} - * The chart that is linked to the JQuery selector element. - */ /** - * Factory function to create a chart in the current JQuery selector - * element. - * - * @function external:JQuery#highcharts - * - * @param {'Chart'|'Map'|'StockChart'|string} [className] - * Name of the factory class in the Highcharts namespace. - * - * @param {Highcharts.Options} [options] - * The chart options structure. - * - * @param {Highcharts.ChartCallbackFunction} [callback] - * Function to run when the chart has loaded and and all external - * images are loaded. Defining a - * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load) - * handler is equivalent. - * - * @return {JQuery} - * The current JQuery selector. - */ - win.jQuery.fn.highcharts = function () { - var args = [].slice.call(arguments); - if (this[0]) { // this[0] is the renderTo div - // Create the chart - if (args[0]) { - new H[ // eslint-disable-line computed-property-spacing, no-new - // Constructor defaults to Chart - isString(args[0]) ? args.shift() : 'Chart'](this[0], args[0], args[1]); - return this; - } - // When called without parameters or with the return argument, - // return an existing chart - return charts[attr(this[0], 'data-highcharts-chart')]; - } - }; - } - // TODO use named exports when supported. - var utilitiesModule = { - addEvent: addEvent, - arrayMax: arrayMax, - arrayMin: arrayMin, - attr: attr, - clamp: clamp, - clearTimeout: internalClearTimeout, - correctFloat: correctFloat, - createElement: createElement, - css: css, - defined: defined, - destroyObjectProperties: destroyObjectProperties, - discardElement: discardElement, - erase: erase, - error: error, - extend: extend, - extendClass: extendClass, - find: find, - fireEvent: fireEvent, - format: format, - getMagnitude: getMagnitude, - getNestedProperty: getNestedProperty, - getOptions: getOptions, - getStyle: getStyle, - inArray: inArray, - isArray: isArray, - isClass: isClass, - isDOMElement: isDOMElement, - isFunction: isFunction, - isNumber: isNumber, - isObject: isObject, - isString: isString, - merge: merge, - normalizeTickInterval: normalizeTickInterval, - numberFormat: numberFormat, - objectEach: objectEach, - offset: offset, - pad: pad, - pick: pick, - pInt: pInt, - relativeLength: relativeLength, - removeEvent: removeEvent, - setOptions: setOptions, - splat: splat, - stableSort: stableSort, - syncTimeout: syncTimeout, - timeUnits: timeUnits, - uniqueKey: uniqueKey, - useSerialIds: useSerialIds, - wrap: wrap - }; - - return utilitiesModule; - }); - _registerModule(_modules, 'Core/Color/Color.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) { - /* * + /** + * Draw and return an arc. * - * (c) 2010-2020 Torstein Honsi + * @sample highcharts/members/renderer-arc/ + * Drawing an arc * - * License: www.highcharts.com/license + * @function Highcharts.SVGRenderer#arc * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @param {number} [x=0] + * Center X position. * - * */ - var isNumber = U.isNumber, - merge = U.merge, - pInt = U.pInt; - /** - * A valid color to be parsed and handled by Highcharts. Highcharts internally - * supports hex colors like `#ffffff`, rgb colors like `rgb(255,255,255)` and - * rgba colors like `rgba(255,255,255,1)`. Other colors may be supported by the - * browsers and displayed correctly, but Highcharts is not able to process them - * and apply concepts like opacity and brightening. + * @param {number} [y=0] + * Center Y position. * - * @typedef {string} Highcharts.ColorString - */ - /** - * A valid color type than can be parsed and handled by Highcharts. It can be a - * color string, a gradient object, or a pattern object. + * @param {number} [r=0] + * The outer radius' of the arc. * - * @typedef {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} Highcharts.ColorType - */ - /** - * Gradient options instead of a solid color. + * @param {number} [innerR=0] + * Inner radius like used in donut charts. * - * @example - * // Linear gradient used as a color option - * color: { - * linearGradient: { x1: 0, x2: 0, y1: 0, y2: 1 }, - * stops: [ - * [0, '#003399'], // start - * [0.5, '#ffffff'], // middle - * [1, '#3366AA'] // end - * ] - * } - * - * @interface Highcharts.GradientColorObject - */ /** - * Holds an object that defines the start position and the end position relative - * to the shape. - * @name Highcharts.GradientColorObject#linearGradient - * @type {Highcharts.LinearGradientColorObject|undefined} - */ /** - * Holds an object that defines the center position and the radius. - * @name Highcharts.GradientColorObject#radialGradient - * @type {Highcharts.RadialGradientColorObject|undefined} - */ /** - * The first item in each tuple is the position in the gradient, where 0 is the - * start of the gradient and 1 is the end of the gradient. Multiple stops can be - * applied. The second item is the color for each stop. This color can also be - * given in the rgba format. - * @name Highcharts.GradientColorObject#stops - * @type {Array<Highcharts.GradientColorStopObject>} - */ - /** - * Color stop tuple. - * - * @see Highcharts.GradientColorObject - * - * @interface Highcharts.GradientColorStopObject - */ /** - * @name Highcharts.GradientColorStopObject#0 - * @type {number} - */ /** - * @name Highcharts.GradientColorStopObject#1 - * @type {Highcharts.ColorString} - */ /** - * @name Highcharts.GradientColorStopObject#color - * @type {Highcharts.Color|undefined} - */ - /** - * Defines the start position and the end position for a gradient relative - * to the shape. Start position (x1, y1) and end position (x2, y2) are relative - * to the shape, where 0 means top/left and 1 is bottom/right. - * - * @interface Highcharts.LinearGradientColorObject - */ /** - * Start horizontal position of the gradient. Float ranges 0-1. - * @name Highcharts.LinearGradientColorObject#x1 - * @type {number} - */ /** - * End horizontal position of the gradient. Float ranges 0-1. - * @name Highcharts.LinearGradientColorObject#x2 - * @type {number} - */ /** - * Start vertical position of the gradient. Float ranges 0-1. - * @name Highcharts.LinearGradientColorObject#y1 - * @type {number} - */ /** - * End vertical position of the gradient. Float ranges 0-1. - * @name Highcharts.LinearGradientColorObject#y2 - * @type {number} - */ - /** - * Defines the center position and the radius for a gradient. - * - * @interface Highcharts.RadialGradientColorObject - */ /** - * Center horizontal position relative to the shape. Float ranges 0-1. - * @name Highcharts.RadialGradientColorObject#cx - * @type {number} - */ /** - * Center vertical position relative to the shape. Float ranges 0-1. - * @name Highcharts.RadialGradientColorObject#cy - * @type {number} - */ /** - * Radius relative to the shape. Float ranges 0-1. - * @name Highcharts.RadialGradientColorObject#r - * @type {number} - */ - ''; // detach doclets above - /* * + * @param {number} [start=0] + * The starting angle of the arc in radians, where 0 is to the right and + * `-Math.PI/2` is up. * - * Class + * @param {number} [end=0] + * The ending angle of the arc in radians, where 0 is to the right and + * `-Math.PI/2` is up. * - * */ - /* eslint-disable no-invalid-this, valid-jsdoc */ + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ /** + * Draw and return an arc. Overloaded function that takes arguments object. + * + * @function Highcharts.SVGRenderer#arc + * + * @param {Highcharts.SVGAttributes} attribs + * Initial SVG attributes. + * + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ + SVGRenderer.prototype.arc = function (x, y, r, innerR, start, end) { + var arc, options; + if (isObject(x)) { + options = x; + y = options.y; + r = options.r; + innerR = options.innerR; + start = options.start; + end = options.end; + x = options.x; + } else { + options = { + innerR: innerR, + start: start, + end: end, + }; + } + // Arcs are defined as symbols for the ability to set + // attributes in attr and animate + arc = this.symbol("arc", x, y, r, r, options); + arc.r = r; // #959 + return arc; + }; /** - * Handle color operations. Some object methods are chainable. + * Draw and return a rectangle. * - * @class - * @name Highcharts.Color + * @function Highcharts.SVGRenderer#rect * - * @param {Highcharts.ColorType} input - * The input color in either rbga or hex format - */ - var Color = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function Color(input) { - // Collection of parsers. This can be extended from the outside by pushing - // parsers to Highcharts.Color.prototype.parsers. - this.parsers = [{ - // RGBA color - // eslint-disable-next-line max-len - regex: /rgba\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]?(?:\.[0-9]+)?)\s*\)/, - parse: function (result) { - return [ - pInt(result[1]), - pInt(result[2]), - pInt(result[3]), - parseFloat(result[4], 10) - ]; - } - }, { - // RGB color - regex: /rgb\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*\)/, - parse: function (result) { - return [pInt(result[1]), pInt(result[2]), pInt(result[3]), 1]; - } - }]; - this.rgba = []; - // Backwards compatibility, allow class overwrite - if (H.Color !== Color) { - return new H.Color(input); - } - // Backwards compatibility, allow instanciation without new (#13053) - if (!(this instanceof Color)) { - return new Color(input); - } - this.init(input); - } - /* * - * - * Static Functions - * - * */ - /** - * Creates a color instance out of a color string or object. - * - * @function Highcharts.Color.parse - * - * @param {Highcharts.ColorType} input - * The input color in either rbga or hex format. - * - * @return {Highcharts.Color} - * Color instance. - */ - Color.parse = function (input) { - return new Color(input); - }; - /* * - * - * Functions - * - * */ - /** - * Parse the input color to rgba array - * - * @private - * @function Highcharts.Color#init - * - * @param {Highcharts.ColorType} input - * The input color in either rbga or hex format - * - * @return {void} - */ - Color.prototype.init = function (input) { - var result, - rgba, - i, - parser, - len; - this.input = input = Color.names[input && input.toLowerCase ? - input.toLowerCase() : - ''] || input; - // Gradients - if (input && input.stops) { - this.stops = input.stops.map(function (stop) { - return new Color(stop[1]); - }); - // Solid colors - } - else { - // Bitmasking as input[0] is not working for legacy IE. - if (input && - input.charAt && - input.charAt() === '#') { - len = input.length; - input = parseInt(input.substr(1), 16); - // Handle long-form, e.g. #AABBCC - if (len === 7) { - rgba = [ - (input & 0xFF0000) >> 16, - (input & 0xFF00) >> 8, - (input & 0xFF), - 1 - ]; - // Handle short-form, e.g. #ABC - // In short form, the value is assumed to be the same - // for both nibbles for each component. e.g. #ABC = #AABBCC - } - else if (len === 4) { - rgba = [ - (((input & 0xF00) >> 4) | - (input & 0xF00) >> 8), - (((input & 0xF0) >> 4) | - (input & 0xF0)), - ((input & 0xF) << 4) | (input & 0xF), - 1 - ]; - } - } - // Otherwise, check regex parsers - if (!rgba) { - i = this.parsers.length; - while (i-- && !rgba) { - parser = this.parsers[i]; - result = parser.regex.exec(input); - if (result) { - rgba = parser.parse(result); - } - } - } - } - this.rgba = rgba || []; - }; - /** - * Return the color or gradient stops in the specified format - * - * @function Highcharts.Color#get - * - * @param {string} [format] - * Possible values are 'a', 'rgb', 'rgba' (default). - * - * @return {Highcharts.ColorType} - * This color as a string or gradient stops. - */ - Color.prototype.get = function (format) { - var input = this.input, - rgba = this.rgba, - ret; - if (typeof this.stops !== 'undefined') { - ret = merge(input); - ret.stops = [].concat(ret.stops); - this.stops.forEach(function (stop, i) { - ret.stops[i] = [ - ret.stops[i][0], - stop.get(format) - ]; - }); - // it's NaN if gradient colors on a column chart - } - else if (rgba && isNumber(rgba[0])) { - if (format === 'rgb' || (!format && rgba[3] === 1)) { - ret = 'rgb(' + rgba[0] + ',' + rgba[1] + ',' + rgba[2] + ')'; - } - else if (format === 'a') { - ret = rgba[3]; - } - else { - ret = 'rgba(' + rgba.join(',') + ')'; - } - } - else { - ret = input; - } - return ret; - }; - /** - * Brighten the color instance. - * - * @function Highcharts.Color#brighten - * - * @param {number} alpha - * The alpha value. - * - * @return {Highcharts.Color} - * This color with modifications. - */ - Color.prototype.brighten = function (alpha) { - var i, - rgba = this.rgba; - if (this.stops) { - this.stops.forEach(function (stop) { - stop.brighten(alpha); - }); - } - else if (isNumber(alpha) && alpha !== 0) { - for (i = 0; i < 3; i++) { - rgba[i] += pInt(alpha * 255); - if (rgba[i] < 0) { - rgba[i] = 0; - } - if (rgba[i] > 255) { - rgba[i] = 255; - } - } - } - return this; - }; - /** - * Set the color's opacity to a given alpha value. - * - * @function Highcharts.Color#setOpacity - * - * @param {number} alpha - * Opacity between 0 and 1. - * - * @return {Highcharts.Color} - * Color with modifications. - */ - Color.prototype.setOpacity = function (alpha) { - this.rgba[3] = alpha; - return this; - }; - /** - * Return an intermediate color between two colors. - * - * @function Highcharts.Color#tweenTo - * - * @param {Highcharts.Color} to - * The color object to tween to. - * - * @param {number} pos - * The intermediate position, where 0 is the from color (current - * color item), and 1 is the `to` color. - * - * @return {Highcharts.ColorString} - * The intermediate color in rgba notation. - */ - Color.prototype.tweenTo = function (to, pos) { - // Check for has alpha, because rgba colors perform worse due to lack of - // support in WebKit. - var fromRgba = this.rgba, - toRgba = to.rgba, - hasAlpha, - ret; - // Unsupported color, return to-color (#3920, #7034) - if (!toRgba.length || !fromRgba || !fromRgba.length) { - ret = to.input || 'none'; - // Interpolate - } - else { - hasAlpha = (toRgba[3] !== 1 || fromRgba[3] !== 1); - ret = (hasAlpha ? 'rgba(' : 'rgb(') + - Math.round(toRgba[0] + (fromRgba[0] - toRgba[0]) * (1 - pos)) + - ',' + - Math.round(toRgba[1] + (fromRgba[1] - toRgba[1]) * (1 - pos)) + - ',' + - Math.round(toRgba[2] + (fromRgba[2] - toRgba[2]) * (1 - pos)) + - (hasAlpha ? - (',' + - (toRgba[3] + (fromRgba[3] - toRgba[3]) * (1 - pos))) : - '') + - ')'; - } - return ret; - }; - /* * - * - * Static Properties - * - * */ - // Collection of named colors. Can be extended from the outside by adding - // colors to Highcharts.Color.names. - Color.names = { - white: '#ffffff', - black: '#000000' - }; - return Color; - }()); - H.Color = Color; - /** - * Creates a color instance out of a color string. + * @param {number} [x] + * Left position. * - * @function Highcharts.color + * @param {number} [y] + * Top position. * - * @param {Highcharts.ColorType} input - * The input color in either rbga or hex format + * @param {number} [width] + * Width of the rectangle. * - * @return {Highcharts.Color} - * Color instance - */ - H.color = Color.parse; - /* * + * @param {number} [height] + * Height of the rectangle. * - * Export + * @param {number} [r] + * Border corner radius. * - * */ - - return Color; - }); - _registerModule(_modules, 'Core/Animation/Fx.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) { - /* * + * @param {number} [strokeWidth] + * A stroke width can be supplied to allow crisp drawing. * - * (c) 2010-2020 Torstein Honsi + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ /** + * Draw and return a rectangle. + * + * @sample highcharts/members/renderer-rect-on-chart/ + * Draw a rectangle in a chart + * @sample highcharts/members/renderer-rect/ + * Draw a rectangle independent from a chart + * + * @function Highcharts.SVGRenderer#rect + * + * @param {Highcharts.SVGAttributes} [attributes] + * General SVG attributes for the rectangle. + * + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ + SVGRenderer.prototype.rect = function ( + x, + y, + width, + height, + r, + strokeWidth + ) { + r = isObject(x) ? x.r : r; + var wrapper = this.createElement("rect"), + attribs = isObject(x) + ? x + : typeof x === "undefined" + ? {} + : { + x: x, + y: y, + width: Math.max(width, 0), + height: Math.max(height, 0), + }; + if (!this.styledMode) { + if (typeof strokeWidth !== "undefined") { + attribs.strokeWidth = strokeWidth; + attribs = wrapper.crisp(attribs); + } + attribs.fill = "none"; + } + if (r) { + attribs.r = r; + } + wrapper.rSetter = function (value, key, element) { + wrapper.r = value; + attr(element, { + rx: value, + ry: value, + }); + }; + wrapper.rGetter = function () { + return wrapper.r; + }; + return wrapper.attr(attribs); + }; + /** + * Resize the {@link SVGRenderer#box} and re-align all aligned child + * elements. * - * License: www.highcharts.com/license + * @sample highcharts/members/renderer-g/ + * Show and hide grouped objects * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @function Highcharts.SVGRenderer#setSize * - * */ - var win = H.win; - var isNumber = U.isNumber, - objectEach = U.objectEach; - /* eslint-disable no-invalid-this, valid-jsdoc */ + * @param {number} width + * The new pixel width. + * + * @param {number} height + * The new pixel height. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animate=true] + * Whether and how to animate. + */ + SVGRenderer.prototype.setSize = function (width, height, animate) { + var renderer = this, + alignedObjects = renderer.alignedObjects, + i = alignedObjects.length; + renderer.width = width; + renderer.height = height; + renderer.boxWrapper.animate( + { + width: width, + height: height, + }, + { + step: function () { + this.attr({ + viewBox: + "0 0 " + this.attr("width") + " " + this.attr("height"), + }); + }, + duration: pick(animate, true) ? void 0 : 0, + } + ); + while (i--) { + alignedObjects[i].align(); + } + }; /** - * An animator object used internally. One instance applies to one property - * (attribute or style prop) on one element. Animation is always initiated - * through {@link SVGElement#animate}. + * Create and return an svg group element. Child + * {@link Highcharts.SVGElement} objects are added to the group by using the + * group as the first parameter in {@link Highcharts.SVGElement#add|add()}. * - * @example - * var rect = renderer.rect(0, 0, 10, 10).add(); - * rect.animate({ width: 100 }); + * @function Highcharts.SVGRenderer#g * - * @private - * @class - * @name Highcharts.Fx + * @param {string} [name] + * The group will be given a class name of `highcharts-{name}`. This + * can be used for styling and scripting. + * + * @return {Highcharts.SVGElement} + * The generated wrapper element. */ - var Fx = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - /** - * - * @param {Highcharts.HTMLDOMElement|Highcharts.SVGElement} elem - * The element to animate. - * - * @param {Partial<Highcharts.AnimationOptionsObject>} options - * Animation options. - * - * @param {string} prop - * The single attribute or CSS property to animate. - */ - function Fx(elem, options, prop) { - this.pos = NaN; - this.options = options; - this.elem = elem; - this.prop = prop; + SVGRenderer.prototype.g = function (name) { + var elem = this.createElement("g"); + return name ? elem.attr({ class: "highcharts-" + name }) : elem; + }; + /** + * Display an image. + * + * @sample highcharts/members/renderer-image-on-chart/ + * Add an image in a chart + * @sample highcharts/members/renderer-image/ + * Add an image independent of a chart + * + * @function Highcharts.SVGRenderer#image + * + * @param {string} src + * The image source. + * + * @param {number} [x] + * The X position. + * + * @param {number} [y] + * The Y position. + * + * @param {number} [width] + * The image width. If omitted, it defaults to the image file width. + * + * @param {number} [height] + * The image height. If omitted it defaults to the image file + * height. + * + * @param {Function} [onload] + * Event handler for image load. + * + * @return {Highcharts.SVGElement} + * The generated wrapper element. + */ + SVGRenderer.prototype.image = function ( + src, + x, + y, + width, + height, + onload + ) { + var attribs = { preserveAspectRatio: "none" }, + elemWrapper, + dummy, + setSVGImageSource = function (el, src) { + // Set the href in the xlink namespace + if (el.setAttributeNS) { + el.setAttributeNS("http://www.w3.org/1999/xlink", "href", src); + } else { + // could be exporting in IE + // using href throws "not supported" in ie7 and under, + // requries regex shim to fix later + el.setAttribute("hc-svg-href", src); + } + }, + onDummyLoad = function (e) { + setSVGImageSource(elemWrapper.element, src); + onload.call(elemWrapper, e); + }; + // optional properties + if (arguments.length > 1) { + extend(attribs, { + x: x, + y: y, + width: width, + height: height, + }); + } + elemWrapper = this.createElement("image").attr(attribs); + // Add load event if supplied + if (onload) { + // We have to use a dummy HTML image since IE support for SVG image + // load events is very buggy. First set a transparent src, wait for + // dummy to load, and then add the real src to the SVG image. + setSVGImageSource( + elemWrapper.element, + "" /* eslint-disable-line */ + ); + dummy = new win.Image(); + addEvent(dummy, "load", onDummyLoad); + dummy.src = src; + if (dummy.complete) { + onDummyLoad({}); } - /* * - * - * Functions - * - * */ - /** - * Set the current step of a path definition on SVGElement. - * - * @function Highcharts.Fx#dSetter - * - * @return {void} - */ - Fx.prototype.dSetter = function () { - var paths = this.paths, - start = paths && paths[0], - end = paths && paths[1], - path = [], - now = this.now || 0; - // Land on the final path without adjustment points appended in the ends - if (now === 1 || !start || !end) { - path = this.toD || []; - } - else if (start.length === end.length && now < 1) { - for (var i = 0; i < end.length; i++) { - // Tween between the start segment and the end segment. Start - // with a copy of the end segment and tween the appropriate - // numerics - var startSeg = start[i]; - var endSeg = end[i]; - var tweenSeg = []; - for (var j = 0; j < endSeg.length; j++) { - var startItem = startSeg[j]; - var endItem = endSeg[j]; - // Tween numbers - if (typeof startItem === 'number' && - typeof endItem === 'number' && - // Arc boolean flags - !(endSeg[0] === 'A' && (j === 4 || j === 5))) { - tweenSeg[j] = startItem + now * (endItem - startItem); - // Strings, take directly from the end segment - } - else { - tweenSeg[j] = endItem; - } - } - path.push(tweenSeg); - } - // If animation is finished or length not matching, land on right value - } - else { - path = end; - } - this.elem.attr('d', path, void 0, true); - }; - /** - * Update the element with the current animation step. - * - * @function Highcharts.Fx#update - * - * @return {void} - */ - Fx.prototype.update = function () { - var elem = this.elem, - prop = this.prop, // if destroyed, it is null - now = this.now, - step = this.options.step; - // Animation setter defined from outside - if (this[prop + 'Setter']) { - this[prop + 'Setter'](); - // Other animations on SVGElement - } - else if (elem.attr) { - if (elem.element) { - elem.attr(prop, now, null, true); - } - // HTML styles, raw HTML content like container size - } - else { - elem.style[prop] = now + this.unit; - } - if (step) { - step.call(elem, now, this); - } - }; + } else { + setSVGImageSource(elemWrapper.element, src); + } + return elemWrapper; + }; + /** + * Draw a symbol out of pre-defined shape paths from + * {@link SVGRenderer#symbols}. + * It is used in Highcharts for point makers, which cake a `symbol` option, + * and label and button backgrounds like in the tooltip and stock flags. + * + * @function Highcharts.SVGRenderer#symbol + * + * @param {string} symbol + * The symbol name. + * + * @param {number} [x] + * The X coordinate for the top left position. + * + * @param {number} [y] + * The Y coordinate for the top left position. + * + * @param {number} [width] + * The pixel width. + * + * @param {number} [height] + * The pixel height. + * + * @param {Highcharts.SymbolOptionsObject} [options] + * Additional options, depending on the actual symbol drawn. + * + * @return {Highcharts.SVGElement} + */ + SVGRenderer.prototype.symbol = function ( + symbol, + x, + y, + width, + height, + options + ) { + var ren = this, + obj, + imageRegex = /^url\((.*?)\)$/, + isImage = imageRegex.test(symbol), + sym = !isImage && (this.symbols[symbol] ? symbol : "circle"), + // get the symbol definition function + symbolFn = sym && this.symbols[sym], + path, + imageSrc, + centerImage; + if (symbolFn) { + // Check if there's a path defined for this symbol + if (typeof x === "number") { + path = symbolFn.call( + this.symbols, + Math.round(x || 0), + Math.round(y || 0), + width || 0, + height || 0, + options + ); + } + obj = this.path(path); + if (!ren.styledMode) { + obj.attr("fill", "none"); + } + // expando properties for use in animate and attr + extend(obj, { + symbolName: sym, + x: x, + y: y, + width: width, + height: height, + }); + if (options) { + extend(obj, options); + } + // Image symbols + } else if (isImage) { + imageSrc = symbol.match(imageRegex)[1]; + // Create the image synchronously, add attribs async + obj = this.image(imageSrc); + // The image width is not always the same as the symbol width. The + // image may be centered within the symbol, as is the case when + // image shapes are used as label backgrounds, for example in flags. + obj.imgwidth = pick( + symbolSizes[imageSrc] && symbolSizes[imageSrc].width, + options && options.width + ); + obj.imgheight = pick( + symbolSizes[imageSrc] && symbolSizes[imageSrc].height, + options && options.height + ); /** - * Run an animation. - * - * @function Highcharts.Fx#run - * - * @param {number} from - * The current value, value to start from. - * - * @param {number} to - * The end value, value to land on. - * - * @param {string} unit - * The property unit, for example `px`. - * - * @return {void} + * Set the size and position */ - Fx.prototype.run = function (from, to, unit) { - var self = this, - options = self.options, - timer = function (gotoEnd) { - return timer.stopped ? false : self.step(gotoEnd); - }, requestAnimationFrame = win.requestAnimationFrame || - function (step) { - setTimeout(step, 13); - }, step = function () { - for (var i = 0; i < H.timers.length; i++) { - if (!H.timers[i]()) { - H.timers.splice(i--, 1); - } - } - if (H.timers.length) { - requestAnimationFrame(step); - } - }; - if (from === to && !this.elem['forceAnimate:' + this.prop]) { - delete options.curAnim[this.prop]; - if (options.complete && Object.keys(options.curAnim).length === 0) { - options.complete.call(this.elem); - } - } - else { // #7166 - this.startTime = +new Date(); - this.start = from; - this.end = to; - this.unit = unit; - this.now = this.start; - this.pos = 0; - timer.elem = this.elem; - timer.prop = this.prop; - if (timer() && H.timers.push(timer) === 1) { - requestAnimationFrame(step); - } - } + centerImage = function () { + obj.attr({ + width: obj.width, + height: obj.height, + }); }; /** - * Run a single step in the animation. - * - * @function Highcharts.Fx#step - * - * @param {boolean} [gotoEnd] - * Whether to go to the endpoint of the animation after abort. - * - * @return {boolean} - * Returns `true` if animation continues. + * Width and height setters that take both the image's physical size + * and the label size into consideration, and translates the image + * to center within the label. */ - Fx.prototype.step = function (gotoEnd) { - var t = +new Date(), - ret, - done, - options = this.options, - elem = this.elem, - complete = options.complete, - duration = options.duration, - curAnim = options.curAnim; - if (elem.attr && !elem.element) { // #2616, element is destroyed - ret = false; - } - else if (gotoEnd || t >= duration + this.startTime) { - this.now = this.end; - this.pos = 1; - this.update(); - curAnim[this.prop] = true; - done = true; - objectEach(curAnim, function (val) { - if (val !== true) { - done = false; - } + ["width", "height"].forEach(function (key) { + obj[key + "Setter"] = function (value, key) { + var attribs = {}, + imgSize = this["img" + key], + trans = key === "width" ? "translateX" : "translateY"; + this[key] = value; + if (defined(imgSize)) { + // Scale and center the image within its container. + // The name `backgroundSize` is taken from the CSS spec, + // but the value `within` is made up. Other possible + // values in the spec, `cover` and `contain`, can be + // implemented if needed. + if ( + options && + options.backgroundSize === "within" && + this.width && + this.height + ) { + imgSize = Math.round( + imgSize * + Math.min( + this.width / this.imgwidth, + this.height / this.imgheight + ) + ); + } + if (this.element) { + this.element.setAttribute(key, imgSize); + } + if (!this.alignByTranslate) { + attribs[trans] = ((this[key] || 0) - imgSize) / 2; + this.attr(attribs); + } + } + }; + }); + if (defined(x)) { + obj.attr({ + x: x, + y: y, + }); + } + obj.isImg = true; + if (defined(obj.imgwidth) && defined(obj.imgheight)) { + centerImage(); + } else { + // Initialize image to be 0 size so export will still function + // if there's no cached sizes. + obj.attr({ width: 0, height: 0 }); + // Create a dummy JavaScript image to get the width and height. + createElement("img", { + onload: function () { + var chart = charts[ren.chartIndex]; + // Special case for SVGs on IE11, the width is not + // accessible until the image is part of the DOM + // (#2854). + if (this.width === 0) { + css(this, { + position: "absolute", + top: "-999em", }); - if (done && complete) { - complete.call(elem); - } - ret = false; - } - else { - this.pos = options.easing((t - this.startTime) / duration); - this.now = this.start + ((this.end - this.start) * this.pos); - this.update(); - ret = true; - } - return ret; - }; - /** - * Prepare start and end values so that the path can be animated one to one. - * - * @function Highcharts.Fx#initPath - * - * @param {Highcharts.SVGElement} elem - * The SVGElement item. - * - * @param {Highcharts.SVGPathArray|undefined} fromD - * Starting path definition. - * - * @param {Highcharts.SVGPathArray} toD - * Ending path definition. - * - * @return {Array<Highcharts.SVGPathArray,Highcharts.SVGPathArray>} - * An array containing start and end paths in array form so that - * they can be animated in parallel. - */ - Fx.prototype.initPath = function (elem, fromD, toD) { - var shift, - startX = elem.startX, - endX = elem.endX, - fullLength, - i, - start = fromD && fromD.slice(), // copy - end = toD.slice(), // copy - isArea = elem.isArea, - positionFactor = isArea ? 2 : 1, - reverse; - if (!start) { - return [end, end]; - } - /** - * If shifting points, prepend a dummy point to the end path. - * @private - * @param {Highcharts.SVGPathArray} arr - array - * @param {Highcharts.SVGPathArray} other - array - * @return {void} - */ - function prepend(arr, other) { - while (arr.length < fullLength) { - // Move to, line to or curve to? - var moveSegment = arr[0], - otherSegment = other[fullLength - arr.length]; - if (otherSegment && moveSegment[0] === 'M') { - if (otherSegment[0] === 'C') { - arr[0] = [ - 'C', - moveSegment[1], - moveSegment[2], - moveSegment[1], - moveSegment[2], - moveSegment[1], - moveSegment[2] - ]; - } - else { - arr[0] = ['L', moveSegment[1], moveSegment[2]]; - } - } - // Prepend a copy of the first point - arr.unshift(moveSegment); - // For areas, the bottom path goes back again to the left, so we - // need to append a copy of the last point. - if (isArea) { - arr.push(arr[arr.length - 1]); - } - } - } - /** - * Copy and append last point until the length matches the end length. - * @private - * @param {Highcharts.SVGPathArray} arr - array - * @param {Highcharts.SVGPathArray} other - array - * @return {void} - */ - function append(arr, other) { - while (arr.length < fullLength) { - // Pull out the slice that is going to be appended or inserted. - // In a line graph, the positionFactor is 1, and the last point - // is sliced out. In an area graph, the positionFactor is 2, - // causing the middle two points to be sliced out, since an area - // path starts at left, follows the upper path then turns and - // follows the bottom back. - var segmentToAdd = arr[arr.length / positionFactor - 1].slice(); - // Disable the first control point of curve segments - if (segmentToAdd[0] === 'C') { - segmentToAdd[1] = segmentToAdd[5]; - segmentToAdd[2] = segmentToAdd[6]; - } - if (!isArea) { - arr.push(segmentToAdd); - } - else { - var lowerSegmentToAdd = arr[arr.length / positionFactor].slice(); - arr.splice(arr.length / 2, 0, segmentToAdd, lowerSegmentToAdd); - } - } - } - // For sideways animation, find out how much we need to shift to get the - // start path Xs to match the end path Xs. - if (startX && endX) { - for (i = 0; i < startX.length; i++) { - // Moving left, new points coming in on right - if (startX[i] === endX[0]) { - shift = i; - break; - // Moving right - } - else if (startX[0] === - endX[endX.length - startX.length + i]) { - shift = i; - reverse = true; - break; - // Fixed from the right side, "scaling" left - } - else if (startX[startX.length - 1] === - endX[endX.length - startX.length + i]) { - shift = startX.length - i; - break; - } - } - if (typeof shift === 'undefined') { - start = []; - } - } - if (start.length && isNumber(shift)) { - // The common target length for the start and end array, where both - // arrays are padded in opposite ends - fullLength = end.length + shift * positionFactor; - if (!reverse) { - prepend(end, start); - append(start, end); - } - else { - prepend(start, end); - append(end, start); - } - } - return [start, end]; - }; - /** - * Handle animation of the color attributes directly. - * - * @function Highcharts.Fx#fillSetter - * - * @return {void} - */ - Fx.prototype.fillSetter = function () { - Fx.prototype.strokeSetter.apply(this, arguments); - }; - /** - * Handle animation of the color attributes directly. - * - * @function Highcharts.Fx#strokeSetter - * - * @return {void} - */ - Fx.prototype.strokeSetter = function () { - this.elem.attr(this.prop, H.color(this.start).tweenTo(H.color(this.end), this.pos), null, true); - }; - return Fx; - }()); - H.Fx = Fx; - - return Fx; - }); - _registerModule(_modules, 'Core/Animation/AnimationUtilities.js', [_modules['Core/Animation/Fx.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Fx, H, U) { - /* * + doc.body.appendChild(this); + } + // Center the image + symbolSizes[imageSrc] = { + width: this.width, + height: this.height, + }; + obj.imgwidth = this.width; + obj.imgheight = this.height; + if (obj.element) { + centerImage(); + } + // Clean up after #2854 workaround. + if (this.parentNode) { + this.parentNode.removeChild(this); + } + // Fire the load event when all external images are + // loaded + ren.imgCount--; + if (!ren.imgCount && chart && !chart.hasLoaded) { + chart.onload(); + } + }, + src: imageSrc, + }); + this.imgCount++; + } + } + return obj; + }; + /** + * Define a clipping rectangle. The clipping rectangle is later applied + * to {@link SVGElement} objects through the {@link SVGElement#clip} + * function. * - * (c) 2010-2020 Torstein Honsi + * @example + * var circle = renderer.circle(100, 100, 100) + * .attr({ fill: 'red' }) + * .add(); + * var clipRect = renderer.clipRect(100, 100, 100, 100); * - * License: www.highcharts.com/license + * // Leave only the lower right quarter visible + * circle.clip(clipRect); * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @function Highcharts.SVGRenderer#clipRect * - * */ - var defined = U.defined, - getStyle = U.getStyle, - isArray = U.isArray, - isNumber = U.isNumber, - isObject = U.isObject, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick; - /** - * Set the global animation to either a given value, or fall back to the given - * chart's animation option. + * @param {number} [x] * - * @function Highcharts.setAnimation + * @param {number} [y] * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>|undefined} animation - * The animation object. + * @param {number} [width] * - * @param {Highcharts.Chart} chart - * The chart instance. + * @param {number} [height] * - * @todo - * This function always relates to a chart, and sets a property on the renderer, - * so it should be moved to the SVGRenderer. + * @return {Highcharts.ClipRectElement} + * A clipping rectangle. */ - var setAnimation = H.setAnimation = function setAnimation(animation, - chart) { - chart.renderer.globalAnimation = pick(animation, - chart.options.chart.animation, - true); + SVGRenderer.prototype.clipRect = function (x, y, width, height) { + var wrapper, + // Add a hyphen at the end to avoid confusion in testing indexes + // -1 and -10, -11 etc (#6550) + id = uniqueKey() + "-", + clipPath = this.createElement("clipPath") + .attr({ + id: id, + }) + .add(this.defs); + wrapper = this.rect(x, y, width, height, 0).add(clipPath); + wrapper.id = id; + wrapper.clipPath = clipPath; + wrapper.count = 0; + return wrapper; }; /** - * Get the animation in object form, where a disabled animation is always - * returned as `{ duration: 0 }`. + * Draw text. The text can contain a subset of HTML, like spans and anchors + * and some basic text styling of these. For more advanced features like + * border and background, use {@link Highcharts.SVGRenderer#label} instead. + * To update the text after render, run `text.attr({ text: 'New text' })`. * - * @function Highcharts.animObject - * - * @param {boolean|Highcharts.AnimationOptionsObject} [animation=0] - * An animation setting. Can be an object with duration, complete and - * easing properties, or a boolean to enable or disable. + * @sample highcharts/members/renderer-text-on-chart/ + * Annotate the chart freely + * @sample highcharts/members/renderer-on-chart/ + * Annotate with a border and in response to the data + * @sample highcharts/members/renderer-text/ + * Formatted text * - * @return {Highcharts.AnimationOptionsObject} - * An object with at least a duration property. - */ - var animObject = H.animObject = function animObject(animation) { - return isObject(animation) ? - H.merge({ duration: 500, - defer: 0 }, - animation) : - { duration: animation ? 500 : 0, - defer: 0 }; + * @function Highcharts.SVGRenderer#text + * + * @param {string} [str] + * The text of (subset) HTML to draw. + * + * @param {number} [x] + * The x position of the text's lower left corner. + * + * @param {number} [y] + * The y position of the text's lower left corner. + * + * @param {boolean} [useHTML=false] + * Use HTML to render the text. + * + * @return {Highcharts.SVGElement} + * The text object. + */ + SVGRenderer.prototype.text = function (str, x, y, useHTML) { + // declare variables + var renderer = this, + wrapper, + attribs = {}; + if (useHTML && (renderer.allowHTML || !renderer.forExport)) { + return renderer.html(str, x, y); + } + attribs.x = Math.round(x || 0); // X always needed for line-wrap logic + if (y) { + attribs.y = Math.round(y); + } + if (defined(str)) { + attribs.text = str; + } + wrapper = renderer.createElement("text").attr(attribs); + if (!useHTML) { + wrapper.xSetter = function (value, key, element) { + var tspans = element.getElementsByTagName("tspan"), + tspan, + parentVal = element.getAttribute(key), + i; + for (i = 0; i < tspans.length; i++) { + tspan = tspans[i]; + // If the x values are equal, the tspan represents a + // linebreak + if (tspan.getAttribute(key) === parentVal) { + tspan.setAttribute(key, value); + } + } + element.setAttribute(key, value); + }; + } + return wrapper; }; /** - * Get the defer as a number value from series animation options. - * - * @function Highcharts.getDeferredAnimation - * - * @param {Highcharts.Chart} chart - * The chart instance. - * - * @param {boolean|Highcharts.AnimationOptionsObject} animation - * An animation setting. Can be an object with duration, complete and - * easing properties, or a boolean to enable or disable. - * - * @param {Highcharts.Series} [series] - * Series to defer animation. - * - * @return {number} - * The numeric value. - */ - var getDeferredAnimation = H.getDeferredAnimation = function (chart, - animation, - series) { - var labelAnimation = animObject(animation); - var s = series ? [series] : chart.series; - var defer = 0; - var duration = 0; - s.forEach(function (series) { - var seriesAnim = animObject(series.options.animation); - defer = animation && defined(animation.defer) ? - labelAnimation.defer : - Math.max(defer, seriesAnim.duration + seriesAnim.defer); - duration = Math.min(labelAnimation.duration, seriesAnim.duration); - }); - // Disable defer for exporting - if (chart.renderer.forExport) { - defer = 0; - } - var anim = { - defer: Math.max(0, - defer - duration), - duration: Math.min(defer, - duration) - }; - return anim; + * Utility to return the baseline offset and total line height from the font + * size. + * + * @function Highcharts.SVGRenderer#fontMetrics + * + * @param {number|string} [fontSize] + * The current font size to inspect. If not given, the font size + * will be found from the DOM element. + * + * @param {Highcharts.SVGElement|Highcharts.SVGDOMElement} [elem] + * The element to inspect for a current font size. + * + * @return {Highcharts.FontMetricsObject} + * The font metrics. + */ + SVGRenderer.prototype.fontMetrics = function (fontSize, elem) { + var lineHeight, baseline; + if ( + (this.styledMode || !/px/.test(fontSize)) && + win.getComputedStyle // old IE doesn't support it + ) { + fontSize = + elem && SVGElement.prototype.getStyle.call(elem, "font-size"); + } else { + fontSize = + fontSize || + // When the elem is a DOM element (#5932) + (elem && elem.style && elem.style.fontSize) || + // Fall back on the renderer style default + (this.style && this.style.fontSize); + } + // Handle different units + if (/px/.test(fontSize)) { + fontSize = pInt(fontSize); + } else { + fontSize = 12; + } + // Empirical values found by comparing font size and bounding box + // height. Applies to the default font family. + // https://jsfiddle.net/highcharts/7xvn7/ + lineHeight = + fontSize < 24 ? fontSize + 3 : Math.round(fontSize * 1.2); + baseline = Math.round(lineHeight * 0.8); + return { + h: lineHeight, + b: baseline, + f: fontSize, + }; }; /** - * The global animate method, which uses Fx to create individual animators. + * Correct X and Y positioning of a label for rotation (#1764). * - * @function Highcharts.animate + * @private + * @function Highcharts.SVGRenderer#rotCorr * - * @param {Highcharts.HTMLDOMElement|Highcharts.SVGElement} el - * The element to animate. + * @param {number} baseline * - * @param {Highcharts.CSSObject|Highcharts.SVGAttributes} params - * An object containing key-value pairs of the properties to animate. - * Supports numeric as pixel-based CSS properties for HTML objects and - * attributes for SVGElements. + * @param {number} rotation * - * @param {Partial<Highcharts.AnimationOptionsObject>} [opt] - * Animation options. + * @param {boolean} [alterY] * - * @return {void} + * @param {Highcharts.PositionObject} */ - var animate = H.animate = function (el, - params, - opt) { - var start, - unit = '', - end, - fx, - args; - if (!isObject(opt)) { // Number or undefined/null - args = arguments; - opt = { - duration: args[2], - easing: args[3], - complete: args[4] - }; - } - if (!isNumber(opt.duration)) { - opt.duration = 400; - } - opt.easing = typeof opt.easing === 'function' ? - opt.easing : - (Math[opt.easing] || Math.easeInOutSine); - opt.curAnim = merge(params); - objectEach(params, function (val, prop) { - // Stop current running animation of this property - stop(el, prop); - fx = new Fx(el, opt, prop); - end = null; - if (prop === 'd' && isArray(params.d)) { - fx.paths = fx.initPath(el, el.pathArray, params.d); - fx.toD = params.d; - start = 0; - end = 1; - } - else if (el.attr) { - start = el.attr(prop); - } - else { - start = parseFloat(getStyle(el, prop)) || 0; - if (prop !== 'opacity') { - unit = 'px'; - } - } - if (!end) { - end = val; - } - if (end && end.match && end.match('px')) { - end = end.replace(/px/g, ''); // #4351 - } - fx.run(start, end, unit); - }); + SVGRenderer.prototype.rotCorr = function (baseline, rotation, alterY) { + var y = baseline; + if (rotation && alterY) { + y = Math.max(y * Math.cos(rotation * deg2rad), 4); + } + return { + x: (-baseline / 3) * Math.sin(rotation * deg2rad), + y: y, + }; }; /** - * Stop running animation. + * Compatibility function to convert the legacy one-dimensional path array + * into an array of segments. * - * @function Highcharts.stop - * - * @param {Highcharts.SVGElement} el - * The SVGElement to stop animation on. - * - * @param {string} [prop] - * The property to stop animating. If given, the stop method will stop a - * single property from animating, while others continue. - * - * @return {void} + * It is used in maps to parse the `path` option, and in SVGRenderer.dSetter + * to support legacy paths from demos. * - * @todo - * A possible extension to this would be to stop a single property, when - * we want to continue animating others. Then assign the prop to the timer - * in the Fx.run method, and check for the prop here. This would be an - * improvement in all cases where we stop the animation from .attr. Instead of - * stopping everything, we can just stop the actual attributes we're setting. - */ - var stop = H.stop = function (el, - prop) { - var i = H.timers.length; - // Remove timers related to this element (#4519) - while (i--) { - if (H.timers[i].elem === el && (!prop || prop === H.timers[i].prop)) { - H.timers[i].stopped = true; // #4667 - } + * @private + * @function Highcharts.SVGRenderer#pathToSegments + */ + SVGRenderer.prototype.pathToSegments = function (path) { + var ret = []; + var segment = []; + var commandLength = { + A: 8, + C: 7, + H: 2, + L: 3, + M: 3, + Q: 5, + S: 5, + T: 3, + V: 2, + }; + // Short, non-typesafe parsing of the one-dimensional array. It splits + // the path on any string. This is not type checked against the tuple + // types, but is shorter, and doesn't require specific checks for any + // command type in SVG. + for (var i = 0; i < path.length; i++) { + // Command skipped, repeat previous or insert L/l for M/m + if ( + isString(segment[0]) && + isNumber(path[i]) && + segment.length === commandLength[segment[0].toUpperCase()] + ) { + path.splice(i, 0, segment[0].replace("M", "L").replace("m", "l")); } - }; - var animationExports = { - animate: animate, - animObject: animObject, - getDeferredAnimation: getDeferredAnimation, - setAnimation: setAnimation, - stop: stop - }; - - return animationExports; - }); - _registerModule(_modules, 'Core/Renderer/SVG/SVGElement.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Color/Color.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (A, Color, H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var animate = A.animate, - animObject = A.animObject, - stop = A.stop; - var deg2rad = H.deg2rad, - doc = H.doc, - hasTouch = H.hasTouch, - isFirefox = H.isFirefox, - noop = H.noop, - svg = H.svg, - SVG_NS = H.SVG_NS, - win = H.win; - var attr = U.attr, - createElement = U.createElement, - css = U.css, - defined = U.defined, - erase = U.erase, - extend = U.extend, - fireEvent = U.fireEvent, - isArray = U.isArray, - isFunction = U.isFunction, - isNumber = U.isNumber, - isString = U.isString, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick, - pInt = U.pInt, - syncTimeout = U.syncTimeout, - uniqueKey = U.uniqueKey; - /** - * The horizontal alignment of an element. - * - * @typedef {"center"|"left"|"right"} Highcharts.AlignValue - */ - /** - * Options to align the element relative to the chart or another box. - * - * @interface Highcharts.AlignObject - */ /** - * Horizontal alignment. Can be one of `left`, `center` and `right`. - * - * @name Highcharts.AlignObject#align - * @type {Highcharts.AlignValue|undefined} - * - * @default left - */ /** - * Vertical alignment. Can be one of `top`, `middle` and `bottom`. - * - * @name Highcharts.AlignObject#verticalAlign - * @type {Highcharts.VerticalAlignValue|undefined} - * - * @default top - */ /** - * Horizontal pixel offset from alignment. - * - * @name Highcharts.AlignObject#x - * @type {number|undefined} - * - * @default 0 - */ /** - * Vertical pixel offset from alignment. - * - * @name Highcharts.AlignObject#y - * @type {number|undefined} - * - * @default 0 - */ /** - * Use the `transform` attribute with translateX and translateY custom - * attributes to align this elements rather than `x` and `y` attributes. - * - * @name Highcharts.AlignObject#alignByTranslate - * @type {boolean|undefined} - * - * @default false - */ - /** - * Bounding box of an element. - * - * @interface Highcharts.BBoxObject - * @extends Highcharts.PositionObject - */ /** - * Height of the bounding box. - * - * @name Highcharts.BBoxObject#height - * @type {number} - */ /** - * Width of the bounding box. - * - * @name Highcharts.BBoxObject#width - * @type {number} - */ /** - * Horizontal position of the bounding box. - * - * @name Highcharts.BBoxObject#x - * @type {number} - */ /** - * Vertical position of the bounding box. - * - * @name Highcharts.BBoxObject#y - * @type {number} - */ - /** - * An object of key-value pairs for SVG attributes. Attributes in Highcharts - * elements for the most parts correspond to SVG, but some are specific to - * Highcharts, like `zIndex`, `rotation`, `rotationOriginX`, - * `rotationOriginY`, `translateX`, `translateY`, `scaleX` and `scaleY`. SVG - * attributes containing a hyphen are _not_ camel-cased, they should be - * quoted to preserve the hyphen. - * - * @example - * { - * 'stroke': '#ff0000', // basic - * 'stroke-width': 2, // hyphenated - * 'rotation': 45 // custom - * 'd': ['M', 10, 10, 'L', 30, 30, 'z'] // path definition, note format - * } - * - * @interface Highcharts.SVGAttributes - */ /** - * @name Highcharts.SVGAttributes#[key:string] - * @type {*} - */ /** - * @name Highcharts.SVGAttributes#d - * @type {string|Highcharts.SVGPathArray|undefined} - */ /** - * @name Highcharts.SVGAttributes#fill - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} - */ /** - * @name Highcharts.SVGAttributes#inverted - * @type {boolean|undefined} - */ /** - * @name Highcharts.SVGAttributes#matrix - * @type {Array<number>|undefined} - */ /** - * @name Highcharts.SVGAttributes#rotation - * @type {number|undefined} - */ /** - * @name Highcharts.SVGAttributes#rotationOriginX - * @type {number|undefined} - */ /** - * @name Highcharts.SVGAttributes#rotationOriginY - * @type {number|undefined} - */ /** - * @name Highcharts.SVGAttributes#scaleX - * @type {number|undefined} - */ /** - * @name Highcharts.SVGAttributes#scaleY - * @type {number|undefined} - */ /** - * @name Highcharts.SVGAttributes#stroke - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} - */ /** - * @name Highcharts.SVGAttributes#style - * @type {string|Highcharts.CSSObject|undefined} - */ /** - * @name Highcharts.SVGAttributes#translateX - * @type {number|undefined} - */ /** - * @name Highcharts.SVGAttributes#translateY - * @type {number|undefined} - */ /** - * @name Highcharts.SVGAttributes#zIndex - * @type {number|undefined} - */ - /** - * An SVG DOM element. The type is a reference to the regular SVGElement in the - * global scope. - * - * @typedef {globals.GlobalSVGElement} Highcharts.SVGDOMElement - * - * @see https://developer.mozilla.org/en-US/docs/Web/API/SVGElement - */ - /** - * The vertical alignment of an element. - * - * @typedef {"bottom"|"middle"|"top"} Highcharts.VerticalAlignValue - */ - ''; // detach doclets above - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The SVGElement prototype is a JavaScript wrapper for SVG elements used in the - * rendering layer of Highcharts. Combined with the - * {@link Highcharts.SVGRenderer} - * object, these prototypes allow freeform annotation in the charts or even in - * HTML pages without instanciating a chart. The SVGElement can also wrap HTML - * labels, when `text` or `label` elements are created with the `useHTML` - * parameter. - * - * The SVGElement instances are created through factory functions on the - * {@link Highcharts.SVGRenderer} - * object, like - * {@link Highcharts.SVGRenderer#rect|rect}, - * {@link Highcharts.SVGRenderer#path|path}, - * {@link Highcharts.SVGRenderer#text|text}, - * {@link Highcharts.SVGRenderer#label|label}, - * {@link Highcharts.SVGRenderer#g|g} - * and more. - * - * @class - * @name Highcharts.SVGElement - */ - var SVGElement = /** @class */ (function () { - function SVGElement() { - /* * - * - * Properties - * - * */ - this.element = void 0; - this.height = void 0; - this.opacity = 1; // Default base for animation - this.renderer = void 0; - this.SVG_NS = SVG_NS; - // Custom attributes used for symbols, these should be filtered out when - // setting SVGElement attributes (#9375). - this.symbolCustomAttribs = [ - 'x', - 'y', - 'width', - 'height', - 'r', - 'start', - 'end', - 'innerR', - 'anchorX', - 'anchorY', - 'rounded' - ]; - this.width = void 0; + // Split on string + if (typeof path[i] === "string") { + if (segment.length) { + ret.push(segment.slice(0)); + } + segment.length = 0; } - /* * - * - * Functions - * - * */ - /** - * Get the current value of an attribute or pseudo attribute, - * used mainly for animation. Called internally from - * the {@link Highcharts.SVGRenderer#attr} function. - * - * @private - * @function Highcharts.SVGElement#_defaultGetter - * - * @param {string} key - * Property key. - * - * @return {number|string} - * Property value. - */ - SVGElement.prototype._defaultGetter = function (key) { - var ret = pick(this[key + 'Value'], // align getter - this[key], - this.element ? this.element.getAttribute(key) : null, 0); - if (/^[\-0-9\.]+$/.test(ret)) { // is numerical - ret = parseFloat(ret); - } - return ret; - }; - /** - * @private - * @function Highcharts.SVGElement#_defaultSetter - * - * @param {string} value - * - * @param {string} key - * - * @param {Highcharts.SVGDOMElement} element - * - * @return {void} - */ - SVGElement.prototype._defaultSetter = function (value, key, element) { - element.setAttribute(key, value); - }; - /** - * Add the element to the DOM. All elements must be added this way. - * - * @sample highcharts/members/renderer-g - * Elements added to a group - * - * @function Highcharts.SVGElement#add - * - * @param {Highcharts.SVGElement} [parent] - * The parent item to add it to. If undefined, the element is added - * to the {@link Highcharts.SVGRenderer.box}. - * - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. - */ - SVGElement.prototype.add = function (parent) { - var renderer = this.renderer, - element = this.element, - inserted; - if (parent) { - this.parentGroup = parent; - } - // Mark as inverted - this.parentInverted = parent && parent.inverted; - // Build formatted text - if (typeof this.textStr !== 'undefined' && - this.element.nodeName === 'text' // Not for SVGLabel instances - ) { - renderer.buildText(this); - } - // Mark as added - this.added = true; - // If we're adding to renderer root, or other elements in the group - // have a z index, we need to handle it - if (!parent || parent.handleZ || this.zIndex) { - inserted = this.zIndexSetter(); - } - // If zIndex is not handled, append at the end - if (!inserted) { - (parent ? - parent.element : - renderer.box).appendChild(element); - } - // fire an event for internal hooks - if (this.onAdd) { - this.onAdd(); - } - return this; - }; - /** - * Add a class name to an element. - * - * @function Highcharts.SVGElement#addClass - * - * @param {string} className - * The new class name to add. - * - * @param {boolean} [replace=false] - * When true, the existing class name(s) will be overwritten with the new - * one. When false, the new one is added. - * - * @return {Highcharts.SVGElement} - * Return the SVG element for chainability. - */ - SVGElement.prototype.addClass = function (className, replace) { - var currentClassName = replace ? '' : (this.attr('class') || ''); - // Trim the string and remove duplicates - className = (className || '') - .split(/ /g) - .reduce(function (newClassName, name) { - if (currentClassName.indexOf(name) === -1) { - newClassName.push(name); - } - return newClassName; - }, (currentClassName ? - [currentClassName] : - [])) - .join(' '); - if (className !== currentClassName) { - this.attr('class', className); - } - return this; - }; - /** - * This method is executed in the end of `attr()`, after setting all - * attributes in the hash. In can be used to efficiently consolidate - * multiple attributes in one SVG property -- e.g., translate, rotate and - * scale are merged in one "transform" attribute in the SVG node. - * - * @private - * @function Highcharts.SVGElement#afterSetters - */ - SVGElement.prototype.afterSetters = function () { - // Update transform. Do this outside the loop to prevent redundant - // updating for batch setting of attributes. - if (this.doTransform) { - this.updateTransform(); - this.doTransform = false; - } - }; - /** - * Align the element relative to the chart or another box. - * - * @function Highcharts.SVGElement#align - * - * @param {Highcharts.AlignObject} [alignOptions] - * The alignment options. The function can be called without this - * parameter in order to re-align an element after the box has been - * updated. - * - * @param {boolean} [alignByTranslate] - * Align element by translation. - * - * @param {string|Highcharts.BBoxObject} [box] - * The box to align to, needs a width and height. When the box is a - * string, it refers to an object in the Renderer. For example, when - * box is `spacingBox`, it refers to `Renderer.spacingBox` which - * holds `width`, `height`, `x` and `y` properties. - * - * @return {Highcharts.SVGElement} Returns the SVGElement for chaining. - */ - SVGElement.prototype.align = function (alignOptions, alignByTranslate, box) { - var align, - vAlign, - x, - y, - attribs = {}, - alignTo, - renderer = this.renderer, - alignedObjects = renderer.alignedObjects, - alignFactor, - vAlignFactor; - // First call on instanciate - if (alignOptions) { - this.alignOptions = alignOptions; - this.alignByTranslate = alignByTranslate; - if (!box || isString(box)) { - this.alignTo = alignTo = box || 'renderer'; - // prevent duplicates, like legendGroup after resize - erase(alignedObjects, this); - alignedObjects.push(this); - box = void 0; // reassign it below + segment.push(path[i]); + } + ret.push(segment.slice(0)); + return ret; + /* + // Fully type-safe version where each tuple type is checked. The + // downside is filesize and a lack of flexibility for unsupported + // commands + const ret: SVGPath = [], + commands = { + A: 7, + C: 6, + H: 1, + L: 2, + M: 2, + Q: 4, + S: 4, + T: 2, + V: 1, + Z: 0 + }; + + let i = 0, + lastI = 0, + lastCommand; + + while (i < path.length) { + const item = path[i]; + + let command; + + if (typeof item === 'string') { + command = item; + i += 1; + } else { + command = lastCommand || 'M'; } - // When called on resize, no arguments are supplied - } - else { - alignOptions = this.alignOptions; - alignByTranslate = this.alignByTranslate; - alignTo = this.alignTo; - } - box = pick(box, renderer[alignTo], renderer); - // Assign variables - align = alignOptions.align; - vAlign = alignOptions.verticalAlign; - // default: left align - x = (box.x || 0) + (alignOptions.x || 0); - // default: top align - y = (box.y || 0) + (alignOptions.y || 0); - // Align - if (align === 'right') { - alignFactor = 1; - } - else if (align === 'center') { - alignFactor = 2; - } - if (alignFactor) { - x += (box.width - (alignOptions.width || 0)) / - alignFactor; - } - attribs[alignByTranslate ? 'translateX' : 'x'] = Math.round(x); - // Vertical align - if (vAlign === 'bottom') { - vAlignFactor = 1; - } - else if (vAlign === 'middle') { - vAlignFactor = 2; - } - if (vAlignFactor) { - y += (box.height - (alignOptions.height || 0)) / - vAlignFactor; + + // Upper case + const commandUC = command.toUpperCase(); + + if (commandUC in commands) { + + // No numeric parameters + if (command === 'Z' || command === 'z') { + ret.push([command]); + + // One numeric parameter + } else { + const val0 = path[i]; + if (typeof val0 === 'number') { + + // Horizontal line to + if (command === 'H' || command === 'h') { + ret.push([command, val0]); + i += 1; + + // Vertical line to + } else if (command === 'V' || command === 'v') { + ret.push([command, val0]); + i += 1; + + // Two numeric parameters + } else { + const val1 = path[i + 1]; + if (typeof val1 === 'number') { + // lineTo + if (command === 'L' || command === 'l') { + ret.push([command, val0, val1]); + i += 2; + + // moveTo + } else if (command === 'M' || command === 'm') { + ret.push([command, val0, val1]); + i += 2; + + // Smooth quadratic bezier + } else if (command === 'T' || command === 't') { + ret.push([command, val0, val1]); + i += 2; + + // Four numeric parameters + } else { + const val2 = path[i + 2], + val3 = path[i + 3]; + if ( + typeof val2 === 'number' && + typeof val3 === 'number' + ) { + // Quadratic bezier to + if ( + command === 'Q' || + command === 'q' + ) { + ret.push([ + command, + val0, + val1, + val2, + val3 + ]); + i += 4; + + // Smooth cubic bezier to + } else if ( + command === 'S' || + command === 's' + ) { + ret.push([ + command, + val0, + val1, + val2, + val3 + ]); + i += 4; + + // Six numeric parameters + } else { + const val4 = path[i + 4], + val5 = path[i + 5]; + + if ( + typeof val4 === 'number' && + typeof val5 === 'number' + ) { + // Curve to + if ( + command === 'C' || + command === 'c' + ) { + ret.push([ + command, + val0, + val1, + val2, + val3, + val4, + val5 + ]); + i += 6; + + // Seven numeric parameters + } else { + const val6 = path[i + 6]; + + // Arc to + if ( + typeof val6 === + 'number' && + ( + command === 'A' || + command === 'a' + ) + ) { + ret.push([ + command, + val0, + val1, + val2, + val3, + val4, + val5, + val6 + ]); + i += 7; + + } + + } + } + } + } + } + } + + } + } + } + } + + // An unmarked command following a moveTo is a lineTo + lastCommand = command === 'M' ? 'L' : command; + + if (i === lastI) { + break; + } + lastI = i; } - attribs[alignByTranslate ? 'translateY' : 'y'] = Math.round(y); - // Animate only if already placed - this[this.placed ? 'animate' : 'attr'](attribs); - this.placed = true; - this.alignAttr = attribs; - return this; - }; + return ret; + */ + }; + /** + * Draw a label, which is an extended text element with support for border + * and background. Highcharts creates a `g` element with a text and a `path` + * or `rect` inside, to make it behave somewhat like a HTML div. Border and + * background are set through `stroke`, `stroke-width` and `fill` attributes + * using the {@link Highcharts.SVGElement#attr|attr} method. To update the + * text after render, run `label.attr({ text: 'New text' })`. + * + * @sample highcharts/members/renderer-label-on-chart/ + * A label on the chart + * + * @function Highcharts.SVGRenderer#label + * + * @param {string} str + * The initial text string or (subset) HTML to render. + * + * @param {number} x + * The x position of the label's left side. + * + * @param {number} [y] + * The y position of the label's top side or baseline, depending on + * the `baseline` parameter. + * + * @param {string} [shape='rect'] + * The shape of the label's border/background, if any. Defaults to + * `rect`. Other possible values are `callout` or other shapes + * defined in {@link Highcharts.SVGRenderer#symbols}. + * + * @param {number} [anchorX] + * In case the `shape` has a pointer, like a flag, this is the + * coordinates it should be pinned to. + * + * @param {number} [anchorY] + * In case the `shape` has a pointer, like a flag, this is the + * coordinates it should be pinned to. + * + * @param {boolean} [useHTML=false] + * Wether to use HTML to render the label. + * + * @param {boolean} [baseline=false] + * Whether to position the label relative to the text baseline, + * like {@link Highcharts.SVGRenderer#text|renderer.text}, or to the + * upper border of the rectangle. + * + * @param {string} [className] + * Class name for the group. + * + * @return {Highcharts.SVGElement} + * The generated label. + */ + SVGRenderer.prototype.label = function ( + str, + x, + y, + shape, + anchorX, + anchorY, + useHTML, + baseline, + className + ) { + return new SVGLabel( + this, + str, + x, + y, + shape, + anchorX, + anchorY, + useHTML, + baseline, + className + ); + }; + return SVGRenderer; + })(); + /** + * A pointer to the renderer's associated Element class. The VMLRenderer + * will have a pointer to VMLElement here. + * + * @name Highcharts.SVGRenderer#Element + * @type {Highcharts.SVGElement} + */ + SVGRenderer.prototype.Element = SVGElement; + /** + * @private + */ + SVGRenderer.prototype.SVG_NS = SVG_NS; + /** + * Dummy function for plugins, called every time the renderer is updated. + * Prior to Highcharts 5, this was used for the canvg renderer. + * + * @deprecated + * @function Highcharts.SVGRenderer#draw + */ + SVGRenderer.prototype.draw = noop; + /** + * A collection of characters mapped to HTML entities. When `useHTML` on an + * element is true, these entities will be rendered correctly by HTML. In + * the SVG pseudo-HTML, they need to be unescaped back to simple characters, + * so for example `<` will render as `<`. + * + * @example + * // Add support for unescaping quotes + * Highcharts.SVGRenderer.prototype.escapes['"'] = '"'; + * + * @name Highcharts.SVGRenderer#escapes + * @type {Highcharts.Dictionary<string>} + */ + SVGRenderer.prototype.escapes = { + "&": "&", + "<": "<", + ">": ">", + "'": "'", + '"': """, + }; + /** + * An extendable collection of functions for defining symbol paths. + * + * @name Highcharts.SVGRenderer#symbols + * @type {Highcharts.SymbolDictionary} + */ + SVGRenderer.prototype.symbols = { + circle: function (x, y, w, h) { + // Return a full arc + return this.arc(x + w / 2, y + h / 2, w / 2, h / 2, { + start: Math.PI * 0.5, + end: Math.PI * 2.5, + open: false, + }); + }, + square: function (x, y, w, h) { + return [ + ["M", x, y], + ["L", x + w, y], + ["L", x + w, y + h], + ["L", x, y + h], + ["Z"], + ]; + }, + triangle: function (x, y, w, h) { + return [ + ["M", x + w / 2, y], + ["L", x + w, y + h], + ["L", x, y + h], + ["Z"], + ]; + }, + "triangle-down": function (x, y, w, h) { + return [["M", x, y], ["L", x + w, y], ["L", x + w / 2, y + h], ["Z"]]; + }, + diamond: function (x, y, w, h) { + return [ + ["M", x + w / 2, y], + ["L", x + w, y + h / 2], + ["L", x + w / 2, y + h], + ["L", x, y + h / 2], + ["Z"], + ]; + }, + arc: function (x, y, w, h, options) { + var arc = []; + if (options) { + var start = options.start || 0, + end = options.end || 0, + rx = options.r || w, + ry = options.r || h || w, + proximity = 0.001, + fullCircle = Math.abs(end - start - 2 * Math.PI) < proximity, + // Substract a small number to prevent cos and sin of start and + // end from becoming equal on 360 arcs (related: #1561) + end = end - proximity, + innerRadius = options.innerR, + open = pick(options.open, fullCircle), + cosStart = Math.cos(start), + sinStart = Math.sin(start), + cosEnd = Math.cos(end), + sinEnd = Math.sin(end), + // Proximity takes care of rounding errors around PI (#6971) + longArc = pick( + options.longArc, + end - start - Math.PI < proximity ? 0 : 1 + ); + arc.push( + ["M", x + rx * cosStart, y + ry * sinStart], + [ + "A", + rx, + ry, + 0, + longArc, + pick(options.clockwise, 1), + x + rx * cosEnd, + y + ry * sinEnd, + ] + ); + if (defined(innerRadius)) { + arc.push( + open + ? ["M", x + innerRadius * cosEnd, y + innerRadius * sinEnd] + : ["L", x + innerRadius * cosEnd, y + innerRadius * sinEnd], + [ + "A", + innerRadius, + innerRadius, + 0, + longArc, + // Clockwise - opposite to the outer arc clockwise + defined(options.clockwise) ? 1 - options.clockwise : 0, + x + innerRadius * cosStart, + y + innerRadius * sinStart, + ] + ); + } + if (!open) { + arc.push(["Z"]); + } + } + return arc; + }, + /** + * Callout shape used for default tooltips, also used for rounded + * rectangles in VML + */ + callout: function (x, y, w, h, options) { + var arrowLength = 6, + halfDistance = 6, + r = Math.min((options && options.r) || 0, w, h), + safeDistance = r + halfDistance, + anchorX = (options && options.anchorX) || 0, + anchorY = (options && options.anchorY) || 0, + path; + path = [ + ["M", x + r, y], + ["L", x + w - r, y], + ["C", x + w, y, x + w, y, x + w, y + r], + ["L", x + w, y + h - r], + ["C", x + w, y + h, x + w, y + h, x + w - r, y + h], + ["L", x + r, y + h], + ["C", x, y + h, x, y + h, x, y + h - r], + ["L", x, y + r], + ["C", x, y, x, y, x + r, y], // top-left corner + ]; + // Anchor on right side + if (anchorX && anchorX > w) { + // Chevron + if (anchorY > y + safeDistance && anchorY < y + h - safeDistance) { + path.splice( + 3, + 1, + ["L", x + w, anchorY - halfDistance], + ["L", x + w + arrowLength, anchorY], + ["L", x + w, anchorY + halfDistance], + ["L", x + w, y + h - r] + ); + // Simple connector + } else { + path.splice( + 3, + 1, + ["L", x + w, h / 2], + ["L", anchorX, anchorY], + ["L", x + w, h / 2], + ["L", x + w, y + h - r] + ); + } + // Anchor on left side + } else if (anchorX && anchorX < 0) { + // Chevron + if (anchorY > y + safeDistance && anchorY < y + h - safeDistance) { + path.splice( + 7, + 1, + ["L", x, anchorY + halfDistance], + ["L", x - arrowLength, anchorY], + ["L", x, anchorY - halfDistance], + ["L", x, y + r] + ); + // Simple connector + } else { + path.splice( + 7, + 1, + ["L", x, h / 2], + ["L", anchorX, anchorY], + ["L", x, h / 2], + ["L", x, y + r] + ); + } + } else if ( + // replace bottom + anchorY && + anchorY > h && + anchorX > x + safeDistance && + anchorX < x + w - safeDistance + ) { + path.splice( + 5, + 1, + ["L", anchorX + halfDistance, y + h], + ["L", anchorX, y + h + arrowLength], + ["L", anchorX - halfDistance, y + h], + ["L", x + r, y + h] + ); + } else if ( + // replace top + anchorY && + anchorY < 0 && + anchorX > x + safeDistance && + anchorX < x + w - safeDistance + ) { + path.splice( + 1, + 1, + ["L", anchorX - halfDistance, y], + ["L", anchorX, y - arrowLength], + ["L", anchorX + halfDistance, y], + ["L", w - r, y] + ); + } + return path; + }, + }; + H.SVGRenderer = SVGRenderer; + H.Renderer = H.SVGRenderer; + + return H.Renderer; + } + ); + _registerModule( + _modules, + "Core/Renderer/HTML/HTMLElement.js", + [ + _modules["Core/Globals.js"], + _modules["Core/Renderer/SVG/SVGElement.js"], + _modules["Core/Utilities.js"], + ], + function (H, SVGElement, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var css = U.css, + defined = U.defined, + extend = U.extend, + pick = U.pick, + pInt = U.pInt; + /** + * Element placebo + * @private + */ + var HTMLElement = SVGElement; + var isFirefox = H.isFirefox; + /* eslint-disable valid-jsdoc */ + // Extend SvgElement for useHTML option. + extend( + HTMLElement.prototype, + /** @lends SVGElement.prototype */ { + /** + * Apply CSS to HTML elements. This is used in text within SVG rendering and + * by the VML renderer + * + * @private + * @function Highcharts.SVGElement#htmlCss + * + * @param {Highcharts.CSSObject} styles + * + * @return {Highcharts.SVGElement} + */ + htmlCss: function (styles) { + var wrapper = this, + element = wrapper.element, + // When setting or unsetting the width style, we need to update + // transform (#8809) + isSettingWidth = + element.tagName === "SPAN" && styles && "width" in styles, + textWidth = pick(isSettingWidth && styles.width, void 0), + doTransform; + if (isSettingWidth) { + delete styles.width; + wrapper.textWidth = textWidth; + doTransform = true; + } + if (styles && styles.textOverflow === "ellipsis") { + styles.whiteSpace = "nowrap"; + styles.overflow = "hidden"; + } + wrapper.styles = extend(wrapper.styles, styles); + css(wrapper.element, styles); + // Now that all styles are applied, to the transform + if (doTransform) { + wrapper.htmlUpdateTransform(); + } + return wrapper; + }, + /** + * VML and useHTML method for calculating the bounding box based on offsets. + * + * @private + * @function Highcharts.SVGElement#htmlGetBBox + * + * @param {boolean} refresh + * Whether to force a fresh value from the DOM or to use the cached + * value. + * + * @return {Highcharts.BBoxObject} + * A hash containing values for x, y, width and height. + */ + htmlGetBBox: function () { + var wrapper = this, + element = wrapper.element; + return { + x: element.offsetLeft, + y: element.offsetTop, + width: element.offsetWidth, + height: element.offsetHeight, + }; + }, + /** + * VML override private method to update elements based on internal + * properties based on SVG transform. + * + * @private + * @function Highcharts.SVGElement#htmlUpdateTransform + * @return {void} + */ + htmlUpdateTransform: function () { + // aligning non added elements is expensive + if (!this.added) { + this.alignOnAdd = true; + return; + } + var wrapper = this, + renderer = wrapper.renderer, + elem = wrapper.element, + translateX = wrapper.translateX || 0, + translateY = wrapper.translateY || 0, + x = wrapper.x || 0, + y = wrapper.y || 0, + align = wrapper.textAlign || "left", + alignCorrection = { + left: 0, + center: 0.5, + right: 1, + }[align], + styles = wrapper.styles, + whiteSpace = styles && styles.whiteSpace; /** * @private - * @function Highcharts.SVGElement#alignSetter - * @param {"left"|"center"|"right"} value - */ - SVGElement.prototype.alignSetter = function (value) { - var convert = { - left: 'start', - center: 'middle', - right: 'end' - }; - if (convert[value]) { - this.alignValue = value; - this.element.setAttribute('text-anchor', convert[value]); - } - }; - /** - * Animate to given attributes or CSS properties. - * - * @sample highcharts/members/element-on/ - * Setting some attributes by animation - * - * @function Highcharts.SVGElement#animate - * - * @param {Highcharts.SVGAttributes} params - * SVG attributes or CSS to animate. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [options] - * Animation options. - * - * @param {Function} [complete] - * Function to perform at the end of animation. - * - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. + * @return {number} */ - SVGElement.prototype.animate = function (params, options, complete) { - var _this = this; - var animOptions = animObject(pick(options, - this.renderer.globalAnimation, - true)), - deferTime = animOptions.defer; - // When the page is hidden save resources in the background by not - // running animation at all (#9749). - if (pick(doc.hidden, doc.msHidden, doc.webkitHidden, false)) { - animOptions.duration = 0; - } - if (animOptions.duration !== 0) { - // allows using a callback with the global animation without - // overwriting it - if (complete) { - animOptions.complete = complete; + function getTextPxLength() { + // Reset multiline/ellipsis in order to read width (#4928, + // #5417) + css(elem, { + width: "", + whiteSpace: whiteSpace || "nowrap", + }); + return elem.offsetWidth; + } + // apply translate + css(elem, { + marginLeft: translateX, + marginTop: translateY, + }); + if (!renderer.styledMode && wrapper.shadows) { + // used in labels/tooltip + wrapper.shadows.forEach(function (shadow) { + css(shadow, { + marginLeft: translateX + 1, + marginTop: translateY + 1, + }); + }); + } + // apply inversion + if (wrapper.inverted) { + // wrapper is a group + [].forEach.call(elem.childNodes, function (child) { + renderer.invertChild(child, elem); + }); + } + if (elem.tagName === "SPAN") { + var rotation = wrapper.rotation, + baseline, + textWidth = wrapper.textWidth && pInt(wrapper.textWidth), + currentTextTransform = [ + rotation, + align, + elem.innerHTML, + wrapper.textWidth, + wrapper.textAlign, + ].join(","); + // Update textWidth. Use the memoized textPxLength if possible, to + // avoid the getTextPxLength function using elem.offsetWidth. + // Calling offsetWidth affects rendering time as it forces layout + // (#7656). + if ( + textWidth !== wrapper.oldTextWidth && + (textWidth > wrapper.oldTextWidth || + (wrapper.textPxLength || getTextPxLength()) > textWidth) && + // Only set the width if the text is able to word-wrap, or + // text-overflow is ellipsis (#9537) + (/[ \-]/.test(elem.textContent || elem.innerText) || + elem.style.textOverflow === "ellipsis") + ) { + // #983, #1254 + css(elem, { + width: textWidth + "px", + display: "block", + whiteSpace: whiteSpace || "normal", // #3331 + }); + wrapper.oldTextWidth = textWidth; + wrapper.hasBoxWidthChanged = true; // #8159 + } else { + wrapper.hasBoxWidthChanged = false; // #8159 + } + // Do the calculations and DOM access only if properties changed + if (currentTextTransform !== wrapper.cTT) { + baseline = renderer.fontMetrics(elem.style.fontSize, elem).b; + // Renderer specific handling of span rotation, but only if we + // have something to update. + if ( + defined(rotation) && + (rotation !== (wrapper.oldRotation || 0) || + align !== wrapper.oldAlign) + ) { + wrapper.setSpanRotation(rotation, alignCorrection, baseline); + } + wrapper.getSpanCorrection( + // Avoid elem.offsetWidth if we can, it affects rendering + // time heavily (#7656) + (!defined(rotation) && wrapper.textPxLength) || // #7920 + elem.offsetWidth, + baseline, + alignCorrection, + rotation, + align + ); + } + // apply position with correction + css(elem, { + left: x + (wrapper.xCorr || 0) + "px", + top: y + (wrapper.yCorr || 0) + "px", + }); + // record current text transform + wrapper.cTT = currentTextTransform; + wrapper.oldRotation = rotation; + wrapper.oldAlign = align; + } + }, + /** + * Set the rotation of an individual HTML span. + * + * @private + * @function Highcharts.SVGElement#setSpanRotation + * @param {number} rotation + * @param {number} alignCorrection + * @param {number} baseline + * @return {void} + */ + setSpanRotation: function (rotation, alignCorrection, baseline) { + var rotationStyle = {}, + cssTransformKey = this.renderer.getTransformKey(); + rotationStyle[cssTransformKey] = rotationStyle.transform = + "rotate(" + rotation + "deg)"; + rotationStyle[ + cssTransformKey + (isFirefox ? "Origin" : "-origin") + ] = rotationStyle.transformOrigin = + alignCorrection * 100 + "% " + baseline + "px"; + css(this.element, rotationStyle); + }, + /** + * Get the correction in X and Y positioning as the element is rotated. + * + * @private + * @function Highcharts.SVGElement#getSpanCorrection + * @param {number} width + * @param {number} baseline + * @param {number} alignCorrection + * @return {void} + */ + getSpanCorrection: function (width, baseline, alignCorrection) { + this.xCorr = -width * alignCorrection; + this.yCorr = -baseline; + }, + } + ); + + return HTMLElement; + } + ); + _registerModule( + _modules, + "Core/Renderer/HTML/HTMLRenderer.js", + [ + _modules["Core/Globals.js"], + _modules["Core/Renderer/SVG/SVGElement.js"], + _modules["Core/Renderer/SVG/SVGRenderer.js"], + _modules["Core/Utilities.js"], + ], + function (H, SVGElement, SVGRenderer, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var isFirefox = H.isFirefox, + isMS = H.isMS, + isWebKit = H.isWebKit, + win = H.win; + var attr = U.attr, + createElement = U.createElement, + extend = U.extend, + pick = U.pick; + /** + * Renderer placebo + * @private + */ + var HTMLRenderer = SVGRenderer; + /* eslint-disable valid-jsdoc */ + // Extend SvgRenderer for useHTML option. + extend( + SVGRenderer.prototype, + /** @lends SVGRenderer.prototype */ { + /** + * @private + * @function Highcharts.SVGRenderer#getTransformKey + * + * @return {string} + */ + getTransformKey: function () { + return isMS && !/Edge/.test(win.navigator.userAgent) + ? "-ms-transform" + : isWebKit + ? "-webkit-transform" + : isFirefox + ? "MozTransform" + : win.opera + ? "-o-transform" + : ""; + }, + /** + * Create HTML text node. This is used by the VML renderer as well as the + * SVG renderer through the useHTML option. + * + * @private + * @function Highcharts.SVGRenderer#html + * + * @param {string} str + * The text of (subset) HTML to draw. + * + * @param {number} x + * The x position of the text's lower left corner. + * + * @param {number} y + * The y position of the text's lower left corner. + * + * @return {Highcharts.HTMLDOMElement} + */ + html: function (str, x, y) { + var wrapper = this.createElement("span"), + element = wrapper.element, + renderer = wrapper.renderer, + isSVG = renderer.isSVG, + addSetters = function (gWrapper, style) { + // These properties are set as attributes on the SVG group, and + // as identical CSS properties on the div. (#3542) + ["opacity", "visibility"].forEach(function (prop) { + gWrapper[prop + "Setter"] = function (value, key, elem) { + var styleObject = gWrapper.div ? gWrapper.div.style : style; + SVGElement.prototype[prop + "Setter"].call( + this, + value, + key, + elem + ); + if (styleObject) { + styleObject[key] = value; } - // If defer option is defined delay the animation #12901 - syncTimeout(function () { - if (_this.element) { - animate(_this, params, animOptions); - } - }, deferTime); + }; + }); + gWrapper.addedSetters = true; + }; + // Text setter + wrapper.textSetter = function (value) { + if (value !== element.innerHTML) { + delete this.bBox; + delete this.oldTextWidth; + } + this.textStr = value; + element.innerHTML = pick(value, ""); + wrapper.doTransform = true; + }; + // Add setters for the element itself (#4938) + if (isSVG) { + // #4938, only for HTML within SVG + addSetters(wrapper, wrapper.element.style); + } + // Various setters which rely on update transform + wrapper.xSetter = + wrapper.ySetter = + wrapper.alignSetter = + wrapper.rotationSetter = + function (value, key) { + if (key === "align") { + // Do not overwrite the SVGElement.align method. Same as VML. + wrapper.alignValue = wrapper.textAlign = value; + } else { + wrapper[key] = value; + } + wrapper.doTransform = true; + }; + // Runs at the end of .attr() + wrapper.afterSetters = function () { + // Update transform. Do this outside the loop to prevent redundant + // updating for batch setting of attributes. + if (this.doTransform) { + this.htmlUpdateTransform(); + this.doTransform = false; + } + }; + // Set the default attributes + wrapper + .attr({ + text: str, + x: Math.round(x), + y: Math.round(y), + }) + .css({ + position: "absolute", + }); + if (!renderer.styledMode) { + wrapper.css({ + fontFamily: this.style.fontFamily, + fontSize: this.style.fontSize, + }); + } + // Keep the whiteSpace style outside the wrapper.styles collection + element.style.whiteSpace = "nowrap"; + // Use the HTML specific .css method + wrapper.css = wrapper.htmlCss; + // This is specific for HTML within SVG + if (isSVG) { + wrapper.add = function (svgGroupWrapper) { + var htmlGroup, + container = renderer.box.parentNode, + parentGroup, + parents = []; + this.parentGroup = svgGroupWrapper; + // Create a mock group to hold the HTML elements + if (svgGroupWrapper) { + htmlGroup = svgGroupWrapper.div; + if (!htmlGroup) { + // Read the parent chain into an array and read from top + // down + parentGroup = svgGroupWrapper; + while (parentGroup) { + parents.push(parentGroup); + // Move up to the next parent group + parentGroup = parentGroup.parentGroup; + } + // Ensure dynamically updating position when any parent + // is translated + parents.reverse().forEach(function (parentGroup) { + var htmlGroupStyle, + cls = attr(parentGroup.element, "class"); + /** + * Common translate setter for X and Y on the HTML + * group. Reverted the fix for #6957 du to + * positioning problems and offline export (#7254, + * #7280, #7529) + * @private + * @param {*} value + * @param {string} key + * @return {void} + */ + function translateSetter(value, key) { + parentGroup[key] = value; + if (key === "translateX") { + htmlGroupStyle.left = value + "px"; + } else { + htmlGroupStyle.top = value + "px"; + } + parentGroup.doTransform = true; + } + // Create a HTML div and append it to the parent div + // to emulate the SVG group structure + htmlGroup = parentGroup.div = + parentGroup.div || + createElement( + "div", + cls ? { className: cls } : void 0, + { + position: "absolute", + left: (parentGroup.translateX || 0) + "px", + top: (parentGroup.translateY || 0) + "px", + display: parentGroup.display, + opacity: parentGroup.opacity, + pointerEvents: + parentGroup.styles && + parentGroup.styles.pointerEvents, // #5595 + // the top group is appended to container + }, + htmlGroup || container + ); + // Shortcut + htmlGroupStyle = htmlGroup.style; + // Set listeners to update the HTML div's position + // whenever the SVG group position is changed. + extend(parentGroup, { + // (#7287) Pass htmlGroup to use + // the related group + classSetter: (function (htmlGroup) { + return function (value) { + this.element.setAttribute("class", value); + htmlGroup.className = value; + }; + })(htmlGroup), + on: function () { + if (parents[0].div) { + // #6418 + wrapper.on.apply( + { element: parents[0].div }, + arguments + ); + } + return parentGroup; + }, + translateXSetter: translateSetter, + translateYSetter: translateSetter, + }); + if (!parentGroup.addedSetters) { + addSetters(parentGroup); + } + }); + } + } else { + htmlGroup = container; } - else { - this.attr(params, void 0, complete); - // Call the end step synchronously - objectEach(params, function (val, prop) { - if (animOptions.step) { - animOptions.step.call(this, val, { prop: prop, pos: 1 }); - } - }, this); + htmlGroup.appendChild(element); + // Shared with VML: + wrapper.added = true; + if (wrapper.alignOnAdd) { + wrapper.htmlUpdateTransform(); } - return this; - }; - /** - * Apply a text outline through a custom CSS property, by copying the text - * element and apply stroke to the copy. Used internally. Contrast checks at - * [example](https://jsfiddle.net/highcharts/43soe9m1/2/). - * - * @example - * // Specific color - * text.css({ - * textOutline: '1px black' - * }); - * // Automatic contrast - * text.css({ - * color: '#000000', // black text - * textOutline: '1px contrast' // => white outline - * }); - * - * @private - * @function Highcharts.SVGElement#applyTextOutline - * - * @param {string} textOutline - * A custom CSS `text-outline` setting, defined by `width color`. - */ - SVGElement.prototype.applyTextOutline = function (textOutline) { - var elem = this.element, - tspans, - hasContrast = textOutline.indexOf('contrast') !== -1, - styles = {}, - color, - strokeWidth, - firstRealChild; - // When the text shadow is set to contrast, use dark stroke for light - // text and vice versa. - if (hasContrast) { - styles.textOutline = textOutline = textOutline.replace(/contrast/g, this.renderer.getContrast(elem.style.fill)); - } - // Extract the stroke width and color - textOutline = textOutline.split(' '); - color = textOutline[textOutline.length - 1]; - strokeWidth = textOutline[0]; - if (strokeWidth && strokeWidth !== 'none' && H.svg) { - this.fakeTS = true; // Fake text shadow - tspans = [].slice.call(elem.getElementsByTagName('tspan')); - // In order to get the right y position of the clone, - // copy over the y setter - this.ySetter = this.xSetter; - // Since the stroke is applied on center of the actual outline, we - // need to double it to get the correct stroke-width outside the - // glyphs. - strokeWidth = strokeWidth.replace(/(^[\d\.]+)(.*?)$/g, function (match, digit, unit) { - return (2 * digit) + unit; - }); - // Remove shadows from previous runs. - this.removeTextOutline(tspans); - // Check if the element contains RTL characters. - // Comparing against Hebrew and Arabic characters, - // excluding Arabic digits. Source: - // https://www.unicode.org/Public/UNIDATA/extracted/DerivedBidiClass.txt - var isRTL_1 = elem.textContent ? - /^[\u0591-\u065F\u066A-\u07FF\uFB1D-\uFDFD\uFE70-\uFEFC]/ - .test(elem.textContent) : false; - // For each of the tspans, create a stroked copy behind it. - firstRealChild = elem.firstChild; - tspans.forEach(function (tspan, y) { - var clone; - // Let the first line start at the correct X position - if (y === 0) { - tspan.setAttribute('x', elem.getAttribute('x')); - y = elem.getAttribute('y'); - tspan.setAttribute('y', y || 0); - if (y === null) { - elem.setAttribute('y', 0); - } - } - // Create the clone and apply outline properties. - // For RTL elements apply outline properties for orginal element - // to prevent outline from overlapping the text. - // For RTL in Firefox keep the orginal order (#10162). - clone = tspan.cloneNode(true); - attr((isRTL_1 && !isFirefox) ? tspan : clone, { - 'class': 'highcharts-text-outline', - fill: color, - stroke: color, - 'stroke-width': strokeWidth, - 'stroke-linejoin': 'round' - }); - elem.insertBefore(clone, firstRealChild); - }); - // Create a whitespace between tspan and clone, - // to fix the display of Arabic characters in Firefox. - if (isRTL_1 && isFirefox && tspans[0]) { - var whitespace = tspans[0].cloneNode(true); - whitespace.textContent = ' '; - elem.insertBefore(whitespace, firstRealChild); - } - } - }; - /** - * @function Highcharts.SVGElement#attr - * @param {string} key - * @return {number|string} - */ /** - * Apply native and custom attributes to the SVG elements. - * - * In order to set the rotation center for rotation, set x and y to 0 and - * use `translateX` and `translateY` attributes to position the element - * instead. - * - * Attributes frequently used in Highcharts are `fill`, `stroke`, - * `stroke-width`. - * - * @sample highcharts/members/renderer-rect/ - * Setting some attributes - * - * @example - * // Set multiple attributes - * element.attr({ - * stroke: 'red', - * fill: 'blue', - * x: 10, - * y: 10 - * }); - * - * // Set a single attribute - * element.attr('stroke', 'red'); - * - * // Get an attribute - * element.attr('stroke'); // => 'red' - * - * @function Highcharts.SVGElement#attr - * - * @param {string|Highcharts.SVGAttributes} [hash] - * The native and custom SVG attributes. - * - * @param {number|string|Highcharts.SVGPathArray} [val] - * If the type of the first argument is `string`, the second can be a - * value, which will serve as a single attribute setter. If the first - * argument is a string and the second is undefined, the function - * serves as a getter and the current value of the property is - * returned. - * - * @param {Function} [complete] - * A callback function to execute after setting the attributes. This - * makes the function compliant and interchangeable with the - * {@link SVGElement#animate} function. - * - * @param {boolean} [continueAnimation=true] - * Used internally when `.attr` is called as part of an animation - * step. Otherwise, calling `.attr` for an attribute will stop - * animation for that attribute. - * - * @return {Highcharts.SVGElement} - * If used as a setter, it returns the current - * {@link Highcharts.SVGElement} so the calls can be chained. If - * used as a getter, the current value of the attribute is returned. - */ - SVGElement.prototype.attr = function (hash, val, complete, continueAnimation) { - var key, - element = this.element, - hasSetSymbolSize, - ret = this, - skipAttr, - setter, - symbolCustomAttribs = this.symbolCustomAttribs; - // single key-value pair - if (typeof hash === 'string' && typeof val !== 'undefined') { - key = hash; - hash = {}; - hash[key] = val; - } - // used as a getter: first argument is a string, second is undefined - if (typeof hash === 'string') { - ret = (this[hash + 'Getter'] || - this._defaultGetter).call(this, hash, element); - // setter - } - else { - objectEach(hash, function eachAttribute(val, key) { - skipAttr = false; - // Unless .attr is from the animator update, stop current - // running animation of this property - if (!continueAnimation) { - stop(this, key); - } - // Special handling of symbol attributes - if (this.symbolName && - symbolCustomAttribs.indexOf(key) !== -1) { - if (!hasSetSymbolSize) { - this.symbolAttr(hash); - hasSetSymbolSize = true; - } - skipAttr = true; - } - if (this.rotation && (key === 'x' || key === 'y')) { - this.doTransform = true; - } - if (!skipAttr) { - setter = (this[key + 'Setter'] || - this._defaultSetter); - setter.call(this, val, key, element); - // Let the shadow follow the main element - if (!this.styledMode && - this.shadows && - /^(width|height|visibility|x|y|d|transform|cx|cy|r)$/.test(key)) { - this.updateShadows(key, val, setter); - } - } - }, this); - this.afterSetters(); - } - // In accordance with animate, run a complete callback - if (complete) { - complete.call(this); - } - return ret; - }; - /** - * Apply a clipping rectangle to this element. - * - * @function Highcharts.SVGElement#clip - * - * @param {Highcharts.ClipRectElement} [clipRect] - * The clipping rectangle. If skipped, the current clip is removed. - * - * @return {Highcharts.SVGElement} - * Returns the SVG element to allow chaining. - */ - SVGElement.prototype.clip = function (clipRect) { - return this.attr('clip-path', clipRect ? - 'url(' + this.renderer.url + '#' + clipRect.id + ')' : - 'none'); - }; - /** - * Calculate the coordinates needed for drawing a rectangle crisply and - * return the calculated attributes. - * - * @function Highcharts.SVGElement#crisp - * - * @param {Highcharts.RectangleObject} rect - * Rectangle to crisp. - * - * @param {number} [strokeWidth] - * The stroke width to consider when computing crisp positioning. It can - * also be set directly on the rect parameter. - * - * @return {Highcharts.RectangleObject} - * The modified rectangle arguments. - */ - SVGElement.prototype.crisp = function (rect, strokeWidth) { - var wrapper = this, - normalizer; - strokeWidth = strokeWidth || rect.strokeWidth || 0; - // Math.round because strokeWidth can sometimes have roundoff errors - normalizer = Math.round(strokeWidth) % 2 / 2; - // normalize for crisp edges - rect.x = Math.floor(rect.x || wrapper.x || 0) + normalizer; - rect.y = Math.floor(rect.y || wrapper.y || 0) + normalizer; - rect.width = Math.floor((rect.width || wrapper.width || 0) - 2 * normalizer); - rect.height = Math.floor((rect.height || wrapper.height || 0) - 2 * normalizer); - if (defined(rect.strokeWidth)) { - rect.strokeWidth = strokeWidth; - } - return rect; - }; - /** - * Build and apply an SVG gradient out of a common JavaScript configuration - * object. This function is called from the attribute setters. An event - * hook is added for supporting other complex color types. - * - * @private - * @function Highcharts.SVGElement#complexColor - * - * @param {Highcharts.GradientColorObject|Highcharts.PatternObject} colorOptions - * The gradient or pattern options structure. - * - * @param {string} prop - * The property to apply, can either be `fill` or `stroke`. - * - * @param {Highcharts.SVGDOMElement} elem - * SVG element to apply the gradient on. - */ - SVGElement.prototype.complexColor = function (colorOptions, prop, elem) { - var renderer = this.renderer, - colorObject, - gradName, - gradAttr, - radAttr, - gradients, - stops, - stopColor, - stopOpacity, - radialReference, - id, - key = [], - value; - fireEvent(this.renderer, 'complexColor', { - args: arguments - }, function () { - // Apply linear or radial gradients - if (colorOptions.radialGradient) { - gradName = 'radialGradient'; - } - else if (colorOptions.linearGradient) { - gradName = 'linearGradient'; - } - if (gradName) { - gradAttr = colorOptions[gradName]; - gradients = renderer.gradients; - stops = colorOptions.stops; - radialReference = elem.radialReference; - // Keep < 2.2 kompatibility - if (isArray(gradAttr)) { - colorOptions[gradName] = gradAttr = { - x1: gradAttr[0], - y1: gradAttr[1], - x2: gradAttr[2], - y2: gradAttr[3], - gradientUnits: 'userSpaceOnUse' - }; - } - // Correct the radial gradient for the radial reference system - if (gradName === 'radialGradient' && - radialReference && - !defined(gradAttr.gradientUnits)) { - // Save the radial attributes for updating - radAttr = gradAttr; - gradAttr = merge(gradAttr, renderer.getRadialAttr(radialReference, radAttr), { gradientUnits: 'userSpaceOnUse' }); - } - // Build the unique key to detect whether we need to create a - // new element (#1282) - objectEach(gradAttr, function (val, n) { - if (n !== 'id') { - key.push(n, val); - } - }); - objectEach(stops, function (val) { - key.push(val); - }); - key = key.join(','); - // Check if a gradient object with the same config object is - // created within this renderer - if (gradients[key]) { - id = gradients[key].attr('id'); - } - else { - // Set the id and create the element - gradAttr.id = id = uniqueKey(); - var gradientObject_1 = gradients[key] = - renderer.createElement(gradName) - .attr(gradAttr) - .add(renderer.defs); - gradientObject_1.radAttr = radAttr; - // The gradient needs to keep a list of stops to be able to - // destroy them - gradientObject_1.stops = []; - stops.forEach(function (stop) { - var stopObject; - if (stop[1].indexOf('rgba') === 0) { - colorObject = Color.parse(stop[1]); - stopColor = colorObject.get('rgb'); - stopOpacity = colorObject.get('a'); - } - else { - stopColor = stop[1]; - stopOpacity = 1; - } - stopObject = renderer.createElement('stop').attr({ - offset: stop[0], - 'stop-color': stopColor, - 'stop-opacity': stopOpacity - }).add(gradientObject_1); - // Add the stop element to the gradient - gradientObject_1.stops.push(stopObject); - }); - } - // Set the reference to the gradient object - value = 'url(' + renderer.url + '#' + id + ')'; - elem.setAttribute(prop, value); - elem.gradient = key; - // Allow the color to be concatenated into tooltips formatters - // etc. (#2995) - colorOptions.toString = function () { - return value; - }; - } - }); - }; - /** - * Set styles for the element. In addition to CSS styles supported by - * native SVG and HTML elements, there are also some custom made for - * Highcharts, like `width`, `ellipsis` and `textOverflow` for SVG text - * elements. - * - * @sample highcharts/members/renderer-text-on-chart/ - * Styled text - * - * @function Highcharts.SVGElement#css - * - * @param {Highcharts.CSSObject} styles - * The new CSS styles. - * - * @return {Highcharts.SVGElement} - * Return the SVG element for chaining. - */ - SVGElement.prototype.css = function (styles) { - var oldStyles = this.styles, newStyles = {}, elem = this.element, textWidth, serializedCss = '', hyphenate, hasNew = !oldStyles, - // These CSS properties are interpreted internally by the SVG - // renderer, but are not supported by SVG and should not be added to - // the DOM. In styled mode, no CSS should find its way to the DOM - // whatsoever (#6173, #6474). - svgPseudoProps = ['textOutline', 'textOverflow', 'width']; - // convert legacy - if (styles && styles.color) { - styles.fill = styles.color; - } - // Filter out existing styles to increase performance (#2640) - if (oldStyles) { - objectEach(styles, function (style, n) { - if (oldStyles && oldStyles[n] !== style) { - newStyles[n] = style; - hasNew = true; - } - }); - } - if (hasNew) { - // Merge the new styles with the old ones - if (oldStyles) { - styles = extend(oldStyles, newStyles); - } - // Get the text width from style - if (styles) { - // Previously set, unset it (#8234) - if (styles.width === null || styles.width === 'auto') { - delete this.textWidth; - // Apply new - } - else if (elem.nodeName.toLowerCase() === 'text' && - styles.width) { - textWidth = this.textWidth = pInt(styles.width); - } - } - // store object - this.styles = styles; - if (textWidth && (!svg && this.renderer.forExport)) { - delete styles.width; - } - // Serialize and set style attribute - if (elem.namespaceURI === this.SVG_NS) { // #7633 - hyphenate = function (a, b) { - return '-' + b.toLowerCase(); - }; - objectEach(styles, function (style, n) { - if (svgPseudoProps.indexOf(n) === -1) { - serializedCss += - n.replace(/([A-Z])/g, hyphenate) + ':' + - style + ';'; - } - }); - if (serializedCss) { - attr(elem, 'style', serializedCss); // #1881 - } - } - else { - css(elem, styles); - } - if (this.added) { - // Rebuild text after added. Cache mechanisms in the buildText - // will prevent building if there are no significant changes. - if (this.element.nodeName === 'text') { - this.renderer.buildText(this); - } - // Apply text outline after added - if (styles && styles.textOutline) { - this.applyTextOutline(styles.textOutline); - } - } - } - return this; - }; - /** - * @private - * @function Highcharts.SVGElement#dashstyleSetter - * @param {string} value - */ - SVGElement.prototype.dashstyleSetter = function (value) { - var i, - strokeWidth = this['stroke-width']; - // If "inherit", like maps in IE, assume 1 (#4981). With HC5 and the new - // strokeWidth function, we should be able to use that instead. - if (strokeWidth === 'inherit') { - strokeWidth = 1; - } - value = value && value.toLowerCase(); - if (value) { - var v = value - .replace('shortdashdotdot', '3,1,1,1,1,1,') - .replace('shortdashdot', '3,1,1,1') - .replace('shortdot', '1,1,') - .replace('shortdash', '3,1,') - .replace('longdash', '8,3,') - .replace(/dot/g, '1,3,') - .replace('dash', '4,3,') - .replace(/,$/, '') - .split(','); // ending comma - i = v.length; - while (i--) { - v[i] = '' + (pInt(v[i]) * pick(strokeWidth, NaN)); - } - value = v.join(',').replace(/NaN/g, 'none'); // #3226 - this.element.setAttribute('stroke-dasharray', value); - } - }; - /** - * Destroy the element and element wrapper and clear up the DOM and event - * hooks. - * - * @function Highcharts.SVGElement#destroy - */ - SVGElement.prototype.destroy = function () { - var wrapper = this, - element = wrapper.element || {}, - renderer = wrapper.renderer, - parentToClean = (renderer.isSVG && - element.nodeName === 'SPAN' && - wrapper.parentGroup || - void 0), - grandParent, - ownerSVGElement = element.ownerSVGElement, - i; - // remove events - element.onclick = element.onmouseout = element.onmouseover = - element.onmousemove = element.point = null; - stop(wrapper); // stop running animations - if (wrapper.clipPath && ownerSVGElement) { - var clipPath_1 = wrapper.clipPath; - // Look for existing references to this clipPath and remove them - // before destroying the element (#6196). - // The upper case version is for Edge - [].forEach.call(ownerSVGElement.querySelectorAll('[clip-path],[CLIP-PATH]'), function (el) { - var clipPathAttr = el.getAttribute('clip-path'); - if (clipPathAttr.indexOf(clipPath_1.element.id) > -1) { - el.removeAttribute('clip-path'); - } - }); - wrapper.clipPath = clipPath_1.destroy(); - } - // Destroy stops in case this is a gradient object @todo old code? - if (wrapper.stops) { - for (i = 0; i < wrapper.stops.length; i++) { - wrapper.stops[i].destroy(); - } - wrapper.stops.length = 0; - wrapper.stops = void 0; - } - // remove element - wrapper.safeRemoveChild(element); - if (!renderer.styledMode) { - wrapper.destroyShadows(); - } - // In case of useHTML, clean up empty containers emulating SVG groups - // (#1960, #2393, #2697). - while (parentToClean && - parentToClean.div && - parentToClean.div.childNodes.length === 0) { - grandParent = parentToClean.parentGroup; - wrapper.safeRemoveChild(parentToClean.div); - delete parentToClean.div; - parentToClean = grandParent; - } - // remove from alignObjects - if (wrapper.alignTo) { - erase(renderer.alignedObjects, wrapper); - } - objectEach(wrapper, function (val, key) { - // Destroy child elements of a group - if (wrapper[key] && - wrapper[key].parentGroup === wrapper && - wrapper[key].destroy) { - wrapper[key].destroy(); - } - // Delete all properties - delete wrapper[key]; + return wrapper; + }; + } + return wrapper; + }, + } + ); + + return HTMLRenderer; + } + ); + _registerModule( + _modules, + "Core/Axis/Tick.js", + [_modules["Core/Globals.js"], _modules["Core/Utilities.js"]], + function (H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + /** + * Optional parameters for the tick. + * @private + * @interface Highcharts.TickParametersObject + */ /** + * Set category for the tick. + * @name Highcharts.TickParametersObject#category + * @type {string|undefined} + */ /** + * @name Highcharts.TickParametersObject#options + * @type {Highcharts.Dictionary<any>|undefined} + */ /** + * Set tickmarkOffset for the tick. + * @name Highcharts.TickParametersObject#tickmarkOffset + * @type {number|undefined} + */ + var clamp = U.clamp, + correctFloat = U.correctFloat, + defined = U.defined, + destroyObjectProperties = U.destroyObjectProperties, + extend = U.extend, + fireEvent = U.fireEvent, + isNumber = U.isNumber, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick; + var deg2rad = H.deg2rad; + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The Tick class. + * + * @class + * @name Highcharts.Tick + * + * @param {Highcharts.Axis} axis + * The axis of the tick. + * + * @param {number} pos + * The position of the tick on the axis in terms of axis values. + * + * @param {string} [type] + * The type of tick, either 'minor' or an empty string + * + * @param {boolean} [noLabel=false] + * Whether to disable the label or not. Defaults to false. + * + * @param {object} [parameters] + * Optional parameters for the tick. + */ + var Tick = /** @class */ (function () { + /* * + * + * Constructors + * + * */ + function Tick(axis, pos, type, noLabel, parameters) { + this.isNew = true; + this.isNewLabel = true; + /** + * The related axis of the tick. + * @name Highcharts.Tick#axis + * @type {Highcharts.Axis} + */ + this.axis = axis; + /** + * The logical position of the tick on the axis in terms of axis values. + * @name Highcharts.Tick#pos + * @type {number} + */ + this.pos = pos; + /** + * The tick type, which can be `"minor"`, or an empty string. + * @name Highcharts.Tick#type + * @type {string} + */ + this.type = type || ""; + this.parameters = parameters || {}; + /** + * The mark offset of the tick on the axis. Usually `undefined`, numeric + * for grid axes. + * @name Highcharts.Tick#tickmarkOffset + * @type {number|undefined} + */ + this.tickmarkOffset = this.parameters.tickmarkOffset; + this.options = this.parameters.options; + fireEvent(this, "init"); + if (!type && !noLabel) { + this.addLabel(); + } + } + /* * + * + * Functions + * + * */ + /** + * Write the tick label. + * + * @private + * @function Highcharts.Tick#addLabel + * @return {void} + */ + Tick.prototype.addLabel = function () { + var tick = this, + axis = tick.axis, + options = axis.options, + chart = axis.chart, + categories = axis.categories, + log = axis.logarithmic, + names = axis.names, + pos = tick.pos, + labelOptions = pick( + tick.options && tick.options.labels, + options.labels + ), + str, + tickPositions = axis.tickPositions, + isFirst = pos === tickPositions[0], + isLast = pos === tickPositions[tickPositions.length - 1], + value = + this.parameters.category || + (categories ? pick(categories[pos], names[pos], pos) : pos), + label = tick.label, + animateLabels = + (!labelOptions.step || labelOptions.step === 1) && + axis.tickInterval === 1, + tickPositionInfo = tickPositions.info, + dateTimeLabelFormat, + dateTimeLabelFormats, + i, + list; + // Set the datetime label format. If a higher rank is set for this + // position, use that. If not, use the general format. + if (axis.dateTime && tickPositionInfo) { + dateTimeLabelFormats = chart.time.resolveDTLFormat( + options.dateTimeLabelFormats[ + (!options.grid && tickPositionInfo.higherRanks[pos]) || + tickPositionInfo.unitName + ] + ); + dateTimeLabelFormat = dateTimeLabelFormats.main; + } + // set properties for access in render method + /** + * True if the tick is the first one on the axis. + * @name Highcharts.Tick#isFirst + * @readonly + * @type {boolean|undefined} + */ + tick.isFirst = isFirst; + /** + * True if the tick is the last one on the axis. + * @name Highcharts.Tick#isLast + * @readonly + * @type {boolean|undefined} + */ + tick.isLast = isLast; + // Get the string + tick.formatCtx = { + axis: axis, + chart: chart, + isFirst: isFirst, + isLast: isLast, + dateTimeLabelFormat: dateTimeLabelFormat, + tickPositionInfo: tickPositionInfo, + value: log ? correctFloat(log.lin2log(value)) : value, + pos: pos, + }; + str = axis.labelFormatter.call(tick.formatCtx, this.formatCtx); + // Set up conditional formatting based on the format list if existing. + list = dateTimeLabelFormats && dateTimeLabelFormats.list; + if (list) { + tick.shortenLabel = function () { + for (i = 0; i < list.length; i++) { + label.attr({ + text: axis.labelFormatter.call( + extend(tick.formatCtx, { dateTimeLabelFormat: list[i] }) + ), }); - return; - }; - /** - * Destroy shadows on the element. - * - * @private - * @function Highcharts.SVGElement#destroyShadows - * - * @return {void} - */ - SVGElement.prototype.destroyShadows = function () { - (this.shadows || []).forEach(function (shadow) { - this.safeRemoveChild(shadow); - }, this); - this.shadows = void 0; - }; - /** - * @private - */ - SVGElement.prototype.destroyTextPath = function (elem, path) { - var textElement = elem.getElementsByTagName('text')[0]; - var tspans; - if (textElement) { - // Remove textPath attributes - textElement.removeAttribute('dx'); - textElement.removeAttribute('dy'); - // Remove ID's: - path.element.setAttribute('id', ''); - // Check if textElement includes textPath, - if (this.textPathWrapper && - textElement.getElementsByTagName('textPath').length) { - // Move nodes to <text> - tspans = this.textPathWrapper.element.childNodes; - // Now move all <tspan>'s to the <textPath> node - while (tspans.length) { - textElement.appendChild(tspans[0]); - } - // Remove <textPath> from the DOM - textElement.removeChild(this.textPathWrapper.element); - } - } - else if (elem.getAttribute('dx') || elem.getAttribute('dy')) { - // Remove textPath attributes from elem - // to get correct text-outline position - elem.removeAttribute('dx'); - elem.removeAttribute('dy'); - } - if (this.textPathWrapper) { - // Set textPathWrapper to undefined and destroy it - this.textPathWrapper = this.textPathWrapper.destroy(); - } - }; - /** - * @private - * @function Highcharts.SVGElement#dSettter - * @param {number|string|Highcharts.SVGPathArray} value - * @param {string} key - * @param {Highcharts.SVGDOMElement} element - */ - SVGElement.prototype.dSetter = function (value, key, element) { - if (isArray(value)) { - // Backwards compatibility, convert one-dimensional array into an - // array of segments - if (typeof value[0] === 'string') { - value = this.renderer.pathToSegments(value); - } - this.pathArray = value; - value = value.reduce(function (acc, seg, i) { - if (!seg || !seg.join) { - return (seg || '').toString(); - } - return (i ? acc + ' ' : '') + seg.join(' '); - }, ''); - } - if (/(NaN| {2}|^$)/.test(value)) { - value = 'M 0 0'; - } - // Check for cache before resetting. Resetting causes disturbance in the - // DOM, causing flickering in some cases in Edge/IE (#6747). Also - // possible performance gain. - if (this[key] !== value) { - element.setAttribute(key, value); - this[key] = value; - } - }; + if ( + label.getBBox().width < + axis.getSlotWidth(tick) - 2 * pick(labelOptions.padding, 5) + ) { + return; + } + } + label.attr({ + text: "", + }); + }; + } + // Call only after first render + if (animateLabels && axis._addedPlotLB) { + tick.moveLabel(str, labelOptions); + } + // First call + if (!defined(label) && !tick.movedLabel) { + /** + * The rendered text label of the tick. + * @name Highcharts.Tick#label + * @type {Highcharts.SVGElement|undefined} + */ + tick.label = label = tick.createLabel( + { x: 0, y: 0 }, + str, + labelOptions + ); + // Base value to detect change for new calls to getBBox + tick.rotation = 0; + // update + } else if (label && label.textStr !== str && !animateLabels) { + // When resetting text, also reset the width if dynamically set + // (#8809) + if ( + label.textWidth && + !(labelOptions.style && labelOptions.style.width) && + !label.styles.width + ) { + label.css({ width: null }); + } + label.attr({ text: str }); + label.textPxLength = label.getBBox().width; + } + }; + /** + * Render and return the label of the tick. + * + * @private + * @function Highcharts.Tick#createLabel + * @param {Highcharts.PositionObject} xy + * @param {string} str + * @param {Highcharts.XAxisLabelsOptions} labelOptions + * @return {Highcharts.SVGElement|undefined} + */ + Tick.prototype.createLabel = function (xy, str, labelOptions) { + var axis = this.axis, + chart = axis.chart, + label = + defined(str) && labelOptions.enabled + ? chart.renderer + .text(str, xy.x, xy.y, labelOptions.useHTML) + .add(axis.labelGroup) + : null; + // Un-rotated length + if (label) { + // Without position absolute, IE export sometimes is wrong + if (!chart.styledMode) { + label.css(merge(labelOptions.style)); + } + label.textPxLength = label.getBBox().width; + } + return label; + }; + /** + * Destructor for the tick prototype + * + * @private + * @function Highcharts.Tick#destroy + * @return {void} + */ + Tick.prototype.destroy = function () { + destroyObjectProperties(this, this.axis); + }; + /** + * Gets the x and y positions for ticks in terms of pixels. + * + * @private + * @function Highcharts.Tick#getPosition + * + * @param {boolean} horiz + * Whether the tick is on an horizontal axis or not. + * + * @param {number} tickPos + * Position of the tick. + * + * @param {number} tickmarkOffset + * Tickmark offset for all ticks. + * + * @param {boolean} [old] + * Whether the axis has changed or not. + * + * @return {Highcharts.PositionObject} + * The tick position. + * + * @fires Highcharts.Tick#event:afterGetPosition + */ + Tick.prototype.getPosition = function ( + horiz, + tickPos, + tickmarkOffset, + old + ) { + var axis = this.axis, + chart = axis.chart, + cHeight = (old && chart.oldChartHeight) || chart.chartHeight, + pos; + pos = { + x: horiz + ? correctFloat( + axis.translate(tickPos + tickmarkOffset, null, null, old) + + axis.transB + ) + : axis.left + + axis.offset + + (axis.opposite + ? ((old && chart.oldChartWidth) || chart.chartWidth) - + axis.right - + axis.left + : 0), + y: horiz + ? cHeight - + axis.bottom + + axis.offset - + (axis.opposite ? axis.height : 0) + : correctFloat( + cHeight - + axis.translate(tickPos + tickmarkOffset, null, null, old) - + axis.transB + ), + }; + // Chrome workaround for #10516 + pos.y = clamp(pos.y, -1e5, 1e5); + fireEvent(this, "afterGetPosition", { pos: pos }); + return pos; + }; + /** + * Get the x, y position of the tick label + * + * @private + * @return {Highcharts.PositionObject} + */ + Tick.prototype.getLabelPosition = function ( + x, + y, + label, + horiz, + labelOptions, + tickmarkOffset, + index, + step + ) { + var axis = this.axis, + transA = axis.transA, + reversed = // #7911 + axis.isLinked && axis.linkedParent + ? axis.linkedParent.reversed + : axis.reversed, + staggerLines = axis.staggerLines, + rotCorr = axis.tickRotCorr || { x: 0, y: 0 }, + yOffset = labelOptions.y, + // Adjust for label alignment if we use reserveSpace: true (#5286) + labelOffsetCorrection = + !horiz && !axis.reserveSpaceDefault + ? -axis.labelOffset * (axis.labelAlign === "center" ? 0.5 : 1) + : 0, + line, + pos = {}; + if (!defined(yOffset)) { + if (axis.side === 0) { + yOffset = label.rotation ? -8 : -label.getBBox().height; + } else if (axis.side === 2) { + yOffset = rotCorr.y + 8; + } else { + // #3140, #3140 + yOffset = + Math.cos(label.rotation * deg2rad) * + (rotCorr.y - label.getBBox(false, 0).height / 2); + } + } + x = + x + + labelOptions.x + + labelOffsetCorrection + + rotCorr.x - + (tickmarkOffset && horiz + ? tickmarkOffset * transA * (reversed ? -1 : 1) + : 0); + y = + y + + yOffset - + (tickmarkOffset && !horiz + ? tickmarkOffset * transA * (reversed ? 1 : -1) + : 0); + // Correct for staggered labels + if (staggerLines) { + line = (index / (step || 1)) % staggerLines; + if (axis.opposite) { + line = staggerLines - line - 1; + } + y += line * (axis.labelOffset / staggerLines); + } + pos.x = x; + pos.y = Math.round(y); + fireEvent(this, "afterGetLabelPosition", { + pos: pos, + tickmarkOffset: tickmarkOffset, + index: index, + }); + return pos; + }; + /** + * Get the offset height or width of the label + * + * @private + * @function Highcharts.Tick#getLabelSize + * @return {number} + */ + Tick.prototype.getLabelSize = function () { + return this.label + ? this.label.getBBox()[this.axis.horiz ? "height" : "width"] + : 0; + }; + /** + * Extendible method to return the path of the marker + * + * @private + * + */ + Tick.prototype.getMarkPath = function ( + x, + y, + tickLength, + tickWidth, + horiz, + renderer + ) { + return renderer.crispLine( + [ + ["M", x, y], + [ + "L", + x + (horiz ? 0 : -tickLength), + y + (horiz ? tickLength : 0), + ], + ], + tickWidth + ); + }; + /** + * Handle the label overflow by adjusting the labels to the left and right + * edge, or hide them if they collide into the neighbour label. + * + * @private + * @function Highcharts.Tick#handleOverflow + * @param {Highcharts.PositionObject} xy + * @return {void} + */ + Tick.prototype.handleOverflow = function (xy) { + var tick = this, + axis = this.axis, + labelOptions = axis.options.labels, + pxPos = xy.x, + chartWidth = axis.chart.chartWidth, + spacing = axis.chart.spacing, + leftBound = pick(axis.labelLeft, Math.min(axis.pos, spacing[3])), + rightBound = pick( + axis.labelRight, + Math.max( + !axis.isRadial ? axis.pos + axis.len : 0, + chartWidth - spacing[1] + ) + ), + label = this.label, + rotation = this.rotation, + factor = { + left: 0, + center: 0.5, + right: 1, + }[axis.labelAlign || label.attr("align")], + labelWidth = label.getBBox().width, + slotWidth = axis.getSlotWidth(tick), + modifiedSlotWidth = slotWidth, + xCorrection = factor, + goRight = 1, + leftPos, + rightPos, + textWidth, + css = {}; + // Check if the label overshoots the chart spacing box. If it does, move + // it. If it now overshoots the slotWidth, add ellipsis. + if ( + !rotation && + pick(labelOptions.overflow, "justify") === "justify" + ) { + leftPos = pxPos - factor * labelWidth; + rightPos = pxPos + (1 - factor) * labelWidth; + if (leftPos < leftBound) { + modifiedSlotWidth = + xy.x + modifiedSlotWidth * (1 - factor) - leftBound; + } else if (rightPos > rightBound) { + modifiedSlotWidth = + rightBound - xy.x + modifiedSlotWidth * factor; + goRight = -1; + } + modifiedSlotWidth = Math.min(slotWidth, modifiedSlotWidth); // #4177 + if (modifiedSlotWidth < slotWidth && axis.labelAlign === "center") { + xy.x += + goRight * + (slotWidth - + modifiedSlotWidth - + xCorrection * + (slotWidth - Math.min(labelWidth, modifiedSlotWidth))); + } + // If the label width exceeds the available space, set a text width + // to be picked up below. Also, if a width has been set before, we + // need to set a new one because the reported labelWidth will be + // limited by the box (#3938). + if ( + labelWidth > modifiedSlotWidth || + (axis.autoRotation && (label.styles || {}).width) + ) { + textWidth = modifiedSlotWidth; + } + // Add ellipsis to prevent rotated labels to be clipped against the edge + // of the chart + } else if (rotation < 0 && pxPos - factor * labelWidth < leftBound) { + textWidth = Math.round( + pxPos / Math.cos(rotation * deg2rad) - leftBound + ); + } else if (rotation > 0 && pxPos + factor * labelWidth > rightBound) { + textWidth = Math.round( + (chartWidth - pxPos) / Math.cos(rotation * deg2rad) + ); + } + if (textWidth) { + if (tick.shortenLabel) { + tick.shortenLabel(); + } else { + css.width = Math.floor(textWidth) + "px"; + if (!(labelOptions.style || {}).textOverflow) { + css.textOverflow = "ellipsis"; + } + label.css(css); + } + } + }; + /** + * Try to replace the label if the same one already exists. + * + * @private + * @function Highcharts.Tick#moveLabel + * @param {string} str + * @param {Highcharts.XAxisLabelsOptions} labelOptions + * + * @return {void} + */ + Tick.prototype.moveLabel = function (str, labelOptions) { + var tick = this, + label = tick.label, + moved = false, + axis = tick.axis, + labelPos, + reversed = axis.reversed, + xPos, + yPos; + if (label && label.textStr === str) { + tick.movedLabel = label; + moved = true; + delete tick.label; + } else { + // Find a label with the same string + objectEach(axis.ticks, function (currentTick) { + if ( + !moved && + !currentTick.isNew && + currentTick !== tick && + currentTick.label && + currentTick.label.textStr === str + ) { + tick.movedLabel = currentTick.label; + moved = true; + currentTick.labelPos = tick.movedLabel.xy; + delete currentTick.label; + } + }); + } + // Create new label if the actual one is moved + if (!moved && (tick.labelPos || label)) { + labelPos = tick.labelPos || label.xy; + xPos = axis.horiz + ? reversed + ? 0 + : axis.width + axis.left + : labelPos.x; + yPos = axis.horiz + ? labelPos.y + : reversed + ? axis.width + axis.left + : 0; + tick.movedLabel = tick.createLabel( + { x: xPos, y: yPos }, + str, + labelOptions + ); + if (tick.movedLabel) { + tick.movedLabel.attr({ opacity: 0 }); + } + } + }; + /** + * Put everything in place + * + * @private + * @param {number} index + * @param {boolean} [old] + * Use old coordinates to prepare an animation into new position + * @param {number} [opacity] + * @return {voids} + */ + Tick.prototype.render = function (index, old, opacity) { + var tick = this, + axis = tick.axis, + horiz = axis.horiz, + pos = tick.pos, + tickmarkOffset = pick(tick.tickmarkOffset, axis.tickmarkOffset), + xy = tick.getPosition(horiz, pos, tickmarkOffset, old), + x = xy.x, + y = xy.y, + reverseCrisp = + (horiz && x === axis.pos + axis.len) || (!horiz && y === axis.pos) + ? -1 + : 1; // #1480, #1687 + opacity = pick(opacity, 1); + this.isActive = true; + // Create the grid line + this.renderGridLine(old, opacity, reverseCrisp); + // create the tick mark + this.renderMark(xy, opacity, reverseCrisp); + // the label is created on init - now move it into place + this.renderLabel(xy, old, opacity, index); + tick.isNew = false; + fireEvent(this, "afterRender"); + }; + /** + * Renders the gridLine. + * + * @private + * @param {boolean} old Whether or not the tick is old + * @param {number} opacity The opacity of the grid line + * @param {number} reverseCrisp Modifier for avoiding overlapping 1 or -1 + * @return {void} + */ + Tick.prototype.renderGridLine = function (old, opacity, reverseCrisp) { + var tick = this, + axis = tick.axis, + options = axis.options, + gridLine = tick.gridLine, + gridLinePath, + attribs = {}, + pos = tick.pos, + type = tick.type, + tickmarkOffset = pick(tick.tickmarkOffset, axis.tickmarkOffset), + renderer = axis.chart.renderer, + gridPrefix = type ? type + "Grid" : "grid", + gridLineWidth = options[gridPrefix + "LineWidth"], + gridLineColor = options[gridPrefix + "LineColor"], + dashStyle = options[gridPrefix + "LineDashStyle"]; + if (!gridLine) { + if (!axis.chart.styledMode) { + attribs.stroke = gridLineColor; + attribs["stroke-width"] = gridLineWidth; + if (dashStyle) { + attribs.dashstyle = dashStyle; + } + } + if (!type) { + attribs.zIndex = 1; + } + if (old) { + opacity = 0; + } /** - * Fade out an element by animating its opacity down to 0, and hide it on - * complete. Used internally for the tooltip. - * - * @function Highcharts.SVGElement#fadeOut - * - * @param {number} [duration=150] - * The fade duration in milliseconds. - */ - SVGElement.prototype.fadeOut = function (duration) { - var elemWrapper = this; - elemWrapper.animate({ - opacity: 0 - }, { - duration: pick(duration, 150), - complete: function () { - // #3088, assuming we're only using this for tooltips - elemWrapper.attr({ y: -9999 }).hide(); - } + * The rendered grid line of the tick. + * @name Highcharts.Tick#gridLine + * @type {Highcharts.SVGElement|undefined} + */ + tick.gridLine = gridLine = renderer + .path() + .attr(attribs) + .addClass("highcharts-" + (type ? type + "-" : "") + "grid-line") + .add(axis.gridGroup); + } + if (gridLine) { + gridLinePath = axis.getPlotLinePath({ + value: pos + tickmarkOffset, + lineWidth: gridLine.strokeWidth() * reverseCrisp, + force: "pass", + old: old, + }); + // If the parameter 'old' is set, the current call will be followed + // by another call, therefore do not do any animations this time + if (gridLinePath) { + gridLine[old || tick.isNew ? "attr" : "animate"]({ + d: gridLinePath, + opacity: opacity, + }); + } + } + }; + /** + * Renders the tick mark. + * + * @private + * @param {Highcharts.PositionObject} xy The position vector of the mark + * @param {number} opacity The opacity of the mark + * @param {number} reverseCrisp Modifier for avoiding overlapping 1 or -1 + * @return {void} + */ + Tick.prototype.renderMark = function (xy, opacity, reverseCrisp) { + var tick = this, + axis = tick.axis, + options = axis.options, + renderer = axis.chart.renderer, + type = tick.type, + tickPrefix = type ? type + "Tick" : "tick", + tickSize = axis.tickSize(tickPrefix), + mark = tick.mark, + isNewMark = !mark, + x = xy.x, + y = xy.y, + tickWidth = pick( + options[tickPrefix + "Width"], + !type && axis.isXAxis ? 1 : 0 + ), // X axis defaults to 1 + tickColor = options[tickPrefix + "Color"]; + if (tickSize) { + // negate the length + if (axis.opposite) { + tickSize[0] = -tickSize[0]; + } + // First time, create it + if (isNewMark) { + /** + * The rendered mark of the tick. + * @name Highcharts.Tick#mark + * @type {Highcharts.SVGElement|undefined} + */ + tick.mark = mark = renderer + .path() + .addClass("highcharts-" + (type ? type + "-" : "") + "tick") + .add(axis.axisGroup); + if (!axis.chart.styledMode) { + mark.attr({ + stroke: tickColor, + "stroke-width": tickWidth, }); + } + } + mark[isNewMark ? "attr" : "animate"]({ + d: tick.getMarkPath( + x, + y, + tickSize[0], + mark.strokeWidth() * reverseCrisp, + axis.horiz, + renderer + ), + opacity: opacity, + }); + } + }; + /** + * Renders the tick label. + * Note: The label should already be created in init(), so it should only + * have to be moved into place. + * + * @private + * @param {Highcharts.PositionObject} xy The position vector of the label + * @param {boolean} old Whether or not the tick is old + * @param {number} opacity The opacity of the label + * @param {number} index The index of the tick + * @return {void} + */ + Tick.prototype.renderLabel = function (xy, old, opacity, index) { + var tick = this, + axis = tick.axis, + horiz = axis.horiz, + options = axis.options, + label = tick.label, + labelOptions = options.labels, + step = labelOptions.step, + tickmarkOffset = pick(tick.tickmarkOffset, axis.tickmarkOffset), + show = true, + x = xy.x, + y = xy.y; + if (label && isNumber(x)) { + label.xy = xy = tick.getLabelPosition( + x, + y, + label, + horiz, + labelOptions, + tickmarkOffset, + index, + step + ); + // Apply show first and show last. If the tick is both first and + // last, it is a single centered tick, in which case we show the + // label anyway (#2100). + if ( + (tick.isFirst && + !tick.isLast && + !pick(options.showFirstLabel, 1)) || + (tick.isLast && !tick.isFirst && !pick(options.showLastLabel, 1)) + ) { + show = false; + // Handle label overflow and show or hide accordingly + } else if ( + horiz && + !labelOptions.step && + !labelOptions.rotation && + !old && + opacity !== 0 + ) { + tick.handleOverflow(xy); + } + // apply step + if (step && index % step) { + // show those indices dividable by step + show = false; + } + // Set the new position, and show or hide + if (show && isNumber(xy.y)) { + xy.opacity = opacity; + label[tick.isNewLabel ? "attr" : "animate"](xy); + tick.isNewLabel = false; + } else { + label.attr("y", -9999); // #1338 + tick.isNewLabel = true; + } + } + }; + /** + * Replace labels with the moved ones to perform animation. Additionally + * destroy unused labels. + * + * @private + * @function Highcharts.Tick#replaceMovedLabel + * @return {void} + */ + Tick.prototype.replaceMovedLabel = function () { + var tick = this, + label = tick.label, + axis = tick.axis, + reversed = axis.reversed, + x, + y; + // Animate and destroy + if (label && !tick.isNew) { + x = axis.horiz + ? reversed + ? axis.left + : axis.width + axis.left + : label.xy.x; + y = axis.horiz + ? label.xy.y + : reversed + ? axis.width + axis.top + : axis.top; + label.animate({ x: x, y: y, opacity: 0 }, void 0, label.destroy); + delete tick.label; + } + axis.isDirty = true; + tick.label = tick.movedLabel; + delete tick.movedLabel; + }; + return Tick; + })(); + H.Tick = Tick; + + return H.Tick; + } + ); + _registerModule( + _modules, + "Core/Time.js", + [_modules["Core/Globals.js"], _modules["Core/Utilities.js"]], + function (Highcharts, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + /** + * Normalized interval. + * + * @interface Highcharts.TimeNormalizedObject + */ /** + * The count. + * + * @name Highcharts.TimeNormalizedObject#count + * @type {number} + */ /** + * The interval in axis values (ms). + * + * @name Highcharts.TimeNormalizedObject#unitRange + * @type {number} + */ + /** + * Function of an additional date format specifier. + * + * @callback Highcharts.TimeFormatCallbackFunction + * + * @param {number} timestamp + * The time to format. + * + * @return {string} + * The formatted portion of the date. + */ + /** + * Additonal time tick information. + * + * @interface Highcharts.TimeTicksInfoObject + * @extends Highcharts.TimeNormalizedObject + */ /** + * @name Highcharts.TimeTicksInfoObject#higherRanks + * @type {Array<string>} + */ /** + * @name Highcharts.TimeTicksInfoObject#totalRange + * @type {number} + */ + /** + * Time ticks. + * + * @interface Highcharts.AxisTickPositionsArray + * @extends global.Array<number> + */ /** + * @name Highcharts.AxisTickPositionsArray#info + * @type {Highcharts.TimeTicksInfoObject|undefined} + */ + /** + * A callback to return the time zone offset for a given datetime. It + * takes the timestamp in terms of milliseconds since January 1 1970, + * and returns the timezone offset in minutes. This provides a hook + * for drawing time based charts in specific time zones using their + * local DST crossover dates, with the help of external libraries. + * + * @callback Highcharts.TimezoneOffsetCallbackFunction + * + * @param {number} timestamp + * Timestamp in terms of milliseconds since January 1 1970. + * + * @return {number} + * Timezone offset in minutes. + */ + /** + * Allows to manually load the `moment.js` library from Highcharts options + * instead of the `window`. + * In case of loading the library from a `script` tag, + * this option is not needed, it will be loaded from there by default. + * + * @type {function} + * @since 8.2.0 + * @apioption time.moment + */ + var defined = U.defined, + error = U.error, + extend = U.extend, + isObject = U.isObject, + merge = U.merge, + objectEach = U.objectEach, + pad = U.pad, + pick = U.pick, + splat = U.splat, + timeUnits = U.timeUnits; + var H = Highcharts, + win = H.win; + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The Time class. Time settings are applied in general for each page using + * `Highcharts.setOptions`, or individually for each Chart item through the + * [time](https://api.highcharts.com/highcharts/time) options set. + * + * The Time object is available from {@link Highcharts.Chart#time}, + * which refers to `Highcharts.time` if no individual time settings are + * applied. + * + * @example + * // Apply time settings globally + * Highcharts.setOptions({ + * time: { + * timezone: 'Europe/London' + * } + * }); + * + * // Apply time settings by instance + * var chart = Highcharts.chart('container', { + * time: { + * timezone: 'America/New_York' + * }, + * series: [{ + * data: [1, 4, 3, 5] + * }] + * }); + * + * // Use the Time object + * console.log( + * 'Current time in New York', + * chart.time.dateFormat('%Y-%m-%d %H:%M:%S', Date.now()) + * ); + * + * @since 6.0.5 + * + * @class + * @name Highcharts.Time + * + * @param {Highcharts.TimeOptions} options + * Time options as defined in [chart.options.time](/highcharts/time). + */ + var Time = /** @class */ (function () { + /* * + * + * Constructors + * + * */ + function Time(options) { + /* * + * + * Properties + * + * */ + this.options = {}; + this.useUTC = false; + this.variableTimezone = false; + this.Date = win.Date; + /** + * Get the time zone offset based on the current timezone information as + * set in the global options. + * + * @function Highcharts.Time#getTimezoneOffset + * + * @param {number} timestamp + * The JavaScript timestamp to inspect. + * + * @return {number} + * The timezone offset in minutes compared to UTC. + */ + this.getTimezoneOffset = this.timezoneOffsetFunction(); + this.update(options); + } + /* * + * + * Functions + * + * */ + /** + * Time units used in `Time.get` and `Time.set` + * + * @typedef {"Date"|"Day"|"FullYear"|"Hours"|"Milliseconds"|"Minutes"|"Month"|"Seconds"} Highcharts.TimeUnitValue + */ + /** + * Get the value of a date object in given units, and subject to the Time + * object's current timezone settings. This function corresponds directly to + * JavaScripts `Date.getXXX / Date.getUTCXXX`, so instead of calling + * `date.getHours()` or `date.getUTCHours()` we will call + * `time.get('Hours')`. + * + * @function Highcharts.Time#get + * + * @param {Highcharts.TimeUnitValue} unit + * @param {Date} date + * + * @return {number} + * The given time unit + */ + Time.prototype.get = function (unit, date) { + if (this.variableTimezone || this.timezoneOffset) { + var realMs = date.getTime(); + var ms = realMs - this.getTimezoneOffset(date); + date.setTime(ms); // Temporary adjust to timezone + var ret = date["getUTC" + unit](); + date.setTime(realMs); // Reset + return ret; + } + // UTC time with no timezone handling + if (this.useUTC) { + return date["getUTC" + unit](); + } + // Else, local time + return date["get" + unit](); + }; + /** + * Set the value of a date object in given units, and subject to the Time + * object's current timezone settings. This function corresponds directly to + * JavaScripts `Date.setXXX / Date.setUTCXXX`, so instead of calling + * `date.setHours(0)` or `date.setUTCHours(0)` we will call + * `time.set('Hours', 0)`. + * + * @function Highcharts.Time#set + * + * @param {Highcharts.TimeUnitValue} unit + * @param {Date} date + * @param {number} value + * + * @return {number} + * The epoch milliseconds of the updated date + */ + Time.prototype.set = function (unit, date, value) { + // UTC time with timezone handling + if (this.variableTimezone || this.timezoneOffset) { + // For lower order time units, just set it directly using UTC + // time + if ( + unit === "Milliseconds" || + unit === "Seconds" || + unit === "Minutes" + ) { + return date["setUTC" + unit](value); + } + // Higher order time units need to take the time zone into + // account + // Adjust by timezone + var offset = this.getTimezoneOffset(date); + var ms = date.getTime() - offset; + date.setTime(ms); + date["setUTC" + unit](value); + var newOffset = this.getTimezoneOffset(date); + ms = date.getTime() + newOffset; + return date.setTime(ms); + } + // UTC time with no timezone handling + if (this.useUTC) { + return date["setUTC" + unit](value); + } + // Else, local time + return date["set" + unit](value); + }; + /** + * Update the Time object with current options. It is called internally on + * initializing Highcharts, after running `Highcharts.setOptions` and on + * `Chart.update`. + * + * @private + * @function Highcharts.Time#update + * + * @param {Highcharts.TimeOptions} options + * + * @return {void} + */ + Time.prototype.update = function (options) { + var useUTC = pick(options && options.useUTC, true), + time = this; + this.options = options = merge(true, this.options || {}, options); + // Allow using a different Date class + this.Date = options.Date || win.Date || Date; + this.useUTC = useUTC; + this.timezoneOffset = useUTC && options.timezoneOffset; + this.getTimezoneOffset = this.timezoneOffsetFunction(); + /* + * The time object has options allowing for variable time zones, meaning + * the axis ticks or series data needs to consider this. + */ + this.variableTimezone = !!( + !useUTC || + options.getTimezoneOffset || + options.timezone + ); + }; + /** + * Make a time and returns milliseconds. Interprets the inputs as UTC time, + * local time or a specific timezone time depending on the current time + * settings. + * + * @function Highcharts.Time#makeTime + * + * @param {number} year + * The year + * + * @param {number} month + * The month. Zero-based, so January is 0. + * + * @param {number} [date=1] + * The day of the month + * + * @param {number} [hours=0] + * The hour of the day, 0-23. + * + * @param {number} [minutes=0] + * The minutes + * + * @param {number} [seconds=0] + * The seconds + * + * @return {number} + * The time in milliseconds since January 1st 1970. + */ + Time.prototype.makeTime = function ( + year, + month, + date, + hours, + minutes, + seconds + ) { + var d, offset, newOffset; + if (this.useUTC) { + d = this.Date.UTC.apply(0, arguments); + offset = this.getTimezoneOffset(d); + d += offset; + newOffset = this.getTimezoneOffset(d); + if (offset !== newOffset) { + d += newOffset - offset; + // A special case for transitioning from summer time to winter time. + // When the clock is set back, the same time is repeated twice, i.e. + // 02:30 am is repeated since the clock is set back from 3 am to + // 2 am. We need to make the same time as local Date does. + } else if ( + offset - 36e5 === this.getTimezoneOffset(d - 36e5) && + !H.isSafari + ) { + d -= 36e5; + } + } else { + d = new this.Date( + year, + month, + pick(date, 1), + pick(hours, 0), + pick(minutes, 0), + pick(seconds, 0) + ).getTime(); + } + return d; + }; + /** + * Sets the getTimezoneOffset function. If the `timezone` option is set, a + * default getTimezoneOffset function with that timezone is returned. If + * a `getTimezoneOffset` option is defined, it is returned. If neither are + * specified, the function using the `timezoneOffset` option or 0 offset is + * returned. + * + * @private + * @function Highcharts.Time#timezoneOffsetFunction + * + * @return {Function} + * A getTimezoneOffset function + */ + Time.prototype.timezoneOffsetFunction = function () { + var time = this, + options = this.options, + moment = options.moment || win.moment; + if (!this.useUTC) { + return function (timestamp) { + return new Date(timestamp.toString()).getTimezoneOffset() * 60000; + }; + } + if (options.timezone) { + if (!moment) { + // getTimezoneOffset-function stays undefined because it depends + // on Moment.js + error(25); + } else { + return function (timestamp) { + return ( + -moment.tz(timestamp, options.timezone).utcOffset() * 60000 + ); + }; + } + } + // If not timezone is set, look for the getTimezoneOffset callback + if (this.useUTC && options.getTimezoneOffset) { + return function (timestamp) { + return options.getTimezoneOffset(timestamp.valueOf()) * 60000; + }; + } + // Last, use the `timezoneOffset` option if set + return function () { + return (time.timezoneOffset || 0) * 60000; + }; + }; + /** + * Formats a JavaScript date timestamp (milliseconds since Jan 1st 1970) + * into a human readable date string. The available format keys are listed + * below. Additional formats can be given in the + * {@link Highcharts.dateFormats} hook. + * + * Supported format keys: + * - `%a`: Short weekday, like 'Mon' + * - `%A`: Long weekday, like 'Monday' + * - `%d`: Two digit day of the month, 01 to 31 + * - `%e`: Day of the month, 1 through 31 + * - `%w`: Day of the week, 0 through 6 + * - `%b`: Short month, like 'Jan' + * - `%B`: Long month, like 'January' + * - `%m`: Two digit month number, 01 through 12 + * - `%y`: Two digits year, like 09 for 2009 + * - `%Y`: Four digits year, like 2009 + * - `%H`: Two digits hours in 24h format, 00 through 23 + * - `%k`: Hours in 24h format, 0 through 23 + * - `%I`: Two digits hours in 12h format, 00 through 11 + * - `%l`: Hours in 12h format, 1 through 12 + * - `%M`: Two digits minutes, 00 through 59 + * - `%p`: Upper case AM or PM + * - `%P`: Lower case AM or PM + * - `%S`: Two digits seconds, 00 through 59 + * - `%L`: Milliseconds (naming from Ruby) + * + * @example + * const time = new Highcharts.Time(); + * const s = time.dateFormat('%Y-%m-%d %H:%M:%S', Date.UTC(2020, 0, 1)); + * console.log(s); // => 2020-01-01 00:00:00 + * + * @function Highcharts.Time#dateFormat + * + * @param {string} format + * The desired format where various time representations are + * prefixed with %. + * + * @param {number} timestamp + * The JavaScript timestamp. + * + * @param {boolean} [capitalize=false] + * Upper case first letter in the return. + * + * @return {string} + * The formatted date. + */ + Time.prototype.dateFormat = function (format, timestamp, capitalize) { + var _a; + if (!defined(timestamp) || isNaN(timestamp)) { + return ( + ((_a = H.defaultOptions.lang) === null || _a === void 0 + ? void 0 + : _a.invalidDate) || "" + ); + } + format = pick(format, "%Y-%m-%d %H:%M:%S"); + var time = this, + date = new this.Date(timestamp), + // get the basic time values + hours = this.get("Hours", date), + day = this.get("Day", date), + dayOfMonth = this.get("Date", date), + month = this.get("Month", date), + fullYear = this.get("FullYear", date), + lang = H.defaultOptions.lang, + langWeekdays = + lang === null || lang === void 0 ? void 0 : lang.weekdays, + shortWeekdays = + lang === null || lang === void 0 ? void 0 : lang.shortWeekdays, + // List all format keys. Custom formats can be added from the + // outside. + replacements = extend( + { + // Day + // Short weekday, like 'Mon' + a: shortWeekdays + ? shortWeekdays[day] + : langWeekdays[day].substr(0, 3), + // Long weekday, like 'Monday' + A: langWeekdays[day], + // Two digit day of the month, 01 to 31 + d: pad(dayOfMonth), + // Day of the month, 1 through 31 + e: pad(dayOfMonth, 2, " "), + // Day of the week, 0 through 6 + w: day, + // Week (none implemented) + // 'W': weekNumber(), + // Month + // Short month, like 'Jan' + b: lang.shortMonths[month], + // Long month, like 'January' + B: lang.months[month], + // Two digit month number, 01 through 12 + m: pad(month + 1), + // Month number, 1 through 12 (#8150) + o: month + 1, + // Year + // Two digits year, like 09 for 2009 + y: fullYear.toString().substr(2, 2), + // Four digits year, like 2009 + Y: fullYear, + // Time + // Two digits hours in 24h format, 00 through 23 + H: pad(hours), + // Hours in 24h format, 0 through 23 + k: hours, + // Two digits hours in 12h format, 00 through 11 + I: pad(hours % 12 || 12), + // Hours in 12h format, 1 through 12 + l: hours % 12 || 12, + // Two digits minutes, 00 through 59 + M: pad(this.get("Minutes", date)), + // Upper case AM or PM + p: hours < 12 ? "AM" : "PM", + // Lower case AM or PM + P: hours < 12 ? "am" : "pm", + // Two digits seconds, 00 through 59 + S: pad(date.getSeconds()), + // Milliseconds (naming from Ruby) + L: pad(Math.floor(timestamp % 1000), 3), + }, + H.dateFormats + ); + // Do the replaces + objectEach(replacements, function (val, key) { + // Regex would do it in one line, but this is faster + while (format.indexOf("%" + key) !== -1) { + format = format.replace( + "%" + key, + typeof val === "function" ? val.call(time, timestamp) : val + ); + } + }); + // Optionally capitalize the string and return + return capitalize + ? format.substr(0, 1).toUpperCase() + format.substr(1) + : format; + }; + /** + * Resolve legacy formats of dateTimeLabelFormats (strings and arrays) into + * an object. + * @private + * @param {string|Array<T>|Highcharts.Dictionary<T>} f - General format description + * @return {Highcharts.Dictionary<T>} - The object definition + */ + Time.prototype.resolveDTLFormat = function (f) { + if (!isObject(f, true)) { + // check for string or array + f = splat(f); + return { + main: f[0], + from: f[1], + to: f[2], }; - /** - * @private - * @function Highcharts.SVGElement#fillSetter - * @param {Highcharts.ColorType} value - * @param {string} key - * @param {Highcharts.SVGDOMElement} element - */ - SVGElement.prototype.fillSetter = function (value, key, element) { - if (typeof value === 'string') { - element.setAttribute(key, value); - } - else if (value) { - this.complexColor(value, key, element); - } - }; - /** - * Get the bounding box (width, height, x and y) for the element. Generally - * used to get rendered text size. Since this is called a lot in charts, - * the results are cached based on text properties, in order to save DOM - * traffic. The returned bounding box includes the rotation, so for example - * a single text line of rotation 90 will report a greater height, and a - * width corresponding to the line-height. - * - * @sample highcharts/members/renderer-on-chart/ - * Draw a rectangle based on a text's bounding box - * - * @function Highcharts.SVGElement#getBBox - * - * @param {boolean} [reload] - * Skip the cache and get the updated DOM bouding box. - * - * @param {number} [rot] - * Override the element's rotation. This is internally used on axis - * labels with a value of 0 to find out what the bounding box would - * be have been if it were not rotated. - * - * @return {Highcharts.BBoxObject} - * The bounding box with `x`, `y`, `width` and `height` properties. - */ - SVGElement.prototype.getBBox = function (reload, rot) { - var wrapper = this, - bBox, // = wrapper.bBox, - renderer = wrapper.renderer, - width, - height, - element = wrapper.element, - styles = wrapper.styles, - fontSize, - textStr = wrapper.textStr, - toggleTextShadowShim, - cache = renderer.cache, - cacheKeys = renderer.cacheKeys, - isSVG = element.namespaceURI === wrapper.SVG_NS, - cacheKey; - var rotation = pick(rot, - wrapper.rotation, 0); - fontSize = renderer.styledMode ? (element && - SVGElement.prototype.getStyle.call(element, 'font-size')) : (styles && styles.fontSize); - // Avoid undefined and null (#7316) - if (defined(textStr)) { - cacheKey = textStr.toString(); - // Since numbers are monospaced, and numerical labels appear a lot - // in a chart, we assume that a label of n characters has the same - // bounding box as others of the same length. Unless there is inner - // HTML in the label. In that case, leave the numbers as is (#5899). - if (cacheKey.indexOf('<') === -1) { - cacheKey = cacheKey.replace(/[0-9]/g, '0'); - } - // Properties that affect bounding box - cacheKey += [ - '', - rotation, - fontSize, - wrapper.textWidth, - styles && styles.textOverflow, - styles && styles.fontWeight // #12163 - ].join(','); - } - if (cacheKey && !reload) { - bBox = cache[cacheKey]; - } - // No cache found - if (!bBox) { - // SVG elements - if (isSVG || renderer.forExport) { - try { // Fails in Firefox if the container has display: none. - // When the text shadow shim is used, we need to hide the - // fake shadows to get the correct bounding box (#3872) - toggleTextShadowShim = this.fakeTS && function (display) { - [].forEach.call(element.querySelectorAll('.highcharts-text-outline'), function (tspan) { - tspan.style.display = display; - }); - }; - // Workaround for #3842, Firefox reporting wrong bounding - // box for shadows - if (isFunction(toggleTextShadowShim)) { - toggleTextShadowShim('none'); - } - bBox = element.getBBox ? - // SVG: use extend because IE9 is not allowed to change - // width and height in case of rotation (below) - extend({}, element.getBBox()) : { - // Legacy IE in export mode - width: element.offsetWidth, - height: element.offsetHeight - }; - // #3842 - if (isFunction(toggleTextShadowShim)) { - toggleTextShadowShim(''); - } - } - catch (e) { - ''; - } - // If the bBox is not set, the try-catch block above failed. The - // other condition is for Opera that returns a width of - // -Infinity on hidden elements. - if (!bBox || bBox.width < 0) { - bBox = { width: 0, height: 0 }; - } - // VML Renderer or useHTML within SVG - } - else { - bBox = wrapper.htmlGetBBox(); - } - // True SVG elements as well as HTML elements in modern browsers - // using the .useHTML option need to compensated for rotation - if (renderer.isSVG) { - width = bBox.width; - height = bBox.height; - // Workaround for wrong bounding box in IE, Edge and Chrome on - // Windows. With Highcharts' default font, IE and Edge report - // a box height of 16.899 and Chrome rounds it to 17. If this - // stands uncorrected, it results in more padding added below - // the text than above when adding a label border or background. - // Also vertical positioning is affected. - // https://jsfiddle.net/highcharts/em37nvuj/ - // (#1101, #1505, #1669, #2568, #6213). - if (isSVG) { - bBox.height = height = ({ - '11px,17': 14, - '13px,20': 16 - }[styles && - styles.fontSize + ',' + Math.round(height)] || - height); - } - // Adjust for rotated text - if (rotation) { - var rad = rotation * deg2rad; - bBox.width = Math.abs(height * Math.sin(rad)) + - Math.abs(width * Math.cos(rad)); - bBox.height = Math.abs(height * Math.cos(rad)) + - Math.abs(width * Math.sin(rad)); - } - } - // Cache it. When loading a chart in a hidden iframe in Firefox and - // IE/Edge, the bounding box height is 0, so don't cache it (#5620). - if (cacheKey && bBox.height > 0) { - // Rotate (#4681) - while (cacheKeys.length > 250) { - delete cache[cacheKeys.shift()]; - } - if (!cache[cacheKey]) { - cacheKeys.push(cacheKey); - } - cache[cacheKey] = bBox; - } - } - return bBox; - }; - /** - * Get the computed style. Only in styled mode. - * - * @example - * chart.series[0].points[0].graphic.getStyle('stroke-width'); // => '1px' - * - * @function Highcharts.SVGElement#getStyle - * - * @param {string} prop - * The property name to check for. - * - * @return {string} - * The current computed value. - */ - SVGElement.prototype.getStyle = function (prop) { - return win - .getComputedStyle(this.element || this, '') - .getPropertyValue(prop); - }; - /** - * Check if an element has the given class name. - * - * @function Highcharts.SVGElement#hasClass - * - * @param {string} className - * The class name to check for. - * - * @return {boolean} - * Whether the class name is found. - */ - SVGElement.prototype.hasClass = function (className) { - return ('' + this.attr('class')) - .split(' ') - .indexOf(className) !== -1; - }; - /** - * Hide the element, similar to setting the `visibility` attribute to - * `hidden`. - * - * @function Highcharts.SVGElement#hide - * - * @param {boolean} [hideByTranslation=false] - * The flag to determine if element should be hidden by moving out - * of the viewport. Used for example for dataLabels. - * - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. - */ - SVGElement.prototype.hide = function (hideByTranslation) { - if (hideByTranslation) { - this.attr({ y: -9999 }); - } - else { - this.attr({ visibility: 'hidden' }); - } - return this; - }; - /** - * @private - */ - SVGElement.prototype.htmlGetBBox = function () { - return { height: 0, width: 0, x: 0, y: 0 }; - }; - /** - * Initialize the SVG element. This function only exists to make the - * initialization process overridable. It should not be called directly. - * - * @function Highcharts.SVGElement#init - * - * @param {Highcharts.SVGRenderer} renderer - * The SVGRenderer instance to initialize to. - * - * @param {string} nodeName - * The SVG node name. - */ - SVGElement.prototype.init = function (renderer, nodeName) { - /** - * The primary DOM node. Each `SVGElement` instance wraps a main DOM - * node, but may also represent more nodes. - * - * @name Highcharts.SVGElement#element - * @type {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} - */ - this.element = nodeName === 'span' ? - createElement(nodeName) : - doc.createElementNS(this.SVG_NS, nodeName); - /** - * The renderer that the SVGElement belongs to. - * - * @name Highcharts.SVGElement#renderer - * @type {Highcharts.SVGRenderer} - */ - this.renderer = renderer; - fireEvent(this, 'afterInit'); - }; - /** - * Invert a group, rotate and flip. This is used internally on inverted - * charts, where the points and graphs are drawn as if not inverted, then - * the series group elements are inverted. - * - * @function Highcharts.SVGElement#invert - * - * @param {boolean} inverted - * Whether to invert or not. An inverted shape can be un-inverted by - * setting it to false. - * - * @return {Highcharts.SVGElement} - * Return the SVGElement for chaining. - */ - SVGElement.prototype.invert = function (inverted) { - var wrapper = this; - wrapper.inverted = inverted; - wrapper.updateTransform(); - return wrapper; - }; - /** - * Add an event listener. This is a simple setter that replaces all other - * events of the same type, opposed to the {@link Highcharts#addEvent} - * function. - * - * @sample highcharts/members/element-on/ - * A clickable rectangle - * - * @function Highcharts.SVGElement#on - * - * @param {string} eventType - * The event type. If the type is `click`, Highcharts will internally - * translate it to a `touchstart` event on touch devices, to prevent the - * browser from waiting for a click event from firing. - * - * @param {Function} handler - * The handler callback. - * - * @return {Highcharts.SVGElement} - * The SVGElement for chaining. - */ - SVGElement.prototype.on = function (eventType, handler) { - var svgElement = this, - element = svgElement.element, - touchStartPos, - touchEventFired; - // touch - if (hasTouch && eventType === 'click') { - element.ontouchstart = function (e) { - // save touch position for later calculation - touchStartPos = { - clientX: e.touches[0].clientX, - clientY: e.touches[0].clientY - }; - }; - // Instead of ontouchstart, event handlers should be called - // on touchend - similar to how current mouseup events are called - element.ontouchend = function (e) { - // hasMoved is a boolean variable containing logic if page - // was scrolled, so if touch position changed more than - // ~4px (value borrowed from general touch handler) - var hasMoved = touchStartPos.clientX ? Math.sqrt(Math.pow(touchStartPos.clientX - e.changedTouches[0].clientX, 2) + - Math.pow(touchStartPos.clientY - e.changedTouches[0].clientY, 2)) >= 4 : false; - if (!hasMoved) { // only call handlers if page was not scrolled - handler.call(element, e); - } - touchEventFired = true; - if (e.cancelable !== false) { - // prevent other events from being fired. #9682 - e.preventDefault(); - } - }; - element.onclick = function (e) { - // Do not call onclick handler if touch event was fired already. - if (!touchEventFired) { - handler.call(element, e); - } - }; - } - else { - // simplest possible event model for internal use - element['on' + eventType] = handler; - } - return this; - }; - /** - * @private - * @function Highcharts.SVGElement#opacitySetter - * @param {string} value - * @param {string} key - * @param {Highcharts.SVGDOMElement} element - */ - SVGElement.prototype.opacitySetter = function (value, key, element) { - // Round off to avoid float errors, like tests where opacity lands on - // 9.86957e-06 instead of 0 - var opacity = Number(Number(value).toFixed(3)); - this.opacity = opacity; - element.setAttribute(key, opacity); - }; - /** - * Remove a class name from the element. - * - * @function Highcharts.SVGElement#removeClass - * - * @param {string|RegExp} className - * The class name to remove. - * - * @return {Highcharts.SVGElement} Returns the SVG element for chainability. - */ - SVGElement.prototype.removeClass = function (className) { - return this.attr('class', ('' + this.attr('class')) - .replace(isString(className) ? - new RegExp("(^| )" + className + "( |$)") : // #12064, #13590 - className, ' ') - .replace(/ +/g, ' ') - .trim()); - }; - /** - * @private - * @param {Array<Highcharts.SVGDOMElement>} tspans - * Text spans. - */ - SVGElement.prototype.removeTextOutline = function (tspans) { - // Iterate from the end to - // support removing items inside the cycle (#6472). - var i = tspans.length, - tspan; - while (i--) { - tspan = tspans[i]; - if (tspan.getAttribute('class') === 'highcharts-text-outline') { - // Remove then erase - erase(tspans, this.element.removeChild(tspan)); - } + } + return f; + }; + /** + * Return an array with time positions distributed on round time values + * right and right after min and max. Used in datetime axes as well as for + * grouping data on a datetime axis. + * + * @function Highcharts.Time#getTimeTicks + * + * @param {Highcharts.TimeNormalizedObject} normalizedInterval + * The interval in axis values (ms) and the count + * + * @param {number} [min] + * The minimum in axis values + * + * @param {number} [max] + * The maximum in axis values + * + * @param {number} [startOfWeek=1] + * + * @return {Highcharts.AxisTickPositionsArray} + */ + Time.prototype.getTimeTicks = function ( + normalizedInterval, + min, + max, + startOfWeek + ) { + var time = this, + Date = time.Date, + tickPositions = [], + i, + higherRanks = {}, + minYear, // used in months and years as a basis for Date.UTC() + // When crossing DST, use the max. Resolves #6278. + minDate = new Date(min), + interval = normalizedInterval.unitRange, + count = normalizedInterval.count || 1, + variableDayLength, + minDay; + startOfWeek = pick(startOfWeek, 1); + if (defined(min)) { + // #1300 + time.set( + "Milliseconds", + minDate, + interval >= timeUnits.second + ? 0 // #3935 + : count * Math.floor(time.get("Milliseconds", minDate) / count) + ); // #3652, #3654 + if (interval >= timeUnits.second) { + // second + time.set( + "Seconds", + minDate, + interval >= timeUnits.minute + ? 0 // #3935 + : count * Math.floor(time.get("Seconds", minDate) / count) + ); + } + if (interval >= timeUnits.minute) { + // minute + time.set( + "Minutes", + minDate, + interval >= timeUnits.hour + ? 0 + : count * Math.floor(time.get("Minutes", minDate) / count) + ); + } + if (interval >= timeUnits.hour) { + // hour + time.set( + "Hours", + minDate, + interval >= timeUnits.day + ? 0 + : count * Math.floor(time.get("Hours", minDate) / count) + ); + } + if (interval >= timeUnits.day) { + // day + time.set( + "Date", + minDate, + interval >= timeUnits.month + ? 1 + : Math.max( + 1, + count * Math.floor(time.get("Date", minDate) / count) + ) + ); + } + if (interval >= timeUnits.month) { + // month + time.set( + "Month", + minDate, + interval >= timeUnits.year + ? 0 + : count * Math.floor(time.get("Month", minDate) / count) + ); + minYear = time.get("FullYear", minDate); + } + if (interval >= timeUnits.year) { + // year + minYear -= minYear % count; + time.set("FullYear", minDate, minYear); + } + // week is a special case that runs outside the hierarchy + if (interval === timeUnits.week) { + // get start of current week, independent of count + minDay = time.get("Day", minDate); + time.set( + "Date", + minDate, + time.get("Date", minDate) - + minDay + + startOfWeek + + // We don't want to skip days that are before + // startOfWeek (#7051) + (minDay < startOfWeek ? -7 : 0) + ); + } + // Get basics for variable time spans + minYear = time.get("FullYear", minDate); + var minMonth = time.get("Month", minDate), + minDateDate = time.get("Date", minDate), + minHours = time.get("Hours", minDate); + // Redefine min to the floored/rounded minimum time (#7432) + min = minDate.getTime(); + // Handle local timezone offset + if (time.variableTimezone) { + // Detect whether we need to take the DST crossover into + // consideration. If we're crossing over DST, the day length may + // be 23h or 25h and we need to compute the exact clock time for + // each tick instead of just adding hours. This comes at a cost, + // so first we find out if it is needed (#4951). + variableDayLength = + // Long range, assume we're crossing over. + max - min > 4 * timeUnits.month || + // Short range, check if min and max are in different time + // zones. + time.getTimezoneOffset(min) !== time.getTimezoneOffset(max); + } + // Iterate and add tick positions at appropriate values + var t = minDate.getTime(); + i = 1; + while (t < max) { + tickPositions.push(t); + // if the interval is years, use Date.UTC to increase years + if (interval === timeUnits.year) { + t = time.makeTime(minYear + i * count, 0); + // if the interval is months, use Date.UTC to increase months + } else if (interval === timeUnits.month) { + t = time.makeTime(minYear, minMonth + i * count); + // if we're using global time, the interval is not fixed as it + // jumps one hour at the DST crossover + } else if ( + variableDayLength && + (interval === timeUnits.day || interval === timeUnits.week) + ) { + t = time.makeTime( + minYear, + minMonth, + minDateDate + i * count * (interval === timeUnits.day ? 1 : 7) + ); + } else if ( + variableDayLength && + interval === timeUnits.hour && + count > 1 + ) { + // make sure higher ranks are preserved across DST (#6797, + // #7621) + t = time.makeTime( + minYear, + minMonth, + minDateDate, + minHours + i * count + ); + // else, the interval is fixed and we use simple addition + } else { + t += interval * count; + } + i++; + } + // push the last time + tickPositions.push(t); + // Handle higher ranks. Mark new days if the time is on midnight + // (#950, #1649, #1760, #3349). Use a reasonable dropout threshold + // to prevent looping over dense data grouping (#6156). + if (interval <= timeUnits.hour && tickPositions.length < 10000) { + tickPositions.forEach(function (t) { + if ( + // Speed optimization, no need to run dateFormat unless + // we're on a full or half hour + t % 1800000 === 0 && + // Check for local or global midnight + time.dateFormat("%H%M%S%L", t) === "000000000" + ) { + higherRanks[t] = "day"; } - }; - /** - * Removes an element from the DOM. - * - * @private - * @function Highcharts.SVGElement#safeRemoveChild - * - * @param {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} element - * The DOM node to remove. - */ - SVGElement.prototype.safeRemoveChild = function (element) { - var parentNode = element.parentNode; - if (parentNode) { - parentNode.removeChild(element); - } - }; - /** - * Set the coordinates needed to draw a consistent radial gradient across - * a shape regardless of positioning inside the chart. Used on pie slices - * to make all the slices have the same radial reference point. - * - * @function Highcharts.SVGElement#setRadialReference - * - * @param {Array<number>} coordinates - * The center reference. The format is `[centerX, centerY, diameter]` in - * pixels. - * - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. - */ - SVGElement.prototype.setRadialReference = function (coordinates) { - var existingGradient = (this.element.gradient && - this.renderer.gradients[this.element.gradient]); - this.element.radialReference = coordinates; - // On redrawing objects with an existing gradient, the gradient needs - // to be repositioned (#3801) - if (existingGradient && existingGradient.radAttr) { - existingGradient.animate(this.renderer.getRadialAttr(coordinates, existingGradient.radAttr)); - } - return this; - }; - /** - * @private - * @function Highcharts.SVGElement#setTextPath - * @param {Highcharts.SVGElement} path - * Path to follow. - * @param {Highcharts.DataLabelsTextPathOptionsObject} textPathOptions - * Options. - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. - */ - SVGElement.prototype.setTextPath = function (path, textPathOptions) { - var elem = this.element, - attribsMap = { - textAnchor: 'text-anchor' - }, - attrs, - adder = false, - textPathElement, - textPathId, - textPathWrapper = this.textPathWrapper, - tspans, - firstTime = !textPathWrapper; - // Defaults - textPathOptions = merge(true, { - enabled: true, - attributes: { - dy: -5, - startOffset: '50%', - textAnchor: 'middle' - } - }, textPathOptions); - attrs = textPathOptions.attributes; - if (path && textPathOptions && textPathOptions.enabled) { - // In case of fixed width for a text, string is rebuilt - // (e.g. ellipsis is applied), so we need to rebuild textPath too - if (textPathWrapper && - textPathWrapper.element.parentNode === null) { - // When buildText functionality was triggered again - // and deletes textPathWrapper parentNode - firstTime = true; - textPathWrapper = textPathWrapper.destroy(); - } - else if (textPathWrapper) { - // Case after drillup when spans were added into - // the DOM outside the textPathWrapper parentGroup - this.removeTextOutline.call(textPathWrapper.parentGroup, [].slice.call(elem.getElementsByTagName('tspan'))); - } - // label() has padding, text() doesn't - if (this.options && this.options.padding) { - attrs.dx = -this.options.padding; - } - if (!textPathWrapper) { - // Create <textPath>, defer the DOM adder - this.textPathWrapper = textPathWrapper = - this.renderer.createElement('textPath'); - adder = true; - } - textPathElement = textPathWrapper.element; - // Set ID for the path - textPathId = path.element.getAttribute('id'); - if (!textPathId) { - path.element.setAttribute('id', textPathId = uniqueKey()); - } - // Change DOM structure, by placing <textPath> tag in <text> - if (firstTime) { - tspans = elem.getElementsByTagName('tspan'); - // Now move all <tspan>'s to the <textPath> node - while (tspans.length) { - // Remove "y" from tspans, as Firefox translates them - tspans[0].setAttribute('y', 0); - // Remove "x" from tspans - if (isNumber(attrs.dx)) { - tspans[0].setAttribute('x', -attrs.dx); - } - textPathElement.appendChild(tspans[0]); - } - } - // Add <textPath> to the DOM - if (adder && - textPathWrapper) { - textPathWrapper.add({ - // label() is placed in a group, text() is standalone - element: this.text ? this.text.element : elem - }); - } - // Set basic options: - // Use `setAttributeNS` because Safari needs this.. - textPathElement.setAttributeNS('http://www.w3.org/1999/xlink', 'href', this.renderer.url + '#' + textPathId); - // Presentation attributes: - // dx/dy options must by set on <text> (parent), - // the rest should be set on <textPath> - if (defined(attrs.dy)) { - textPathElement.parentNode - .setAttribute('dy', attrs.dy); - delete attrs.dy; - } - if (defined(attrs.dx)) { - textPathElement.parentNode - .setAttribute('dx', attrs.dx); - delete attrs.dx; - } - // Additional attributes - objectEach(attrs, function (val, key) { - textPathElement.setAttribute(attribsMap[key] || key, val); - }); - // Remove translation, text that follows path does not need that - elem.removeAttribute('transform'); - // Remove shadows and text outlines - this.removeTextOutline.call(textPathWrapper, [].slice.call(elem.getElementsByTagName('tspan'))); - // Remove background and border for label(), see #10545 - // Alternatively, we can disable setting background rects in - // series.drawDataLabels() - if (this.text && !this.renderer.styledMode) { - this.attr({ - fill: 'none', - 'stroke-width': 0 - }); - } - // Disable some functions - this.updateTransform = noop; - this.applyTextOutline = noop; - } - else if (textPathWrapper) { - // Reset to prototype - delete this.updateTransform; - delete this.applyTextOutline; - // Restore DOM structure: - this.destroyTextPath(elem, path); - // Bring attributes back - this.updateTransform(); - // Set textOutline back for text() - if (this.options && this.options.rotation) { - this.applyTextOutline(this.options.style.textOutline); - } - } - return this; - }; - /** - * Add a shadow to the element. Must be called after the element is added to - * the DOM. In styled mode, this method is not used, instead use `defs` and - * filters. - * - * @example - * renderer.rect(10, 100, 100, 100) - * .attr({ fill: 'red' }) - * .shadow(true); - * - * @function Highcharts.SVGElement#shadow - * - * @param {boolean|Highcharts.ShadowOptionsObject} [shadowOptions] - * The shadow options. If `true`, the default options are applied. If - * `false`, the current shadow will be removed. - * - * @param {Highcharts.SVGElement} [group] - * The SVG group element where the shadows will be applied. The - * default is to add it to the same parent as the current element. - * Internally, this is ised for pie slices, where all the shadows are - * added to an element behind all the slices. - * - * @param {boolean} [cutOff] - * Used internally for column shadows. - * - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. - */ - SVGElement.prototype.shadow = function (shadowOptions, group, cutOff) { - var shadows = [], - i, - shadow, - element = this.element, - strokeWidth, - shadowElementOpacity, - update = false, - oldShadowOptions = this.oldShadowOptions, - // compensate for inverted plot area - transform; - var defaultShadowOptions = { - color: '#000000', - offsetX: 1, - offsetY: 1, - opacity: 0.15, - width: 3 - }; - var options; - if (shadowOptions === true) { - options = defaultShadowOptions; - } - else if (typeof shadowOptions === 'object') { - options = extend(defaultShadowOptions, shadowOptions); - } - // Update shadow when options change (#12091). - if (options) { - // Go over each key to look for change - if (options && oldShadowOptions) { - objectEach(options, function (value, key) { - if (value !== oldShadowOptions[key]) { - update = true; - } - }); - } - if (update) { - this.destroyShadows(); - } - this.oldShadowOptions = options; - } - if (!options) { - this.destroyShadows(); - } - else if (!this.shadows) { - shadowElementOpacity = options.opacity / options.width; - transform = this.parentInverted ? - 'translate(-1,-1)' : - "translate(" + options.offsetX + ", " + options.offsetY + ")"; - for (i = 1; i <= options.width; i++) { - shadow = element.cloneNode(false); - strokeWidth = (options.width * 2) + 1 - (2 * i); - attr(shadow, { - stroke: (shadowOptions.color || - '#000000'), - 'stroke-opacity': shadowElementOpacity * i, - 'stroke-width': strokeWidth, - transform: transform, - fill: 'none' - }); - shadow.setAttribute('class', (shadow.getAttribute('class') || '') + ' highcharts-shadow'); - if (cutOff) { - attr(shadow, 'height', Math.max(attr(shadow, 'height') - strokeWidth, 0)); - shadow.cutHeight = strokeWidth; - } - if (group) { - group.element.appendChild(shadow); - } - else if (element.parentNode) { - element.parentNode.insertBefore(shadow, element); - } - shadows.push(shadow); - } - this.shadows = shadows; - } - return this; - }; - /** - * Show the element after it has been hidden. - * - * @function Highcharts.SVGElement#show - * - * @param {boolean} [inherit=false] - * Set the visibility attribute to `inherit` rather than `visible`. - * The difference is that an element with `visibility="visible"` - * will be visible even if the parent is hidden. - * - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. - */ - SVGElement.prototype.show = function (inherit) { - return this.attr({ visibility: inherit ? 'inherit' : 'visible' }); - }; - /** - * WebKit and Batik have problems with a stroke-width of zero, so in this - * case we remove the stroke attribute altogether. #1270, #1369, #3065, - * #3072. - * - * @private - * @function Highcharts.SVGElement#strokeSetter - * @param {number|string} value - * @param {string} key - * @param {Highcharts.SVGDOMElement} element - */ - SVGElement.prototype.strokeSetter = function (value, key, element) { - this[key] = value; - // Only apply the stroke attribute if the stroke width is defined and - // larger than 0 - if (this.stroke && this['stroke-width']) { - // Use prototype as instance may be overridden - SVGElement.prototype.fillSetter.call(this, this.stroke, 'stroke', element); - element.setAttribute('stroke-width', this['stroke-width']); - this.hasStroke = true; - } - else if (key === 'stroke-width' && value === 0 && this.hasStroke) { - element.removeAttribute('stroke'); - this.hasStroke = false; - } - else if (this.renderer.styledMode && this['stroke-width']) { - element.setAttribute('stroke-width', this['stroke-width']); - this.hasStroke = true; - } - }; - /** - * Get the computed stroke width in pixel values. This is used extensively - * when drawing shapes to ensure the shapes are rendered crisp and - * positioned correctly relative to each other. Using - * `shape-rendering: crispEdges` leaves us less control over positioning, - * for example when we want to stack columns next to each other, or position - * things pixel-perfectly within the plot box. - * - * The common pattern when placing a shape is: - * - Create the SVGElement and add it to the DOM. In styled mode, it will - * now receive a stroke width from the style sheet. In classic mode we - * will add the `stroke-width` attribute. - * - Read the computed `elem.strokeWidth()`. - * - Place it based on the stroke width. - * - * @function Highcharts.SVGElement#strokeWidth - * - * @return {number} - * The stroke width in pixels. Even if the given stroke widtch (in CSS or by - * attributes) is based on `em` or other units, the pixel size is returned. - */ - SVGElement.prototype.strokeWidth = function () { - // In non-styled mode, read the stroke width as set by .attr - if (!this.renderer.styledMode) { - return this['stroke-width'] || 0; - } - // In styled mode, read computed stroke width - var val = this.getStyle('stroke-width'), - ret = 0, - dummy; - // Read pixel values directly - if (val.indexOf('px') === val.length - 2) { - ret = pInt(val); - // Other values like em, pt etc need to be measured - } - else if (val !== '') { - dummy = doc.createElementNS(SVG_NS, 'rect'); - attr(dummy, { - width: val, - 'stroke-width': 0 - }); - this.element.parentNode.appendChild(dummy); - ret = dummy.getBBox().width; - dummy.parentNode.removeChild(dummy); - } - return ret; - }; + }); + } + } + // record information on the chosen unit - for dynamic label formatter + tickPositions.info = extend(normalizedInterval, { + higherRanks: higherRanks, + totalRange: interval * count, + }); + return tickPositions; + }; + return Time; + })(); + H.Time = Time; + + return H.Time; + } + ); + _registerModule( + _modules, + "Core/Options.js", + [ + _modules["Core/Globals.js"], + _modules["Core/Color/Color.js"], + _modules["Core/Time.js"], + _modules["Core/Utilities.js"], + ], + function (H, Color, Time, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var isTouchDevice = H.isTouchDevice, + svg = H.svg; + var color = Color.parse; + var merge = U.merge; + /** + * @typedef {"plotBox"|"spacingBox"} Highcharts.ButtonRelativeToValue + */ + /** + * Gets fired when a series is added to the chart after load time, using the + * `addSeries` method. Returning `false` prevents the series from being added. + * + * @callback Highcharts.ChartAddSeriesCallbackFunction + * + * @param {Highcharts.Chart} this + * The chart on which the event occured. + * + * @param {Highcharts.ChartAddSeriesEventObject} event + * The event that occured. + */ + /** + * Contains common event information. Through the `options` property you can + * access the series options that were passed to the `addSeries` method. + * + * @interface Highcharts.ChartAddSeriesEventObject + */ /** + * The series options that were passed to the `addSeries` method. + * @name Highcharts.ChartAddSeriesEventObject#options + * @type {Highcharts.SeriesOptionsType} + */ /** + * Prevents the default behaviour of the event. + * @name Highcharts.ChartAddSeriesEventObject#preventDefault + * @type {Function} + */ /** + * The event target. + * @name Highcharts.ChartAddSeriesEventObject#target + * @type {Highcharts.Chart} + */ /** + * The event type. + * @name Highcharts.ChartAddSeriesEventObject#type + * @type {"addSeries"} + */ + /** + * Gets fired when clicking on the plot background. + * + * @callback Highcharts.ChartClickCallbackFunction + * + * @param {Highcharts.Chart} this + * The chart on which the event occured. + * + * @param {Highcharts.PointerEventObject} event + * The event that occured. + */ + /** + * Contains an axes of the clicked spot. + * + * @interface Highcharts.ChartClickEventAxisObject + */ /** + * Axis at the clicked spot. + * @name Highcharts.ChartClickEventAxisObject#axis + * @type {Highcharts.Axis} + */ /** + * Axis value at the clicked spot. + * @name Highcharts.ChartClickEventAxisObject#value + * @type {number} + */ + /** + * Contains information about the clicked spot on the chart. Remember the unit + * of a datetime axis is milliseconds since 1970-01-01 00:00:00. + * + * @interface Highcharts.ChartClickEventObject + * @extends Highcharts.PointerEventObject + */ /** + * Information about the x-axis on the clicked spot. + * @name Highcharts.ChartClickEventObject#xAxis + * @type {Array<Highcharts.ChartClickEventAxisObject>} + */ /** + * Information about the y-axis on the clicked spot. + * @name Highcharts.ChartClickEventObject#yAxis + * @type {Array<Highcharts.ChartClickEventAxisObject>} + */ /** + * Information about the z-axis on the clicked spot. + * @name Highcharts.ChartClickEventObject#zAxis + * @type {Array<Highcharts.ChartClickEventAxisObject>|undefined} + */ + /** + * Gets fired when the chart is finished loading. + * + * @callback Highcharts.ChartLoadCallbackFunction + * + * @param {Highcharts.Chart} this + * The chart on which the event occured. + * + * @param {global.Event} event + * The event that occured. + */ + /** + * Fires when the chart is redrawn, either after a call to `chart.redraw()` or + * after an axis, series or point is modified with the `redraw` option set to + * `true`. + * + * @callback Highcharts.ChartRedrawCallbackFunction + * + * @param {Highcharts.Chart} this + * The chart on which the event occured. + * + * @param {global.Event} event + * The event that occured. + */ + /** + * Gets fired after initial load of the chart (directly after the `load` event), + * and after each redraw (directly after the `redraw` event). + * + * @callback Highcharts.ChartRenderCallbackFunction + * + * @param {Highcharts.Chart} this + * The chart on which the event occured. + * + * @param {global.Event} event + * The event that occured. + */ + /** + * Gets fired when an area of the chart has been selected. The default action + * for the selection event is to zoom the chart to the selected area. It can be + * prevented by calling `event.preventDefault()` or return false. + * + * @callback Highcharts.ChartSelectionCallbackFunction + * + * @param {Highcharts.Chart} this + * The chart on which the event occured. + * + * @param {global.ChartSelectionContextObject} event + * Event informations + * + * @return {boolean|undefined} + * Return false to prevent the default action, usually zoom. + */ + /** + * The primary axes are `xAxis[0]` and `yAxis[0]`. Remember the unit of a + * datetime axis is milliseconds since 1970-01-01 00:00:00. + * + * @interface Highcharts.ChartSelectionContextObject + * @extends global.Event + */ /** + * Arrays containing the axes of each dimension and each axis' min and max + * values. + * @name Highcharts.ChartSelectionContextObject#xAxis + * @type {Array<Highcharts.ChartSelectionAxisContextObject>} + */ /** + * Arrays containing the axes of each dimension and each axis' min and max + * values. + * @name Highcharts.ChartSelectionContextObject#yAxis + * @type {Array<Highcharts.ChartSelectionAxisContextObject>} + */ + /** + * Axis context of the selection. + * + * @interface Highcharts.ChartSelectionAxisContextObject + */ /** + * The selected Axis. + * @name Highcharts.ChartSelectionAxisContextObject#axis + * @type {Highcharts.Axis} + */ /** + * The maximum axis value, either automatic or set manually. + * @name Highcharts.ChartSelectionAxisContextObject#max + * @type {number} + */ /** + * The minimum axis value, either automatic or set manually. + * @name Highcharts.ChartSelectionAxisContextObject#min + * @type {number} + */ + (""); // detach doclets above + /* ************************************************************************** * + * Handle the options * + * ************************************************************************** */ + /** + * Global default settings. + * + * @name Highcharts.defaultOptions + * @type {Highcharts.Options} + */ /** + * @optionparent + */ + H.defaultOptions = { + /** + * An array containing the default colors for the chart's series. When + * all colors are used, new colors are pulled from the start again. + * + * Default colors can also be set on a series or series.type basis, + * see [column.colors](#plotOptions.column.colors), + * [pie.colors](#plotOptions.pie.colors). + * + * In styled mode, the colors option doesn't exist. Instead, colors + * are defined in CSS and applied either through series or point class + * names, or through the [chart.colorCount](#chart.colorCount) option. + * + * + * ### Legacy + * + * In Highcharts 3.x, the default colors were: + * ```js + * colors: ['#2f7ed8', '#0d233a', '#8bbc21', '#910000', '#1aadce', + * '#492970', '#f28f43', '#77a1e5', '#c42525', '#a6c96a'] + * ``` + * + * In Highcharts 2.x, the default colors were: + * ```js + * colors: ['#4572A7', '#AA4643', '#89A54E', '#80699B', '#3D96AE', + * '#DB843D', '#92A8CD', '#A47D7C', '#B5CA92'] + * ``` + * + * @sample {highcharts} highcharts/chart/colors/ + * Assign a global color theme + * + * @type {Array<Highcharts.ColorString>} + * @default ["#7cb5ec", "#434348", "#90ed7d", "#f7a35c", "#8085e9", + * "#f15c80", "#e4d354", "#2b908f", "#f45b5b", "#91e8e1"] + */ + colors: + "#7cb5ec #434348 #90ed7d #f7a35c #8085e9 #f15c80 #e4d354 #2b908f #f45b5b #91e8e1".split( + " " + ), + /** + * Styled mode only. Configuration object for adding SVG definitions for + * reusable elements. See [gradients, shadows and + * patterns](https://www.highcharts.com/docs/chart-design-and-style/gradients-shadows-and-patterns) + * for more information and code examples. + * + * @type {*} + * @since 5.0.0 + * @apioption defs + */ + /** + * @ignore-option + */ + symbols: ["circle", "diamond", "square", "triangle", "triangle-down"], + /** + * The language object is global and it can't be set on each chart + * initialization. Instead, use `Highcharts.setOptions` to set it before any + * chart is initialized. + * + * ```js + * Highcharts.setOptions({ + * lang: { + * months: [ + * 'Janvier', 'Février', 'Mars', 'Avril', + * 'Mai', 'Juin', 'Juillet', 'Août', + * 'Septembre', 'Octobre', 'Novembre', 'Décembre' + * ], + * weekdays: [ + * 'Dimanche', 'Lundi', 'Mardi', 'Mercredi', + * 'Jeudi', 'Vendredi', 'Samedi' + * ] + * } + * }); + * ``` + */ + lang: { + /** + * The loading text that appears when the chart is set into the loading + * state following a call to `chart.showLoading`. + */ + loading: "Loading...", + /** + * An array containing the months names. Corresponds to the `%B` format + * in `Highcharts.dateFormat()`. + * + * @type {Array<string>} + * @default ["January", "February", "March", "April", "May", "June", + * "July", "August", "September", "October", "November", + * "December"] + */ + months: [ + "January", + "February", + "March", + "April", + "May", + "June", + "July", + "August", + "September", + "October", + "November", + "December", + ], + /** + * An array containing the months names in abbreviated form. Corresponds + * to the `%b` format in `Highcharts.dateFormat()`. + * + * @type {Array<string>} + * @default ["Jan", "Feb", "Mar", "Apr", "May", "Jun", + * "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"] + */ + shortMonths: [ + "Jan", + "Feb", + "Mar", + "Apr", + "May", + "Jun", + "Jul", + "Aug", + "Sep", + "Oct", + "Nov", + "Dec", + ], + /** + * An array containing the weekday names. + * + * @type {Array<string>} + * @default ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", + * "Friday", "Saturday"] + */ + weekdays: [ + "Sunday", + "Monday", + "Tuesday", + "Wednesday", + "Thursday", + "Friday", + "Saturday", + ], + /** + * Short week days, starting Sunday. If not specified, Highcharts uses + * the first three letters of the `lang.weekdays` option. + * + * @sample highcharts/lang/shortweekdays/ + * Finnish two-letter abbreviations + * + * @type {Array<string>} + * @since 4.2.4 + * @apioption lang.shortWeekdays + */ + /** + * What to show in a date field for invalid dates. Defaults to an empty + * string. + * + * @type {string} + * @since 4.1.8 + * @product highcharts highstock + * @apioption lang.invalidDate + */ + /** + * The title appearing on hovering the zoom in button. The text itself + * defaults to "+" and can be changed in the button options. + * + * @type {string} + * @default Zoom in + * @product highmaps + * @apioption lang.zoomIn + */ + /** + * The title appearing on hovering the zoom out button. The text itself + * defaults to "-" and can be changed in the button options. + * + * @type {string} + * @default Zoom out + * @product highmaps + * @apioption lang.zoomOut + */ + /** + * The default decimal point used in the `Highcharts.numberFormat` + * method unless otherwise specified in the function arguments. + * + * @since 1.2.2 + */ + decimalPoint: ".", + /** + * [Metric prefixes](https://en.wikipedia.org/wiki/Metric_prefix) used + * to shorten high numbers in axis labels. Replacing any of the + * positions with `null` causes the full number to be written. Setting + * `numericSymbols` to `null` disables shortening altogether. + * + * @sample {highcharts} highcharts/lang/numericsymbols/ + * Replacing the symbols with text + * @sample {highstock} highcharts/lang/numericsymbols/ + * Replacing the symbols with text + * + * @type {Array<string>} + * @default ["k", "M", "G", "T", "P", "E"] + * @since 2.3.0 + */ + numericSymbols: ["k", "M", "G", "T", "P", "E"], + /** + * The magnitude of [numericSymbols](#lang.numericSymbol) replacements. + * Use 10000 for Japanese, Korean and various Chinese locales, which + * use symbols for 10^4, 10^8 and 10^12. + * + * @sample highcharts/lang/numericsymbolmagnitude/ + * 10000 magnitude for Japanese + * + * @type {number} + * @default 1000 + * @since 5.0.3 + * @apioption lang.numericSymbolMagnitude + */ + /** + * The text for the label appearing when a chart is zoomed. + * + * @since 1.2.4 + */ + resetZoom: "Reset zoom", + /** + * The tooltip title for the label appearing when a chart is zoomed. + * + * @since 1.2.4 + */ + resetZoomTitle: "Reset zoom level 1:1", + /** + * The default thousands separator used in the `Highcharts.numberFormat` + * method unless otherwise specified in the function arguments. Defaults + * to a single space character, which is recommended in + * [ISO 31-0](https://en.wikipedia.org/wiki/ISO_31-0#Numbers) and works + * across Anglo-American and continental European languages. + * + * @default \u0020 + * @since 1.2.2 + */ + thousandsSep: " ", + }, + /** + * Global options that don't apply to each chart. These options, like + * the `lang` options, must be set using the `Highcharts.setOptions` + * method. + * + * ```js + * Highcharts.setOptions({ + * global: { + * useUTC: false + * } + * }); + * ``` + */ + /** + * _Canvg rendering for Android 2.x is removed as of Highcharts 5.0\. + * Use the [libURL](#exporting.libURL) option to configure exporting._ + * + * The URL to the additional file to lazy load for Android 2.x devices. + * These devices don't support SVG, so we download a helper file that + * contains [canvg](https://github.com/canvg/canvg), its dependency + * rbcolor, and our own CanVG Renderer class. To avoid hotlinking to + * our site, you can install canvas-tools.js on your own server and + * change this option accordingly. + * + * @deprecated + * + * @type {string} + * @default https://code.highcharts.com/{version}/modules/canvas-tools.js + * @product highcharts highmaps + * @apioption global.canvasToolsURL + */ + /** + * This option is deprecated since v6.0.5. Instead, use + * [time.useUTC](#time.useUTC) that supports individual time settings + * per chart. + * + * @deprecated + * + * @type {boolean} + * @apioption global.useUTC + */ + /** + * This option is deprecated since v6.0.5. Instead, use + * [time.Date](#time.Date) that supports individual time settings + * per chart. + * + * @deprecated + * + * @type {Function} + * @product highcharts highstock + * @apioption global.Date + */ + /** + * This option is deprecated since v6.0.5. Instead, use + * [time.getTimezoneOffset](#time.getTimezoneOffset) that supports + * individual time settings per chart. + * + * @deprecated + * + * @type {Function} + * @product highcharts highstock + * @apioption global.getTimezoneOffset + */ + /** + * This option is deprecated since v6.0.5. Instead, use + * [time.timezone](#time.timezone) that supports individual time + * settings per chart. + * + * @deprecated + * + * @type {string} + * @product highcharts highstock + * @apioption global.timezone + */ + /** + * This option is deprecated since v6.0.5. Instead, use + * [time.timezoneOffset](#time.timezoneOffset) that supports individual + * time settings per chart. + * + * @deprecated + * + * @type {number} + * @product highcharts highstock + * @apioption global.timezoneOffset + */ + global: {}, + /** + * Time options that can apply globally or to individual charts. These + * settings affect how `datetime` axes are laid out, how tooltips are + * formatted, how series + * [pointIntervalUnit](#plotOptions.series.pointIntervalUnit) works and how + * the Highstock range selector handles time. + * + * The common use case is that all charts in the same Highcharts object + * share the same time settings, in which case the global settings are set + * using `setOptions`. + * + * ```js + * // Apply time settings globally + * Highcharts.setOptions({ + * time: { + * timezone: 'Europe/London' + * } + * }); + * // Apply time settings by instance + * var chart = Highcharts.chart('container', { + * time: { + * timezone: 'America/New_York' + * }, + * series: [{ + * data: [1, 4, 3, 5] + * }] + * }); + * + * // Use the Time object + * console.log( + * 'Current time in New York', + * chart.time.dateFormat('%Y-%m-%d %H:%M:%S', Date.now()) + * ); + * ``` + * + * Since v6.0.5, the time options were moved from the `global` obect to the + * `time` object, and time options can be set on each individual chart. + * + * @sample {highcharts|highstock} + * highcharts/time/timezone/ + * Set the timezone globally + * @sample {highcharts} + * highcharts/time/individual/ + * Set the timezone per chart instance + * @sample {highstock} + * stock/time/individual/ + * Set the timezone per chart instance + * + * @since 6.0.5 + * @optionparent time + */ + time: { + /** + * A custom `Date` class for advanced date handling. For example, + * [JDate](https://github.com/tahajahangir/jdate) can be hooked in to + * handle Jalali dates. + * + * @type {*} + * @since 4.0.4 + * @product highcharts highstock gantt + */ + Date: void 0, + /** + * A callback to return the time zone offset for a given datetime. It + * takes the timestamp in terms of milliseconds since January 1 1970, + * and returns the timezone offset in minutes. This provides a hook + * for drawing time based charts in specific time zones using their + * local DST crossover dates, with the help of external libraries. + * + * @see [global.timezoneOffset](#global.timezoneOffset) + * + * @sample {highcharts|highstock} highcharts/time/gettimezoneoffset/ + * Use moment.js to draw Oslo time regardless of browser locale + * + * @type {Highcharts.TimezoneOffsetCallbackFunction} + * @since 4.1.0 + * @product highcharts highstock gantt + */ + getTimezoneOffset: void 0, + /** + * Requires [moment.js](https://momentjs.com/). If the timezone option + * is specified, it creates a default + * [getTimezoneOffset](#time.getTimezoneOffset) function that looks + * up the specified timezone in moment.js. If moment.js is not included, + * this throws a Highcharts error in the console, but does not crash the + * chart. + * + * @see [getTimezoneOffset](#time.getTimezoneOffset) + * + * @sample {highcharts|highstock} highcharts/time/timezone/ + * Europe/Oslo + * + * @type {string} + * @since 5.0.7 + * @product highcharts highstock gantt + */ + timezone: void 0, + /** + * The timezone offset in minutes. Positive values are west, negative + * values are east of UTC, as in the ECMAScript + * [getTimezoneOffset](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/getTimezoneOffset) + * method. Use this to display UTC based data in a predefined time zone. + * + * @see [time.getTimezoneOffset](#time.getTimezoneOffset) + * + * @sample {highcharts|highstock} highcharts/time/timezoneoffset/ + * Timezone offset + * + * @since 3.0.8 + * @product highcharts highstock gantt + */ + timezoneOffset: 0, + /** + * Whether to use UTC time for axis scaling, tickmark placement and + * time display in `Highcharts.dateFormat`. Advantages of using UTC + * is that the time displays equally regardless of the user agent's + * time zone settings. Local time can be used when the data is loaded + * in real time or when correct Daylight Saving Time transitions are + * required. + * + * @sample {highcharts} highcharts/time/useutc-true/ + * True by default + * @sample {highcharts} highcharts/time/useutc-false/ + * False + */ + useUTC: true, + }, + /** + * General options for the chart. + */ + chart: { + /** + * Default `mapData` for all series. If set to a string, it functions + * as an index into the `Highcharts.maps` array. Otherwise it is + * interpreted as map data. + * + * @see [mapData](#series.map.mapData) + * + * @sample maps/demo/geojson + * Loading geoJSON data + * @sample maps/chart/topojson + * Loading topoJSON converted to geoJSON + * + * @type {string|Array<*>|Highcharts.GeoJSON} + * @since 5.0.0 + * @product highmaps + * @apioption chart.map + */ + /** + * Set lat/lon transformation definitions for the chart. If not defined, + * these are extracted from the map data. + * + * @type {*} + * @since 5.0.0 + * @product highmaps + * @apioption chart.mapTransforms + */ + /** + * When using multiple axis, the ticks of two or more opposite axes + * will automatically be aligned by adding ticks to the axis or axes + * with the least ticks, as if `tickAmount` were specified. + * + * This can be prevented by setting `alignTicks` to false. If the grid + * lines look messy, it's a good idea to hide them for the secondary + * axis by setting `gridLineWidth` to 0. + * + * If `startOnTick` or `endOnTick` in an Axis options are set to false, + * then the `alignTicks ` will be disabled for the Axis. + * + * Disabled for logarithmic axes. + * + * @sample {highcharts} highcharts/chart/alignticks-true/ + * True by default + * @sample {highcharts} highcharts/chart/alignticks-false/ + * False + * @sample {highstock} stock/chart/alignticks-true/ + * True by default + * @sample {highstock} stock/chart/alignticks-false/ + * False + * + * @type {boolean} + * @default true + * @product highcharts highstock gantt + * @apioption chart.alignTicks + */ + /** + * Set the overall animation for all chart updating. Animation can be + * disabled throughout the chart by setting it to false here. It can + * be overridden for each individual API method as a function parameter. + * The only animation not affected by this option is the initial series + * animation, see [plotOptions.series.animation]( + * #plotOptions.series.animation). + * + * The animation can either be set as a boolean or a configuration + * object. If `true`, it will use the 'swing' jQuery easing and a + * duration of 500 ms. If used as a configuration object, the following + * properties are supported: + * + * - `defer`: The animation delay time in milliseconds. + * + * - `duration`: The duration of the animation in milliseconds. + * + * - `easing`: A string reference to an easing function set on the + * `Math` object. See + * [the easing demo](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-animation-easing/). + * + * When zooming on a series with less than 100 points, the chart redraw + * will be done with animation, but in case of more data points, it is + * necessary to set this option to ensure animation on zoom. + * + * @sample {highcharts} highcharts/chart/animation-none/ + * Updating with no animation + * @sample {highcharts} highcharts/chart/animation-duration/ + * With a longer duration + * @sample {highcharts} highcharts/chart/animation-easing/ + * With a jQuery UI easing + * @sample {highmaps} maps/chart/animation-none/ + * Updating with no animation + * @sample {highmaps} maps/chart/animation-duration/ + * With a longer duration + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + * @default undefined + * @apioption chart.animation + */ + /** + * A CSS class name to apply to the charts container `div`, allowing + * unique CSS styling for each chart. + * + * @type {string} + * @apioption chart.className + */ + /** + * Event listeners for the chart. + * + * @apioption chart.events + */ + /** + * Fires when a series is added to the chart after load time, using the + * `addSeries` method. One parameter, `event`, is passed to the + * function, containing common event information. Through + * `event.options` you can access the series options that were passed to + * the `addSeries` method. Returning false prevents the series from + * being added. + * + * @sample {highcharts} highcharts/chart/events-addseries/ + * Alert on add series + * @sample {highstock} stock/chart/events-addseries/ + * Alert on add series + * + * @type {Highcharts.ChartAddSeriesCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Chart + * @apioption chart.events.addSeries + */ + /** + * Fires when clicking on the plot background. One parameter, `event`, + * is passed to the function, containing common event information. + * + * Information on the clicked spot can be found through `event.xAxis` + * and `event.yAxis`, which are arrays containing the axes of each + * dimension and each axis' value at the clicked spot. The primary axes + * are `event.xAxis[0]` and `event.yAxis[0]`. Remember the unit of a + * datetime axis is milliseconds since 1970-01-01 00:00:00. + * + * ```js + * click: function(e) { + * console.log( + * Highcharts.dateFormat('%Y-%m-%d %H:%M:%S', e.xAxis[0].value), + * e.yAxis[0].value + * ) + * } + * ``` + * + * @sample {highcharts} highcharts/chart/events-click/ + * Alert coordinates on click + * @sample {highcharts} highcharts/chart/events-container/ + * Alternatively, attach event to container + * @sample {highstock} stock/chart/events-click/ + * Alert coordinates on click + * @sample {highstock} highcharts/chart/events-container/ + * Alternatively, attach event to container + * @sample {highmaps} maps/chart/events-click/ + * Record coordinates on click + * @sample {highmaps} highcharts/chart/events-container/ + * Alternatively, attach event to container + * + * @type {Highcharts.ChartClickCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Chart + * @apioption chart.events.click + */ + /** + * Fires when the chart is finished loading. Since v4.2.2, it also waits + * for images to be loaded, for example from point markers. One + * parameter, `event`, is passed to the function, containing common + * event information. + * + * There is also a second parameter to the chart constructor where a + * callback function can be passed to be executed on chart.load. + * + * @sample {highcharts} highcharts/chart/events-load/ + * Alert on chart load + * @sample {highstock} stock/chart/events-load/ + * Alert on chart load + * @sample {highmaps} maps/chart/events-load/ + * Add series on chart load + * + * @type {Highcharts.ChartLoadCallbackFunction} + * @context Highcharts.Chart + * @apioption chart.events.load + */ + /** + * Fires when the chart is redrawn, either after a call to + * `chart.redraw()` or after an axis, series or point is modified with + * the `redraw` option set to `true`. One parameter, `event`, is passed + * to the function, containing common event information. + * + * @sample {highcharts} highcharts/chart/events-redraw/ + * Alert on chart redraw + * @sample {highstock} stock/chart/events-redraw/ + * Alert on chart redraw when adding a series or moving the + * zoomed range + * @sample {highmaps} maps/chart/events-redraw/ + * Set subtitle on chart redraw + * + * @type {Highcharts.ChartRedrawCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Chart + * @apioption chart.events.redraw + */ + /** + * Fires after initial load of the chart (directly after the `load` + * event), and after each redraw (directly after the `redraw` event). + * + * @type {Highcharts.ChartRenderCallbackFunction} + * @since 5.0.7 + * @context Highcharts.Chart + * @apioption chart.events.render + */ + /** + * Fires when an area of the chart has been selected. Selection is + * enabled by setting the chart's zoomType. One parameter, `event`, is + * passed to the function, containing common event information. The + * default action for the selection event is to zoom the chart to the + * selected area. It can be prevented by calling + * `event.preventDefault()` or return false. + * + * Information on the selected area can be found through `event.xAxis` + * and `event.yAxis`, which are arrays containing the axes of each + * dimension and each axis' min and max values. The primary axes are + * `event.xAxis[0]` and `event.yAxis[0]`. Remember the unit of a + * datetime axis is milliseconds since 1970-01-01 00:00:00. + * + * ```js + * selection: function(event) { + * // log the min and max of the primary, datetime x-axis + * console.log( + * Highcharts.dateFormat( + * '%Y-%m-%d %H:%M:%S', + * event.xAxis[0].min + * ), + * Highcharts.dateFormat( + * '%Y-%m-%d %H:%M:%S', + * event.xAxis[0].max + * ) + * ); + * // log the min and max of the y axis + * console.log(event.yAxis[0].min, event.yAxis[0].max); + * } + * ``` + * + * @sample {highcharts} highcharts/chart/events-selection/ + * Report on selection and reset + * @sample {highcharts} highcharts/chart/events-selection-points/ + * Select a range of points through a drag selection + * @sample {highstock} stock/chart/events-selection/ + * Report on selection and reset + * @sample {highstock} highcharts/chart/events-selection-points/ + * Select a range of points through a drag selection + * (Highcharts) + * + * @type {Highcharts.ChartSelectionCallbackFunction} + * @apioption chart.events.selection + */ + /** + * The margin between the outer edge of the chart and the plot area. + * The numbers in the array designate top, right, bottom and left + * respectively. Use the options `marginTop`, `marginRight`, + * `marginBottom` and `marginLeft` for shorthand setting of one option. + * + * By default there is no margin. The actual space is dynamically + * calculated from the offset of axis labels, axis title, title, + * subtitle and legend in addition to the `spacingTop`, `spacingRight`, + * `spacingBottom` and `spacingLeft` options. + * + * @sample {highcharts} highcharts/chart/margins-zero/ + * Zero margins + * @sample {highstock} stock/chart/margin-zero/ + * Zero margins + * + * @type {number|Array<number>} + * @apioption chart.margin + */ + /** + * The margin between the bottom outer edge of the chart and the plot + * area. Use this to set a fixed pixel value for the margin as opposed + * to the default dynamic margin. See also `spacingBottom`. + * + * @sample {highcharts} highcharts/chart/marginbottom/ + * 100px bottom margin + * @sample {highstock} stock/chart/marginbottom/ + * 100px bottom margin + * @sample {highmaps} maps/chart/margin/ + * 100px margins + * + * @type {number} + * @since 2.0 + * @apioption chart.marginBottom + */ + /** + * The margin between the left outer edge of the chart and the plot + * area. Use this to set a fixed pixel value for the margin as opposed + * to the default dynamic margin. See also `spacingLeft`. + * + * @sample {highcharts} highcharts/chart/marginleft/ + * 150px left margin + * @sample {highstock} stock/chart/marginleft/ + * 150px left margin + * @sample {highmaps} maps/chart/margin/ + * 100px margins + * + * @type {number} + * @since 2.0 + * @apioption chart.marginLeft + */ + /** + * The margin between the right outer edge of the chart and the plot + * area. Use this to set a fixed pixel value for the margin as opposed + * to the default dynamic margin. See also `spacingRight`. + * + * @sample {highcharts} highcharts/chart/marginright/ + * 100px right margin + * @sample {highstock} stock/chart/marginright/ + * 100px right margin + * @sample {highmaps} maps/chart/margin/ + * 100px margins + * + * @type {number} + * @since 2.0 + * @apioption chart.marginRight + */ + /** + * The margin between the top outer edge of the chart and the plot area. + * Use this to set a fixed pixel value for the margin as opposed to + * the default dynamic margin. See also `spacingTop`. + * + * @sample {highcharts} highcharts/chart/margintop/ 100px top margin + * @sample {highstock} stock/chart/margintop/ + * 100px top margin + * @sample {highmaps} maps/chart/margin/ + * 100px margins + * + * @type {number} + * @since 2.0 + * @apioption chart.marginTop + */ + /** + * Callback function to override the default function that formats all + * the numbers in the chart. Returns a string with the formatted number. + * + * @sample highcharts/members/highcharts-numberformat + * Arabic digits in Highcharts + * @type {Highcharts.NumberFormatterCallbackFunction} + * @since 8.0.0 + * @apioption chart.numberFormatter + */ + /** + * Allows setting a key to switch between zooming and panning. Can be + * one of `alt`, `ctrl`, `meta` (the command key on Mac and Windows + * key on Windows) or `shift`. The keys are mapped directly to the key + * properties of the click event argument (`event.altKey`, + * `event.ctrlKey`, `event.metaKey` and `event.shiftKey`). + * + * @type {string} + * @since 4.0.3 + * @product highcharts gantt + * @validvalue ["alt", "ctrl", "meta", "shift"] + * @apioption chart.panKey + */ + /** + * Allow panning in a chart. Best used with [panKey](#chart.panKey) + * to combine zooming and panning. + * + * On touch devices, when the [tooltip.followTouchMove]( + * #tooltip.followTouchMove) option is `true` (default), panning + * requires two fingers. To allow panning with one finger, set + * `followTouchMove` to `false`. + * + * @sample {highcharts} highcharts/chart/pankey/ Zooming and panning + * @sample {highstock} stock/chart/panning/ Zooming and xy panning + * + * @product highcharts highstock gantt + * @apioption chart.panning + */ + /** + * Enable or disable chart panning. + * + * @type {boolean} + * @default {highcharts} false + * @default {highstock} true + * @apioption chart.panning.enabled + */ + /** + * Decides in what dimensions the user can pan the chart. Can be + * one of `x`, `y`, or `xy`. + * + * @sample {highcharts} highcharts/chart/panning-type + * Zooming and xy panning + * + * @type {string} + * @validvalue ["x", "y", "xy"] + * @default x + * @apioption chart.panning.type + */ + /** + * Equivalent to [zoomType](#chart.zoomType), but for multitouch + * gestures only. By default, the `pinchType` is the same as the + * `zoomType` setting. However, pinching can be enabled separately in + * some cases, for example in stock charts where a mouse drag pans the + * chart, while pinching is enabled. When [tooltip.followTouchMove]( + * #tooltip.followTouchMove) is true, pinchType only applies to + * two-finger touches. + * + * @type {string} + * @default {highcharts} undefined + * @default {highstock} x + * @since 3.0 + * @product highcharts highstock gantt + * @validvalue ["x", "y", "xy"] + * @apioption chart.pinchType + */ + /** + * Whether to apply styled mode. When in styled mode, no presentational + * attributes or CSS are applied to the chart SVG. Instead, CSS rules + * are required to style the chart. The default style sheet is + * available from `https://code.highcharts.com/css/highcharts.css`. + * + * @type {boolean} + * @default false + * @since 7.0 + * @apioption chart.styledMode + */ + styledMode: false, + /** + * The corner radius of the outer chart border. + * + * @sample {highcharts} highcharts/chart/borderradius/ + * 20px radius + * @sample {highstock} stock/chart/border/ + * 10px radius + * @sample {highmaps} maps/chart/border/ + * Border options + * + */ + borderRadius: 0, + /** + * In styled mode, this sets how many colors the class names + * should rotate between. With ten colors, series (or points) are + * given class names like `highcharts-color-0`, `highcharts-color-0` + * [...] `highcharts-color-9`. The equivalent in non-styled mode + * is to set colors using the [colors](#colors) setting. + * + * @since 5.0.0 + */ + colorCount: 10, + /** + * Alias of `type`. + * + * @sample {highcharts} highcharts/chart/defaultseriestype/ + * Bar + * + * @deprecated + * + * @product highcharts + */ + defaultSeriesType: "line", + /** + * If true, the axes will scale to the remaining visible series once + * one series is hidden. If false, hiding and showing a series will + * not affect the axes or the other series. For stacks, once one series + * within the stack is hidden, the rest of the stack will close in + * around it even if the axis is not affected. + * + * @sample {highcharts} highcharts/chart/ignorehiddenseries-true/ + * True by default + * @sample {highcharts} highcharts/chart/ignorehiddenseries-false/ + * False + * @sample {highcharts} highcharts/chart/ignorehiddenseries-true-stacked/ + * True with stack + * @sample {highstock} stock/chart/ignorehiddenseries-true/ + * True by default + * @sample {highstock} stock/chart/ignorehiddenseries-false/ + * False + * + * @since 1.2.0 + * @product highcharts highstock gantt + */ + ignoreHiddenSeries: true, + /** + * Whether to invert the axes so that the x axis is vertical and y axis + * is horizontal. When `true`, the x axis is [reversed](#xAxis.reversed) + * by default. + * + * @productdesc {highcharts} + * If a bar series is present in the chart, it will be inverted + * automatically. Inverting the chart doesn't have an effect if there + * are no cartesian series in the chart, or if the chart is + * [polar](#chart.polar). + * + * @sample {highcharts} highcharts/chart/inverted/ + * Inverted line + * @sample {highstock} stock/navigator/inverted/ + * Inverted stock chart + * + * @type {boolean} + * @default false + * @product highcharts highstock gantt + * @apioption chart.inverted + */ + /** + * The distance between the outer edge of the chart and the content, + * like title or legend, or axis title and labels if present. The + * numbers in the array designate top, right, bottom and left + * respectively. Use the options spacingTop, spacingRight, spacingBottom + * and spacingLeft options for shorthand setting of one option. + * + * @type {Array<number>} + * @see [chart.margin](#chart.margin) + * @default [10, 10, 15, 10] + * @since 3.0.6 + */ + spacing: [10, 10, 15, 10], + /** + * The button that appears after a selection zoom, allowing the user + * to reset zoom. + */ + resetZoomButton: { + /** + * What frame the button placement should be related to. Can be + * either `plotBox` or `spacingBox`. + * + * @sample {highcharts} highcharts/chart/resetzoombutton-relativeto/ + * Relative to the chart + * @sample {highstock} highcharts/chart/resetzoombutton-relativeto/ + * Relative to the chart + * + * @type {Highcharts.ButtonRelativeToValue} + * @default plot + * @since 2.2 + * @apioption chart.resetZoomButton.relativeTo + */ + /** + * A collection of attributes for the button. The object takes SVG + * attributes like `fill`, `stroke`, `stroke-width` or `r`, the + * border radius. The theme also supports `style`, a collection of + * CSS properties for the text. Equivalent attributes for the hover + * state are given in `theme.states.hover`. + * + * @sample {highcharts} highcharts/chart/resetzoombutton-theme/ + * Theming the button + * @sample {highstock} highcharts/chart/resetzoombutton-theme/ + * Theming the button + * + * @type {Highcharts.SVGAttributes} + * @since 2.2 + */ + theme: { + /** @internal */ + zIndex: 6, + }, /** - * If one of the symbol size affecting parameters are changed, - * check all the others only once for each call to an element's - * .attr() method - * - * @private - * @function Highcharts.SVGElement#symbolAttr - * - * @param {Highcharts.SVGAttributes} hash - * The attributes to set. - */ - SVGElement.prototype.symbolAttr = function (hash) { - var wrapper = this; - [ - 'x', - 'y', - 'r', - 'start', - 'end', - 'width', - 'height', - 'innerR', - 'anchorX', - 'anchorY', - 'clockwise' - ].forEach(function (key) { - wrapper[key] = pick(hash[key], wrapper[key]); - }); - wrapper.attr({ - d: wrapper.renderer.symbols[wrapper.symbolName](wrapper.x, wrapper.y, wrapper.width, wrapper.height, wrapper) - }); - }; + * The position of the button. + * + * @sample {highcharts} highcharts/chart/resetzoombutton-position/ + * Above the plot area + * @sample {highstock} highcharts/chart/resetzoombutton-position/ + * Above the plot area + * @sample {highmaps} highcharts/chart/resetzoombutton-position/ + * Above the plot area + * + * @type {Highcharts.AlignObject} + * @since 2.2 + */ + position: { + /** + * The horizontal alignment of the button. + */ + align: "right", + /** + * The horizontal offset of the button. + */ + x: -10, + /** + * The vertical alignment of the button. + * + * @type {Highcharts.VerticalAlignValue} + * @default top + * @apioption chart.resetZoomButton.position.verticalAlign + */ + /** + * The vertical offset of the button. + */ + y: 10, + }, + }, + /** + * The pixel width of the plot area border. + * + * @sample {highcharts} highcharts/chart/plotborderwidth/ + * 1px border + * @sample {highstock} stock/chart/plotborder/ + * 2px border + * @sample {highmaps} maps/chart/plotborder/ + * Plot border options + * + * @type {number} + * @default 0 + * @apioption chart.plotBorderWidth + */ + /** + * Whether to apply a drop shadow to the plot area. Requires that + * plotBackgroundColor be set. The shadow can be an object configuration + * containing `color`, `offsetX`, `offsetY`, `opacity` and `width`. + * + * @sample {highcharts} highcharts/chart/plotshadow/ + * Plot shadow + * @sample {highstock} stock/chart/plotshadow/ + * Plot shadow + * @sample {highmaps} maps/chart/plotborder/ + * Plot border options + * + * @type {boolean|Highcharts.CSSObject} + * @default false + * @apioption chart.plotShadow + */ + /** + * When true, cartesian charts like line, spline, area and column are + * transformed into the polar coordinate system. This produces _polar + * charts_, also known as _radar charts_. + * + * @sample {highcharts} highcharts/demo/polar/ + * Polar chart + * @sample {highcharts} highcharts/demo/polar-wind-rose/ + * Wind rose, stacked polar column chart + * @sample {highcharts} highcharts/demo/polar-spider/ + * Spider web chart + * @sample {highcharts} highcharts/parallel-coordinates/polar/ + * Star plot, multivariate data in a polar chart + * + * @type {boolean} + * @default false + * @since 2.3.0 + * @product highcharts + * @requires highcharts-more + * @apioption chart.polar + */ + /** + * Whether to reflow the chart to fit the width of the container div + * on resizing the window. + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * True by default + * @sample {highcharts} highcharts/chart/reflow-false/ + * False + * @sample {highstock} stock/chart/reflow-true/ + * True by default + * @sample {highstock} stock/chart/reflow-false/ + * False + * @sample {highmaps} maps/chart/reflow-true/ + * True by default + * @sample {highmaps} maps/chart/reflow-false/ + * False + * + * @type {boolean} + * @default true + * @since 2.1 + * @apioption chart.reflow + */ + /** + * The HTML element where the chart will be rendered. If it is a string, + * the element by that id is used. The HTML element can also be passed + * by direct reference, or as the first argument of the chart + * constructor, in which case the option is not needed. + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * String + * @sample {highcharts} highcharts/chart/renderto-object/ + * Object reference + * @sample {highcharts} highcharts/chart/renderto-jquery/ + * Object reference through jQuery + * @sample {highstock} stock/chart/renderto-string/ + * String + * @sample {highstock} stock/chart/renderto-object/ + * Object reference + * @sample {highstock} stock/chart/renderto-jquery/ + * Object reference through jQuery + * + * @type {string|Highcharts.HTMLDOMElement} + * @apioption chart.renderTo + */ + /** + * The background color of the marker square when selecting (zooming + * in on) an area of the chart. + * + * @see In styled mode, the selection marker fill is set with the + * `.highcharts-selection-marker` class. + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @default rgba(51,92,173,0.25) + * @since 2.1.7 + * @apioption chart.selectionMarkerFill + */ + /** + * Whether to apply a drop shadow to the outer chart area. Requires + * that backgroundColor be set. The shadow can be an object + * configuration containing `color`, `offsetX`, `offsetY`, `opacity` and + * `width`. + * + * @sample {highcharts} highcharts/chart/shadow/ + * Shadow + * @sample {highstock} stock/chart/shadow/ + * Shadow + * @sample {highmaps} maps/chart/border/ + * Chart border and shadow + * + * @type {boolean|Highcharts.CSSObject} + * @default false + * @apioption chart.shadow + */ + /** + * Whether to show the axes initially. This only applies to empty charts + * where series are added dynamically, as axes are automatically added + * to cartesian series. + * + * @sample {highcharts} highcharts/chart/showaxes-false/ + * False by default + * @sample {highcharts} highcharts/chart/showaxes-true/ + * True + * + * @type {boolean} + * @since 1.2.5 + * @product highcharts gantt + * @apioption chart.showAxes + */ + /** + * The space between the bottom edge of the chart and the content (plot + * area, axis title and labels, title, subtitle or legend in top + * position). + * + * @sample {highcharts} highcharts/chart/spacingbottom/ + * Spacing bottom set to 100 + * @sample {highstock} stock/chart/spacingbottom/ + * Spacing bottom set to 100 + * @sample {highmaps} maps/chart/spacing/ + * Spacing 100 all around + * + * @type {number} + * @default 15 + * @since 2.1 + * @apioption chart.spacingBottom + */ + /** + * The space between the left edge of the chart and the content (plot + * area, axis title and labels, title, subtitle or legend in top + * position). + * + * @sample {highcharts} highcharts/chart/spacingleft/ + * Spacing left set to 100 + * @sample {highstock} stock/chart/spacingleft/ + * Spacing left set to 100 + * @sample {highmaps} maps/chart/spacing/ + * Spacing 100 all around + * + * @type {number} + * @default 10 + * @since 2.1 + * @apioption chart.spacingLeft + */ + /** + * The space between the right edge of the chart and the content (plot + * area, axis title and labels, title, subtitle or legend in top + * position). + * + * @sample {highcharts} highcharts/chart/spacingright-100/ + * Spacing set to 100 + * @sample {highcharts} highcharts/chart/spacingright-legend/ + * Legend in right position with default spacing + * @sample {highstock} stock/chart/spacingright/ + * Spacing set to 100 + * @sample {highmaps} maps/chart/spacing/ + * Spacing 100 all around + * + * @type {number} + * @default 10 + * @since 2.1 + * @apioption chart.spacingRight + */ + /** + * The space between the top edge of the chart and the content (plot + * area, axis title and labels, title, subtitle or legend in top + * position). + * + * @sample {highcharts} highcharts/chart/spacingtop-100/ + * A top spacing of 100 + * @sample {highcharts} highcharts/chart/spacingtop-10/ + * Floating chart title makes the plot area align to the default + * spacingTop of 10. + * @sample {highstock} stock/chart/spacingtop/ + * A top spacing of 100 + * @sample {highmaps} maps/chart/spacing/ + * Spacing 100 all around + * + * @type {number} + * @default 10 + * @since 2.1 + * @apioption chart.spacingTop + */ + /** + * Additional CSS styles to apply inline to the container `div`. Note + * that since the default font styles are applied in the renderer, it + * is ignorant of the individual chart options and must be set globally. + * + * @see In styled mode, general chart styles can be set with the + * `.highcharts-root` class. + * @sample {highcharts} highcharts/chart/style-serif-font/ + * Using a serif type font + * @sample {highcharts} highcharts/css/em/ + * Styled mode with relative font sizes + * @sample {highstock} stock/chart/style/ + * Using a serif type font + * @sample {highmaps} maps/chart/style-serif-font/ + * Using a serif type font + * + * @type {Highcharts.CSSObject} + * @default {"fontFamily": "\"Lucida Grande\", \"Lucida Sans Unicode\", Verdana, Arial, Helvetica, sans-serif","fontSize":"12px"} + * @apioption chart.style + */ + /** + * The default series type for the chart. Can be any of the chart types + * listed under [plotOptions](#plotOptions) and [series](#series) or can + * be a series provided by an additional module. + * + * In TypeScript this option has no effect in sense of typing and + * instead the `type` option must always be set in the series. + * + * @sample {highcharts} highcharts/chart/type-bar/ + * Bar + * @sample {highstock} stock/chart/type/ + * Areaspline + * @sample {highmaps} maps/chart/type-mapline/ + * Mapline + * + * @type {string} + * @default {highcharts} line + * @default {highstock} line + * @default {highmaps} map + * @since 2.1.0 + * @apioption chart.type + */ + /** + * Decides in what dimensions the user can zoom by dragging the mouse. + * Can be one of `x`, `y` or `xy`. + * + * @see [panKey](#chart.panKey) + * + * @sample {highcharts} highcharts/chart/zoomtype-none/ + * None by default + * @sample {highcharts} highcharts/chart/zoomtype-x/ + * X + * @sample {highcharts} highcharts/chart/zoomtype-y/ + * Y + * @sample {highcharts} highcharts/chart/zoomtype-xy/ + * Xy + * @sample {highstock} stock/demo/basic-line/ + * None by default + * @sample {highstock} stock/chart/zoomtype-x/ + * X + * @sample {highstock} stock/chart/zoomtype-y/ + * Y + * @sample {highstock} stock/chart/zoomtype-xy/ + * Xy + * + * @type {string} + * @product highcharts highstock gantt + * @validvalue ["x", "y", "xy"] + * @apioption chart.zoomType + */ + /** + * An explicit width for the chart. By default (when `null`) the width + * is calculated from the offset width of the containing element. + * + * @sample {highcharts} highcharts/chart/width/ + * 800px wide + * @sample {highstock} stock/chart/width/ + * 800px wide + * @sample {highmaps} maps/chart/size/ + * Chart with explicit size + * + * @type {null|number|string} + */ + width: null, + /** + * An explicit height for the chart. If a _number_, the height is + * given in pixels. If given a _percentage string_ (for example + * `'56%'`), the height is given as the percentage of the actual chart + * width. This allows for preserving the aspect ratio across responsive + * sizes. + * + * By default (when `null`) the height is calculated from the offset + * height of the containing element, or 400 pixels if the containing + * element's height is 0. + * + * @sample {highcharts} highcharts/chart/height/ + * 500px height + * @sample {highstock} stock/chart/height/ + * 300px height + * @sample {highmaps} maps/chart/size/ + * Chart with explicit size + * @sample highcharts/chart/height-percent/ + * Highcharts with percentage height + * + * @type {null|number|string} + */ + height: null, + /** + * The color of the outer chart border. + * + * @see In styled mode, the stroke is set with the + * `.highcharts-background` class. + * + * @sample {highcharts} highcharts/chart/bordercolor/ + * Brown border + * @sample {highstock} stock/chart/border/ + * Brown border + * @sample {highmaps} maps/chart/border/ + * Border options + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + */ + borderColor: "#335cad", + /** + * The pixel width of the outer chart border. + * + * @see In styled mode, the stroke is set with the + * `.highcharts-background` class. + * + * @sample {highcharts} highcharts/chart/borderwidth/ + * 5px border + * @sample {highstock} stock/chart/border/ + * 2px border + * @sample {highmaps} maps/chart/border/ + * Border options + * + * @type {number} + * @default 0 + * @apioption chart.borderWidth + */ + /** + * The background color or gradient for the outer chart area. + * + * @see In styled mode, the background is set with the + * `.highcharts-background` class. + * + * @sample {highcharts} highcharts/chart/backgroundcolor-color/ + * Color + * @sample {highcharts} highcharts/chart/backgroundcolor-gradient/ + * Gradient + * @sample {highstock} stock/chart/backgroundcolor-color/ + * Color + * @sample {highstock} stock/chart/backgroundcolor-gradient/ + * Gradient + * @sample {highmaps} maps/chart/backgroundcolor-color/ + * Color + * @sample {highmaps} maps/chart/backgroundcolor-gradient/ + * Gradient + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + */ + backgroundColor: "#ffffff", + /** + * The background color or gradient for the plot area. + * + * @see In styled mode, the plot background is set with the + * `.highcharts-plot-background` class. + * + * @sample {highcharts} highcharts/chart/plotbackgroundcolor-color/ + * Color + * @sample {highcharts} highcharts/chart/plotbackgroundcolor-gradient/ + * Gradient + * @sample {highstock} stock/chart/plotbackgroundcolor-color/ + * Color + * @sample {highstock} stock/chart/plotbackgroundcolor-gradient/ + * Gradient + * @sample {highmaps} maps/chart/plotbackgroundcolor-color/ + * Color + * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/ + * Gradient + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @apioption chart.plotBackgroundColor + */ + /** + * The URL for an image to use as the plot background. To set an image + * as the background for the entire chart, set a CSS background image + * to the container element. Note that for the image to be applied to + * exported charts, its URL needs to be accessible by the export server. + * + * @see In styled mode, a plot background image can be set with the + * `.highcharts-plot-background` class and a [custom pattern]( + * https://www.highcharts.com/docs/chart-design-and-style/ + * gradients-shadows-and-patterns). + * + * @sample {highcharts} highcharts/chart/plotbackgroundimage/ + * Skies + * @sample {highstock} stock/chart/plotbackgroundimage/ + * Skies + * + * @type {string} + * @apioption chart.plotBackgroundImage + */ + /** + * The color of the inner chart or plot area border. + * + * @see In styled mode, a plot border stroke can be set with the + * `.highcharts-plot-border` class. + * + * @sample {highcharts} highcharts/chart/plotbordercolor/ + * Blue border + * @sample {highstock} stock/chart/plotborder/ + * Blue border + * @sample {highmaps} maps/chart/plotborder/ + * Plot border options + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + */ + plotBorderColor: "#cccccc", + }, + /** + * The chart's main title. + * + * @sample {highmaps} maps/title/title/ + * Title options demonstrated + */ + title: { + /** + * When the title is floating, the plot area will not move to make space + * for it. + * + * @sample {highcharts} highcharts/chart/zoomtype-none/ + * False by default + * @sample {highcharts} highcharts/title/floating/ + * True - title on top of the plot area + * @sample {highstock} stock/chart/title-floating/ + * True - title on top of the plot area + * + * @type {boolean} + * @default false + * @since 2.1 + * @apioption title.floating + */ + /** + * CSS styles for the title. Use this for font styling, but use `align`, + * `x` and `y` for text alignment. + * + * In styled mode, the title style is given in the `.highcharts-title` + * class. + * + * @sample {highcharts} highcharts/title/style/ + * Custom color and weight + * @sample {highstock} stock/chart/title-style/ + * Custom color and weight + * @sample highcharts/css/titles/ + * Styled mode + * + * @type {Highcharts.CSSObject} + * @default {highcharts|highmaps} { "color": "#333333", "fontSize": "18px" } + * @default {highstock} { "color": "#333333", "fontSize": "16px" } + * @apioption title.style + */ + /** + * Whether to + * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the text. + * + * @type {boolean} + * @default false + * @apioption title.useHTML + */ + /** + * The vertical alignment of the title. Can be one of `"top"`, + * `"middle"` and `"bottom"`. When a value is given, the title behaves + * as if [floating](#title.floating) were `true`. + * + * @sample {highcharts} highcharts/title/verticalalign/ + * Chart title in bottom right corner + * @sample {highstock} stock/chart/title-verticalalign/ + * Chart title in bottom right corner + * + * @type {Highcharts.VerticalAlignValue} + * @since 2.1 + * @apioption title.verticalAlign + */ + /** + * The x position of the title relative to the alignment within + * `chart.spacingLeft` and `chart.spacingRight`. + * + * @sample {highcharts} highcharts/title/align/ + * Aligned to the plot area (x = 70px = margin left - spacing + * left) + * @sample {highstock} stock/chart/title-align/ + * Aligned to the plot area (x = 50px = margin left - spacing + * left) + * + * @type {number} + * @default 0 + * @since 2.0 + * @apioption title.x + */ + /** + * The y position of the title relative to the alignment within + * [chart.spacingTop](#chart.spacingTop) and [chart.spacingBottom]( + * #chart.spacingBottom). By default it depends on the font size. + * + * @sample {highcharts} highcharts/title/y/ + * Title inside the plot area + * @sample {highstock} stock/chart/title-verticalalign/ + * Chart title in bottom right corner + * + * @type {number} + * @since 2.0 + * @apioption title.y + */ + /** + * The title of the chart. To disable the title, set the `text` to + * `undefined`. + * + * @sample {highcharts} highcharts/title/text/ + * Custom title + * @sample {highstock} stock/chart/title-text/ + * Custom title + * + * @default {highcharts|highmaps} Chart title + * @default {highstock} undefined + */ + text: "Chart title", + /** + * The horizontal alignment of the title. Can be one of "left", "center" + * and "right". + * + * @sample {highcharts} highcharts/title/align/ + * Aligned to the plot area (x = 70px = margin left - spacing + * left) + * @sample {highstock} stock/chart/title-align/ + * Aligned to the plot area (x = 50px = margin left - spacing + * left) + * + * @type {Highcharts.AlignValue} + * @since 2.0 + */ + align: "center", + /** + * The margin between the title and the plot area, or if a subtitle + * is present, the margin between the subtitle and the plot area. + * + * @sample {highcharts} highcharts/title/margin-50/ + * A chart title margin of 50 + * @sample {highcharts} highcharts/title/margin-subtitle/ + * The same margin applied with a subtitle + * @sample {highstock} stock/chart/title-margin/ + * A chart title margin of 50 + * + * @since 2.1 + */ + margin: 15, + /** + * Adjustment made to the title width, normally to reserve space for + * the exporting burger menu. + * + * @sample highcharts/title/widthadjust/ + * Wider menu, greater padding + * + * @since 4.2.5 + */ + widthAdjust: -44, + }, + /** + * The chart's subtitle. This can be used both to display a subtitle below + * the main title, and to display random text anywhere in the chart. The + * subtitle can be updated after chart initialization through the + * `Chart.setTitle` method. + * + * @sample {highmaps} maps/title/subtitle/ + * Subtitle options demonstrated + */ + subtitle: { + /** + * When the subtitle is floating, the plot area will not move to make + * space for it. + * + * @sample {highcharts} highcharts/subtitle/floating/ + * Floating title and subtitle + * @sample {highstock} stock/chart/subtitle-footnote + * Footnote floating at bottom right of plot area + * + * @type {boolean} + * @default false + * @since 2.1 + * @apioption subtitle.floating + */ + /** + * CSS styles for the title. + * + * In styled mode, the subtitle style is given in the + * `.highcharts-subtitle` class. + * + * @sample {highcharts} highcharts/subtitle/style/ + * Custom color and weight + * @sample {highcharts} highcharts/css/titles/ + * Styled mode + * @sample {highstock} stock/chart/subtitle-style + * Custom color and weight + * @sample {highstock} highcharts/css/titles/ + * Styled mode + * @sample {highmaps} highcharts/css/titles/ + * Styled mode + * + * @type {Highcharts.CSSObject} + * @default {"color": "#666666"} + * @apioption subtitle.style + */ + /** + * Whether to + * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the text. + * + * @type {boolean} + * @default false + * @apioption subtitle.useHTML + */ + /** + * The vertical alignment of the title. Can be one of `"top"`, + * `"middle"` and `"bottom"`. When middle, the subtitle behaves as + * floating. + * + * @sample {highcharts} highcharts/subtitle/verticalalign/ + * Footnote at the bottom right of plot area + * @sample {highstock} stock/chart/subtitle-footnote + * Footnote at the bottom right of plot area + * + * @type {Highcharts.VerticalAlignValue} + * @since 2.1 + * @apioption subtitle.verticalAlign + */ + /** + * The x position of the subtitle relative to the alignment within + * `chart.spacingLeft` and `chart.spacingRight`. + * + * @sample {highcharts} highcharts/subtitle/align/ + * Footnote at right of plot area + * @sample {highstock} stock/chart/subtitle-footnote + * Footnote at the bottom right of plot area + * + * @type {number} + * @default 0 + * @since 2.0 + * @apioption subtitle.x + */ + /** + * The y position of the subtitle relative to the alignment within + * `chart.spacingTop` and `chart.spacingBottom`. By default the subtitle + * is laid out below the title unless the title is floating. + * + * @sample {highcharts} highcharts/subtitle/verticalalign/ + * Footnote at the bottom right of plot area + * @sample {highstock} stock/chart/subtitle-footnote + * Footnote at the bottom right of plot area + * + * @type {number} + * @since 2.0 + * @apioption subtitle.y + */ + /** + * The subtitle of the chart. + * + * @sample {highcharts|highstock} highcharts/subtitle/text/ + * Custom subtitle + * @sample {highcharts|highstock} highcharts/subtitle/text-formatted/ + * Formatted and linked text. + */ + text: "", + /** + * The horizontal alignment of the subtitle. Can be one of "left", + * "center" and "right". + * + * @sample {highcharts} highcharts/subtitle/align/ + * Footnote at right of plot area + * @sample {highstock} stock/chart/subtitle-footnote + * Footnote at bottom right of plot area + * + * @type {Highcharts.AlignValue} + * @since 2.0 + */ + align: "center", + /** + * Adjustment made to the subtitle width, normally to reserve space + * for the exporting burger menu. + * + * @see [title.widthAdjust](#title.widthAdjust) + * + * @sample highcharts/title/widthadjust/ + * Wider menu, greater padding + * + * @since 4.2.5 + */ + widthAdjust: -44, + }, + /** + * The chart's caption, which will render below the chart and will be part + * of exported charts. The caption can be updated after chart initialization + * through the `Chart.update` or `Chart.caption.update` methods. + * + * @sample highcharts/caption/text/ + * A chart with a caption + * @since 7.2.0 + */ + caption: { + /** + * When the caption is floating, the plot area will not move to make + * space for it. + * + * @type {boolean} + * @default false + * @apioption caption.floating + */ + /** + * The margin between the caption and the plot area. + */ + margin: 15, + /** + * CSS styles for the caption. + * + * In styled mode, the caption style is given in the + * `.highcharts-caption` class. + * + * @sample {highcharts} highcharts/css/titles/ + * Styled mode + * + * @type {Highcharts.CSSObject} + * @default {"color": "#666666"} + * @apioption caption.style + */ + /** + * Whether to + * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the text. + * + * @type {boolean} + * @default false + * @apioption caption.useHTML + */ + /** + * The x position of the caption relative to the alignment within + * `chart.spacingLeft` and `chart.spacingRight`. + * + * @type {number} + * @default 0 + * @apioption caption.x + */ + /** + * The y position of the caption relative to the alignment within + * `chart.spacingTop` and `chart.spacingBottom`. + * + * @type {number} + * @apioption caption.y + */ + /** + * The caption text of the chart. + * + * @sample {highcharts} highcharts/caption/text/ + * Custom caption + */ + text: "", + /** + * The horizontal alignment of the caption. Can be one of "left", + * "center" and "right". + * + * @type {Highcharts.AlignValue} + */ + align: "left", + /** + * The vertical alignment of the caption. Can be one of `"top"`, + * `"middle"` and `"bottom"`. When middle, the caption behaves as + * floating. + * + * @type {Highcharts.VerticalAlignValue} + */ + verticalAlign: "bottom", + }, + /** + * The plotOptions is a wrapper object for config objects for each series + * type. The config objects for each series can also be overridden for + * each series item as given in the series array. + * + * Configuration options for the series are given in three levels. Options + * for all series in a chart are given in the [plotOptions.series]( + * #plotOptions.series) object. Then options for all series of a specific + * type are given in the plotOptions of that type, for example + * `plotOptions.line`. Next, options for one single series are given in + * [the series array](#series). + */ + plotOptions: {}, + /** + * HTML labels that can be positioned anywhere in the chart area. + * + * This option is deprecated since v7.1.2. Instead, use + * [annotations](#annotations) that support labels. + * + * @deprecated + * @product highcharts highstock + */ + labels: { + /** + * An HTML label that can be positioned anywhere in the chart area. + * + * @deprecated + * @type {Array<*>} + * @apioption labels.items + */ + /** + * Inner HTML or text for the label. + * + * @deprecated + * @type {string} + * @apioption labels.items.html + */ + /** + * CSS styles for each label. To position the label, use left and top + * like this: + * ```js + * style: { + * left: '100px', + * top: '100px' + * } + * ``` + * + * @deprecated + * @type {Highcharts.CSSObject} + * @apioption labels.items.style + */ + /** + * Shared CSS styles for all labels. + * + * @deprecated + * @type {Highcharts.CSSObject} + * @default {"color": "#333333", "position": "absolute"} + */ + style: { /** - * @private - * @function Highcharts.SVGElement#textSetter - * @param {string} value + * @ignore-option */ - SVGElement.prototype.textSetter = function (value) { - if (value !== this.textStr) { - // Delete size caches when the text changes - // delete this.bBox; // old code in series-label - delete this.textPxLength; - this.textStr = value; - if (this.added) { - this.renderer.buildText(this); - } - } - }; + position: "absolute", /** - * @private - * @function Highcharts.SVGElement#titleSetter - * @param {string} value + * @ignore-option */ - SVGElement.prototype.titleSetter = function (value) { - var titleNode = this.element.getElementsByTagName('title')[0]; - if (!titleNode) { - titleNode = doc.createElementNS(this.SVG_NS, 'title'); - this.element.appendChild(titleNode); - } - // Remove text content if it exists - if (titleNode.firstChild) { - titleNode.removeChild(titleNode.firstChild); - } - titleNode.appendChild(doc.createTextNode( - // #3276, #3895 - String(pick(value, '')) - .replace(/<[^>]*>/g, '') - .replace(/</g, '<') - .replace(/>/g, '>'))); - }; - /** - * Bring the element to the front. Alternatively, a new zIndex can be set. - * - * @sample highcharts/members/element-tofront/ - * Click an element to bring it to front - * - * @function Highcharts.SVGElement#toFront - * - * @return {Highcharts.SVGElement} - * Returns the SVGElement for chaining. + color: "#333333", + }, + }, + /** + * The legend is a box containing a symbol and name for each series + * item or point item in the chart. Each series (or points in case + * of pie charts) is represented by a symbol and its name in the legend. + * + * It is possible to override the symbol creator function and create + * [custom legend symbols](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/studies/legend-custom-symbol/). + * + * @productdesc {highmaps} + * A Highmaps legend by default contains one legend item per series, but if + * a `colorAxis` is defined, the axis will be displayed in the legend. + * Either as a gradient, or as multiple legend items for `dataClasses`. + */ + legend: { + /** + * The background color of the legend. + * + * @see In styled mode, the legend background fill can be applied with + * the `.highcharts-legend-box` class. + * + * @sample {highcharts} highcharts/legend/backgroundcolor/ + * Yellowish background + * @sample {highstock} stock/legend/align/ + * Various legend options + * @sample {highmaps} maps/legend/border-background/ + * Border and background options + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @apioption legend.backgroundColor + */ + /** + * The width of the drawn border around the legend. + * + * @see In styled mode, the legend border stroke width can be applied + * with the `.highcharts-legend-box` class. + * + * @sample {highcharts} highcharts/legend/borderwidth/ + * 2px border width + * @sample {highstock} stock/legend/align/ + * Various legend options + * @sample {highmaps} maps/legend/border-background/ + * Border and background options + * + * @type {number} + * @default 0 + * @apioption legend.borderWidth + */ + /** + * Enable or disable the legend. There is also a series-specific option, + * [showInLegend](#plotOptions.series.showInLegend), that can hide the + * series from the legend. In some series types this is `false` by + * default, so it must set to `true` in order to show the legend for the + * series. + * + * @sample {highcharts} highcharts/legend/enabled-false/ Legend disabled + * @sample {highstock} stock/legend/align/ Various legend options + * @sample {highmaps} maps/legend/enabled-false/ Legend disabled + * + * @default {highstock} false + * @default {highmaps} true + * @default {gantt} false + */ + enabled: true, + /** + * The horizontal alignment of the legend box within the chart area. + * Valid values are `left`, `center` and `right`. + * + * In the case that the legend is aligned in a corner position, the + * `layout` option will determine whether to place it above/below + * or on the side of the plot area. + * + * @sample {highcharts} highcharts/legend/align/ + * Legend at the right of the chart + * @sample {highstock} stock/legend/align/ + * Various legend options + * @sample {highmaps} maps/legend/alignment/ + * Legend alignment + * + * @type {Highcharts.AlignValue} + * @since 2.0 + */ + align: "center", + /** + * If the [layout](legend.layout) is `horizontal` and the legend items + * span over two lines or more, whether to align the items into vertical + * columns. Setting this to `false` makes room for more items, but will + * look more messy. + * + * @since 6.1.0 + */ + alignColumns: true, + /** + * When the legend is floating, the plot area ignores it and is allowed + * to be placed below it. + * + * @sample {highcharts} highcharts/legend/floating-false/ + * False by default + * @sample {highcharts} highcharts/legend/floating-true/ + * True + * @sample {highmaps} maps/legend/alignment/ + * Floating legend + * + * @type {boolean} + * @default false + * @since 2.1 + * @apioption legend.floating + */ + /** + * The layout of the legend items. Can be one of `horizontal` or + * `vertical` or `proximate`. When `proximate`, the legend items will be + * placed as close as possible to the graphs they're representing, + * except in inverted charts or when the legend position doesn't allow + * it. + * + * @sample {highcharts} highcharts/legend/layout-horizontal/ + * Horizontal by default + * @sample {highcharts} highcharts/legend/layout-vertical/ + * Vertical + * @sample highcharts/legend/layout-proximate + * Labels proximate to the data + * @sample {highstock} stock/legend/layout-horizontal/ + * Horizontal by default + * @sample {highmaps} maps/legend/padding-itemmargin/ + * Vertical with data classes + * @sample {highmaps} maps/legend/layout-vertical/ + * Vertical with color axis gradient + * + * @validvalue ["horizontal", "vertical", "proximate"] + */ + layout: "horizontal", + /** + * In a legend with horizontal layout, the itemDistance defines the + * pixel distance between each item. + * + * @sample {highcharts} highcharts/legend/layout-horizontal/ + * 50px item distance + * @sample {highstock} highcharts/legend/layout-horizontal/ + * 50px item distance + * + * @type {number} + * @default {highcharts} 20 + * @default {highstock} 20 + * @default {highmaps} 8 + * @since 3.0.3 + * @apioption legend.itemDistance + */ + /** + * The pixel bottom margin for each legend item. + * + * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/ + * Padding and item margins demonstrated + * @sample {highmaps} maps/legend/padding-itemmargin/ + * Padding and item margins demonstrated + * + * @type {number} + * @default 0 + * @since 2.2.0 + * @apioption legend.itemMarginBottom + */ + /** + * The pixel top margin for each legend item. + * + * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/ + * Padding and item margins demonstrated + * @sample {highmaps} maps/legend/padding-itemmargin/ + * Padding and item margins demonstrated + * + * @type {number} + * @default 0 + * @since 2.2.0 + * @apioption legend.itemMarginTop + */ + /** + * The width for each legend item. By default the items are laid out + * successively. In a [horizontal layout](legend.layout), if the items + * are laid out across two rows or more, they will be vertically aligned + * depending on the [legend.alignColumns](legend.alignColumns) option. + * + * @sample {highcharts} highcharts/legend/itemwidth-default/ + * Undefined by default + * @sample {highcharts} highcharts/legend/itemwidth-80/ + * 80 for aligned legend items + * + * @type {number} + * @since 2.0 + * @apioption legend.itemWidth + */ + /** + * A [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) + * for each legend label. Available variables relates to properties on + * the series, or the point in case of pies. + * + * @type {string} + * @default {name} + * @since 1.3 + * @apioption legend.labelFormat + */ + /* eslint-disable valid-jsdoc */ + /** + * Callback function to format each of the series' labels. The `this` + * keyword refers to the series object, or the point object in case of + * pie charts. By default the series or point name is printed. + * + * @productdesc {highmaps} + * In Highmaps the context can also be a data class in case of a + * `colorAxis`. + * + * @sample {highcharts} highcharts/legend/labelformatter/ + * Add text + * @sample {highmaps} maps/legend/labelformatter/ + * Data classes with label formatter + * + * @type {Highcharts.FormatterCallbackFunction<Point|Series>} + */ + labelFormatter: function () { + /** eslint-enable valid-jsdoc */ + return this.name; + }, + /** + * Line height for the legend items. Deprecated as of 2.1\. Instead, + * the line height for each item can be set using + * `itemStyle.lineHeight`, and the padding between items using + * `itemMarginTop` and `itemMarginBottom`. + * + * @sample {highcharts} highcharts/legend/lineheight/ + * Setting padding + * + * @deprecated + * + * @type {number} + * @default 16 + * @since 2.0 + * @product highcharts gantt + * @apioption legend.lineHeight + */ + /** + * If the plot area sized is calculated automatically and the legend is + * not floating, the legend margin is the space between the legend and + * the axis labels or plot area. + * + * @sample {highcharts} highcharts/legend/margin-default/ + * 12 pixels by default + * @sample {highcharts} highcharts/legend/margin-30/ + * 30 pixels + * + * @type {number} + * @default 12 + * @since 2.1 + * @apioption legend.margin + */ + /** + * Maximum pixel height for the legend. When the maximum height is + * extended, navigation will show. + * + * @type {number} + * @since 2.3.0 + * @apioption legend.maxHeight + */ + /** + * The color of the drawn border around the legend. + * + * @see In styled mode, the legend border stroke can be applied with the + * `.highcharts-legend-box` class. + * + * @sample {highcharts} highcharts/legend/bordercolor/ + * Brown border + * @sample {highstock} stock/legend/align/ + * Various legend options + * @sample {highmaps} maps/legend/border-background/ + * Border and background options + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + */ + borderColor: "#999999", + /** + * The border corner radius of the legend. + * + * @sample {highcharts} highcharts/legend/borderradius-default/ + * Square by default + * @sample {highcharts} highcharts/legend/borderradius-round/ + * 5px rounded + * @sample {highmaps} maps/legend/border-background/ + * Border and background options + */ + borderRadius: 0, + /** + * Options for the paging or navigation appearing when the legend is + * overflown. Navigation works well on screen, but not in static + * exported images. One way of working around that is to + * [increase the chart height in + * export](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/legend/navigation-enabled-false/). + */ + navigation: { + /** + * How to animate the pages when navigating up or down. A value of + * `true` applies the default navigation given in the + * `chart.animation` option. Additional options can be given as an + * object containing values for easing and duration. + * + * @sample {highcharts} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * @sample {highstock} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + * @default true + * @since 2.2.4 + * @apioption legend.navigation.animation */ - SVGElement.prototype.toFront = function () { - var element = this.element; - element.parentNode.appendChild(element); - return this; - }; /** - * Move an object and its children by x and y values. - * - * @function Highcharts.SVGElement#translate + * The pixel size of the up and down arrows in the legend paging + * navigation. * - * @param {number} x - * The x value. + * @sample {highcharts} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * @sample {highstock} highcharts/legend/navigation/ + * Legend page navigation demonstrated * - * @param {number} y - * The y value. - * - * @return {Highcharts.SVGElement} + * @type {number} + * @default 12 + * @since 2.2.4 + * @apioption legend.navigation.arrowSize */ - SVGElement.prototype.translate = function (x, y) { - return this.attr({ - translateX: x, - translateY: y - }); - }; /** - * Update the shadow elements with new attributes. - * - * @private - * @function Highcharts.SVGElement#updateShadows - * - * @param {string} key - * The attribute name. - * - * @param {number} value - * The value of the attribute. + * Whether to enable the legend navigation. In most cases, disabling + * the navigation results in an unwanted overflow. * - * @param {Function} setter - * The setter function, inherited from the parent wrapper. - */ - SVGElement.prototype.updateShadows = function (key, value, setter) { - var shadows = this.shadows; - if (shadows) { - var i = shadows.length; - while (i--) { - setter.call(shadows[i], key === 'height' ? - Math.max(value - (shadows[i].cutHeight || 0), 0) : - key === 'd' ? this.d : value, key, shadows[i]); - } - } - }; - /** - * Update the transform attribute based on internal properties. Deals with - * the custom `translateX`, `translateY`, `rotation`, `scaleX` and `scaleY` - * attributes and updates the SVG `transform` attribute. + * See also the [adapt chart to legend]( + * https://www.highcharts.com/products/plugin-registry/single/8/Adapt-Chart-To-Legend) + * plugin for a solution to extend the chart height to make room for + * the legend, optionally in exported charts only. * - * @private - * @function Highcharts.SVGElement#updateTransform + * @type {boolean} + * @default true + * @since 4.2.4 + * @apioption legend.navigation.enabled */ - SVGElement.prototype.updateTransform = function () { - var wrapper = this, - translateX = wrapper.translateX || 0, - translateY = wrapper.translateY || 0, - scaleX = wrapper.scaleX, - scaleY = wrapper.scaleY, - inverted = wrapper.inverted, - rotation = wrapper.rotation, - matrix = wrapper.matrix, - element = wrapper.element, - transform; - // Flipping affects translate as adjustment for flipping around the - // group's axis - if (inverted) { - translateX += wrapper.width; - translateY += wrapper.height; - } - // Apply translate. Nearly all transformed elements have translation, - // so instead of checking for translate = 0, do it always (#1767, - // #1846). - transform = ['translate(' + translateX + ',' + translateY + ')']; - // apply matrix - if (defined(matrix)) { - transform.push('matrix(' + matrix.join(',') + ')'); - } - // apply rotation - if (inverted) { - transform.push('rotate(90) scale(-1,1)'); - } - else if (rotation) { // text rotation - transform.push('rotate(' + rotation + ' ' + - pick(this.rotationOriginX, element.getAttribute('x'), 0) + - ' ' + - pick(this.rotationOriginY, element.getAttribute('y') || 0) + ')'); - } - // apply scale - if (defined(scaleX) || defined(scaleY)) { - transform.push('scale(' + pick(scaleX, 1) + ' ' + pick(scaleY, 1) + ')'); - } - if (transform.length) { - element.setAttribute('transform', transform.join(' ')); - } - }; /** - * @private - * @function Highcharts.SVGElement#visibilitySetter + * Text styles for the legend page navigation. * - * @param {string} value + * @see In styled mode, the navigation items are styled with the + * `.highcharts-legend-navigation` class. * - * @param {string} key + * @sample {highcharts} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * @sample {highstock} highcharts/legend/navigation/ + * Legend page navigation demonstrated * - * @param {Highcharts.SVGDOMElement} element + * @type {Highcharts.CSSObject} + * @since 2.2.4 + * @apioption legend.navigation.style + */ + /** + * The color for the active up or down arrow in the legend page + * navigation. + * + * @see In styled mode, the active arrow be styled with the + * `.highcharts-legend-nav-active` class. + * + * @sample {highcharts} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * @sample {highstock} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 2.2.4 + */ + activeColor: "#003399", + /** + * The color of the inactive up or down arrow in the legend page + * navigation. . + * + * @see In styled mode, the inactive arrow be styled with the + * `.highcharts-legend-nav-inactive` class. + * + * @sample {highcharts} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * @sample {highstock} highcharts/legend/navigation/ + * Legend page navigation demonstrated + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 2.2.4 + */ + inactiveColor: "#cccccc", + }, + /** + * The inner padding of the legend box. + * + * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/ + * Padding and item margins demonstrated + * @sample {highmaps} maps/legend/padding-itemmargin/ + * Padding and item margins demonstrated + * + * @type {number} + * @default 8 + * @since 2.2.0 + * @apioption legend.padding + */ + /** + * Whether to reverse the order of the legend items compared to the + * order of the series or points as defined in the configuration object. + * + * @see [yAxis.reversedStacks](#yAxis.reversedStacks), + * [series.legendIndex](#series.legendIndex) + * + * @sample {highcharts} highcharts/legend/reversed/ + * Stacked bar with reversed legend + * + * @type {boolean} + * @default false + * @since 1.2.5 + * @apioption legend.reversed + */ + /** + * Whether to show the symbol on the right side of the text rather than + * the left side. This is common in Arabic and Hebraic. + * + * @sample {highcharts} highcharts/legend/rtl/ + * Symbol to the right + * + * @type {boolean} + * @default false + * @since 2.2 + * @apioption legend.rtl + */ + /** + * CSS styles for the legend area. In the 1.x versions the position + * of the legend area was determined by CSS. In 2.x, the position is + * determined by properties like `align`, `verticalAlign`, `x` and `y`, + * but the styles are still parsed for backwards compatibility. + * + * @deprecated + * + * @type {Highcharts.CSSObject} + * @product highcharts highstock + * @apioption legend.style + */ + /** + * CSS styles for each legend item. Only a subset of CSS is supported, + * notably those options related to text. The default `textOverflow` + * property makes long texts truncate. Set it to `undefined` to wrap + * text instead. A `width` property can be added to control the text + * width. + * + * @see In styled mode, the legend items can be styled with the + * `.highcharts-legend-item` class. + * + * @sample {highcharts} highcharts/legend/itemstyle/ + * Bold black text + * @sample {highmaps} maps/legend/itemstyle/ + * Item text styles + * + * @type {Highcharts.CSSObject} + * @default {"color": "#333333", "cursor": "pointer", "fontSize": "12px", "fontWeight": "bold", "textOverflow": "ellipsis"} + */ + itemStyle: { + /** + * @ignore + */ + color: "#333333", + /** + * @ignore + */ + cursor: "pointer", + /** + * @ignore + */ + fontSize: "12px", + /** + * @ignore + */ + fontWeight: "bold", + /** + * @ignore + */ + textOverflow: "ellipsis", + }, + /** + * CSS styles for each legend item in hover mode. Only a subset of + * CSS is supported, notably those options related to text. Properties + * are inherited from `style` unless overridden here. + * + * @see In styled mode, the hovered legend items can be styled with + * the `.highcharts-legend-item:hover` pesudo-class. + * + * @sample {highcharts} highcharts/legend/itemhoverstyle/ + * Red on hover + * @sample {highmaps} maps/legend/itemstyle/ + * Item text styles + * + * @type {Highcharts.CSSObject} + * @default {"color": "#000000"} + */ + itemHoverStyle: { + /** + * @ignore + */ + color: "#000000", + }, + /** + * CSS styles for each legend item when the corresponding series or + * point is hidden. Only a subset of CSS is supported, notably those + * options related to text. Properties are inherited from `style` + * unless overridden here. + * + * @see In styled mode, the hidden legend items can be styled with + * the `.highcharts-legend-item-hidden` class. + * + * @sample {highcharts} highcharts/legend/itemhiddenstyle/ + * Darker gray color + * + * @type {Highcharts.CSSObject} + * @default {"color": "#cccccc"} + */ + itemHiddenStyle: { + /** + * @ignore + */ + color: "#cccccc", + }, + /** + * Whether to apply a drop shadow to the legend. A `backgroundColor` + * also needs to be applied for this to take effect. The shadow can be + * an object configuration containing `color`, `offsetX`, `offsetY`, + * `opacity` and `width`. + * + * @sample {highcharts} highcharts/legend/shadow/ + * White background and drop shadow + * @sample {highstock} stock/legend/align/ + * Various legend options + * @sample {highmaps} maps/legend/border-background/ + * Border and background options + * + * @type {boolean|Highcharts.CSSObject} + */ + shadow: false, + /** + * Default styling for the checkbox next to a legend item when + * `showCheckbox` is true. + * + * @type {Highcharts.CSSObject} + * @default {"width": "13px", "height": "13px", "position":"absolute"} + */ + itemCheckboxStyle: { + /** + * @ignore + */ + position: "absolute", + /** + * @ignore + */ + width: "13px", + /** + * @ignore + */ + height: "13px", + }, + // itemWidth: undefined, + /** + * When this is true, the legend symbol width will be the same as + * the symbol height, which in turn defaults to the font size of the + * legend items. + * + * @since 5.0.0 + */ + squareSymbol: true, + /** + * The pixel height of the symbol for series types that use a rectangle + * in the legend. Defaults to the font size of legend items. + * + * @productdesc {highmaps} + * In Highmaps, when the symbol is the gradient of a vertical color + * axis, the height defaults to 200. + * + * @sample {highmaps} maps/legend/layout-vertical-sized/ + * Sized vertical gradient + * @sample {highmaps} maps/legend/padding-itemmargin/ + * No distance between data classes + * + * @type {number} + * @since 3.0.8 + * @apioption legend.symbolHeight + */ + /** + * The border radius of the symbol for series types that use a rectangle + * in the legend. Defaults to half the `symbolHeight`. + * + * @sample {highcharts} highcharts/legend/symbolradius/ + * Round symbols + * @sample {highstock} highcharts/legend/symbolradius/ + * Round symbols + * @sample {highmaps} highcharts/legend/symbolradius/ + * Round symbols + * + * @type {number} + * @since 3.0.8 + * @apioption legend.symbolRadius + */ + /** + * The pixel width of the legend item symbol. When the `squareSymbol` + * option is set, this defaults to the `symbolHeight`, otherwise 16. + * + * @productdesc {highmaps} + * In Highmaps, when the symbol is the gradient of a horizontal color + * axis, the width defaults to 200. + * + * @sample {highcharts} highcharts/legend/symbolwidth/ + * Greater symbol width and padding + * @sample {highmaps} maps/legend/padding-itemmargin/ + * Padding and item margins demonstrated + * @sample {highmaps} maps/legend/layout-vertical-sized/ + * Sized vertical gradient + * + * @type {number} + * @apioption legend.symbolWidth + */ + /** + * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the legend item texts. + * + * Prior to 4.1.7, when using HTML, [legend.navigation]( + * #legend.navigation) was disabled. + * + * @type {boolean} + * @default false + * @apioption legend.useHTML + */ + /** + * The width of the legend box. If a number is set, it translates to + * pixels. Since v7.0.2 it allows setting a percent string of the full + * chart width, for example `40%`. + * + * Defaults to the full chart width for legends below or above the + * chart, half the chart width for legends to the left and right. + * + * @sample {highcharts} highcharts/legend/width/ + * Aligned to the plot area + * @sample {highcharts} highcharts/legend/width-percent/ + * A percent of the chart width + * + * @type {number|string} + * @since 2.0 + * @apioption legend.width + */ + /** + * The pixel padding between the legend item symbol and the legend + * item text. + * + * @sample {highcharts} highcharts/legend/symbolpadding/ + * Greater symbol width and padding + */ + symbolPadding: 5, + /** + * The vertical alignment of the legend box. Can be one of `top`, + * `middle` or `bottom`. Vertical position can be further determined + * by the `y` option. + * + * In the case that the legend is aligned in a corner position, the + * `layout` option will determine whether to place it above/below + * or on the side of the plot area. + * + * When the [layout](#legend.layout) option is `proximate`, the + * `verticalAlign` option doesn't apply. + * + * @sample {highcharts} highcharts/legend/verticalalign/ + * Legend 100px from the top of the chart + * @sample {highstock} stock/legend/align/ + * Various legend options + * @sample {highmaps} maps/legend/alignment/ + * Legend alignment + * + * @type {Highcharts.VerticalAlignValue} + * @since 2.0 + */ + verticalAlign: "bottom", + // width: undefined, + /** + * The x offset of the legend relative to its horizontal alignment + * `align` within chart.spacingLeft and chart.spacingRight. Negative + * x moves it to the left, positive x moves it to the right. + * + * @sample {highcharts} highcharts/legend/width/ + * Aligned to the plot area + * + * @since 2.0 + */ + x: 0, + /** + * The vertical offset of the legend relative to it's vertical alignment + * `verticalAlign` within chart.spacingTop and chart.spacingBottom. + * Negative y moves it up, positive y moves it down. + * + * @sample {highcharts} highcharts/legend/verticalalign/ + * Legend 100px from the top of the chart + * @sample {highstock} stock/legend/align/ + * Various legend options + * @sample {highmaps} maps/legend/alignment/ + * Legend alignment + * + * @since 2.0 + */ + y: 0, + /** + * A title to be added on top of the legend. + * + * @sample {highcharts} highcharts/legend/title/ + * Legend title + * @sample {highmaps} maps/legend/alignment/ + * Legend with title + * + * @since 3.0 + */ + title: { + /** + * A text or HTML string for the title. * - * @return {void} + * @type {string} + * @since 3.0 + * @apioption legend.title.text */ - SVGElement.prototype.visibilitySetter = function (value, key, element) { - // IE9-11 doesn't handle visibilty:inherit well, so we remove the - // attribute instead (#2881, #3909) - if (value === 'inherit') { - element.removeAttribute(key); - } - else if (this[key] !== value) { // #6747 - element.setAttribute(key, value); - } - this[key] = value; - }; /** - * @private - * @function Highcharts.SVGElement#xGetter + * Generic CSS styles for the legend title. * - * @param {string} key + * @see In styled mode, the legend title is styled with the + * `.highcharts-legend-title` class. * - * @return {number|string|null} + * @type {Highcharts.CSSObject} + * @default {"fontWeight": "bold"} + * @since 3.0 */ - SVGElement.prototype.xGetter = function (key) { - if (this.element.nodeName === 'circle') { - if (key === 'x') { - key = 'cx'; - } - else if (key === 'y') { - key = 'cy'; - } - } - return this._defaultGetter(key); - }; - /** - * @private - * @function Highcharts.SVGElement#zIndexSetter - * @param {number} [value] - * @param {string} [key] - * @return {boolean} - */ - SVGElement.prototype.zIndexSetter = function (value, key) { - var renderer = this.renderer, - parentGroup = this.parentGroup, - parentWrapper = parentGroup || renderer, - parentNode = parentWrapper.element || renderer.box, - childNodes, - otherElement, - otherZIndex, - element = this.element, - inserted = false, - undefinedOtherZIndex, - svgParent = parentNode === renderer.box, - run = this.added, - i; - if (defined(value)) { - // So we can read it for other elements in the group - element.setAttribute('data-z-index', value); - value = +value; - if (this[key] === value) { - // Only update when needed (#3865) - run = false; - } - } - else if (defined(this[key])) { - element.removeAttribute('data-z-index'); - } - this[key] = value; - // Insert according to this and other elements' zIndex. Before .add() is - // called, nothing is done. Then on add, or by later calls to - // zIndexSetter, the node is placed on the right place in the DOM. - if (run) { - value = this.zIndex; - if (value && parentGroup) { - parentGroup.handleZ = true; - } - childNodes = parentNode.childNodes; - for (i = childNodes.length - 1; i >= 0 && !inserted; i--) { - otherElement = childNodes[i]; - otherZIndex = otherElement.getAttribute('data-z-index'); - undefinedOtherZIndex = !defined(otherZIndex); - if (otherElement !== element) { - if ( - // Negative zIndex versus no zIndex: - // On all levels except the highest. If the parent is - // <svg>, then we don't want to put items before <desc> - // or <defs> - value < 0 && - undefinedOtherZIndex && - !svgParent && - !i) { - parentNode.insertBefore(element, childNodes[i]); - inserted = true; - } - else if ( - // Insert after the first element with a lower zIndex - pInt(otherZIndex) <= value || - // If negative zIndex, add this before first undefined - // zIndex element - (undefinedOtherZIndex && - (!defined(value) || value >= 0))) { - parentNode.insertBefore(element, childNodes[i + 1] || null // null for oldIE export - ); - inserted = true; - } - } - } - if (!inserted) { - parentNode.insertBefore(element, childNodes[svgParent ? 3 : 0] || null // null for oldIE - ); - inserted = true; - } - } - return inserted; - }; - return SVGElement; - }()); - // Some shared setters and getters - SVGElement.prototype['stroke-widthSetter'] = SVGElement.prototype.strokeSetter; - SVGElement.prototype.yGetter = SVGElement.prototype.xGetter; - SVGElement.prototype.matrixSetter = - SVGElement.prototype.rotationOriginXSetter = - SVGElement.prototype.rotationOriginYSetter = - SVGElement.prototype.rotationSetter = - SVGElement.prototype.scaleXSetter = - SVGElement.prototype.scaleYSetter = - SVGElement.prototype.translateXSetter = - SVGElement.prototype.translateYSetter = - SVGElement.prototype.verticalAlignSetter = function (value, key) { - this[key] = value; - this.doTransform = true; - }; - H.SVGElement = SVGElement; + style: { + /** + * @ignore + */ + fontWeight: "bold", + }, + }, + }, + /** + * The loading options control the appearance of the loading screen + * that covers the plot area on chart operations. This screen only + * appears after an explicit call to `chart.showLoading()`. It is a + * utility for developers to communicate to the end user that something + * is going on, for example while retrieving new data via an XHR connection. + * The "Loading..." text itself is not part of this configuration + * object, but part of the `lang` object. + */ + loading: { + /** + * The duration in milliseconds of the fade out effect. + * + * @sample highcharts/loading/hideduration/ + * Fade in and out over a second + * + * @type {number} + * @default 100 + * @since 1.2.0 + * @apioption loading.hideDuration + */ + /** + * The duration in milliseconds of the fade in effect. + * + * @sample highcharts/loading/hideduration/ + * Fade in and out over a second + * + * @type {number} + * @default 100 + * @since 1.2.0 + * @apioption loading.showDuration + */ + /** + * CSS styles for the loading label `span`. + * + * @see In styled mode, the loading label is styled with the + * `.highcharts-loading-inner` class. + * + * @sample {highcharts|highmaps} highcharts/loading/labelstyle/ + * Vertically centered + * @sample {highstock} stock/loading/general/ + * Label styles + * + * @type {Highcharts.CSSObject} + * @default {"fontWeight": "bold", "position": "relative", "top": "45%"} + * @since 1.2.0 + */ + labelStyle: { + /** + * @ignore + */ + fontWeight: "bold", + /** + * @ignore + */ + position: "relative", + /** + * @ignore + */ + top: "45%", + }, + /** + * CSS styles for the loading screen that covers the plot area. + * + * In styled mode, the loading label is styled with the + * `.highcharts-loading` class. + * + * @sample {highcharts|highmaps} highcharts/loading/style/ + * Gray plot area, white text + * @sample {highstock} stock/loading/general/ + * Gray plot area, white text + * + * @type {Highcharts.CSSObject} + * @default {"position": "absolute", "backgroundColor": "#ffffff", "opacity": 0.5, "textAlign": "center"} + * @since 1.2.0 + */ + style: { + /** + * @ignore + */ + position: "absolute", + /** + * @ignore + */ + backgroundColor: "#ffffff", + /** + * @ignore + */ + opacity: 0.5, + /** + * @ignore + */ + textAlign: "center", + }, + }, + /** + * Options for the tooltip that appears when the user hovers over a + * series or point. + * + * @declare Highcharts.TooltipOptions + */ + tooltip: { + /** + * The color of the tooltip border. When `undefined`, the border takes + * the color of the corresponding series or point. + * + * @sample {highcharts} highcharts/tooltip/bordercolor-default/ + * Follow series by default + * @sample {highcharts} highcharts/tooltip/bordercolor-black/ + * Black border + * @sample {highstock} stock/tooltip/general/ + * Styled tooltip + * @sample {highmaps} maps/tooltip/background-border/ + * Background and border demo + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @apioption tooltip.borderColor + */ + /** + * A CSS class name to apply to the tooltip's container div, + * allowing unique CSS styling for each chart. + * + * @type {string} + * @apioption tooltip.className + */ + /** + * Since 4.1, the crosshair definitions are moved to the Axis object + * in order for a better separation from the tooltip. See + * [xAxis.crosshair](#xAxis.crosshair). + * + * @sample {highcharts} highcharts/tooltip/crosshairs-x/ + * Enable a crosshair for the x value + * + * @deprecated + * + * @type {*} + * @default true + * @apioption tooltip.crosshairs + */ + /** + * Distance from point to tooltip in pixels. + * + * @type {number} + * @default 16 + * @apioption tooltip.distance + */ + /** + * Whether the tooltip should follow the mouse as it moves across + * columns, pie slices and other point types with an extent. + * By default it behaves this way for pie, polygon, map, sankey + * and wordcloud series by override in the `plotOptions` + * for those series types. + * + * For touch moves to behave the same way, [followTouchMove]( + * #tooltip.followTouchMove) must be `true` also. + * + * @type {boolean} + * @default {highcharts} false + * @default {highstock} false + * @default {highmaps} true + * @since 3.0 + * @apioption tooltip.followPointer + */ + /** + * Whether the tooltip should update as the finger moves on a touch + * device. If this is `true` and [chart.panning](#chart.panning) is + * set,`followTouchMove` will take over one-finger touches, so the user + * needs to use two fingers for zooming and panning. + * + * Note the difference to [followPointer](#tooltip.followPointer) that + * only defines the _position_ of the tooltip. If `followPointer` is + * false in for example a column series, the tooltip will show above or + * below the column, but as `followTouchMove` is true, the tooltip will + * jump from column to column as the user swipes across the plot area. + * + * @type {boolean} + * @default {highcharts} true + * @default {highstock} true + * @default {highmaps} false + * @since 3.0.1 + * @apioption tooltip.followTouchMove + */ + /** + * Callback function to format the text of the tooltip from scratch. In + * case of single or [shared](#tooltip.shared) tooltips, a string should + * be returned. In case of [split](#tooltip.split) tooltips, it should + * return an array where the first item is the header, and subsequent + * items are mapped to the points. Return `false` to disable tooltip for + * a specific point on series. + * + * A subset of HTML is supported. Unless `useHTML` is true, the HTML of + * the tooltip is parsed and converted to SVG, therefore this isn't a + * complete HTML renderer. The following HTML tags are supported: `b`, + * `br`, `em`, `i`, `span`, `strong`. Spans can be styled with a `style` + * attribute, but only text-related CSS, that is shared with SVG, is + * handled. + * + * The available data in the formatter differ a bit depending on whether + * the tooltip is shared or split, or belongs to a single point. In a + * shared/split tooltip, all properties except `x`, which is common for + * all points, are kept in an array, `this.points`. + * + * Available data are: + * + * - **this.percentage (not shared) /** + * **this.points[i].percentage (shared)**: + * Stacked series and pies only. The point's percentage of the total. + * + * - **this.point (not shared) / this.points[i].point (shared)**: + * The point object. The point name, if defined, is available through + * `this.point.name`. + * + * - **this.points**: + * In a shared tooltip, this is an array containing all other + * properties for each point. + * + * - **this.series (not shared) / this.points[i].series (shared)**: + * The series object. The series name is available through + * `this.series.name`. + * + * - **this.total (not shared) / this.points[i].total (shared)**: + * Stacked series only. The total value at this point's x value. + * + * - **this.x**: + * The x value. This property is the same regardless of the tooltip + * being shared or not. + * + * - **this.y (not shared) / this.points[i].y (shared)**: + * The y value. + * + * @sample {highcharts} highcharts/tooltip/formatter-simple/ + * Simple string formatting + * @sample {highcharts} highcharts/tooltip/formatter-shared/ + * Formatting with shared tooltip + * @sample {highcharts|highstock} highcharts/tooltip/formatter-split/ + * Formatting with split tooltip + * @sample highcharts/tooltip/formatter-conditional-default/ + * Extending default formatter + * @sample {highstock} stock/tooltip/formatter/ + * Formatting with shared tooltip + * @sample {highmaps} maps/tooltip/formatter/ + * String formatting + * + * @type {Highcharts.TooltipFormatterCallbackFunction} + * @apioption tooltip.formatter + */ + /** + * Callback function to format the text of the tooltip for + * visible null points. + * Works analogously to [formatter](#tooltip.formatter). + * + * @sample highcharts/plotoptions/series-nullformat + * Format data label and tooltip for null point. + * + * @type {Highcharts.TooltipFormatterCallbackFunction} + * @apioption tooltip.nullFormatter + */ + /** + * The number of milliseconds to wait until the tooltip is hidden when + * mouse out from a point or chart. + * + * @type {number} + * @default 500 + * @since 3.0 + * @apioption tooltip.hideDelay + */ + /** + * Whether to allow the tooltip to render outside the chart's SVG + * element box. By default (`false`), the tooltip is rendered within the + * chart's SVG element, which results in the tooltip being aligned + * inside the chart area. For small charts, this may result in clipping + * or overlapping. When `true`, a separate SVG element is created and + * overlaid on the page, allowing the tooltip to be aligned inside the + * page itself. + * + * Defaults to `true` if `chart.scrollablePlotArea` is activated, + * otherwise `false`. + * + * @sample highcharts/tooltip/outside + * Small charts with tooltips outside + * + * @type {boolean|undefined} + * @default undefined + * @since 6.1.1 + * @apioption tooltip.outside + */ + /** + * A callback function for formatting the HTML output for a single point + * in the tooltip. Like the `pointFormat` string, but with more + * flexibility. + * + * @type {Highcharts.FormatterCallbackFunction<Highcharts.Point>} + * @since 4.1.0 + * @context Highcharts.Point + * @apioption tooltip.pointFormatter + */ + /** + * A callback function to place the tooltip in a default position. The + * callback receives three parameters: `labelWidth`, `labelHeight` and + * `point`, where point contains values for `plotX` and `plotY` telling + * where the reference point is in the plot area. Add `chart.plotLeft` + * and `chart.plotTop` to get the full coordinates. + * + * Since v7, when [tooltip.split](#tooltip.split) option is enabled, + * positioner is called for each of the boxes separately, including + * xAxis header. xAxis header is not a point, instead `point` argument + * contains info: + * `{ plotX: Number, plotY: Number, isHeader: Boolean }` + * + * + * The return should be an object containing x and y values, for example + * `{ x: 100, y: 100 }`. + * + * @sample {highcharts} highcharts/tooltip/positioner/ + * A fixed tooltip position + * @sample {highstock} stock/tooltip/positioner/ + * A fixed tooltip position on top of the chart + * @sample {highmaps} maps/tooltip/positioner/ + * A fixed tooltip position + * @sample {highstock} stock/tooltip/split-positioner/ + * Split tooltip with fixed positions + * @sample {highstock} stock/tooltip/positioner-scrollable-plotarea/ + * Scrollable plot area combined with tooltip positioner + * + * @type {Highcharts.TooltipPositionerCallbackFunction} + * @since 2.2.4 + * @apioption tooltip.positioner + */ + /** + * The name of a symbol to use for the border around the tooltip. Can + * be one of: `"callout"`, `"circle"`, or `"square"`. When + * [tooltip.split](#tooltip.split) + * option is enabled, shape is applied to all boxes except header, which + * is controlled by + * [tooltip.headerShape](#tooltip.headerShape). + * + * Custom callbacks for symbol path generation can also be added to + * `Highcharts.SVGRenderer.prototype.symbols` the same way as for + * [series.marker.symbol](plotOptions.line.marker.symbol). + * + * @type {Highcharts.TooltipShapeValue} + * @default callout + * @since 4.0 + * @apioption tooltip.shape + */ + /** + * The name of a symbol to use for the border around the tooltip + * header. Applies only when [tooltip.split](#tooltip.split) is + * enabled. + * + * Custom callbacks for symbol path generation can also be added to + * `Highcharts.SVGRenderer.prototype.symbols` the same way as for + * [series.marker.symbol](plotOptions.line.marker.symbol). + * + * @see [tooltip.shape](#tooltip.shape) + * + * @sample {highstock} stock/tooltip/split-positioner/ + * Different shapes for header and split boxes + * + * @type {Highcharts.TooltipShapeValue} + * @default callout + * @validvalue ["callout", "square"] + * @since 7.0 + * @apioption tooltip.headerShape + */ + /** + * When the tooltip is shared, the entire plot area will capture mouse + * movement or touch events. Tooltip texts for series types with ordered + * data (not pie, scatter, flags etc) will be shown in a single bubble. + * This is recommended for single series charts and for tablet/mobile + * optimized charts. + * + * See also [tooltip.split](#tooltip.split), that is better suited for + * charts with many series, especially line-type series. The + * `tooltip.split` option takes precedence over `tooltip.shared`. + * + * @sample {highcharts} highcharts/tooltip/shared-false/ + * False by default + * @sample {highcharts} highcharts/tooltip/shared-true/ + * True + * @sample {highcharts} highcharts/tooltip/shared-x-crosshair/ + * True with x axis crosshair + * @sample {highcharts} highcharts/tooltip/shared-true-mixed-types/ + * True with mixed series types + * + * @type {boolean} + * @default false + * @since 2.1 + * @product highcharts highstock + * @apioption tooltip.shared + */ + /** + * Split the tooltip into one label per series, with the header close + * to the axis. This is recommended over [shared](#tooltip.shared) + * tooltips for charts with multiple line series, generally making them + * easier to read. This option takes precedence over `tooltip.shared`. + * + * @productdesc {highstock} In Highstock, tooltips are split by default + * since v6.0.0. Stock charts typically contain multi-dimension points + * and multiple panes, making split tooltips the preferred layout over + * the previous `shared` tooltip. + * + * @sample highcharts/tooltip/split/ + * Split tooltip + * @sample {highcharts|highstock} highcharts/tooltip/formatter-split/ + * Split tooltip and custom formatter callback + * + * @type {boolean} + * @default {highcharts} false + * @default {highstock} true + * @since 5.0.0 + * @product highcharts highstock + * @apioption tooltip.split + */ + /** + * Prevents the tooltip from switching or closing, when touched or + * pointed. + * + * @sample highcharts/tooltip/stickoncontact/ + * Tooltip sticks on pointer contact + * + * @type {boolean} + * @since 8.0.1 + * @apioption tooltip.stickOnContact + */ + /** + * Use HTML to render the contents of the tooltip instead of SVG. Using + * HTML allows advanced formatting like tables and images in the + * tooltip. It is also recommended for rtl languages as it works around + * rtl bugs in early Firefox. + * + * @sample {highcharts|highstock} highcharts/tooltip/footerformat/ + * A table for value alignment + * @sample {highcharts|highstock} highcharts/tooltip/fullhtml/ + * Full HTML tooltip + * @sample {highmaps} maps/tooltip/usehtml/ + * Pure HTML tooltip + * + * @type {boolean} + * @default false + * @since 2.2 + * @apioption tooltip.useHTML + */ + /** + * How many decimals to show in each series' y value. This is + * overridable in each series' tooltip options object. The default is to + * preserve all decimals. + * + * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/ + * Set decimals, prefix and suffix for the value + * @sample {highmaps} maps/tooltip/valuedecimals/ + * Set decimals, prefix and suffix for the value + * + * @type {number} + * @since 2.2 + * @apioption tooltip.valueDecimals + */ + /** + * A string to prepend to each series' y value. Overridable in each + * series' tooltip options object. + * + * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/ + * Set decimals, prefix and suffix for the value + * @sample {highmaps} maps/tooltip/valuedecimals/ + * Set decimals, prefix and suffix for the value + * + * @type {string} + * @since 2.2 + * @apioption tooltip.valuePrefix + */ + /** + * A string to append to each series' y value. Overridable in each + * series' tooltip options object. + * + * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/ + * Set decimals, prefix and suffix for the value + * @sample {highmaps} maps/tooltip/valuedecimals/ + * Set decimals, prefix and suffix for the value + * + * @type {string} + * @since 2.2 + * @apioption tooltip.valueSuffix + */ + /** + * The format for the date in the tooltip header if the X axis is a + * datetime axis. The default is a best guess based on the smallest + * distance between points in the chart. + * + * @sample {highcharts} highcharts/tooltip/xdateformat/ + * A different format + * + * @type {string} + * @product highcharts highstock gantt + * @apioption tooltip.xDateFormat + */ + /** + * How many decimals to show for the `point.change` value when the + * `series.compare` option is set. This is overridable in each series' + * tooltip options object. The default is to preserve all decimals. + * + * @type {number} + * @since 1.0.1 + * @product highstock + * @apioption tooltip.changeDecimals + */ + /** + * Enable or disable the tooltip. + * + * @sample {highcharts} highcharts/tooltip/enabled/ + * Disabled + * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/ + * Disable tooltip and show values on chart instead + */ + enabled: true, + /** + * Enable or disable animation of the tooltip. + * + * @type {boolean} + * @default true + * @since 2.3.0 + */ + animation: svg, + /** + * The radius of the rounded border corners. + * + * @sample {highcharts} highcharts/tooltip/bordercolor-default/ + * 5px by default + * @sample {highcharts} highcharts/tooltip/borderradius-0/ + * Square borders + * @sample {highmaps} maps/tooltip/background-border/ + * Background and border demo + */ + borderRadius: 3, + /** + * For series on a datetime axes, the date format in the tooltip's + * header will by default be guessed based on the closest data points. + * This member gives the default string representations used for + * each unit. For an overview of the replacement codes, see + * [dateFormat](/class-reference/Highcharts#dateFormat). + * + * @see [xAxis.dateTimeLabelFormats](#xAxis.dateTimeLabelFormats) + * + * @type {Highcharts.Dictionary<string>} + * @product highcharts highstock gantt + */ + dateTimeLabelFormats: { + /** @internal */ + millisecond: "%A, %b %e, %H:%M:%S.%L", + /** @internal */ + second: "%A, %b %e, %H:%M:%S", + /** @internal */ + minute: "%A, %b %e, %H:%M", + /** @internal */ + hour: "%A, %b %e, %H:%M", + /** @internal */ + day: "%A, %b %e, %Y", + /** @internal */ + week: "Week from %A, %b %e, %Y", + /** @internal */ + month: "%B %Y", + /** @internal */ + year: "%Y", + }, + /** + * A string to append to the tooltip format. + * + * @sample {highcharts} highcharts/tooltip/footerformat/ + * A table for value alignment + * @sample {highmaps} maps/tooltip/format/ + * Format demo + * + * @since 2.2 + */ + footerFormat: "", + /** + * Padding inside the tooltip, in pixels. + * + * @since 5.0.0 + */ + padding: 8, + /** + * Proximity snap for graphs or single points. It defaults to 10 for + * mouse-powered devices and 25 for touch devices. + * + * Note that in most cases the whole plot area captures the mouse + * movement, and in these cases `tooltip.snap` doesn't make sense. This + * applies when [stickyTracking](#plotOptions.series.stickyTracking) + * is `true` (default) and when the tooltip is [shared](#tooltip.shared) + * or [split](#tooltip.split). + * + * @sample {highcharts} highcharts/tooltip/bordercolor-default/ + * 10 px by default + * @sample {highcharts} highcharts/tooltip/snap-50/ + * 50 px on graph + * + * @type {number} + * @default 10/25 + * @since 1.2.0 + * @product highcharts highstock + */ + snap: isTouchDevice ? 25 : 10, + /** + * The HTML of the tooltip header line. Variables are enclosed by + * curly brackets. Available variables are `point.key`, `series.name`, + * `series.color` and other members from the `point` and `series` + * objects. The `point.key` variable contains the category name, x + * value or datetime string depending on the type of axis. For datetime + * axes, the `point.key` date format can be set using + * `tooltip.xDateFormat`. + * + * @sample {highcharts} highcharts/tooltip/footerformat/ + * An HTML table in the tooltip + * @sample {highstock} highcharts/tooltip/footerformat/ + * An HTML table in the tooltip + * @sample {highmaps} maps/tooltip/format/ + * Format demo + * + * @type {string} + * @apioption tooltip.headerFormat + */ + headerFormat: '<span style="font-size: 10px">{point.key}</span><br/>', + /** + * The HTML of the null point's line in the tooltip. Works analogously + * to [pointFormat](#tooltip.pointFormat). + * + * @sample {highcharts} highcharts/plotoptions/series-nullformat + * Format data label and tooltip for null point. + * + * @type {string} + * @apioption tooltip.nullFormat + */ + /** + * The HTML of the point's line in the tooltip. Variables are enclosed + * by curly brackets. Available variables are `point.x`, `point.y`, + * `series.name` and `series.color` and other properties on the same + * form. Furthermore, `point.y` can be extended by the + * `tooltip.valuePrefix` and `tooltip.valueSuffix` variables. This can + * also be overridden for each series, which makes it a good hook for + * displaying units. + * + * In styled mode, the dot is colored by a class name rather + * than the point color. + * + * @sample {highcharts} highcharts/tooltip/pointformat/ + * A different point format with value suffix + * @sample {highmaps} maps/tooltip/format/ + * Format demo + * + * @type {string} + * @since 2.2 + * @apioption tooltip.pointFormat + */ + pointFormat: + '<span style="color:{point.color}">\u25CF</span> {series.name}: <b>{point.y}</b><br/>', + /** + * The background color or gradient for the tooltip. + * + * In styled mode, the stroke width is set in the + * `.highcharts-tooltip-box` class. + * + * @sample {highcharts} highcharts/tooltip/backgroundcolor-solid/ + * Yellowish background + * @sample {highcharts} highcharts/tooltip/backgroundcolor-gradient/ + * Gradient + * @sample {highcharts} highcharts/css/tooltip-border-background/ + * Tooltip in styled mode + * @sample {highstock} stock/tooltip/general/ + * Custom tooltip + * @sample {highstock} highcharts/css/tooltip-border-background/ + * Tooltip in styled mode + * @sample {highmaps} maps/tooltip/background-border/ + * Background and border demo + * @sample {highmaps} highcharts/css/tooltip-border-background/ + * Tooltip in styled mode + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + */ + backgroundColor: color("#f7f7f7").setOpacity(0.85).get(), + /** + * The pixel width of the tooltip border. + * + * In styled mode, the stroke width is set in the + * `.highcharts-tooltip-box` class. + * + * @sample {highcharts} highcharts/tooltip/bordercolor-default/ + * 2px by default + * @sample {highcharts} highcharts/tooltip/borderwidth/ + * No border (shadow only) + * @sample {highcharts} highcharts/css/tooltip-border-background/ + * Tooltip in styled mode + * @sample {highstock} stock/tooltip/general/ + * Custom tooltip + * @sample {highstock} highcharts/css/tooltip-border-background/ + * Tooltip in styled mode + * @sample {highmaps} maps/tooltip/background-border/ + * Background and border demo + * @sample {highmaps} highcharts/css/tooltip-border-background/ + * Tooltip in styled mode + */ + borderWidth: 1, + /** + * Whether to apply a drop shadow to the tooltip. + * + * @sample {highcharts} highcharts/tooltip/bordercolor-default/ + * True by default + * @sample {highcharts} highcharts/tooltip/shadow/ + * False + * @sample {highmaps} maps/tooltip/positioner/ + * Fixed tooltip position, border and shadow disabled + * + * @type {boolean|Highcharts.ShadowOptionsObject} + */ + shadow: true, + /** + * CSS styles for the tooltip. The tooltip can also be styled through + * the CSS class `.highcharts-tooltip`. + * + * Note that the default `pointerEvents` style makes the tooltip ignore + * mouse events, so in order to use clickable tooltips, this value must + * be set to `auto`. + * + * @sample {highcharts} highcharts/tooltip/style/ + * Greater padding, bold text + * + * @type {Highcharts.CSSObject} + */ + style: { + /** @internal */ + color: "#333333", + /** @internal */ + cursor: "default", + /** @internal */ + fontSize: "12px", + /** @internal */ + whiteSpace: "nowrap", + }, + }, + /** + * Highchart by default puts a credits label in the lower right corner + * of the chart. This can be changed using these options. + */ + credits: { + /** + * Credits for map source to be concatenated with conventional credit + * text. By default this is a format string that collects copyright + * information from the map if available. + * + * @see [mapTextFull](#credits.mapTextFull) + * @see [text](#credits.text) + * + * @type {string} + * @default \u00a9 <a href="{geojson.copyrightUrl}">{geojson.copyrightShort}</a> + * @since 4.2.2 + * @product highmaps + * @apioption credits.mapText + */ + /** + * Detailed credits for map source to be displayed on hover of credits + * text. By default this is a format string that collects copyright + * information from the map if available. + * + * @see [mapText](#credits.mapText) + * @see [text](#credits.text) + * + * @type {string} + * @default {geojson.copyright} + * @since 4.2.2 + * @product highmaps + * @apioption credits.mapTextFull + */ + /** + * Whether to show the credits text. + * + * @sample {highcharts} highcharts/credits/enabled-false/ + * Credits disabled + * @sample {highstock} stock/credits/enabled/ + * Credits disabled + * @sample {highmaps} maps/credits/enabled-false/ + * Credits disabled + */ + enabled: true, + /** + * The URL for the credits label. + * + * @sample {highcharts} highcharts/credits/href/ + * Custom URL and text + * @sample {highmaps} maps/credits/customized/ + * Custom URL and text + */ + href: "https://www.highcharts.com?credits", + /** + * Position configuration for the credits label. + * + * @sample {highcharts} highcharts/credits/position-left/ + * Left aligned + * @sample {highcharts} highcharts/credits/position-left/ + * Left aligned + * @sample {highmaps} maps/credits/customized/ + * Left aligned + * @sample {highmaps} maps/credits/customized/ + * Left aligned + * + * @type {Highcharts.AlignObject} + * @since 2.1 + */ + position: { + /** @internal */ + align: "right", + /** @internal */ + x: -10, + /** @internal */ + verticalAlign: "bottom", + /** @internal */ + y: -5, + }, + /** + * CSS styles for the credits label. + * + * @see In styled mode, credits styles can be set with the + * `.highcharts-credits` class. + * + * @type {Highcharts.CSSObject} + */ + style: { + /** @internal */ + cursor: "pointer", + /** @internal */ + color: "#999999", + /** @internal */ + fontSize: "9px", + }, + /** + * The text for the credits label. + * + * @productdesc {highmaps} + * If a map is loaded as GeoJSON, the text defaults to + * `Highcharts @ {map-credits}`. Otherwise, it defaults to + * `Highcharts.com`. + * + * @sample {highcharts} highcharts/credits/href/ + * Custom URL and text + * @sample {highmaps} maps/credits/customized/ + * Custom URL and text + */ + text: "Highcharts.com", + }, + }; + /* eslint-disable spaced-comment */ - return H.SVGElement; - }); - _registerModule(_modules, 'Core/Renderer/SVG/SVGLabel.js', [_modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Utilities.js']], function (SVGElement, U) { + (""); + /** + * Global `Time` object with default options. Since v6.0.5, time settings can be + * applied individually for each chart. If no individual settings apply, this + * `Time` object is shared by all instances. + * + * @name Highcharts.time + * @type {Highcharts.Time} + */ + H.time = new Time(merge(H.defaultOptions.global, H.defaultOptions.time)); + /** + * Formats a JavaScript date timestamp (milliseconds since Jan 1st 1970) into a + * human readable date string. The format is a subset of the formats for PHP's + * [strftime](https://www.php.net/manual/en/function.strftime.php) function. + * Additional formats can be given in the {@link Highcharts.dateFormats} hook. + * + * Since v6.0.5, all internal dates are formatted through the + * {@link Highcharts.Chart#time} instance to respect chart-level time settings. + * The `Highcharts.dateFormat` function only reflects global time settings set + * with `setOptions`. + * + * Supported format keys: + * - `%a`: Short weekday, like 'Mon' + * - `%A`: Long weekday, like 'Monday' + * - `%d`: Two digit day of the month, 01 to 31 + * - `%e`: Day of the month, 1 through 31 + * - `%w`: Day of the week, 0 through 6 + * - `%b`: Short month, like 'Jan' + * - `%B`: Long month, like 'January' + * - `%m`: Two digit month number, 01 through 12 + * - `%y`: Two digits year, like 09 for 2009 + * - `%Y`: Four digits year, like 2009 + * - `%H`: Two digits hours in 24h format, 00 through 23 + * - `%k`: Hours in 24h format, 0 through 23 + * - `%I`: Two digits hours in 12h format, 00 through 11 + * - `%l`: Hours in 12h format, 1 through 12 + * - `%M`: Two digits minutes, 00 through 59 + * - `%p`: Upper case AM or PM + * - `%P`: Lower case AM or PM + * - `%S`: Two digits seconds, 00 through 59 + * - `%L`: Milliseconds (naming from Ruby) + * + * @function Highcharts.dateFormat + * + * @param {string} format + * The desired format where various time representations are prefixed + * with `%`. + * + * @param {number} timestamp + * The JavaScript timestamp. + * + * @param {boolean} [capitalize=false] + * Upper case first letter in the return. + * + * @return {string} + * The formatted date. + */ + H.dateFormat = function (format, timestamp, capitalize) { + return H.time.dateFormat(format, timestamp, capitalize); + }; + var optionsModule = { + dateFormat: H.dateFormat, + defaultOptions: H.defaultOptions, + time: H.time, + }; + + return optionsModule; + } + ); + _registerModule( + _modules, + "Core/Axis/Axis.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Color/Color.js"], + _modules["Core/Globals.js"], + _modules["Core/Axis/Tick.js"], + _modules["Core/Utilities.js"], + _modules["Core/Options.js"], + ], + function (A, Color, H, Tick, U, O) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animObject = A.animObject; + var addEvent = U.addEvent, + arrayMax = U.arrayMax, + arrayMin = U.arrayMin, + clamp = U.clamp, + correctFloat = U.correctFloat, + defined = U.defined, + destroyObjectProperties = U.destroyObjectProperties, + error = U.error, + extend = U.extend, + fireEvent = U.fireEvent, + format = U.format, + getMagnitude = U.getMagnitude, + isArray = U.isArray, + isFunction = U.isFunction, + isNumber = U.isNumber, + isString = U.isString, + merge = U.merge, + normalizeTickInterval = U.normalizeTickInterval, + objectEach = U.objectEach, + pick = U.pick, + relativeLength = U.relativeLength, + removeEvent = U.removeEvent, + splat = U.splat, + syncTimeout = U.syncTimeout; + /** + * Options for the path on the Axis to be calculated. + * @interface Highcharts.AxisPlotLinePathOptionsObject + */ /** + * Axis value. + * @name Highcharts.AxisPlotLinePathOptionsObject#value + * @type {number|undefined} + */ /** + * Line width used for calculation crisp line coordinates. Defaults to 1. + * @name Highcharts.AxisPlotLinePathOptionsObject#lineWidth + * @type {number|undefined} + */ /** + * If `false`, the function will return null when it falls outside the axis + * bounds. If `true`, the function will return a path aligned to the plot area + * sides if it falls outside. If `pass`, it will return a path outside. + * @name Highcharts.AxisPlotLinePathOptionsObject#force + * @type {string|boolean|undefined} + */ /** + * Used in Highstock. When `true`, plot paths (crosshair, plotLines, gridLines) + * will be rendered on all axes when defined on the first axis. + * @name Highcharts.AxisPlotLinePathOptionsObject#acrossPanes + * @type {boolean|undefined} + */ /** + * Use old coordinates (for resizing and rescaling). + * If not set, defaults to `false`. + * @name Highcharts.AxisPlotLinePathOptionsObject#old + * @type {boolean|undefined} + */ /** + * If given, return the plot line path of a pixel position on the axis. + * @name Highcharts.AxisPlotLinePathOptionsObject#translatedValue + * @type {number|undefined} + */ /** + * Used in Polar axes. Reverse the positions for concatenation of polygonal + * plot bands + * @name Highcharts.AxisPlotLinePathOptionsObject#reverse + * @type {boolean|undefined} + */ + /** + * Options for crosshairs on axes. + * + * @product highstock + * + * @typedef {Highcharts.XAxisCrosshairOptions|Highcharts.YAxisCrosshairOptions} Highcharts.AxisCrosshairOptions + */ + /** + * @typedef {"navigator"|"pan"|"rangeSelectorButton"|"rangeSelectorInput"|"scrollbar"|"traverseUpButton"|"zoom"} Highcharts.AxisExtremesTriggerValue + */ + /** + * @callback Highcharts.AxisEventCallbackFunction + * + * @param {Highcharts.Axis} this + */ + /** + * @callback Highcharts.AxisLabelsFormatterCallbackFunction + * + * @param {Highcharts.AxisLabelsFormatterContextObject<number>} this + * + * @param {Highcharts.AxisLabelsFormatterContextObject<string>} that + * + * @return {string} + */ + /** + * @interface Highcharts.AxisLabelsFormatterContextObject<T> + */ /** + * @name Highcharts.AxisLabelsFormatterContextObject<T>#axis + * @type {Highcharts.Axis} + */ /** + * @name Highcharts.AxisLabelsFormatterContextObject<T>#chart + * @type {Highcharts.Chart} + */ /** + * @name Highcharts.AxisLabelsFormatterContextObject<T>#isFirst + * @type {boolean} + */ /** + * @name Highcharts.AxisLabelsFormatterContextObject<T>#isLast + * @type {boolean} + */ /** + * @name Highcharts.AxisLabelsFormatterContextObject<T>#pos + * @type {number} + */ /** + * This can be either a numeric value or a category string. + * @name Highcharts.AxisLabelsFormatterContextObject<T>#value + * @type {T} + */ + /** + * Options for axes. + * + * @typedef {Highcharts.XAxisOptions|Highcharts.YAxisOptions|Highcharts.ZAxisOptions} Highcharts.AxisOptions + */ + /** + * @callback Highcharts.AxisPointBreakEventCallbackFunction + * + * @param {Highcharts.Axis} this + * + * @param {Highcharts.AxisPointBreakEventObject} evt + */ + /** + * @interface Highcharts.AxisPointBreakEventObject + */ /** + * @name Highcharts.AxisPointBreakEventObject#brk + * @type {Highcharts.Dictionary<number>} + */ /** + * @name Highcharts.AxisPointBreakEventObject#point + * @type {Highcharts.Point} + */ /** + * @name Highcharts.AxisPointBreakEventObject#preventDefault + * @type {Function} + */ /** + * @name Highcharts.AxisPointBreakEventObject#target + * @type {Highcharts.SVGElement} + */ /** + * @name Highcharts.AxisPointBreakEventObject#type + * @type {"pointBreak"|"pointInBreak"} + */ + /** + * @callback Highcharts.AxisSetExtremesEventCallbackFunction + * + * @param {Highcharts.Axis} this + * + * @param {Highcharts.AxisSetExtremesEventObject} evt + */ + /** + * @interface Highcharts.AxisSetExtremesEventObject + * @extends Highcharts.ExtremesObject + */ /** + * @name Highcharts.AxisSetExtremesEventObject#preventDefault + * @type {Function} + */ /** + * @name Highcharts.AxisSetExtremesEventObject#target + * @type {Highcharts.SVGElement} + */ /** + * @name Highcharts.AxisSetExtremesEventObject#trigger + * @type {Highcharts.AxisExtremesTriggerValue|string} + */ /** + * @name Highcharts.AxisSetExtremesEventObject#type + * @type {"setExtremes"} + */ + /** + * @callback Highcharts.AxisTickPositionerCallbackFunction + * + * @param {Highcharts.Axis} this + * + * @return {Highcharts.AxisTickPositionsArray} + */ + /** + * @interface Highcharts.AxisTickPositionsArray + * @augments Array<number> + */ + /** + * @typedef {"high"|"low"|"middle"} Highcharts.AxisTitleAlignValue + */ + /** + * @typedef {Highcharts.XAxisTitleOptions|Highcharts.YAxisTitleOptions|Highcharts.ZAxisTitleOptions} Highcharts.AxisTitleOptions + */ + /** + * @typedef {"linear"|"logarithmic"|"datetime"|"category"|"treegrid"} Highcharts.AxisTypeValue + */ + /** + * The returned object literal from the {@link Highcharts.Axis#getExtremes} + * function. + * + * @interface Highcharts.ExtremesObject + */ /** + * The maximum value of the axis' associated series. + * @name Highcharts.ExtremesObject#dataMax + * @type {number} + */ /** + * The minimum value of the axis' associated series. + * @name Highcharts.ExtremesObject#dataMin + * @type {number} + */ /** + * The maximum axis value, either automatic or set manually. If the `max` option + * is not set, `maxPadding` is 0 and `endOnTick` is false, this value will be + * the same as `dataMax`. + * @name Highcharts.ExtremesObject#max + * @type {number} + */ /** + * The minimum axis value, either automatic or set manually. If the `min` option + * is not set, `minPadding` is 0 and `startOnTick` is false, this value will be + * the same as `dataMin`. + * @name Highcharts.ExtremesObject#min + * @type {number} + */ /** + * The user defined maximum, either from the `max` option or from a zoom or + * `setExtremes` action. + * @name Highcharts.ExtremesObject#userMax + * @type {number} + */ /** + * The user defined minimum, either from the `min` option or from a zoom or + * `setExtremes` action. + * @name Highcharts.ExtremesObject#userMin + * @type {number} + */ + /** + * Formatter function for the text of a crosshair label. + * + * @callback Highcharts.XAxisCrosshairLabelFormatterCallbackFunction + * + * @param {Highcharts.Axis} this + * Axis context + * + * @param {number} value + * Y value of the data point + * + * @return {string} + */ + var defaultOptions = O.defaultOptions; + var deg2rad = H.deg2rad; + /** + * Create a new axis object. Called internally when instanciating a new chart or + * adding axes by {@link Highcharts.Chart#addAxis}. + * + * A chart can have from 0 axes (pie chart) to multiples. In a normal, single + * series cartesian chart, there is one X axis and one Y axis. + * + * The X axis or axes are referenced by {@link Highcharts.Chart.xAxis}, which is + * an array of Axis objects. If there is only one axis, it can be referenced + * through `chart.xAxis[0]`, and multiple axes have increasing indices. The same + * pattern goes for Y axes. + * + * If you need to get the axes from a series object, use the `series.xAxis` and + * `series.yAxis` properties. These are not arrays, as one series can only be + * associated to one X and one Y axis. + * + * A third way to reference the axis programmatically is by `id`. Add an `id` in + * the axis configuration options, and get the axis by + * {@link Highcharts.Chart#get}. + * + * Configuration options for the axes are given in options.xAxis and + * options.yAxis. + * + * @class + * @name Highcharts.Axis + * + * @param {Highcharts.Chart} chart + * The Chart instance to apply the axis on. + * + * @param {Highcharts.AxisOptions} userOptions + * Axis options. + */ + var Axis = /** @class */ (function () { /* * * - * (c) 2010-2020 Torstein Honsi + * Constructors * - * License: www.highcharts.com/license + * */ + function Axis(chart, userOptions) { + this.alternateBands = void 0; + this.bottom = void 0; + this.categories = void 0; + this.chart = void 0; + this.closestPointRange = void 0; + this.coll = void 0; + this.hasNames = void 0; + this.hasVisibleSeries = void 0; + this.height = void 0; + this.isLinked = void 0; + this.labelEdge = void 0; // @todo + this.labelFormatter = void 0; + this.left = void 0; + this.len = void 0; + this.max = void 0; + this.maxLabelLength = void 0; + this.min = void 0; + this.minorTickInterval = void 0; + this.minorTicks = void 0; + this.minPixelPadding = void 0; + this.names = void 0; + this.offset = void 0; + this.oldMax = void 0; + this.oldMin = void 0; + this.options = void 0; + this.overlap = void 0; + this.paddedTicks = void 0; + this.plotLinesAndBands = void 0; + this.plotLinesAndBandsGroups = void 0; + this.pointRange = void 0; + this.pointRangePadding = void 0; + this.pos = void 0; + this.positiveValuesOnly = void 0; + this.right = void 0; + this.series = void 0; + this.side = void 0; + this.tickAmount = void 0; + this.tickInterval = void 0; + this.tickmarkOffset = void 0; + this.tickPositions = void 0; + this.tickRotCorr = void 0; + this.ticks = void 0; + this.top = void 0; + this.transA = void 0; + this.transB = void 0; + this.translationSlope = void 0; + this.userOptions = void 0; + this.visible = void 0; + this.width = void 0; + this.zoomEnabled = void 0; + this.init(chart, userOptions); + } + /* * * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * Functions * * */ - var __extends = (this && this.__extends) || (function () { - var extendStatics = function (d, - b) { - extendStatics = Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && function (d, - b) { d.__proto__ = b; }) || - function (d, - b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; - return extendStatics(d, b); - }; - return function (d, b) { - extendStatics(d, b); - function __() { this.constructor = d; } - d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); - }; - })(); - var defined = U.defined, - extend = U.extend, - isNumber = U.isNumber, - merge = U.merge, - removeEvent = U.removeEvent; /** - * SVG label to render text. - * @private - * @class - * @name Highcharts.SVGLabel - * @augments Highcharts.SVGElement - */ - var SVGLabel = /** @class */ (function (_super) { - __extends(SVGLabel, _super); - /* * - * - * Constructors - * - * */ - function SVGLabel(renderer, str, x, y, shape, anchorX, anchorY, useHTML, baseline, className) { - var _this = _super.call(this) || this; - _this.init(renderer, 'g'); - _this.textStr = str; - _this.x = x; - _this.y = y; - _this.anchorX = anchorX; - _this.anchorY = anchorY; - _this.baseline = baseline; - _this.className = className; - if (className !== 'button') { - _this.addClass('highcharts-label'); - } - if (className) { - _this.addClass('highcharts-' + className); - } - _this.text = renderer.text('', 0, 0, useHTML) - .attr({ - zIndex: 1 - }); - // Validate the shape argument - var hasBGImage; - if (typeof shape === 'string') { - hasBGImage = /^url\((.*?)\)$/.test(shape); - if (_this.renderer.symbols[shape] || hasBGImage) { - _this.symbolKey = shape; - } - } - _this.bBox = SVGLabel.emptyBBox; - _this.padding = 3; - _this.paddingLeft = 0; - _this.baselineOffset = 0; - _this.needsBox = renderer.styledMode || hasBGImage; - _this.deferredAttr = {}; - _this.alignFactor = 0; - return _this; - } - /* * - * - * Functions - * - * */ - SVGLabel.prototype.alignSetter = function (value) { - var alignFactor = { - left: 0, - center: 0.5, - right: 1 - }[value]; - if (alignFactor !== this.alignFactor) { - this.alignFactor = alignFactor; - // Bounding box exists, means we're dynamically changing - if (this.bBox && isNumber(this.xSetting)) { - this.attr({ x: this.xSetting }); // #5134 - } - } - }; - SVGLabel.prototype.anchorXSetter = function (value, key) { - this.anchorX = value; - this.boxAttr(key, Math.round(value) - this.getCrispAdjust() - this.xSetting); - }; - SVGLabel.prototype.anchorYSetter = function (value, key) { - this.anchorY = value; - this.boxAttr(key, value - this.ySetting); - }; - /* - * Set a box attribute, or defer it if the box is not yet created - */ - SVGLabel.prototype.boxAttr = function (key, value) { - if (this.box) { - this.box.attr(key, value); - } - else { - this.deferredAttr[key] = value; - } - }; - /* - * Pick up some properties and apply them to the text instead of the - * wrapper. - */ - SVGLabel.prototype.css = function (styles) { - if (styles) { - var textStyles = {}, - isWidth, - isFontStyle; - // Create a copy to avoid altering the original object - // (#537) - styles = merge(styles); - SVGLabel.textProps.forEach(function (prop) { - if (typeof styles[prop] !== 'undefined') { - textStyles[prop] = styles[prop]; - delete styles[prop]; - } - }); - this.text.css(textStyles); - isWidth = 'width' in textStyles; - isFontStyle = 'fontSize' in textStyles || - 'fontWeight' in textStyles; - // Update existing text, box (#9400, #12163) - if (isWidth || isFontStyle) { - this.updateBoxSize(); - // Keep updated (#9400, #12163) - if (isFontStyle) { - this.updateTextPadding(); - } - } - } - return SVGElement.prototype.css.call(this, styles); - }; - /* - * Destroy and release memory. - */ - SVGLabel.prototype.destroy = function () { - // Added by button implementation - removeEvent(this.element, 'mouseenter'); - removeEvent(this.element, 'mouseleave'); - if (this.text) { - this.text.destroy(); - } - if (this.box) { - this.box = this.box.destroy(); - } - // Call base implementation to destroy the rest - SVGElement.prototype.destroy.call(this); - return void 0; - }; - SVGLabel.prototype.fillSetter = function (value, key) { - if (value) { - this.needsBox = true; - } - // for animation getter (#6776) - this.fill = value; - this.boxAttr(key, value); - }; - /* - * Return the bounding box of the box, not the group. - */ - SVGLabel.prototype.getBBox = function () { - var bBox = this.bBox; - var padding = this.padding; - return { - width: bBox.width + 2 * padding, - height: bBox.height + 2 * padding, - x: bBox.x - padding, - y: bBox.y - padding - }; - }; - SVGLabel.prototype.getCrispAdjust = function () { - return this.renderer.styledMode && this.box ? - this.box.strokeWidth() % 2 / 2 : - (this['stroke-width'] ? parseInt(this['stroke-width'], 10) : 0) % 2 / 2; - }; - SVGLabel.prototype.heightSetter = function (value) { - this.heightSetting = value; - }; - // Event handling. In case of useHTML, we need to make sure that events - // are captured on the span as well, and that mouseenter/mouseleave - // between the SVG group and the HTML span are not treated as real - // enter/leave events. #13310. - SVGLabel.prototype.on = function (eventType, handler) { - var label = this; - var text = label.text; - var span = text && text.element.tagName === 'SPAN' ? text : void 0; - var selectiveHandler; - if (span) { - selectiveHandler = function (e) { - if ((eventType === 'mouseenter' || - eventType === 'mouseleave') && - e.relatedTarget instanceof Element && - (label.element.contains(e.relatedTarget) || - span.element.contains(e.relatedTarget))) { - return; - } - handler.call(label.element, e); - }; - span.on(eventType, selectiveHandler); - } - SVGElement.prototype.on.call(label, eventType, selectiveHandler || handler); - return label; - }; - /* - * After the text element is added, get the desired size of the border - * box and add it before the text in the DOM. - */ - SVGLabel.prototype.onAdd = function () { - var str = this.textStr; - this.text.add(this); - this.attr({ - // Alignment is available now (#3295, 0 not rendered if given - // as a value) - text: (defined(str) ? str : ''), - x: this.x, - y: this.y - }); - if (this.box && defined(this.anchorX)) { - this.attr({ - anchorX: this.anchorX, - anchorY: this.anchorY - }); - } - }; - SVGLabel.prototype.paddingSetter = function (value) { - if (defined(value) && value !== this.padding) { - this.padding = value; - this.updateTextPadding(); - } - }; - SVGLabel.prototype.paddingLeftSetter = function (value) { - if (defined(value) && value !== this.paddingLeft) { - this.paddingLeft = value; - this.updateTextPadding(); - } - }; - SVGLabel.prototype.rSetter = function (value, key) { - this.boxAttr(key, value); - }; - SVGLabel.prototype.shadow = function (b) { - if (b && !this.renderer.styledMode) { - this.updateBoxSize(); - if (this.box) { - this.box.shadow(b); - } - } - return this; - }; - SVGLabel.prototype.strokeSetter = function (value, key) { - // for animation getter (#6776) - this.stroke = value; - this.boxAttr(key, value); - }; - SVGLabel.prototype['stroke-widthSetter'] = function (value, key) { - if (value) { - this.needsBox = true; - } - this['stroke-width'] = value; - this.boxAttr(key, value); - }; - SVGLabel.prototype['text-alignSetter'] = function (value) { - this.textAlign = value; - }; - SVGLabel.prototype.textSetter = function (text) { - if (typeof text !== 'undefined') { - // Must use .attr to ensure transforms are done (#10009) - this.text.attr({ text: text }); - } - this.updateBoxSize(); - this.updateTextPadding(); - }; - /* - * This function runs after the label is added to the DOM (when the bounding - * box is available), and after the text of the label is updated to detect - * the new bounding box and reflect it in the border box. - */ - SVGLabel.prototype.updateBoxSize = function () { - var style = this.text.element.style, - crispAdjust, - attribs = {}; - var padding = this.padding; - var paddingLeft = this.paddingLeft; - // #12165 error when width is null (auto) - // #12163 when fontweight: bold, recalculate bBox withot cache - // #3295 && 3514 box failure when string equals 0 - var bBox = ((!isNumber(this.widthSetting) || !isNumber(this.heightSetting) || this.textAlign) && - defined(this.text.textStr)) ? - this.text.getBBox() : SVGLabel.emptyBBox; - this.width = ((this.widthSetting || bBox.width || 0) + - 2 * padding + - paddingLeft); - this.height = (this.heightSetting || bBox.height || 0) + 2 * padding; - // Update the label-scoped y offset. Math.min because of inline - // style (#9400) - this.baselineOffset = padding + Math.min(this.renderer.fontMetrics(style && style.fontSize, this.text).b, - // When the height is 0, there is no bBox, so go with the font - // metrics. Highmaps CSS demos. - bBox.height || Infinity); - if (this.needsBox) { - // Create the border box if it is not already present - if (!this.box) { - // Symbol definition exists (#5324) - var box = this.box = this.symbolKey ? - this.renderer.symbol(this.symbolKey) : - this.renderer.rect(); - box.addClass(// Don't use label className for buttons - (this.className === 'button' ? '' : 'highcharts-label-box') + - (this.className ? ' highcharts-' + this.className + '-box' : '')); - box.add(this); - crispAdjust = this.getCrispAdjust(); - attribs.x = crispAdjust; - attribs.y = (this.baseline ? -this.baselineOffset : 0) + crispAdjust; - } - // Apply the box attributes - attribs.width = Math.round(this.width); - attribs.height = Math.round(this.height); - this.box.attr(extend(attribs, this.deferredAttr)); - this.deferredAttr = {}; - } - this.bBox = bBox; - }; - /* - * This function runs after setting text or padding, but only if padding - * is changed. - */ - SVGLabel.prototype.updateTextPadding = function () { - var text = this.text; - // Determine y based on the baseline - var textY = this.baseline ? 0 : this.baselineOffset; - var textX = this.paddingLeft + this.padding; - // compensate for alignment - if (defined(this.widthSetting) && - this.bBox && - (this.textAlign === 'center' || this.textAlign === 'right')) { - textX += { center: 0.5, right: 1 }[this.textAlign] * - (this.widthSetting - this.bBox.width); - } - // update if anything changed - if (textX !== text.x || textY !== text.y) { - text.attr('x', textX); - // #8159 - prevent misplaced data labels in treemap - // (useHTML: true) - if (text.hasBoxWidthChanged) { - this.bBox = text.getBBox(true); - this.updateBoxSize(); - } - if (typeof textY !== 'undefined') { - text.attr('y', textY); - } - } - // record current values - text.x = textX; - text.y = textY; - }; - SVGLabel.prototype.widthSetter = function (value) { - // width:auto => null - this.widthSetting = isNumber(value) ? value : void 0; - }; - SVGLabel.prototype.xSetter = function (value) { - this.x = value; // for animation getter - if (this.alignFactor) { - value -= this.alignFactor * ((this.widthSetting || this.bBox.width) + - 2 * this.padding); - // Force animation even when setting to the same value (#7898) - this['forceAnimate:x'] = true; - } - this.xSetting = Math.round(value); - this.attr('translateX', this.xSetting); - }; - SVGLabel.prototype.ySetter = function (value) { - this.ySetting = this.y = Math.round(value); - this.attr('translateY', this.ySetting); - }; - /* * - * - * Static Properties - * - * */ - SVGLabel.emptyBBox = { width: 0, height: 0, x: 0, y: 0 }; - /* * - * - * Properties - * - * */ - /** - * For labels, these CSS properties are applied to the `text` node directly. - * - * @private - * @name Highcharts.SVGLabel#textProps - * @type {Array<string>} - */ - SVGLabel.textProps = [ - 'color', 'cursor', 'direction', 'fontFamily', 'fontSize', 'fontStyle', - 'fontWeight', 'lineHeight', 'textAlign', 'textDecoration', - 'textOutline', 'textOverflow', 'width' - ]; - return SVGLabel; - }(SVGElement)); - - return SVGLabel; - }); - _registerModule(_modules, 'Core/Renderer/SVG/SVGRenderer.js', [_modules['Core/Color/Color.js'], _modules['Core/Globals.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Renderer/SVG/SVGLabel.js'], _modules['Core/Utilities.js']], function (Color, H, SVGElement, SVGLabel, U) { - /* * + * Overrideable function to initialize the axis. * - * (c) 2010-2020 Torstein Honsi + * @see {@link Axis} * - * License: www.highcharts.com/license + * @function Highcharts.Axis#init * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @param {Highcharts.Chart} chart + * The Chart instance to apply the axis on. * - * */ - var addEvent = U.addEvent, - attr = U.attr, - createElement = U.createElement, - css = U.css, - defined = U.defined, - destroyObjectProperties = U.destroyObjectProperties, - extend = U.extend, - isArray = U.isArray, - isNumber = U.isNumber, - isObject = U.isObject, - isString = U.isString, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick, - pInt = U.pInt, - splat = U.splat, - uniqueKey = U.uniqueKey; - /** - * A clipping rectangle that can be applied to one or more {@link SVGElement} - * instances. It is instanciated with the {@link SVGRenderer#clipRect} function - * and applied with the {@link SVGElement#clip} function. + * @param {Highcharts.AxisOptions} userOptions + * Axis options. * - * @example - * var circle = renderer.circle(100, 100, 100) - * .attr({ fill: 'red' }) - * .add(); - * var clipRect = renderer.clipRect(100, 100, 100, 100); + * @fires Highcharts.Axis#event:afterInit + * @fires Highcharts.Axis#event:init + */ + Axis.prototype.init = function (chart, userOptions) { + var isXAxis = userOptions.isX, + axis = this; + /** + * The Chart that the axis belongs to. + * + * @name Highcharts.Axis#chart + * @type {Highcharts.Chart} + */ + axis.chart = chart; + /** + * Whether the axis is horizontal. + * + * @name Highcharts.Axis#horiz + * @type {boolean|undefined} + */ + axis.horiz = chart.inverted && !axis.isZAxis ? !isXAxis : isXAxis; + /** + * Whether the axis is the x-axis. + * + * @name Highcharts.Axis#isXAxis + * @type {boolean|undefined} + */ + axis.isXAxis = isXAxis; + /** + * The collection where the axis belongs, for example `xAxis`, `yAxis` + * or `colorAxis`. Corresponds to properties on Chart, for example + * {@link Chart.xAxis}. + * + * @name Highcharts.Axis#coll + * @type {string} + */ + axis.coll = axis.coll || (isXAxis ? "xAxis" : "yAxis"); + fireEvent(this, "init", { userOptions: userOptions }); + axis.opposite = userOptions.opposite; // needed in setOptions + /** + * The side on which the axis is rendered. 0 is top, 1 is right, 2 + * is bottom and 3 is left. + * + * @name Highcharts.Axis#side + * @type {number} + */ + axis.side = + userOptions.side || + (axis.horiz + ? axis.opposite + ? 0 + : 2 // top : bottom + : axis.opposite + ? 1 + : 3); // right : left + /** + * Current options for the axis after merge of defaults and user's + * options. + * + * @name Highcharts.Axis#options + * @type {Highcharts.AxisOptions} + */ + axis.setOptions(userOptions); + var options = this.options, + type = options.type; + axis.labelFormatter = + options.labels.formatter || + // can be overwritten by dynamic format + axis.defaultLabelFormatter; + /** + * User's options for this axis without defaults. + * + * @name Highcharts.Axis#userOptions + * @type {Highcharts.AxisOptions} + */ + axis.userOptions = userOptions; + axis.minPixelPadding = 0; + /** + * Whether the axis is reversed. Based on the `axis.reversed`, + * option, but inverted charts have reversed xAxis by default. + * + * @name Highcharts.Axis#reversed + * @type {boolean} + */ + axis.reversed = options.reversed; + axis.visible = options.visible !== false; + axis.zoomEnabled = options.zoomEnabled !== false; + // Initial categories + axis.hasNames = type === "category" || options.categories === true; + /** + * If categories are present for the axis, names are used instead of + * numbers for that axis. + * + * Since Highcharts 3.0, categories can also be extracted by giving each + * point a name and setting axis type to `category`. However, if you + * have multiple series, best practice remains defining the `categories` + * array. + * + * @see [xAxis.categories](/highcharts/xAxis.categories) + * + * @name Highcharts.Axis#categories + * @type {Array<string>} + * @readonly + */ + axis.categories = options.categories || axis.hasNames; + if (!axis.names) { + // Preserve on update (#3830) + axis.names = []; + axis.names.keys = {}; + } + // Placeholder for plotlines and plotbands groups + axis.plotLinesAndBandsGroups = {}; + // Shorthand types + axis.positiveValuesOnly = !!axis.logarithmic; + // Flag, if axis is linked to another axis + axis.isLinked = defined(options.linkedTo); + /** + * List of major ticks mapped by postition on axis. + * + * @see {@link Highcharts.Tick} + * + * @name Highcharts.Axis#ticks + * @type {Highcharts.Dictionary<Highcharts.Tick>} + */ + axis.ticks = {}; + axis.labelEdge = []; + /** + * List of minor ticks mapped by position on the axis. + * + * @see {@link Highcharts.Tick} + * + * @name Highcharts.Axis#minorTicks + * @type {Highcharts.Dictionary<Highcharts.Tick>} + */ + axis.minorTicks = {}; + // List of plotLines/Bands + axis.plotLinesAndBands = []; + // Alternate bands + axis.alternateBands = {}; + // Axis metrics + axis.len = 0; + axis.minRange = axis.userMinRange = + options.minRange || options.maxZoom; + axis.range = options.range; + axis.offset = options.offset || 0; + /** + * The maximum value of the axis. In a logarithmic axis, this is the + * logarithm of the real value, and the real value can be obtained from + * {@link Axis#getExtremes}. + * + * @name Highcharts.Axis#max + * @type {number|null} + */ + axis.max = null; + /** + * The minimum value of the axis. In a logarithmic axis, this is the + * logarithm of the real value, and the real value can be obtained from + * {@link Axis#getExtremes}. + * + * @name Highcharts.Axis#min + * @type {number|null} + */ + axis.min = null; + /** + * The processed crosshair options. + * + * @name Highcharts.Axis#crosshair + * @type {boolean|Highcharts.AxisCrosshairOptions} + */ + axis.crosshair = pick( + options.crosshair, + splat(chart.options.tooltip.crosshairs)[isXAxis ? 0 : 1], + false + ); + var events = axis.options.events; + // Register. Don't add it again on Axis.update(). + if (chart.axes.indexOf(axis) === -1) { + // + if (isXAxis) { + // #2713 + chart.axes.splice(chart.xAxis.length, 0, axis); + } else { + chart.axes.push(axis); + } + chart[axis.coll].push(axis); + } + /** + * All series associated to the axis. + * + * @name Highcharts.Axis#series + * @type {Array<Highcharts.Series>} + */ + axis.series = axis.series || []; // populated by Series + // Reversed axis + if ( + chart.inverted && + !axis.isZAxis && + isXAxis && + typeof axis.reversed === "undefined" + ) { + axis.reversed = true; + } + axis.labelRotation = axis.options.labels.rotation; + // register event listeners + objectEach(events, function (event, eventType) { + if (isFunction(event)) { + addEvent(axis, eventType, event); + } + }); + fireEvent(this, "afterInit"); + }; + /** + * Merge and set options. * - * // Leave only the lower right quarter visible - * circle.clip(clipRect); + * @private + * @function Highcharts.Axis#setOptions * - * @typedef {Highcharts.SVGElement} Highcharts.ClipRectElement - */ - /** - * The font metrics. + * @param {Highcharts.AxisOptions} userOptions + * Axis options. * - * @interface Highcharts.FontMetricsObject - */ /** - * The baseline relative to the top of the box. - * - * @name Highcharts.FontMetricsObject#b - * @type {number} - */ /** - * The font size. - * - * @name Highcharts.FontMetricsObject#f - * @type {number} - */ /** - * The line height. - * - * @name Highcharts.FontMetricsObject#h - * @type {number} - */ - /** - * An object containing `x` and `y` properties for the position of an element. - * - * @interface Highcharts.PositionObject - */ /** - * X position of the element. - * @name Highcharts.PositionObject#x - * @type {number} - */ /** - * Y position of the element. - * @name Highcharts.PositionObject#y - * @type {number} - */ - /** - * A rectangle. - * - * @interface Highcharts.RectangleObject - */ /** - * Height of the rectangle. - * @name Highcharts.RectangleObject#height - * @type {number} - */ /** - * Width of the rectangle. - * @name Highcharts.RectangleObject#width - * @type {number} - */ /** - * Horizontal position of the rectangle. - * @name Highcharts.RectangleObject#x - * @type {number} - */ /** - * Vertical position of the rectangle. - * @name Highcharts.RectangleObject#y - * @type {number} - */ - /** - * The shadow options. - * - * @interface Highcharts.ShadowOptionsObject - */ /** - * The shadow color. - * @name Highcharts.ShadowOptionsObject#color - * @type {Highcharts.ColorString|undefined} - * @default #000000 - */ /** - * The horizontal offset from the element. - * - * @name Highcharts.ShadowOptionsObject#offsetX - * @type {number|undefined} - * @default 1 - */ /** - * The vertical offset from the element. - * @name Highcharts.ShadowOptionsObject#offsetY - * @type {number|undefined} - * @default 1 - */ /** - * The shadow opacity. - * - * @name Highcharts.ShadowOptionsObject#opacity - * @type {number|undefined} - * @default 0.15 - */ /** - * The shadow width or distance from the element. - * @name Highcharts.ShadowOptionsObject#width - * @type {number|undefined} - * @default 3 - */ - /** - * @interface Highcharts.SizeObject - */ /** - * @name Highcharts.SizeObject#height - * @type {number} - */ /** - * @name Highcharts.SizeObject#width - * @type {number} - */ + * @fires Highcharts.Axis#event:afterSetOptions + */ + Axis.prototype.setOptions = function (userOptions) { + this.options = merge( + Axis.defaultOptions, + this.coll === "yAxis" && Axis.defaultYAxisOptions, + [ + Axis.defaultTopAxisOptions, + Axis.defaultRightAxisOptions, + Axis.defaultBottomAxisOptions, + Axis.defaultLeftAxisOptions, + ][this.side], + merge( + // if set in setOptions (#1053): + defaultOptions[this.coll], + userOptions + ) + ); + fireEvent(this, "afterSetOptions", { userOptions: userOptions }); + }; /** - * Serialized form of an SVG definition, including children. Some key - * property names are reserved: tagName, textContent, and children. + * The default label formatter. The context is a special config object for + * the label. In apps, use the + * [labels.formatter](https://api.highcharts.com/highcharts/xAxis.labels.formatter) + * instead, except when a modification is needed. * - * @interface Highcharts.SVGDefinitionObject - */ /** - * @name Highcharts.SVGDefinitionObject#[key:string] - * @type {boolean|number|string|Array<Highcharts.SVGDefinitionObject>|undefined} - */ /** - * @name Highcharts.SVGDefinitionObject#children - * @type {Array<Highcharts.SVGDefinitionObject>|undefined} - */ /** - * @name Highcharts.SVGDefinitionObject#tagName - * @type {string|undefined} - */ /** - * @name Highcharts.SVGDefinitionObject#textContent - * @type {string|undefined} - */ - /** - * Array of path commands, that will go into the `d` attribute of an SVG - * element. + * @function Highcharts.Axis#defaultLabelFormatter * - * @typedef {Array<(Array<Highcharts.SVGPathCommand>|Array<Highcharts.SVGPathCommand,number>|Array<Highcharts.SVGPathCommand,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number,number,number,number>)>} Highcharts.SVGPathArray - */ + * @param {Highcharts.AxisLabelsFormatterContextObject<number>|Highcharts.AxisLabelsFormatterContextObject<string>} this + * Formatter context of axis label. + * + * @return {string} + * The formatted label content. + */ + Axis.prototype.defaultLabelFormatter = function () { + var axis = this.axis, + value = isNumber(this.value) ? this.value : NaN, + time = axis.chart.time, + categories = axis.categories, + dateTimeLabelFormat = this.dateTimeLabelFormat, + lang = defaultOptions.lang, + numericSymbols = lang.numericSymbols, + numSymMagnitude = lang.numericSymbolMagnitude || 1000, + i = numericSymbols && numericSymbols.length, + multi, + ret, + formatOption = axis.options.labels.format, + // make sure the same symbol is added for all labels on a linear + // axis + numericSymbolDetector = axis.logarithmic + ? Math.abs(value) + : axis.tickInterval; + var chart = this.chart; + var numberFormatter = chart.numberFormatter; + if (formatOption) { + ret = format(formatOption, this, chart); + } else if (categories) { + ret = "" + this.value; + } else if (dateTimeLabelFormat) { + // datetime axis + ret = time.dateFormat(dateTimeLabelFormat, value); + } else if (i && numericSymbolDetector >= 1000) { + // Decide whether we should add a numeric symbol like k (thousands) + // or M (millions). If we are to enable this in tooltip or other + // places as well, we can move this logic to the numberFormatter and + // enable it by a parameter. + while (i-- && typeof ret === "undefined") { + multi = Math.pow(numSymMagnitude, i + 1); + if ( + // Only accept a numeric symbol when the distance is more + // than a full unit. So for example if the symbol is k, we + // don't accept numbers like 0.5k. + numericSymbolDetector >= multi && + // Accept one decimal before the symbol. Accepts 0.5k but + // not 0.25k. How does this work with the previous? + (value * 10) % multi === 0 && + numericSymbols[i] !== null && + value !== 0 + ) { + // #5480 + ret = numberFormatter(value / multi, -1) + numericSymbols[i]; + } + } + } + if (typeof ret === "undefined") { + if (Math.abs(value) >= 10000) { + // add thousands separators + ret = numberFormatter(value, -1); + } else { + // small numbers + ret = numberFormatter(value, -1, void 0, ""); // #2466 + } + } + return ret; + }; /** - * Possible path commands in an SVG path array. Valid values are `A`, `C`, `H`, - * `L`, `M`, `Q`, `S`, `T`, `V`, `Z`. + * Get the minimum and maximum for the series of each axis. The function + * analyzes the axis series and updates `this.dataMin` and `this.dataMax`. * - * @typedef {string} Highcharts.SVGPathCommand - * @validvalue ["a","c","h","l","m","q","s","t","v","z","A","C","H","L","M","Q","S","T","V","Z"] - */ + * @private + * @function Highcharts.Axis#getSeriesExtremes + * + * @fires Highcharts.Axis#event:afterGetSeriesExtremes + * @fires Highcharts.Axis#event:getSeriesExtremes + */ + Axis.prototype.getSeriesExtremes = function () { + var axis = this, + chart = axis.chart, + xExtremes; + fireEvent(this, "getSeriesExtremes", null, function () { + axis.hasVisibleSeries = false; + // Reset properties in case we're redrawing (#3353) + axis.dataMin = axis.dataMax = axis.threshold = null; + axis.softThreshold = !axis.isXAxis; + if (axis.stacking) { + axis.stacking.buildStacks(); + } + // loop through this axis' series + axis.series.forEach(function (series) { + if (series.visible || !chart.options.chart.ignoreHiddenSeries) { + var seriesOptions = series.options, + xData, + threshold = seriesOptions.threshold, + seriesDataMin, + seriesDataMax; + axis.hasVisibleSeries = true; + // Validate threshold in logarithmic axes + if (axis.positiveValuesOnly && threshold <= 0) { + threshold = null; + } + // Get dataMin and dataMax for X axes + if (axis.isXAxis) { + xData = series.xData; + if (xData.length) { + var isPositive = function (number) { + return number > 0; + }; + xData = axis.logarithmic + ? xData.filter(axis.validatePositiveValue) + : xData; + xExtremes = series.getXExtremes(xData); + // If xData contains values which is not numbers, + // then filter them out. To prevent performance hit, + // we only do this after we have already found + // seriesDataMin because in most cases all data is + // valid. #5234. + seriesDataMin = xExtremes.min; + seriesDataMax = xExtremes.max; + if ( + !isNumber(seriesDataMin) && + // #5010: + !(seriesDataMin instanceof Date) + ) { + xData = xData.filter(isNumber); + xExtremes = series.getXExtremes(xData); + // Do it again with valid data + seriesDataMin = xExtremes.min; + seriesDataMax = xExtremes.max; + } + if (xData.length) { + axis.dataMin = Math.min( + pick(axis.dataMin, seriesDataMin), + seriesDataMin + ); + axis.dataMax = Math.max( + pick(axis.dataMax, seriesDataMax), + seriesDataMax + ); + } + } + // Get dataMin and dataMax for Y axes, as well as handle + // stacking and processed data + } else { + // Get this particular series extremes + var dataExtremes = series.applyExtremes(); + // Get the dataMin and dataMax so far. If percentage is + // used, the min and max are always 0 and 100. If + // seriesDataMin and seriesDataMax is null, then series + // doesn't have active y data, we continue with nulls + if (isNumber(dataExtremes.dataMin)) { + seriesDataMin = dataExtremes.dataMin; + axis.dataMin = Math.min( + pick(axis.dataMin, seriesDataMin), + seriesDataMin + ); + } + if (isNumber(dataExtremes.dataMax)) { + seriesDataMax = dataExtremes.dataMax; + axis.dataMax = Math.max( + pick(axis.dataMax, seriesDataMax), + seriesDataMax + ); + } + // Adjust to threshold + if (defined(threshold)) { + axis.threshold = threshold; + } + // If any series has a hard threshold, it takes + // precedence + if (!seriesOptions.softThreshold || axis.positiveValuesOnly) { + axis.softThreshold = false; + } + } + } + }); + }); + fireEvent(this, "afterGetSeriesExtremes"); + }; /** - * An extendable collection of functions for defining symbol paths. Symbols are - * used internally for point markers, button and label borders and backgrounds, - * or custom shapes. Extendable by adding to {@link SVGRenderer#symbols}. + * Translate from axis value to pixel position on the chart, or back. Use + * the `toPixels` and `toValue` functions in applications. * - * @interface Highcharts.SymbolDictionary - */ /** - * @name Highcharts.SymbolDictionary#[key:string] - * @type {Function|undefined} - */ /** - * @name Highcharts.SymbolDictionary#arc - * @type {Function|undefined} - */ /** - * @name Highcharts.SymbolDictionary#callout - * @type {Function|undefined} - */ /** - * @name Highcharts.SymbolDictionary#circle - * @type {Function|undefined} - */ /** - * @name Highcharts.SymbolDictionary#diamond - * @type {Function|undefined} - */ /** - * @name Highcharts.SymbolDictionary#square - * @type {Function|undefined} - */ /** - * @name Highcharts.SymbolDictionary#triangle - * @type {Function|undefined} - */ - /** - * Can be one of `arc`, `callout`, `circle`, `diamond`, `square`, `triangle`, - * and `triangle-down`. Symbols are used internally for point markers, button - * and label borders and backgrounds, or custom shapes. Extendable by adding to - * {@link SVGRenderer#symbols}. + * @private + * @function Highcharts.Axis#translate + * + * @param {number} val + * TO-DO: parameter description + * + * @param {boolean|null} [backwards] + * TO-DO: parameter description + * + * @param {boolean|null} [cvsCoord] + * TO-DO: parameter description + * + * @param {boolean|null} [old] + * TO-DO: parameter description + * + * @param {boolean} [handleLog] + * TO-DO: parameter description + * + * @param {number} [pointPlacement] + * TO-DO: parameter description + * + * @return {number|undefined} + */ + Axis.prototype.translate = function ( + val, + backwards, + cvsCoord, + old, + handleLog, + pointPlacement + ) { + var axis = this.linkedParent || this, // #1417 + sign = 1, + cvsOffset = 0, + localA = old ? axis.oldTransA : axis.transA, + localMin = old ? axis.oldMin : axis.min, + returnValue = 0, + minPixelPadding = axis.minPixelPadding, + doPostTranslate = + (axis.isOrdinal || + (axis.brokenAxis && axis.brokenAxis.hasBreaks) || + (axis.logarithmic && handleLog)) && + axis.lin2val; + if (!localA) { + localA = axis.transA; + } + // In vertical axes, the canvas coordinates start from 0 at the top like + // in SVG. + if (cvsCoord) { + sign *= -1; // canvas coordinates inverts the value + cvsOffset = axis.len; + } + // Handle reversed axis + if (axis.reversed) { + sign *= -1; + cvsOffset -= sign * (axis.sector || axis.len); + } + // From pixels to value + if (backwards) { + // reverse translation + val = val * sign + cvsOffset; + val -= minPixelPadding; + // from chart pixel to value: + returnValue = val / localA + localMin; + if (doPostTranslate) { + // log and ordinal axes + returnValue = axis.lin2val(returnValue); + } + // From value to pixels + } else { + if (doPostTranslate) { + // log and ordinal axes + val = axis.val2lin(val); + } + returnValue = isNumber(localMin) + ? sign * (val - localMin) * localA + + cvsOffset + + sign * minPixelPadding + + (isNumber(pointPlacement) ? localA * pointPlacement : 0) + : void 0; + } + return returnValue; + }; + /** + * Translate a value in terms of axis units into pixels within the chart. + * + * @function Highcharts.Axis#toPixels * - * @typedef {"arc"|"callout"|"circle"|"diamond"|"square"|"triangle"|"triangle-down"} Highcharts.SymbolKeyValue + * @param {number} value + * A value in terms of axis units. + * + * @param {boolean} paneCoordinates + * Whether to return the pixel coordinate relative to the chart or just the + * axis/pane itself. + * + * @return {number} + * Pixel position of the value on the chart or axis. */ + Axis.prototype.toPixels = function (value, paneCoordinates) { + return ( + this.translate(value, false, !this.horiz, null, true) + + (paneCoordinates ? 0 : this.pos) + ); + }; /** - * Additional options, depending on the actual symbol drawn. + * Translate a pixel position along the axis to a value in terms of axis + * units. * - * @interface Highcharts.SymbolOptionsObject - */ /** - * The anchor X position for the `callout` symbol. This is where the chevron - * points to. - * - * @name Highcharts.SymbolOptionsObject#anchorX - * @type {number|undefined} - */ /** - * The anchor Y position for the `callout` symbol. This is where the chevron - * points to. - * - * @name Highcharts.SymbolOptionsObject#anchorY - * @type {number|undefined} - */ /** - * The end angle of an `arc` symbol. - * - * @name Highcharts.SymbolOptionsObject#end - * @type {number|undefined} - */ /** - * Whether to draw `arc` symbol open or closed. - * - * @name Highcharts.SymbolOptionsObject#open - * @type {boolean|undefined} - */ /** - * The radius of an `arc` symbol, or the border radius for the `callout` symbol. - * - * @name Highcharts.SymbolOptionsObject#r - * @type {number|undefined} - */ /** - * The start angle of an `arc` symbol. - * - * @name Highcharts.SymbolOptionsObject#start - * @type {number|undefined} - */ - /* eslint-disable no-invalid-this, valid-jsdoc */ - var charts = H.charts, - deg2rad = H.deg2rad, - doc = H.doc, - isFirefox = H.isFirefox, - isMS = H.isMS, - isWebKit = H.isWebKit, - noop = H.noop, - svg = H.svg, - SVG_NS = H.SVG_NS, - symbolSizes = H.symbolSizes, - win = H.win; - /** - * Allows direct access to the Highcharts rendering layer in order to draw - * primitive shapes like circles, rectangles, paths or text directly on a chart, - * or independent from any chart. The SVGRenderer represents a wrapper object - * for SVG in modern browsers. Through the VMLRenderer, part of the `oldie.js` - * module, it also brings vector graphics to IE <= 8. - * - * An existing chart's renderer can be accessed through {@link Chart.renderer}. - * The renderer can also be used completely decoupled from a chart. - * - * @sample highcharts/members/renderer-on-chart - * Annotating a chart programmatically. - * @sample highcharts/members/renderer-basic - * Independent SVG drawing. + * @function Highcharts.Axis#toValue * - * @example - * // Use directly without a chart object. - * var renderer = new Highcharts.Renderer(parentNode, 600, 400); + * @param {number} pixel + * The pixel value coordinate. * - * @class - * @name Highcharts.SVGRenderer + * @param {boolean} [paneCoordinates=false] + * Whether the input pixel is relative to the chart or just the axis/pane + * itself. * - * @param {Highcharts.HTMLDOMElement} container - * Where to put the SVG in the web page. + * @return {number} + * The axis value. + */ + Axis.prototype.toValue = function (pixel, paneCoordinates) { + return this.translate( + pixel - (paneCoordinates ? 0 : this.pos), + true, + !this.horiz, + null, + true + ); + }; + /** + * Create the path for a plot line that goes from the given value on + * this axis, across the plot to the opposite side. Also used internally for + * grid lines and crosshairs. + * + * @function Highcharts.Axis#getPlotLinePath + * + * @param {Highcharts.AxisPlotLinePathOptionsObject} options + * Options for the path. + * + * @return {Highcharts.SVGPathArray|null} + * The SVG path definition for the plot line. + */ + Axis.prototype.getPlotLinePath = function (options) { + var axis = this, + chart = axis.chart, + axisLeft = axis.left, + axisTop = axis.top, + old = options.old, + value = options.value, + translatedValue = options.translatedValue, + lineWidth = options.lineWidth, + force = options.force, + x1, + y1, + x2, + y2, + cHeight = (old && chart.oldChartHeight) || chart.chartHeight, + cWidth = (old && chart.oldChartWidth) || chart.chartWidth, + skip, + transB = axis.transB, + evt; + // eslint-disable-next-line valid-jsdoc + /** + * Check if x is between a and b. If not, either move to a/b + * or skip, depending on the force parameter. + * @private + */ + function between(x, a, b) { + if ((force !== "pass" && x < a) || x > b) { + if (force) { + x = clamp(x, a, b); + } else { + skip = true; + } + } + return x; + } + evt = { + value: value, + lineWidth: lineWidth, + old: old, + force: force, + acrossPanes: options.acrossPanes, + translatedValue: translatedValue, + }; + fireEvent(this, "getPlotLinePath", evt, function (e) { + translatedValue = pick( + translatedValue, + axis.translate(value, null, null, old) + ); + // Keep the translated value within sane bounds, and avoid Infinity + // to fail the isNumber test (#7709). + translatedValue = clamp(translatedValue, -1e5, 1e5); + x1 = x2 = Math.round(translatedValue + transB); + y1 = y2 = Math.round(cHeight - translatedValue - transB); + if (!isNumber(translatedValue)) { + // no min or max + skip = true; + force = false; // #7175, don't force it when path is invalid + } else if (axis.horiz) { + y1 = axisTop; + y2 = cHeight - axis.bottom; + x1 = x2 = between(x1, axisLeft, axisLeft + axis.width); + } else { + x1 = axisLeft; + x2 = cWidth - axis.right; + y1 = y2 = between(y1, axisTop, axisTop + axis.height); + } + e.path = + skip && !force + ? null + : chart.renderer.crispLine( + [ + ["M", x1, y1], + ["L", x2, y2], + ], + lineWidth || 1 + ); + }); + return evt.path; + }; + /** + * Internal function to et the tick positions of a linear axis to round + * values like whole tens or every five. + * + * @function Highcharts.Axis#getLinearTickPositions + * + * @param {number} tickInterval + * The normalized tick interval. + * + * @param {number} min + * Axis minimum. + * + * @param {number} max + * Axis maximum. + * + * @return {Array<number>} + * An array of axis values where ticks should be placed. + */ + Axis.prototype.getLinearTickPositions = function ( + tickInterval, + min, + max + ) { + var pos, + lastPos, + roundedMin = correctFloat( + Math.floor(min / tickInterval) * tickInterval + ), + roundedMax = correctFloat( + Math.ceil(max / tickInterval) * tickInterval + ), + tickPositions = [], + precision; + // When the precision is higher than what we filter out in + // correctFloat, skip it (#6183). + if (correctFloat(roundedMin + tickInterval) === roundedMin) { + precision = 20; + } + // For single points, add a tick regardless of the relative position + // (#2662, #6274) + if (this.single) { + return [min]; + } + // Populate the intermediate values + pos = roundedMin; + while (pos <= roundedMax) { + // Place the tick on the rounded value + tickPositions.push(pos); + // Always add the raw tickInterval, not the corrected one. + pos = correctFloat(pos + tickInterval, precision); + // If the interval is not big enough in the current min - max range + // to actually increase the loop variable, we need to break out to + // prevent endless loop. Issue #619 + if (pos === lastPos) { + break; + } + // Record the last value + lastPos = pos; + } + return tickPositions; + }; + /** + * Resolve the new minorTicks/minorTickInterval options into the legacy + * loosely typed minorTickInterval option. * - * @param {number} width - * The width of the SVG. + * @function Highcharts.Axis#getMinorTickInterval * - * @param {number} height - * The height of the SVG. + * @return {number|"auto"|null} + */ + Axis.prototype.getMinorTickInterval = function () { + var options = this.options; + if (options.minorTicks === true) { + return pick(options.minorTickInterval, "auto"); + } + if (options.minorTicks === false) { + return null; + } + return options.minorTickInterval; + }; + /** + * Internal function to return the minor tick positions. For logarithmic + * axes, the same logic as for major ticks is reused. + * + * @function Highcharts.Axis#getMinorTickPositions + * + * @return {Array<number>} + * An array of axis values where ticks should be placed. + */ + Axis.prototype.getMinorTickPositions = function () { + var axis = this, + options = axis.options, + tickPositions = axis.tickPositions, + minorTickInterval = axis.minorTickInterval, + minorTickPositions = [], + pos, + pointRangePadding = axis.pointRangePadding || 0, + min = axis.min - pointRangePadding, // #1498 + max = axis.max + pointRangePadding, // #1498 + range = max - min; + // If minor ticks get too dense, they are hard to read, and may cause + // long running script. So we don't draw them. + if (range && range / minorTickInterval < axis.len / 3) { + // #3875 + var logarithmic_1 = axis.logarithmic; + if (logarithmic_1) { + // For each interval in the major ticks, compute the minor ticks + // separately. + this.paddedTicks.forEach(function (_pos, i, paddedTicks) { + if (i) { + minorTickPositions.push.apply( + minorTickPositions, + logarithmic_1.getLogTickPositions( + minorTickInterval, + paddedTicks[i - 1], + paddedTicks[i], + true + ) + ); + } + }); + } else if ( + axis.dateTime && + this.getMinorTickInterval() === "auto" + ) { + // #1314 + minorTickPositions = minorTickPositions.concat( + axis.getTimeTicks( + axis.dateTime.normalizeTimeTickInterval(minorTickInterval), + min, + max, + options.startOfWeek + ) + ); + } else { + for ( + pos = min + ((tickPositions[0] - min) % minorTickInterval); + pos <= max; + pos += minorTickInterval + ) { + // Very, very, tight grid lines (#5771) + if (pos === minorTickPositions[0]) { + break; + } + minorTickPositions.push(pos); + } + } + } + if (minorTickPositions.length !== 0) { + axis.trimTicks(minorTickPositions); // #3652 #3743 #1498 #6330 + } + return minorTickPositions; + }; + /** + * Adjust the min and max for the minimum range. Keep in mind that the + * series data is not yet processed, so we don't have information on data + * cropping and grouping, or updated `axis.pointRange` or + * `series.pointRange`. The data can't be processed until we have finally + * established min and max. * - * @param {Highcharts.CSSObject} [style] - * The box style, if not in styleMode + * @private + * @function Highcharts.Axis#adjustForMinRange + */ + Axis.prototype.adjustForMinRange = function () { + var axis = this, + options = axis.options, + min = axis.min, + max = axis.max, + log = axis.logarithmic, + zoomOffset, + spaceAvailable, + closestDataRange, + i, + distance, + xData, + loopLength, + minArgs, + maxArgs, + minRange; + // Set the automatic minimum range based on the closest point distance + if (axis.isXAxis && typeof axis.minRange === "undefined" && !log) { + if (defined(options.min) || defined(options.max)) { + axis.minRange = null; // don't do this again + } else { + // Find the closest distance between raw data points, as opposed + // to closestPointRange that applies to processed points + // (cropped and grouped) + axis.series.forEach(function (series) { + xData = series.xData; + loopLength = series.xIncrement ? 1 : xData.length - 1; + for (i = loopLength; i > 0; i--) { + distance = xData[i] - xData[i - 1]; + if ( + typeof closestDataRange === "undefined" || + distance < closestDataRange + ) { + closestDataRange = distance; + } + } + }); + axis.minRange = Math.min( + closestDataRange * 5, + axis.dataMax - axis.dataMin + ); + } + } + // if minRange is exceeded, adjust + if (max - min < axis.minRange) { + spaceAvailable = axis.dataMax - axis.dataMin >= axis.minRange; + minRange = axis.minRange; + zoomOffset = (minRange - max + min) / 2; + // if min and max options have been set, don't go beyond it + minArgs = [min - zoomOffset, pick(options.min, min - zoomOffset)]; + // If space is available, stay within the data range + if (spaceAvailable) { + minArgs[2] = axis.logarithmic + ? axis.logarithmic.log2lin(axis.dataMin) + : axis.dataMin; + } + min = arrayMax(minArgs); + maxArgs = [min + minRange, pick(options.max, min + minRange)]; + // If space is availabe, stay within the data range + if (spaceAvailable) { + maxArgs[2] = log ? log.log2lin(axis.dataMax) : axis.dataMax; + } + max = arrayMin(maxArgs); + // now if the max is adjusted, adjust the min back + if (max - min < minRange) { + minArgs[0] = max - minRange; + minArgs[1] = pick(options.min, max - minRange); + min = arrayMax(minArgs); + } + } + // Record modified extremes + axis.min = min; + axis.max = max; + }; + // eslint-disable-next-line valid-jsdoc + /** + * Find the closestPointRange across all series. * - * @param {boolean} [forExport=false] - * Whether the rendered content is intended for export. + * @private + * @function Highcharts.Axis#getClosest + */ + Axis.prototype.getClosest = function () { + var ret; + if (this.categories) { + ret = 1; + } else { + this.series.forEach(function (series) { + var seriesClosest = series.closestPointRange, + visible = + series.visible || + !series.chart.options.chart.ignoreHiddenSeries; + if ( + !series.noSharedTooltip && + defined(seriesClosest) && + visible + ) { + ret = defined(ret) + ? Math.min(ret, seriesClosest) + : seriesClosest; + } + }); + } + return ret; + }; + /** + * When a point name is given and no x, search for the name in the existing + * categories, or if categories aren't provided, search names or create a + * new category (#2522). + * @private + * @function Highcharts.Axis#nameToX * - * @param {boolean} [allowHTML=true] - * Whether the renderer is allowed to include HTML text, which will be - * projected on top of the SVG. + * @param {Highcharts.Point} point + * The point to inspect. * - * @param {boolean} [styledMode=false] - * Whether the renderer belongs to a chart that is in styled mode. - * If it does, it will avoid setting presentational attributes in - * some cases, but not when set explicitly through `.attr` and `.css` - * etc. - */ - var SVGRenderer = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function SVGRenderer(container, width, height, style, forExport, allowHTML, styledMode) { - /* * - * - * Properties - * - * */ - this.alignedObjects = void 0; - /** - * The root `svg` node of the renderer. - * - * @name Highcharts.SVGRenderer#box - * @type {Highcharts.SVGDOMElement} - */ - this.box = void 0; - /** - * The wrapper for the root `svg` node of the renderer. - * - * @name Highcharts.SVGRenderer#boxWrapper - * @type {Highcharts.SVGElement} - */ - this.boxWrapper = void 0; - this.cache = void 0; - this.cacheKeys = void 0; - this.chartIndex = void 0; - /** - * A pointer to the `defs` node of the root SVG. - * - * @name Highcharts.SVGRenderer#defs - * @type {Highcharts.SVGElement} - */ - this.defs = void 0; - this.globalAnimation = void 0; - this.gradients = void 0; - this.height = void 0; - this.imgCount = void 0; - this.isSVG = void 0; - this.style = void 0; - /** - * Page url used for internal references. - * - * @private - * @name Highcharts.SVGRenderer#url - * @type {string} - */ - this.url = void 0; - this.width = void 0; - this.init(container, width, height, style, forExport, allowHTML, styledMode); + * @return {number} + * The X value that the point is given. + */ + Axis.prototype.nameToX = function (point) { + var explicitCategories = isArray(this.categories), + names = explicitCategories ? this.categories : this.names, + nameX = point.options.x, + x; + point.series.requireSorting = false; + if (!defined(nameX)) { + nameX = + this.options.uniqueNames === false + ? point.series.autoIncrement() + : explicitCategories + ? names.indexOf(point.name) + : pick(names.keys[point.name], -1); + } + if (nameX === -1) { + // Not found in currenct categories + if (!explicitCategories) { + x = names.length; } - /* * - * - * Functions - * - * */ - /** - * Initialize the SVGRenderer. Overridable initializer function that takes - * the same parameters as the constructor. - * - * @function Highcharts.SVGRenderer#init - * - * @param {Highcharts.HTMLDOMElement} container - * Where to put the SVG in the web page. - * - * @param {number} width - * The width of the SVG. - * - * @param {number} height - * The height of the SVG. - * - * @param {Highcharts.CSSObject} [style] - * The box style, if not in styleMode - * - * @param {boolean} [forExport=false] - * Whether the rendered content is intended for export. - * - * @param {boolean} [allowHTML=true] - * Whether the renderer is allowed to include HTML text, which will be - * projected on top of the SVG. - * - * @param {boolean} [styledMode=false] - * Whether the renderer belongs to a chart that is in styled mode. If it - * does, it will avoid setting presentational attributes in some cases, but - * not when set explicitly through `.attr` and `.css` etc. - */ - SVGRenderer.prototype.init = function (container, width, height, style, forExport, allowHTML, styledMode) { - var renderer = this, - boxWrapper, - element, - desc; - boxWrapper = renderer.createElement('svg') - .attr({ - version: '1.1', - 'class': 'highcharts-root' - }); - if (!styledMode) { - boxWrapper.css(this.getStyle(style)); - } - element = boxWrapper.element; - container.appendChild(element); - // Always use ltr on the container, otherwise text-anchor will be - // flipped and text appear outside labels, buttons, tooltip etc (#3482) - attr(container, 'dir', 'ltr'); - // For browsers other than IE, add the namespace attribute (#1978) - if (container.innerHTML.indexOf('xmlns') === -1) { - attr(element, 'xmlns', this.SVG_NS); - } - // object properties - renderer.isSVG = true; - this.box = element; - this.boxWrapper = boxWrapper; - renderer.alignedObjects = []; - // #24, #672, #1070 - this.url = ((isFirefox || isWebKit) && - doc.getElementsByTagName('base').length) ? - win.location.href - .split('#')[0] // remove the hash - .replace(/<[^>]*>/g, '') // wing cut HTML - // escape parantheses and quotes - .replace(/([\('\)])/g, '\\$1') - // replace spaces (needed for Safari only) - .replace(/ /g, '%20') : - ''; - // Add description - desc = this.createElement('desc').add(); - desc.element.appendChild(doc.createTextNode('Created with Highcharts 8.2.2')); - renderer.defs = this.createElement('defs').add(); - renderer.allowHTML = allowHTML; - renderer.forExport = forExport; - renderer.styledMode = styledMode; - renderer.gradients = {}; // Object where gradient SvgElements are stored - renderer.cache = {}; // Cache for numerical bounding boxes - renderer.cacheKeys = []; - renderer.imgCount = 0; - renderer.setSize(width, height, false); - // Issue 110 workaround: - // In Firefox, if a div is positioned by percentage, its pixel position - // may land between pixels. The container itself doesn't display this, - // but an SVG element inside this container will be drawn at subpixel - // precision. In order to draw sharp lines, this must be compensated - // for. This doesn't seem to work inside iframes though (like in - // jsFiddle). - var subPixelFix, - rect; - if (isFirefox && container.getBoundingClientRect) { - subPixelFix = function () { - css(container, { left: 0, top: 0 }); - rect = container.getBoundingClientRect(); - css(container, { - left: (Math.ceil(rect.left) - rect.left) + 'px', - top: (Math.ceil(rect.top) - rect.top) + 'px' - }); - }; - // run the fix now - subPixelFix(); - // run it on resize - renderer.unSubPixelFix = addEvent(win, 'resize', subPixelFix); + } else { + x = nameX; + } + // Write the last point's name to the names array + if (typeof x !== "undefined") { + this.names[x] = point.name; + // Backwards mapping is much faster than array searching (#7725) + this.names.keys[point.name] = x; + } + return x; + }; + /** + * When changes have been done to series data, update the axis.names. + * + * @private + * @function Highcharts.Axis#updateNames + */ + Axis.prototype.updateNames = function () { + var axis = this, + names = this.names, + i = names.length; + if (i > 0) { + Object.keys(names.keys).forEach(function (key) { + delete names.keys[key]; + }); + names.length = 0; + this.minRange = this.userMinRange; // Reset + (this.series || []).forEach(function (series) { + // Reset incrementer (#5928) + series.xIncrement = null; + // When adding a series, points are not yet generated + if (!series.points || series.isDirtyData) { + // When we're updating the series with data that is longer + // than it was, and cropThreshold is passed, we need to make + // sure that the axis.max is increased _before_ running the + // premature processData. Otherwise this early iteration of + // processData will crop the points to axis.max, and the + // names array will be too short (#5857). + axis.max = Math.max(axis.max, series.xData.length - 1); + series.processData(); + series.generatePoints(); + } + series.data.forEach(function (point, i) { + var x; + if ( + point && + point.options && + typeof point.name !== "undefined" // #9562 + ) { + x = axis.nameToX(point); + if (typeof x !== "undefined" && x !== point.x) { + point.x = x; + series.xData[i] = x; + } } - }; - /** - * General method for adding a definition to the SVG `defs` tag. Can be used - * for gradients, fills, filters etc. Styled mode only. A hook for adding - * general definitions to the SVG's defs tag. Definitions can be referenced - * from the CSS by its `id`. Read more in - * [gradients, shadows and patterns](https://www.highcharts.com/docs/chart-design-and-style/gradients-shadows-and-patterns). - * Styled mode only. - * - * @function Highcharts.SVGRenderer#definition - * - * @param {Highcharts.SVGDefinitionObject} def - * A serialized form of an SVG definition, including children. - * - * @return {Highcharts.SVGElement} - * The inserted node. - */ - SVGRenderer.prototype.definition = function (def) { - var ren = this; - /** - * @private - * @param {Highcharts.SVGDefinitionObject} config - SVG definition - * @param {Highcharts.SVGElement} [parent] - parent node - */ - function recurse(config, parent) { - var ret; - splat(config).forEach(function (item) { - var node = ren.createElement(item.tagName), - attr = {}; - // Set attributes - objectEach(item, function (val, key) { - if (key !== 'tagName' && - key !== 'children' && - key !== 'textContent') { - attr[key] = val; - } - }); - node.attr(attr); - // Add to the tree - node.add(parent || ren.defs); - // Add text content - if (item.textContent) { - node.element.appendChild(doc.createTextNode(item.textContent)); - } - // Recurse - recurse(item.children || [], node); - ret = node; - }); - // Return last node added (on top level it's the only one) - return ret; - } - return recurse(def); - }; - /** - * Get the global style setting for the renderer. - * - * @private - * @function Highcharts.SVGRenderer#getStyle - * - * @param {Highcharts.CSSObject} style - * Style settings. - * - * @return {Highcharts.CSSObject} - * The style settings mixed with defaults. - */ - SVGRenderer.prototype.getStyle = function (style) { - this.style = extend({ - fontFamily: '"Lucida Grande", "Lucida Sans Unicode", ' + - 'Arial, Helvetica, sans-serif', - fontSize: '12px' - }, style); - return this.style; - }; - /** - * Apply the global style on the renderer, mixed with the default styles. - * - * @function Highcharts.SVGRenderer#setStyle - * - * @param {Highcharts.CSSObject} style - * CSS to apply. - */ - SVGRenderer.prototype.setStyle = function (style) { - this.boxWrapper.css(this.getStyle(style)); - }; - /** - * Detect whether the renderer is hidden. This happens when one of the - * parent elements has `display: none`. Used internally to detect when we - * needto render preliminarily in another div to get the text bounding boxes - * right. - * - * @function Highcharts.SVGRenderer#isHidden - * - * @return {boolean} - * True if it is hidden. - */ - SVGRenderer.prototype.isHidden = function () { - return !this.boxWrapper.getBBox().width; - }; - /** - * Destroys the renderer and its allocated members. - * - * @function Highcharts.SVGRenderer#destroy - * - * @return {null} - */ - SVGRenderer.prototype.destroy = function () { - var renderer = this, - rendererDefs = renderer.defs; - renderer.box = null; - renderer.boxWrapper = renderer.boxWrapper.destroy(); - // Call destroy on all gradient elements - destroyObjectProperties(renderer.gradients || {}); - renderer.gradients = null; - // Defs are null in VMLRenderer - // Otherwise, destroy them here. - if (rendererDefs) { - renderer.defs = rendererDefs.destroy(); - } - // Remove sub pixel fix handler (#982) - if (renderer.unSubPixelFix) { - renderer.unSubPixelFix(); - } - renderer.alignedObjects = null; - return null; - }; - /** - * Create a wrapper for an SVG element. Serves as a factory for - * {@link SVGElement}, but this function is itself mostly called from - * primitive factories like {@link SVGRenderer#path}, {@link - * SVGRenderer#rect} or {@link SVGRenderer#text}. - * - * @function Highcharts.SVGRenderer#createElement - * - * @param {string} nodeName - * The node name, for example `rect`, `g` etc. - * - * @return {Highcharts.SVGElement} - * The generated SVGElement. - */ - SVGRenderer.prototype.createElement = function (nodeName) { - var wrapper = new this.Element(); - wrapper.init(this, nodeName); - return wrapper; - }; - /** - * Get converted radial gradient attributes according to the radial - * reference. Used internally from the {@link SVGElement#colorGradient} - * function. - * - * @private - * @function Highcharts.SVGRenderer#getRadialAttr - */ - SVGRenderer.prototype.getRadialAttr = function (radialReference, gradAttr) { - return { - cx: (radialReference[0] - radialReference[2] / 2) + - gradAttr.cx * radialReference[2], - cy: (radialReference[1] - radialReference[2] / 2) + - gradAttr.cy * radialReference[2], - r: gradAttr.r * radialReference[2] - }; - }; - /** - * Truncate the text node contents to a given length. Used when the css - * width is set. If the `textOverflow` is `ellipsis`, the text is truncated - * character by character to the given length. If not, the text is - * word-wrapped line by line. - * - * @private - * @function Highcharts.SVGRenderer#truncate - * - * @return {boolean} - * True if tspan is too long. - */ - SVGRenderer.prototype.truncate = function (wrapper, tspan, text, words, startAt, width, getString) { - var renderer = this, - rotation = wrapper.rotation, - str, - // Word wrap can not be truncated to shorter than one word, ellipsis - // text can be completely blank. - minIndex = words ? 1 : 0, - maxIndex = (text || words).length, - currentIndex = maxIndex, - // Cache the lengths to avoid checking the same twice - lengths = [], - updateTSpan = function (s) { - if (tspan.firstChild) { - tspan.removeChild(tspan.firstChild); - } - if (s) { - tspan.appendChild(doc.createTextNode(s)); - } - }, getSubStringLength = function (charEnd, concatenatedEnd) { - // charEnd is useed when finding the character-by-character - // break for ellipsis, concatenatedEnd is used for word-by-word - // break for word wrapping. - var end = concatenatedEnd || charEnd; - if (typeof lengths[end] === 'undefined') { - // Modern browsers - if (tspan.getSubStringLength) { - // Fails with DOM exception on unit-tests/legend/members - // of unknown reason. Desired width is 0, text content - // is "5" and end is 1. - try { - lengths[end] = startAt + - tspan.getSubStringLength(0, words ? end + 1 : end); - } - catch (e) { - ''; - } - // Legacy - } - else if (renderer.getSpanWidth) { // #9058 jsdom - updateTSpan(getString(text || words, charEnd)); - lengths[end] = startAt + - renderer.getSpanWidth(wrapper, tspan); - } - } - return lengths[end]; - }, actualWidth, truncated; - wrapper.rotation = 0; // discard rotation when computing box - actualWidth = getSubStringLength(tspan.textContent.length); - truncated = startAt + actualWidth > width; - if (truncated) { - // Do a binary search for the index where to truncate the text - while (minIndex <= maxIndex) { - currentIndex = Math.ceil((minIndex + maxIndex) / 2); - // When checking words for word-wrap, we need to build the - // string and measure the subStringLength at the concatenated - // word length. - if (words) { - str = getString(words, currentIndex); - } - actualWidth = getSubStringLength(currentIndex, str && str.length - 1); - if (minIndex === maxIndex) { - // Complete - minIndex = maxIndex + 1; - } - else if (actualWidth > width) { - // Too large. Set max index to current. - maxIndex = currentIndex - 1; - } - else { - // Within width. Set min index to current. - minIndex = currentIndex; - } - } - // If max index was 0 it means the shortest possible text was also - // too large. For ellipsis that means only the ellipsis, while for - // word wrap it means the whole first word. - if (maxIndex === 0) { - // Remove ellipsis - updateTSpan(''); - // If the new text length is one less than the original, we don't - // need the ellipsis - } - else if (!(text && maxIndex === text.length - 1)) { - updateTSpan(str || getString(text || words, currentIndex)); - } - } - // When doing line wrapping, prepare for the next line by removing the - // items from this line. - if (words) { - words.splice(0, currentIndex); - } - wrapper.actualWidth = actualWidth; - wrapper.rotation = rotation; // Apply rotation again. - return truncated; - }; - /** - * Parse a simple HTML string into SVG tspans. Called internally when text - * is set on an SVGElement. The function supports a subset of HTML tags, CSS - * text features like `width`, `text-overflow`, `white-space`, and also - * attributes like `href` and `style`. - * - * @private - * @function Highcharts.SVGRenderer#buildText - * - * @param {Highcharts.SVGElement} wrapper - * The parent SVGElement. - */ - SVGRenderer.prototype.buildText = function (wrapper) { - var textNode = wrapper.element, renderer = this, forExport = renderer.forExport, textStr = pick(wrapper.textStr, '').toString(), hasMarkup = textStr.indexOf('<') !== -1, lines, childNodes = textNode.childNodes, truncated, parentX = attr(textNode, 'x'), textStyles = wrapper.styles, width = wrapper.textWidth, textLineHeight = textStyles && textStyles.lineHeight, textOutline = textStyles && textStyles.textOutline, ellipsis = textStyles && textStyles.textOverflow === 'ellipsis', noWrap = textStyles && textStyles.whiteSpace === 'nowrap', fontSize = textStyles && textStyles.fontSize, textCache, isSubsequentLine, i = childNodes.length, tempParent = width && !wrapper.added && this.box, getLineHeight = function (tspan) { - var fontSizeStyle; - if (!renderer.styledMode) { - fontSizeStyle = - /(px|em)$/.test(tspan && tspan.style.fontSize) ? - tspan.style.fontSize : - (fontSize || renderer.style.fontSize || 12); - } - return textLineHeight ? - pInt(textLineHeight) : - renderer.fontMetrics(fontSizeStyle, - // Get the computed size from parent if not explicit - (tspan.getAttribute('style') ? tspan : textNode)).h; - }, unescapeEntities = function (inputStr, except) { - objectEach(renderer.escapes, function (value, key) { - if (!except || except.indexOf(value) === -1) { - inputStr = inputStr.toString().replace(new RegExp(value, 'g'), key); - } - }); - return inputStr; - }, parseAttribute = function (s, attr) { - var start, - delimiter; - start = s.indexOf('<'); - s = s.substring(start, s.indexOf('>') - start); - start = s.indexOf(attr + '='); - if (start !== -1) { - start = start + attr.length + 1; - delimiter = s.charAt(start); - if (delimiter === '"' || delimiter === "'") { // eslint-disable-line quotes - s = s.substring(start + 1); - return s.substring(0, s.indexOf(delimiter)); - } - } - }; - var regexMatchBreaks = /<br.*?>/g; - // The buildText code is quite heavy, so if we're not changing something - // that affects the text, skip it (#6113). - textCache = [ - textStr, - ellipsis, - noWrap, - textLineHeight, - textOutline, - fontSize, - width - ].join(','); - if (textCache === wrapper.textCache) { - return; - } - wrapper.textCache = textCache; - // Remove old text - while (i--) { - textNode.removeChild(childNodes[i]); - } - // Skip tspans, add text directly to text node. The forceTSpan is a hook - // used in text outline hack. - if (!hasMarkup && - !textOutline && - !ellipsis && - !width && - (textStr.indexOf(' ') === -1 || - (noWrap && !regexMatchBreaks.test(textStr)))) { - textNode.appendChild(doc.createTextNode(unescapeEntities(textStr))); - // Complex strings, add more logic - } - else { - if (tempParent) { - // attach it to the DOM to read offset width - tempParent.appendChild(textNode); - } - if (hasMarkup) { - lines = renderer.styledMode ? (textStr - .replace(/<(b|strong)>/g, '<span class="highcharts-strong">') - .replace(/<(i|em)>/g, '<span class="highcharts-emphasized">')) : (textStr - .replace(/<(b|strong)>/g, '<span style="font-weight:bold">') - .replace(/<(i|em)>/g, '<span style="font-style:italic">')); - lines = lines - .replace(/<a/g, '<span') - .replace(/<\/(b|strong|i|em|a)>/g, '</span>') - .split(regexMatchBreaks); - } - else { - lines = [textStr]; - } - // Trim empty lines (#5261) - lines = lines.filter(function (line) { - return line !== ''; - }); - // build the lines - lines.forEach(function (line, lineNo) { - var spans, - spanNo = 0, - lineLength = 0; - line = line - // Trim to prevent useless/costly process on the spaces - // (#5258) - .replace(/^\s+|\s+$/g, '') - .replace(/<span/g, '|||<span') - .replace(/<\/span>/g, '</span>|||'); - spans = line.split('|||'); - spans.forEach(function buildTextSpans(span) { - if (span !== '' || spans.length === 1) { - var attributes = {}, - tspan = doc.createElementNS(renderer.SVG_NS, 'tspan'), - a, - classAttribute, - styleAttribute, // #390 - hrefAttribute; - classAttribute = parseAttribute(span, 'class'); - if (classAttribute) { - attr(tspan, 'class', classAttribute); - } - styleAttribute = parseAttribute(span, 'style'); - if (styleAttribute) { - styleAttribute = styleAttribute.replace(/(;| |^)color([ :])/, '$1fill$2'); - attr(tspan, 'style', styleAttribute); - } - // For anchors, wrap the tspan in an <a> tag and apply - // the href attribute as is (#13559). Not for export - // (#1529) - hrefAttribute = parseAttribute(span, 'href'); - if (hrefAttribute && !forExport) { - if ( - // Stop JavaScript links, vulnerable to XSS - hrefAttribute.split(':')[0].toLowerCase() - .indexOf('javascript') === -1) { - a = doc.createElementNS(renderer.SVG_NS, 'a'); - attr(a, 'href', hrefAttribute); - attr(tspan, 'class', 'highcharts-anchor'); - a.appendChild(tspan); - if (!renderer.styledMode) { - css(tspan, { cursor: 'pointer' }); - } - } - } - // Strip away unsupported HTML tags (#7126) - span = unescapeEntities(span.replace(/<[a-zA-Z\/](.|\n)*?>/g, '') || ' '); - // Nested tags aren't supported, and cause crash in - // Safari (#1596) - if (span !== ' ') { - // add the text node - tspan.appendChild(doc.createTextNode(span)); - // First span in a line, align it to the left - if (!spanNo) { - if (lineNo && parentX !== null) { - attributes.x = parentX; - } - } - else { - attributes.dx = 0; // #16 - } - // add attributes - attr(tspan, attributes); - // Append it - textNode.appendChild(a || tspan); - // first span on subsequent line, add the line - // height - if (!spanNo && isSubsequentLine) { - // allow getting the right offset height in - // exporting in IE - if (!svg && forExport) { - css(tspan, { display: 'block' }); - } - // Set the line height based on the font size of - // either the text element or the tspan element - attr(tspan, 'dy', getLineHeight(tspan)); - } - // Check width and apply soft breaks or ellipsis - if (width) { - var words = span.replace(/([^\^])-/g, '$1- ').split(' '), // #1273 - hasWhiteSpace = !noWrap && (spans.length > 1 || - lineNo || - words.length > 1), - wrapLineNo = 0, - dy = getLineHeight(tspan); - if (ellipsis) { - truncated = renderer.truncate(wrapper, tspan, span, void 0, 0, - // Target width - Math.max(0, - // Substract the font face to make - // room for the ellipsis itself - width - parseInt(fontSize || 12, 10)), - // Build the text to test for - function (text, currentIndex) { - return text.substring(0, currentIndex) + '\u2026'; - }); - } - else if (hasWhiteSpace) { - while (words.length) { - // For subsequent lines, create tspans - // with the same style attributes as the - // parent text node. - if (words.length && - !noWrap && - wrapLineNo > 0) { - tspan = doc.createElementNS(SVG_NS, 'tspan'); - attr(tspan, { - dy: dy, - x: parentX - }); - if (styleAttribute) { // #390 - attr(tspan, 'style', styleAttribute); - } - // Start by appending the full - // remaining text - tspan.appendChild(doc.createTextNode(words.join(' ') - .replace(/- /g, '-'))); - textNode.appendChild(tspan); - } - // For each line, truncate the remaining - // words into the line length. - renderer.truncate(wrapper, tspan, null, words, wrapLineNo === 0 ? lineLength : 0, width, - // Build the text to test for - function (text, currentIndex) { - return words - .slice(0, currentIndex) - .join(' ') - .replace(/- /g, '-'); - }); - lineLength = wrapper.actualWidth; - wrapLineNo++; - } - } - } - spanNo++; - } - } - }); - // To avoid beginning lines that doesn't add to the textNode - // (#6144) - isSubsequentLine = (isSubsequentLine || - textNode.childNodes.length); - }); - if (ellipsis && truncated) { - wrapper.attr('title', unescapeEntities(wrapper.textStr || '', ['<', '>']) // #7179 - ); - } - if (tempParent) { - tempParent.removeChild(textNode); - } - // Apply the text outline - if (isString(textOutline) && wrapper.applyTextOutline) { - wrapper.applyTextOutline(textOutline); - } - } - }; - /** - * Returns white for dark colors and black for bright colors. - * - * @function Highcharts.SVGRenderer#getContrast - * - * @param {Highcharts.ColorString} rgba - * The color to get the contrast for. - * - * @return {Highcharts.ColorString} - * The contrast color, either `#000000` or `#FFFFFF`. - */ - SVGRenderer.prototype.getContrast = function (rgba) { - rgba = Color.parse(rgba).rgba; - // The threshold may be discussed. Here's a proposal for adding - // different weight to the color channels (#6216) - rgba[0] *= 1; // red - rgba[1] *= 1.2; // green - rgba[2] *= 0.5; // blue - return rgba[0] + rgba[1] + rgba[2] > - 1.8 * 255 ? - '#000000' : - '#FFFFFF'; - }; - /** - * Create a button with preset states. - * - * @function Highcharts.SVGRenderer#button - * - * @param {string} text - * The text or HTML to draw. - * - * @param {number} x - * The x position of the button's left side. - * - * @param {number} y - * The y position of the button's top side. - * - * @param {Highcharts.EventCallbackFunction<Highcharts.SVGElement>} callback - * The function to execute on button click or touch. - * - * @param {Highcharts.SVGAttributes} [normalState] - * SVG attributes for the normal state. - * - * @param {Highcharts.SVGAttributes} [hoverState] - * SVG attributes for the hover state. - * - * @param {Highcharts.SVGAttributes} [pressedState] - * SVG attributes for the pressed state. - * - * @param {Highcharts.SVGAttributes} [disabledState] - * SVG attributes for the disabled state. - * - * @param {Highcharts.SymbolKeyValue} [shape=rect] - * The shape type. - * - * @param {boolean} [useHTML=false] - * Wether to use HTML to render the label. - * - * @return {Highcharts.SVGElement} - * The button element. - */ - SVGRenderer.prototype.button = function (text, x, y, callback, normalState, hoverState, pressedState, disabledState, shape, useHTML) { - var label = this.label(text, - x, - y, - shape, - void 0, - void 0, - useHTML, - void 0, 'button'), - curState = 0, - styledMode = this.styledMode, - // Make a copy of normalState (#13798) - // (reference to options.rangeSelector.buttonTheme) - normalState = normalState ? merge(normalState) : normalState, - userNormalStyle = normalState && normalState.style || {}; - // Remove stylable attributes - if (normalState && normalState.style) { - delete normalState.style; - } - // Default, non-stylable attributes - label.attr(merge({ padding: 8, r: 2 }, normalState)); - if (!styledMode) { - // Presentational - var normalStyle, - hoverStyle, - pressedStyle, - disabledStyle; - // Normal state - prepare the attributes - normalState = merge({ - fill: '#f7f7f7', - stroke: '#cccccc', - 'stroke-width': 1, - style: { - color: '#333333', - cursor: 'pointer', - fontWeight: 'normal' - } - }, { - style: userNormalStyle - }, normalState); - normalStyle = normalState.style; - delete normalState.style; - // Hover state - hoverState = merge(normalState, { - fill: '#e6e6e6' - }, hoverState); - hoverStyle = hoverState.style; - delete hoverState.style; - // Pressed state - pressedState = merge(normalState, { - fill: '#e6ebf5', - style: { - color: '#000000', - fontWeight: 'bold' - } - }, pressedState); - pressedStyle = pressedState.style; - delete pressedState.style; - // Disabled state - disabledState = merge(normalState, { - style: { - color: '#cccccc' - } - }, disabledState); - disabledStyle = disabledState.style; - delete disabledState.style; - } - // Add the events. IE9 and IE10 need mouseover and mouseout to funciton - // (#667). - addEvent(label.element, isMS ? 'mouseover' : 'mouseenter', function () { - if (curState !== 3) { - label.setState(1); - } - }); - addEvent(label.element, isMS ? 'mouseout' : 'mouseleave', function () { - if (curState !== 3) { - label.setState(curState); - } - }); - label.setState = function (state) { - // Hover state is temporary, don't record it - if (state !== 1) { - label.state = curState = state; - } - // Update visuals - label - .removeClass(/highcharts-button-(normal|hover|pressed|disabled)/) - .addClass('highcharts-button-' + - ['normal', 'hover', 'pressed', 'disabled'][state || 0]); - if (!styledMode) { - label - .attr([ - normalState, - hoverState, - pressedState, - disabledState - ][state || 0]) - .css([ - normalStyle, - hoverStyle, - pressedStyle, - disabledStyle - ][state || 0]); - } - }; - // Presentational attributes - if (!styledMode) { - label - .attr(normalState) - .css(extend({ cursor: 'default' }, normalStyle)); - } - return label - .on('click', function (e) { - if (curState !== 3) { - callback.call(label, e); - } - }); - }; - /** - * Make a straight line crisper by not spilling out to neighbour pixels. - * - * @function Highcharts.SVGRenderer#crispLine - * - * @param {Highcharts.SVGPathArray} points - * The original points on the format `[['M', 0, 0], ['L', 100, 0]]`. - * - * @param {number} width - * The width of the line. - * - * @param {string} roundingFunction - * The rounding function name on the `Math` object, can be one of - * `round`, `floor` or `ceil`. - * - * @return {Highcharts.SVGPathArray} - * The original points array, but modified to render crisply. - */ - SVGRenderer.prototype.crispLine = function (points, width, roundingFunction) { - if (roundingFunction === void 0) { roundingFunction = 'round'; } - var start = points[0]; - var end = points[1]; - // Normalize to a crisp line - if (start[1] === end[1]) { - // Substract due to #1129. Now bottom and left axis gridlines behave - // the same. - start[1] = end[1] = - Math[roundingFunction](start[1]) - (width % 2 / 2); - } - if (start[2] === end[2]) { - start[2] = end[2] = - Math[roundingFunction](start[2]) + (width % 2 / 2); - } - return points; - }; - /** - * Draw a path, wraps the SVG `path` element. - * - * @sample highcharts/members/renderer-path-on-chart/ - * Draw a path in a chart - * @sample highcharts/members/renderer-path/ - * Draw a path independent from a chart - * - * @example - * var path = renderer.path(['M', 10, 10, 'L', 30, 30, 'z']) - * .attr({ stroke: '#ff00ff' }) - * .add(); - * - * @function Highcharts.SVGRenderer#path - * - * @param {Highcharts.SVGPathArray} [path] - * An SVG path definition in array form. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - * - */ /** - * Draw a path, wraps the SVG `path` element. - * - * @function Highcharts.SVGRenderer#path - * - * @param {Highcharts.SVGAttributes} [attribs] - * The initial attributes. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ - SVGRenderer.prototype.path = function (path) { - var attribs = (this.styledMode ? {} : { - fill: 'none' - }); - if (isArray(path)) { - attribs.d = path; - } - else if (isObject(path)) { // attributes - extend(attribs, path); - } - return this.createElement('path').attr(attribs); - }; - /** - * Draw a circle, wraps the SVG `circle` element. - * - * @sample highcharts/members/renderer-circle/ - * Drawing a circle - * - * @function Highcharts.SVGRenderer#circle - * - * @param {number} [x] - * The center x position. - * - * @param {number} [y] - * The center y position. - * - * @param {number} [r] - * The radius. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ /** - * Draw a circle, wraps the SVG `circle` element. - * - * @function Highcharts.SVGRenderer#circle - * - * @param {Highcharts.SVGAttributes} [attribs] - * The initial attributes. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ - SVGRenderer.prototype.circle = function (x, y, r) { - var attribs = (isObject(x) ? - x : - typeof x === 'undefined' ? {} : { x: x, y: y, r: r }), wrapper = this.createElement('circle'); - // Setting x or y translates to cx and cy - wrapper.xSetter = wrapper.ySetter = function (value, key, element) { - element.setAttribute('c' + key, value); - }; - return wrapper.attr(attribs); - }; - /** - * Draw and return an arc. - * - * @sample highcharts/members/renderer-arc/ - * Drawing an arc - * - * @function Highcharts.SVGRenderer#arc - * - * @param {number} [x=0] - * Center X position. - * - * @param {number} [y=0] - * Center Y position. - * - * @param {number} [r=0] - * The outer radius' of the arc. - * - * @param {number} [innerR=0] - * Inner radius like used in donut charts. - * - * @param {number} [start=0] - * The starting angle of the arc in radians, where 0 is to the right and - * `-Math.PI/2` is up. - * - * @param {number} [end=0] - * The ending angle of the arc in radians, where 0 is to the right and - * `-Math.PI/2` is up. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ /** - * Draw and return an arc. Overloaded function that takes arguments object. - * - * @function Highcharts.SVGRenderer#arc - * - * @param {Highcharts.SVGAttributes} attribs - * Initial SVG attributes. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ - SVGRenderer.prototype.arc = function (x, y, r, innerR, start, end) { - var arc, - options; - if (isObject(x)) { - options = x; - y = options.y; - r = options.r; - innerR = options.innerR; - start = options.start; - end = options.end; - x = options.x; + }); + }); + } + }; + /** + * Update translation information. + * + * @private + * @function Highcharts.Axis#setAxisTranslation + * + * @param {boolean} [saveOld] + * TO-DO: parameter description + * + * @fires Highcharts.Axis#event:afterSetAxisTranslation + */ + Axis.prototype.setAxisTranslation = function (saveOld) { + var axis = this, + range = axis.max - axis.min, + pointRange = axis.axisPointRange || 0, + closestPointRange, + minPointOffset = 0, + pointRangePadding = 0, + linkedParent = axis.linkedParent, + ordinalCorrection, + hasCategories = !!axis.categories, + transA = axis.transA, + isXAxis = axis.isXAxis; + // Adjust translation for padding. Y axis with categories need to go + // through the same (#1784). + if (isXAxis || hasCategories || pointRange) { + // Get the closest points + closestPointRange = axis.getClosest(); + if (linkedParent) { + minPointOffset = linkedParent.minPointOffset; + pointRangePadding = linkedParent.pointRangePadding; + } else { + axis.series.forEach(function (series) { + var seriesPointRange = hasCategories + ? 1 + : isXAxis + ? pick(series.options.pointRange, closestPointRange, 0) + : axis.axisPointRange || 0, // #2806 + pointPlacement = series.options.pointPlacement; + pointRange = Math.max(pointRange, seriesPointRange); + if (!axis.single || hasCategories) { + // TODO: series should internally set x- and y- + // pointPlacement to simplify this logic. + var isPointPlacementAxis = series.is("xrange") + ? !isXAxis + : isXAxis; + // minPointOffset is the value padding to the left of + // the axis in order to make room for points with a + // pointRange, typically columns. When the + // pointPlacement option is 'between' or 'on', this + // padding does not apply. + minPointOffset = Math.max( + minPointOffset, + isPointPlacementAxis && isString(pointPlacement) + ? 0 + : seriesPointRange / 2 + ); + // Determine the total padding needed to the length of + // the axis to make room for the pointRange. If the + // series' pointPlacement is 'on', no padding is added. + pointRangePadding = Math.max( + pointRangePadding, + isPointPlacementAxis && pointPlacement === "on" + ? 0 + : seriesPointRange + ); + } + }); + } + // Record minPointOffset and pointRangePadding + ordinalCorrection = + axis.ordinal && axis.ordinal.slope && closestPointRange + ? axis.ordinal.slope / closestPointRange + : 1; // #988, #1853 + axis.minPointOffset = minPointOffset = + minPointOffset * ordinalCorrection; + axis.pointRangePadding = pointRangePadding = + pointRangePadding * ordinalCorrection; + // pointRange means the width reserved for each point, like in a + // column chart + axis.pointRange = Math.min( + pointRange, + axis.single && hasCategories ? 1 : range + ); + // closestPointRange means the closest distance between points. In + // columns it is mostly equal to pointRange, but in lines pointRange + // is 0 while closestPointRange is some other value + if (isXAxis) { + axis.closestPointRange = closestPointRange; + } + } + // Secondary values + if (saveOld) { + axis.oldTransA = transA; + } + axis.translationSlope = + axis.transA = + transA = + axis.staticScale || axis.len / (range + pointRangePadding || 1); + // Translation addend + axis.transB = axis.horiz ? axis.left : axis.bottom; + axis.minPixelPadding = transA * minPointOffset; + fireEvent(this, "afterSetAxisTranslation"); + }; + /** + * @private + * @function Highcharts.Axis#minFromRange + * + * @return {number} + */ + Axis.prototype.minFromRange = function () { + var axis = this; + return axis.max - axis.range; + }; + /** + * Set the tick positions to round values and optionally extend the extremes + * to the nearest tick. + * + * @private + * @function Highcharts.Axis#setTickInterval + * + * @param {boolean} secondPass + * TO-DO: parameter description + * + * @fires Highcharts.Axis#event:foundExtremes + */ + Axis.prototype.setTickInterval = function (secondPass) { + var axis = this, + chart = axis.chart, + log = axis.logarithmic, + options = axis.options, + isXAxis = axis.isXAxis, + isLinked = axis.isLinked, + maxPadding = options.maxPadding, + minPadding = options.minPadding, + length, + linkedParentExtremes, + tickIntervalOption = options.tickInterval, + minTickInterval, + tickPixelIntervalOption = options.tickPixelInterval, + categories = axis.categories, + threshold = isNumber(axis.threshold) ? axis.threshold : null, + softThreshold = axis.softThreshold, + thresholdMin, + thresholdMax, + hardMin, + hardMax; + if (!axis.dateTime && !categories && !isLinked) { + this.getTickAmount(); + } + // Min or max set either by zooming/setExtremes or initial options + hardMin = pick(axis.userMin, options.min); + hardMax = pick(axis.userMax, options.max); + // Linked axis gets the extremes from the parent axis + if (isLinked) { + axis.linkedParent = chart[axis.coll][options.linkedTo]; + linkedParentExtremes = axis.linkedParent.getExtremes(); + axis.min = pick( + linkedParentExtremes.min, + linkedParentExtremes.dataMin + ); + axis.max = pick( + linkedParentExtremes.max, + linkedParentExtremes.dataMax + ); + if (options.type !== axis.linkedParent.options.type) { + // Can't link axes of different type + error(11, 1, chart); + } + // Initial min and max from the extreme data values + } else { + // Adjust to hard threshold + if (softThreshold && defined(threshold)) { + if (axis.dataMin >= threshold) { + thresholdMin = threshold; + minPadding = 0; + } else if (axis.dataMax <= threshold) { + thresholdMax = threshold; + maxPadding = 0; + } + } + axis.min = pick(hardMin, thresholdMin, axis.dataMin); + axis.max = pick(hardMax, thresholdMax, axis.dataMax); + } + if (log) { + if ( + axis.positiveValuesOnly && + !secondPass && + Math.min(axis.min, pick(axis.dataMin, axis.min)) <= 0 + ) { + // #978 + // Can't plot negative values on log axis + error(10, 1, chart); + } + // The correctFloat cures #934, float errors on full tens. But it + // was too aggressive for #4360 because of conversion back to lin, + // therefore use precision 15. + axis.min = correctFloat(log.log2lin(axis.min), 16); + axis.max = correctFloat(log.log2lin(axis.max), 16); + } + // handle zoomed range + if (axis.range && defined(axis.max)) { + // #618, #6773: + axis.userMin = + axis.min = + hardMin = + Math.max(axis.dataMin, axis.minFromRange()); + axis.userMax = hardMax = axis.max; + axis.range = null; // don't use it when running setExtremes + } + // Hook for Highstock Scroller. Consider combining with beforePadding. + fireEvent(axis, "foundExtremes"); + // Hook for adjusting this.min and this.max. Used by bubble series. + if (axis.beforePadding) { + axis.beforePadding(); + } + // adjust min and max for the minimum range + axis.adjustForMinRange(); + // Pad the values to get clear of the chart's edges. To avoid + // tickInterval taking the padding into account, we do this after + // computing tick interval (#1337). + if ( + !categories && + !axis.axisPointRange && + !(axis.stacking && axis.stacking.usePercentage) && + !isLinked && + defined(axis.min) && + defined(axis.max) + ) { + length = axis.max - axis.min; + if (length) { + if (!defined(hardMin) && minPadding) { + axis.min -= length * minPadding; + } + if (!defined(hardMax) && maxPadding) { + axis.max += length * maxPadding; + } + } + } + // Handle options for floor, ceiling, softMin and softMax (#6359) + if (!isNumber(axis.userMin)) { + if (isNumber(options.softMin) && options.softMin < axis.min) { + axis.min = hardMin = options.softMin; // #6894 + } + if (isNumber(options.floor)) { + axis.min = Math.max(axis.min, options.floor); + } + } + if (!isNumber(axis.userMax)) { + if (isNumber(options.softMax) && options.softMax > axis.max) { + axis.max = hardMax = options.softMax; // #6894 + } + if (isNumber(options.ceiling)) { + axis.max = Math.min(axis.max, options.ceiling); + } + } + // When the threshold is soft, adjust the extreme value only if the data + // extreme and the padded extreme land on either side of the threshold. + // For example, a series of [0, 1, 2, 3] would make the yAxis add a tick + // for -1 because of the default minPadding and startOnTick options. + // This is prevented by the softThreshold option. + if (softThreshold && defined(axis.dataMin)) { + threshold = threshold || 0; + if ( + !defined(hardMin) && + axis.min < threshold && + axis.dataMin >= threshold + ) { + axis.min = axis.options.minRange + ? Math.min(threshold, axis.max - axis.minRange) + : threshold; + } else if ( + !defined(hardMax) && + axis.max > threshold && + axis.dataMax <= threshold + ) { + axis.max = axis.options.minRange + ? Math.max(threshold, axis.min + axis.minRange) + : threshold; + } + } + // get tickInterval + if ( + axis.min === axis.max || + typeof axis.min === "undefined" || + typeof axis.max === "undefined" + ) { + axis.tickInterval = 1; + } else if ( + isLinked && + !tickIntervalOption && + tickPixelIntervalOption === + axis.linkedParent.options.tickPixelInterval + ) { + axis.tickInterval = tickIntervalOption = + axis.linkedParent.tickInterval; + } else { + axis.tickInterval = pick( + tickIntervalOption, + this.tickAmount + ? (axis.max - axis.min) / Math.max(this.tickAmount - 1, 1) + : void 0, + // For categoried axis, 1 is default, for linear axis use + // tickPix + categories + ? 1 + : // don't let it be more than the data range + ((axis.max - axis.min) * tickPixelIntervalOption) / + Math.max(axis.len, tickPixelIntervalOption) + ); + } + // Now we're finished detecting min and max, crop and group series data. + // This is in turn needed in order to find tick positions in ordinal + // axes. + if (isXAxis && !secondPass) { + axis.series.forEach(function (series) { + series.processData( + axis.min !== axis.oldMin || axis.max !== axis.oldMax + ); + }); + } + // set the translation factor used in translate function + axis.setAxisTranslation(true); + // hook for ordinal axes and radial axes + fireEvent(this, "initialAxisTranslation"); + // In column-like charts, don't cramp in more ticks than there are + // points (#1943, #4184) + if (axis.pointRange && !tickIntervalOption) { + axis.tickInterval = Math.max(axis.pointRange, axis.tickInterval); + } + // Before normalizing the tick interval, handle minimum tick interval. + // This applies only if tickInterval is not defined. + minTickInterval = pick( + options.minTickInterval, + // In datetime axes, don't go below the data interval, except when + // there are scatter-like series involved (#13369). + axis.dateTime && + !axis.series.some(function (s) { + return s.noSharedTooltip; + }) + ? axis.closestPointRange + : 0 + ); + if (!tickIntervalOption && axis.tickInterval < minTickInterval) { + axis.tickInterval = minTickInterval; + } + // for linear axes, get magnitude and normalize the interval + if (!axis.dateTime && !axis.logarithmic && !tickIntervalOption) { + axis.tickInterval = normalizeTickInterval( + axis.tickInterval, + void 0, + getMagnitude(axis.tickInterval), + pick( + options.allowDecimals, + // If the tick interval is greather than 0.5, avoid + // decimals, as linear axes are often used to render + // discrete values. #3363. If a tick amount is set, allow + // decimals by default, as it increases the chances for a + // good fit. + axis.tickInterval < 0.5 || this.tickAmount !== void 0 + ), + !!this.tickAmount + ); + } + // Prevent ticks from getting so close that we can't draw the labels + if (!this.tickAmount) { + axis.tickInterval = axis.unsquish(); + } + this.setTickPositions(); + }; + /** + * Now we have computed the normalized tickInterval, get the tick positions. + * + * @private + * @function Highcharts.Axis#setTickPositions + * + * @fires Highcharts.Axis#event:afterSetTickPositions + */ + Axis.prototype.setTickPositions = function () { + var axis = this, + options = this.options, + tickPositions, + tickPositionsOption = options.tickPositions, + minorTickIntervalOption = this.getMinorTickInterval(), + tickPositioner = options.tickPositioner, + hasVerticalPanning = this.hasVerticalPanning(), + isColorAxis = this.coll === "colorAxis", + startOnTick = + (isColorAxis || !hasVerticalPanning) && options.startOnTick, + endOnTick = + (isColorAxis || !hasVerticalPanning) && options.endOnTick; + // Set the tickmarkOffset + this.tickmarkOffset = + this.categories && + options.tickmarkPlacement === "between" && + this.tickInterval === 1 + ? 0.5 + : 0; // #3202 + // get minorTickInterval + this.minorTickInterval = + minorTickIntervalOption === "auto" && this.tickInterval + ? this.tickInterval / 5 + : minorTickIntervalOption; + // When there is only one point, or all points have the same value on + // this axis, then min and max are equal and tickPositions.length is 0 + // or 1. In this case, add some padding in order to center the point, + // but leave it with one tick. #1337. + this.single = + this.min === this.max && + defined(this.min) && + !this.tickAmount && + // Data is on integer (#6563) + (parseInt(this.min, 10) === this.min || + // Between integers and decimals are not allowed (#6274) + options.allowDecimals !== false); + /** + * Contains the current positions that are laid out on the axis. The + * positions are numbers in terms of axis values. In a category axis + * they are integers, in a datetime axis they are also integers, but + * designating milliseconds. + * + * This property is read only - for modifying the tick positions, use + * the `tickPositioner` callback or [axis.tickPositions( + * https://api.highcharts.com/highcharts/xAxis.tickPositions) option + * instead. + * + * @name Highcharts.Axis#tickPositions + * @type {Highcharts.AxisTickPositionsArray|undefined} + */ + this.tickPositions = + // Find the tick positions. Work on a copy (#1565) + tickPositions = tickPositionsOption && tickPositionsOption.slice(); + if (!tickPositions) { + // Too many ticks (#6405). Create a friendly warning and provide two + // ticks so at least we can show the data series. + if ( + (!axis.ordinal || !axis.ordinal.positions) && + (this.max - this.min) / this.tickInterval > + Math.max(2 * this.len, 200) + ) { + tickPositions = [this.min, this.max]; + error(19, false, this.chart); + } else if (axis.dateTime) { + tickPositions = axis.getTimeTicks( + axis.dateTime.normalizeTimeTickInterval( + this.tickInterval, + options.units + ), + this.min, + this.max, + options.startOfWeek, + axis.ordinal && axis.ordinal.positions, + this.closestPointRange, + true + ); + } else if (axis.logarithmic) { + tickPositions = axis.logarithmic.getLogTickPositions( + this.tickInterval, + this.min, + this.max + ); + } else { + tickPositions = this.getLinearTickPositions( + this.tickInterval, + this.min, + this.max + ); + } + // Too dense ticks, keep only the first and last (#4477) + if (tickPositions.length > this.len) { + tickPositions = [tickPositions[0], tickPositions.pop()]; + // Reduce doubled value (#7339) + if (tickPositions[0] === tickPositions[1]) { + tickPositions.length = 1; + } + } + this.tickPositions = tickPositions; + // Run the tick positioner callback, that allows modifying auto tick + // positions. + if (tickPositioner) { + tickPositioner = tickPositioner.apply(axis, [this.min, this.max]); + if (tickPositioner) { + this.tickPositions = tickPositions = tickPositioner; + } + } + } + // Reset min/max or remove extremes based on start/end on tick + this.paddedTicks = tickPositions.slice(0); // Used for logarithmic minor + this.trimTicks(tickPositions, startOnTick, endOnTick); + if (!this.isLinked) { + // Substract half a unit (#2619, #2846, #2515, #3390), + // but not in case of multiple ticks (#6897) + if ( + this.single && + tickPositions.length < 2 && + !this.categories && + !this.series.some(function (s) { + return ( + s.is("heatmap") && s.options.pointPlacement === "between" + ); + }) + ) { + this.min -= 0.5; + this.max += 0.5; + } + if (!tickPositionsOption && !tickPositioner) { + this.adjustTickAmount(); + } + } + fireEvent(this, "afterSetTickPositions"); + }; + /** + * Handle startOnTick and endOnTick by either adapting to padding min/max or + * rounded min/max. Also handle single data points. + * + * @private + * @function Highcharts.Axis#trimTicks + * + * @param {Array<number>} tickPositions + * TO-DO: parameter description + * + * @param {boolean} [startOnTick] + * TO-DO: parameter description + * + * @param {boolean} [endOnTick] + * TO-DO: parameter description + */ + Axis.prototype.trimTicks = function ( + tickPositions, + startOnTick, + endOnTick + ) { + var roundedMin = tickPositions[0], + roundedMax = tickPositions[tickPositions.length - 1], + minPointOffset = (!this.isOrdinal && this.minPointOffset) || 0; // (#12716) + fireEvent(this, "trimTicks"); + if (!this.isLinked) { + if (startOnTick && roundedMin !== -Infinity) { + // #6502 + this.min = roundedMin; + } else { + while (this.min - minPointOffset > tickPositions[0]) { + tickPositions.shift(); + } + } + if (endOnTick) { + this.max = roundedMax; + } else { + while ( + this.max + minPointOffset < + tickPositions[tickPositions.length - 1] + ) { + tickPositions.pop(); + } + } + // If no tick are left, set one tick in the middle (#3195) + if ( + tickPositions.length === 0 && + defined(roundedMin) && + !this.options.tickPositions + ) { + tickPositions.push((roundedMax + roundedMin) / 2); + } + } + }; + /** + * Check if there are multiple axes in the same pane. + * + * @private + * @function Highcharts.Axis#alignToOthers + * + * @return {boolean|undefined} + * True if there are other axes. + */ + Axis.prototype.alignToOthers = function () { + var axis = this, + others = + // Whether there is another axis to pair with this one + {}, + hasOther, + options = axis.options; + if ( + // Only if alignTicks is true + this.chart.options.chart.alignTicks !== false && + options.alignTicks !== false && + // Disabled when startOnTick or endOnTick are false (#7604) + options.startOnTick !== false && + options.endOnTick !== false && + // Don't try to align ticks on a log axis, they are not evenly + // spaced (#6021) + !axis.logarithmic + ) { + this.chart[this.coll].forEach(function (axis) { + var otherOptions = axis.options, + horiz = axis.horiz, + key = [ + horiz ? otherOptions.left : otherOptions.top, + otherOptions.width, + otherOptions.height, + otherOptions.pane, + ].join(","); + if (axis.series.length) { + // #4442 + if (others[key]) { + hasOther = true; // #4201 + } else { + others[key] = 1; + } + } + }); + } + return hasOther; + }; + /** + * Find the max ticks of either the x and y axis collection, and record it + * in `this.tickAmount`. + * + * @private + * @function Highcharts.Axis#getTickAmount + */ + Axis.prototype.getTickAmount = function () { + var axis = this, + options = this.options, + tickAmount = options.tickAmount, + tickPixelInterval = options.tickPixelInterval; + if ( + !defined(options.tickInterval) && + !tickAmount && + this.len < tickPixelInterval && + !this.isRadial && + !axis.logarithmic && + options.startOnTick && + options.endOnTick + ) { + tickAmount = 2; + } + if (!tickAmount && this.alignToOthers()) { + // Add 1 because 4 tick intervals require 5 ticks (including first + // and last) + tickAmount = Math.ceil(this.len / tickPixelInterval) + 1; + } + // For tick amounts of 2 and 3, compute five ticks and remove the + // intermediate ones. This prevents the axis from adding ticks that are + // too far away from the data extremes. + if (tickAmount < 4) { + this.finalTickAmt = tickAmount; + tickAmount = 5; + } + this.tickAmount = tickAmount; + }; + /** + * When using multiple axes, adjust the number of ticks to match the highest + * number of ticks in that group. + * + * @private + * @function Highcharts.Axis#adjustTickAmount + */ + Axis.prototype.adjustTickAmount = function () { + var axis = this, + axisOptions = axis.options, + tickInterval = axis.tickInterval, + tickPositions = axis.tickPositions, + tickAmount = axis.tickAmount, + finalTickAmt = axis.finalTickAmt, + currentTickAmount = tickPositions && tickPositions.length, + threshold = pick(axis.threshold, axis.softThreshold ? 0 : null), + min, + len, + i; + if (axis.hasData()) { + if (currentTickAmount < tickAmount) { + min = axis.min; + while (tickPositions.length < tickAmount) { + // Extend evenly for both sides unless we're on the + // threshold (#3965) + if (tickPositions.length % 2 || min === threshold) { + // to the end + tickPositions.push( + correctFloat( + tickPositions[tickPositions.length - 1] + tickInterval + ) + ); + } else { + // to the start + tickPositions.unshift( + correctFloat(tickPositions[0] - tickInterval) + ); + } + } + axis.transA *= (currentTickAmount - 1) / (tickAmount - 1); + // Do not crop when ticks are not extremes (#9841) + axis.min = axisOptions.startOnTick + ? tickPositions[0] + : Math.min(axis.min, tickPositions[0]); + axis.max = axisOptions.endOnTick + ? tickPositions[tickPositions.length - 1] + : Math.max(axis.max, tickPositions[tickPositions.length - 1]); + // We have too many ticks, run second pass to try to reduce ticks + } else if (currentTickAmount > tickAmount) { + axis.tickInterval *= 2; + axis.setTickPositions(); + } + // The finalTickAmt property is set in getTickAmount + if (defined(finalTickAmt)) { + i = len = tickPositions.length; + while (i--) { + if ( + // Remove every other tick + (finalTickAmt === 3 && i % 2 === 1) || + // Remove all but first and last + (finalTickAmt <= 2 && i > 0 && i < len - 1) + ) { + tickPositions.splice(i, 1); } - else { - options = { - innerR: innerR, - start: start, - end: end - }; - } - // Arcs are defined as symbols for the ability to set - // attributes in attr and animate - arc = this.symbol('arc', x, y, r, r, options); - arc.r = r; // #959 - return arc; - }; - /** - * Draw and return a rectangle. - * - * @function Highcharts.SVGRenderer#rect - * - * @param {number} [x] - * Left position. - * - * @param {number} [y] - * Top position. - * - * @param {number} [width] - * Width of the rectangle. - * - * @param {number} [height] - * Height of the rectangle. - * - * @param {number} [r] - * Border corner radius. - * - * @param {number} [strokeWidth] - * A stroke width can be supplied to allow crisp drawing. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ /** - * Draw and return a rectangle. - * - * @sample highcharts/members/renderer-rect-on-chart/ - * Draw a rectangle in a chart - * @sample highcharts/members/renderer-rect/ - * Draw a rectangle independent from a chart - * - * @function Highcharts.SVGRenderer#rect - * - * @param {Highcharts.SVGAttributes} [attributes] - * General SVG attributes for the rectangle. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ - SVGRenderer.prototype.rect = function (x, y, width, height, r, strokeWidth) { - r = isObject(x) ? x.r : r; - var wrapper = this.createElement('rect'), - attribs = isObject(x) ? - x : - typeof x === 'undefined' ? - {} : - { - x: x, - y: y, - width: Math.max(width, 0), - height: Math.max(height, 0) - }; - if (!this.styledMode) { - if (typeof strokeWidth !== 'undefined') { - attribs.strokeWidth = strokeWidth; - attribs = wrapper.crisp(attribs); - } - attribs.fill = 'none'; - } - if (r) { - attribs.r = r; - } - wrapper.rSetter = function (value, key, element) { - wrapper.r = value; - attr(element, { - rx: value, - ry: value - }); - }; - wrapper.rGetter = function () { - return wrapper.r; - }; - return wrapper.attr(attribs); - }; - /** - * Resize the {@link SVGRenderer#box} and re-align all aligned child - * elements. - * - * @sample highcharts/members/renderer-g/ - * Show and hide grouped objects - * - * @function Highcharts.SVGRenderer#setSize - * - * @param {number} width - * The new pixel width. - * - * @param {number} height - * The new pixel height. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animate=true] - * Whether and how to animate. - */ - SVGRenderer.prototype.setSize = function (width, height, animate) { - var renderer = this, - alignedObjects = renderer.alignedObjects, - i = alignedObjects.length; - renderer.width = width; - renderer.height = height; - renderer.boxWrapper.animate({ - width: width, - height: height - }, { - step: function () { - this.attr({ - viewBox: '0 0 ' + this.attr('width') + ' ' + - this.attr('height') - }); - }, - duration: pick(animate, true) ? void 0 : 0 + } + axis.finalTickAmt = void 0; + } + } + }; + /** + * Set the scale based on data min and max, user set min and max or options. + * + * @private + * @function Highcharts.Axis#setScale + * + * @fires Highcharts.Axis#event:afterSetScale + */ + Axis.prototype.setScale = function () { + var axis = this, + isDirtyAxisLength, + isDirtyData = false, + isXAxisDirty = false; + axis.series.forEach(function (series) { + var _a; + isDirtyData = isDirtyData || series.isDirtyData || series.isDirty; + // When x axis is dirty, we need new data extremes for y as + // well: + isXAxisDirty = + isXAxisDirty || + ((_a = series.xAxis) === null || _a === void 0 + ? void 0 + : _a.isDirty) || + false; + }); + axis.oldMin = axis.min; + axis.oldMax = axis.max; + axis.oldAxisLength = axis.len; + // set the new axisLength + axis.setAxisSize(); + isDirtyAxisLength = axis.len !== axis.oldAxisLength; + // do we really need to go through all this? + if ( + isDirtyAxisLength || + isDirtyData || + isXAxisDirty || + axis.isLinked || + axis.forceRedraw || + axis.userMin !== axis.oldUserMin || + axis.userMax !== axis.oldUserMax || + axis.alignToOthers() + ) { + if (axis.stacking) { + axis.stacking.resetStacks(); + } + axis.forceRedraw = false; + // get data extremes if needed + axis.getSeriesExtremes(); + // get fixed positions based on tickInterval + axis.setTickInterval(); + // record old values to decide whether a rescale is necessary later + // on (#540) + axis.oldUserMin = axis.userMin; + axis.oldUserMax = axis.userMax; + // Mark as dirty if it is not already set to dirty and extremes have + // changed. #595. + if (!axis.isDirty) { + axis.isDirty = + isDirtyAxisLength || + axis.min !== axis.oldMin || + axis.max !== axis.oldMax; + } + } else if (axis.stacking) { + axis.stacking.cleanStacks(); + } + // Recalculate panning state object, when the data + // has changed. It is required when vertical panning is enabled. + if (isDirtyData && axis.panningState) { + axis.panningState.isDirty = true; + } + fireEvent(this, "afterSetScale"); + }; + /** + * Set the minimum and maximum of the axes after render time. If the + * `startOnTick` and `endOnTick` options are true, the minimum and maximum + * values are rounded off to the nearest tick. To prevent this, these + * options can be set to false before calling setExtremes. Also, setExtremes + * will not allow a range lower than the `minRange` option, which by default + * is the range of five points. + * + * @sample highcharts/members/axis-setextremes/ + * Set extremes from a button + * @sample highcharts/members/axis-setextremes-datetime/ + * Set extremes on a datetime axis + * @sample highcharts/members/axis-setextremes-off-ticks/ + * Set extremes off ticks + * @sample stock/members/axis-setextremes/ + * Set extremes in Highstock + * @sample maps/members/axis-setextremes/ + * Set extremes in Highmaps + * + * @function Highcharts.Axis#setExtremes + * + * @param {number} [newMin] + * The new minimum value. + * + * @param {number} [newMax] + * The new maximum value. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart or wait for an explicit call to + * {@link Highcharts.Chart#redraw} + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] + * Enable or modify animations. + * + * @param {*} [eventArguments] + * Arguments to be accessed in event handler. + * + * @fires Highcharts.Axis#event:setExtremes + */ + Axis.prototype.setExtremes = function ( + newMin, + newMax, + redraw, + animation, + eventArguments + ) { + var axis = this, + chart = axis.chart; + redraw = pick(redraw, true); // defaults to true + axis.series.forEach(function (serie) { + delete serie.kdTree; + }); + // Extend the arguments with min and max + eventArguments = extend(eventArguments, { + min: newMin, + max: newMax, + }); + // Fire the event + fireEvent(axis, "setExtremes", eventArguments, function () { + axis.userMin = newMin; + axis.userMax = newMax; + axis.eventArgs = eventArguments; + if (redraw) { + chart.redraw(animation); + } + }); + }; + /** + * Overridable method for zooming chart. Pulled out in a separate method to + * allow overriding in stock charts. + * @private + * @function Highcharts.Axis#zoom + * + * @param {number} newMin + * TO-DO: parameter description + * + * @param {number} newMax + * TO-DO: parameter description + * + * @return {boolean} + */ + Axis.prototype.zoom = function (newMin, newMax) { + var axis = this, + dataMin = this.dataMin, + dataMax = this.dataMax, + options = this.options, + min = Math.min(dataMin, pick(options.min, dataMin)), + max = Math.max(dataMax, pick(options.max, dataMax)), + evt = { + newMin: newMin, + newMax: newMax, + }; + fireEvent(this, "zoom", evt, function (e) { + // Use e.newMin and e.newMax - event handlers may have altered them + var newMin = e.newMin, + newMax = e.newMax; + if (newMin !== axis.min || newMax !== axis.max) { + // #5790 + // Prevent pinch zooming out of range. Check for defined is for + // #1946. #1734. + if (!axis.allowZoomOutside) { + // #6014, sometimes newMax will be smaller than min (or + // newMin will be larger than max). + if (defined(dataMin)) { + if (newMin < min) { + newMin = min; + } + if (newMin > max) { + newMin = max; + } + } + if (defined(dataMax)) { + if (newMax < min) { + newMax = min; + } + if (newMax > max) { + newMax = max; + } + } + } + // In full view, displaying the reset zoom button is not + // required + axis.displayBtn = + typeof newMin !== "undefined" || typeof newMax !== "undefined"; + // Do it + axis.setExtremes(newMin, newMax, false, void 0, { + trigger: "zoom", + }); + } + e.zoomed = true; + }); + return evt.zoomed; + }; + /** + * Update the axis metrics. + * + * @private + * @function Highcharts.Axis#setAxisSize + */ + Axis.prototype.setAxisSize = function () { + var chart = this.chart, + options = this.options, + // [top, right, bottom, left] + offsets = options.offsets || [0, 0, 0, 0], + horiz = this.horiz, + // Check for percentage based input values. Rounding fixes problems + // with column overflow and plot line filtering (#4898, #4899) + width = (this.width = Math.round( + relativeLength( + pick(options.width, chart.plotWidth - offsets[3] + offsets[1]), + chart.plotWidth + ) + )), + height = (this.height = Math.round( + relativeLength( + pick( + options.height, + chart.plotHeight - offsets[0] + offsets[2] + ), + chart.plotHeight + ) + )), + top = (this.top = Math.round( + relativeLength( + pick(options.top, chart.plotTop + offsets[0]), + chart.plotHeight, + chart.plotTop + ) + )), + left = (this.left = Math.round( + relativeLength( + pick(options.left, chart.plotLeft + offsets[3]), + chart.plotWidth, + chart.plotLeft + ) + )); + // Expose basic values to use in Series object and navigator + this.bottom = chart.chartHeight - height - top; + this.right = chart.chartWidth - width - left; + // Direction agnostic properties + this.len = Math.max(horiz ? width : height, 0); // Math.max fixes #905 + this.pos = horiz ? left : top; // distance from SVG origin + }; + /** + * Get the current extremes for the axis. + * + * @sample highcharts/members/axis-getextremes/ + * Report extremes by click on a button + * @sample maps/members/axis-getextremes/ + * Get extremes in Highmaps + * + * @function Highcharts.Axis#getExtremes + * + * @return {Highcharts.ExtremesObject} + * An object containing extremes information. + */ + Axis.prototype.getExtremes = function () { + var axis = this; + var log = axis.logarithmic; + return { + min: log ? correctFloat(log.lin2log(axis.min)) : axis.min, + max: log ? correctFloat(log.lin2log(axis.max)) : axis.max, + dataMin: axis.dataMin, + dataMax: axis.dataMax, + userMin: axis.userMin, + userMax: axis.userMax, + }; + }; + /** + * Get the zero plane either based on zero or on the min or max value. + * Used in bar and area plots. + * + * @function Highcharts.Axis#getThreshold + * + * @param {number} threshold + * The threshold in axis values. + * + * @return {number|undefined} + * The translated threshold position in terms of pixels, and corrected to + * stay within the axis bounds. + */ + Axis.prototype.getThreshold = function (threshold) { + var axis = this, + log = axis.logarithmic, + realMin = log ? log.lin2log(axis.min) : axis.min, + realMax = log ? log.lin2log(axis.max) : axis.max; + if (threshold === null || threshold === -Infinity) { + threshold = realMin; + } else if (threshold === Infinity) { + threshold = realMax; + } else if (realMin > threshold) { + threshold = realMin; + } else if (realMax < threshold) { + threshold = realMax; + } + return axis.translate(threshold, 0, 1, 0, 1); + }; + /** + * Compute auto alignment for the axis label based on which side the axis is + * on and the given rotation for the label. + * + * @private + * @function Highcharts.Axis#autoLabelAlign + * + * @param {number} rotation + * The rotation in degrees as set by either the `rotation` or `autoRotation` + * options. + * + * @return {Highcharts.AlignValue} + * Can be `"center"`, `"left"` or `"right"`. + */ + Axis.prototype.autoLabelAlign = function (rotation) { + var angle = (pick(rotation, 0) - this.side * 90 + 720) % 360, + evt = { align: "center" }; + fireEvent(this, "autoLabelAlign", evt, function (e) { + if (angle > 15 && angle < 165) { + e.align = "right"; + } else if (angle > 195 && angle < 345) { + e.align = "left"; + } + }); + return evt.align; + }; + /** + * Get the tick length and width for the axis based on axis options. + * @private + * @function Highcharts.Axis#tickSize + * + * @param {string} [prefix] + * 'tick' or 'minorTick' + * + * @return {Array<number,number>|undefined} + * An array of tickLength and tickWidth + */ + Axis.prototype.tickSize = function (prefix) { + var options = this.options, + tickLength = + options[prefix === "tick" ? "tickLength" : "minorTickLength"], + tickWidth = pick( + options[prefix === "tick" ? "tickWidth" : "minorTickWidth"], + // Default to 1 on linear and datetime X axes + prefix === "tick" && this.isXAxis && !this.categories ? 1 : 0 + ), + e, + tickSize; + if (tickWidth && tickLength) { + // Negate the length + if (options[prefix + "Position"] === "inside") { + tickLength = -tickLength; + } + tickSize = [tickLength, tickWidth]; + } + e = { tickSize: tickSize }; + fireEvent(this, "afterTickSize", e); + return e.tickSize; + }; + /** + * Return the size of the labels. + * + * @private + * @function Highcharts.Axis#labelMetrics + * + * @return {Highcharts.FontMetricsObject} + */ + Axis.prototype.labelMetrics = function () { + var index = (this.tickPositions && this.tickPositions[0]) || 0; + return this.chart.renderer.fontMetrics( + this.options.labels.style && this.options.labels.style.fontSize, + this.ticks[index] && this.ticks[index].label + ); + }; + /** + * Prevent the ticks from getting so close we can't draw the labels. On a + * horizontal axis, this is handled by rotating the labels, removing ticks + * and adding ellipsis. On a vertical axis remove ticks and add ellipsis. + * + * @private + * @function Highcharts.Axis#unsquish + * + * @return {number} + */ + Axis.prototype.unsquish = function () { + var labelOptions = this.options.labels, + horiz = this.horiz, + tickInterval = this.tickInterval, + newTickInterval = tickInterval, + slotSize = + this.len / + (((this.categories ? 1 : 0) + this.max - this.min) / + tickInterval), + rotation, + rotationOption = labelOptions.rotation, + labelMetrics = this.labelMetrics(), + step, + bestScore = Number.MAX_VALUE, + autoRotation, + range = this.max - this.min, + // Return the multiple of tickInterval that is needed to avoid + // collision + getStep = function (spaceNeeded) { + var step = spaceNeeded / (slotSize || 1); + step = step > 1 ? Math.ceil(step) : 1; + // Guard for very small or negative angles (#9835) + if ( + step * tickInterval > range && + spaceNeeded !== Infinity && + slotSize !== Infinity && + range + ) { + step = Math.ceil(range / tickInterval); + } + return correctFloat(step * tickInterval); + }; + if (horiz) { + autoRotation = + !labelOptions.staggerLines && + !labelOptions.step && // #3971 + (defined(rotationOption) + ? [rotationOption] + : slotSize < pick(labelOptions.autoRotationLimit, 80) && + labelOptions.autoRotation); + if (autoRotation) { + // Loop over the given autoRotation options, and determine + // which gives the best score. The best score is that with + // the lowest number of steps and a rotation closest + // to horizontal. + autoRotation.forEach(function (rot) { + var score; + if ( + rot === rotationOption || + (rot && rot >= -90 && rot <= 90) + ) { + // #3891 + step = getStep( + Math.abs(labelMetrics.h / Math.sin(deg2rad * rot)) + ); + score = step + Math.abs(rot / 360); + if (score < bestScore) { + bestScore = score; + rotation = rot; + newTickInterval = step; + } + } + }); + } + } else if (!labelOptions.step) { + // #4411 + newTickInterval = getStep(labelMetrics.h); + } + this.autoRotation = autoRotation; + this.labelRotation = pick(rotation, rotationOption); + return newTickInterval; + }; + /** + * Get the general slot width for labels/categories on this axis. This may + * change between the pre-render (from Axis.getOffset) and the final tick + * rendering and placement. + * + * @private + * @function Highcharts.Axis#getSlotWidth + * + * @param {Highcharts.Tick} [tick] Optionally, calculate the slot width + * basing on tick label. It is used in highcharts-3d module, where the slots + * has different widths depending on perspective angles. + * + * @return {number} + * The pixel width allocated to each axis label. + */ + Axis.prototype.getSlotWidth = function (tick) { + var _a; + // #5086, #1580, #1931 + var chart = this.chart, + horiz = this.horiz, + labelOptions = this.options.labels, + slotCount = Math.max( + this.tickPositions.length - (this.categories ? 0 : 1), + 1 + ), + marginLeft = chart.margin[3]; + // Used by grid axis + if (tick && isNumber(tick.slotWidth)) { + // #13221, can be 0 + return tick.slotWidth; + } + if (horiz && labelOptions && (labelOptions.step || 0) < 2) { + if (labelOptions.rotation) { + // #4415 + return 0; + } + return ((this.staggerLines || 1) * this.len) / slotCount; + } + if (!horiz) { + // #7028 + var cssWidth = + (_a = + labelOptions === null || labelOptions === void 0 + ? void 0 + : labelOptions.style) === null || _a === void 0 + ? void 0 + : _a.width; + if (cssWidth !== void 0) { + return parseInt(cssWidth, 10); + } + if (marginLeft) { + return marginLeft - chart.spacing[3]; + } + } + // Last resort, a fraction of the available size + return chart.chartWidth * 0.33; + }; + /** + * Render the axis labels and determine whether ellipsis or rotation need to + * be applied. + * + * @private + * @function Highcharts.Axis#renderUnsquish + */ + Axis.prototype.renderUnsquish = function () { + var chart = this.chart, + renderer = chart.renderer, + tickPositions = this.tickPositions, + ticks = this.ticks, + labelOptions = this.options.labels, + labelStyleOptions = (labelOptions && labelOptions.style) || {}, + horiz = this.horiz, + slotWidth = this.getSlotWidth(), + innerWidth = Math.max( + 1, + Math.round(slotWidth - 2 * (labelOptions.padding || 5)) + ), + attr = {}, + labelMetrics = this.labelMetrics(), + textOverflowOption = + labelOptions.style && labelOptions.style.textOverflow, + commonWidth, + commonTextOverflow, + maxLabelLength = 0, + label, + i, + pos; + // Set rotation option unless it is "auto", like in gauges + if (!isString(labelOptions.rotation)) { + // #4443: + attr.rotation = labelOptions.rotation || 0; + } + // Get the longest label length + tickPositions.forEach(function (tick) { + tick = ticks[tick]; + // Replace label - sorting animation + if (tick.movedLabel) { + tick.replaceMovedLabel(); + } + if ( + tick && + tick.label && + tick.label.textPxLength > maxLabelLength + ) { + maxLabelLength = tick.label.textPxLength; + } + }); + this.maxLabelLength = maxLabelLength; + // Handle auto rotation on horizontal axis + if (this.autoRotation) { + // Apply rotation only if the label is too wide for the slot, and + // the label is wider than its height. + if ( + maxLabelLength > innerWidth && + maxLabelLength > labelMetrics.h + ) { + attr.rotation = this.labelRotation; + } else { + this.labelRotation = 0; + } + // Handle word-wrap or ellipsis on vertical axis + } else if (slotWidth) { + // For word-wrap or ellipsis + commonWidth = innerWidth; + if (!textOverflowOption) { + commonTextOverflow = "clip"; + // On vertical axis, only allow word wrap if there is room + // for more lines. + i = tickPositions.length; + while (!horiz && i--) { + pos = tickPositions[i]; + label = ticks[pos].label; + if (label) { + // Reset ellipsis in order to get the correct + // bounding box (#4070) + if ( + label.styles && + label.styles.textOverflow === "ellipsis" + ) { + label.css({ textOverflow: "clip" }); + // Set the correct width in order to read + // the bounding box height (#4678, #5034) + } else if (label.textPxLength > slotWidth) { + label.css({ width: slotWidth + "px" }); + } + if ( + label.getBBox().height > + this.len / tickPositions.length - + (labelMetrics.h - labelMetrics.f) + ) { + label.specificTextOverflow = "ellipsis"; + } + } + } + } + } + // Add ellipsis if the label length is significantly longer than ideal + if (attr.rotation) { + commonWidth = + maxLabelLength > chart.chartHeight * 0.5 + ? chart.chartHeight * 0.33 + : maxLabelLength; + if (!textOverflowOption) { + commonTextOverflow = "ellipsis"; + } + } + // Set the explicit or automatic label alignment + this.labelAlign = + labelOptions.align || this.autoLabelAlign(this.labelRotation); + if (this.labelAlign) { + attr.align = this.labelAlign; + } + // Apply general and specific CSS + tickPositions.forEach(function (pos) { + var tick = ticks[pos], + label = tick && tick.label, + widthOption = labelStyleOptions.width, + css = {}; + if (label) { + // This needs to go before the CSS in old IE (#4502) + label.attr(attr); + if (tick.shortenLabel) { + tick.shortenLabel(); + } else if ( + commonWidth && + !widthOption && + // Setting width in this case messes with the bounding box + // (#7975) + labelStyleOptions.whiteSpace !== "nowrap" && + // Speed optimizing, #7656 + (commonWidth < label.textPxLength || + // Resetting CSS, #4928 + label.element.tagName === "SPAN") + ) { + css.width = commonWidth + "px"; + if (!textOverflowOption) { + css.textOverflow = + label.specificTextOverflow || commonTextOverflow; + } + label.css(css); + // Reset previously shortened label (#8210) + } else if ( + label.styles && + label.styles.width && + !css.width && + !widthOption + ) { + label.css({ width: null }); + } + delete label.specificTextOverflow; + tick.rotation = attr.rotation; + } + }, this); + // Note: Why is this not part of getLabelPosition? + this.tickRotCorr = renderer.rotCorr( + labelMetrics.b, + this.labelRotation || 0, + this.side !== 0 + ); + }; + /** + * Return true if the axis has associated data. + * + * @function Highcharts.Axis#hasData + * + * @return {boolean} + * True if the axis has associated visible series and those series have + * either valid data points or explicit `min` and `max` settings. + */ + Axis.prototype.hasData = function () { + return ( + this.series.some(function (s) { + return s.hasData(); + }) || + (this.options.showEmpty && defined(this.min) && defined(this.max)) + ); + }; + /** + * Adds the title defined in axis.options.title. + * + * @function Highcharts.Axis#addTitle + * + * @param {boolean} [display] + * Whether or not to display the title. + */ + Axis.prototype.addTitle = function (display) { + var axis = this, + renderer = axis.chart.renderer, + horiz = axis.horiz, + opposite = axis.opposite, + options = axis.options, + axisTitleOptions = options.title, + textAlign, + styledMode = axis.chart.styledMode; + if (!axis.axisTitle) { + textAlign = axisTitleOptions.textAlign; + if (!textAlign) { + textAlign = ( + horiz + ? { + low: "left", + middle: "center", + high: "right", + } + : { + low: opposite ? "right" : "left", + middle: "center", + high: opposite ? "left" : "right", + } + )[axisTitleOptions.align]; + } + axis.axisTitle = renderer + .text(axisTitleOptions.text, 0, 0, axisTitleOptions.useHTML) + .attr({ + zIndex: 7, + rotation: axisTitleOptions.rotation || 0, + align: textAlign, + }) + .addClass("highcharts-axis-title"); + // #7814, don't mutate style option + if (!styledMode) { + axis.axisTitle.css(merge(axisTitleOptions.style)); + } + axis.axisTitle.add(axis.axisGroup); + axis.axisTitle.isNew = true; + } + // Max width defaults to the length of the axis + if (!styledMode && !axisTitleOptions.style.width && !axis.isRadial) { + axis.axisTitle.css({ + width: axis.len + "px", + }); + } + // hide or show the title depending on whether showEmpty is set + axis.axisTitle[display ? "show" : "hide"](display); + }; + /** + * Generates a tick for initial positioning. + * + * @private + * @function Highcharts.Axis#generateTick + * + * @param {number} pos + * The tick position in axis values. + * + * @param {number} [i] + * The index of the tick in {@link Axis.tickPositions}. + */ + Axis.prototype.generateTick = function (pos) { + var axis = this; + var ticks = axis.ticks; + if (!ticks[pos]) { + ticks[pos] = new Tick(axis, pos); + } else { + ticks[pos].addLabel(); // update labels depending on tick interval + } + }; + /** + * Render the tick labels to a preliminary position to get their sizes + * + * @private + * @function Highcharts.Axis#getOffset + * + * @fires Highcharts.Axis#event:afterGetOffset + */ + Axis.prototype.getOffset = function () { + var axis = this, + chart = axis.chart, + renderer = chart.renderer, + options = axis.options, + tickPositions = axis.tickPositions, + ticks = axis.ticks, + horiz = axis.horiz, + side = axis.side, + invertedSide = + chart.inverted && !axis.isZAxis ? [1, 0, 3, 2][side] : side, + hasData, + showAxis, + titleOffset = 0, + titleOffsetOption, + titleMargin = 0, + axisTitleOptions = options.title, + labelOptions = options.labels, + labelOffset = 0, // reset + labelOffsetPadded, + axisOffset = chart.axisOffset, + clipOffset = chart.clipOffset, + clip, + directionFactor = [-1, 1, 1, -1][side], + className = options.className, + axisParent = axis.axisParent, // Used in color axis + lineHeightCorrection; + // For reuse in Axis.render + hasData = axis.hasData(); + axis.showAxis = showAxis = hasData || pick(options.showEmpty, true); + // Set/reset staggerLines + axis.staggerLines = axis.horiz && labelOptions.staggerLines; + // Create the axisGroup and gridGroup elements on first iteration + if (!axis.axisGroup) { + axis.gridGroup = renderer + .g("grid") + .attr({ zIndex: options.gridZIndex || 1 }) + .addClass( + "highcharts-" + + this.coll.toLowerCase() + + "-grid " + + (className || "") + ) + .add(axisParent); + axis.axisGroup = renderer + .g("axis") + .attr({ zIndex: options.zIndex || 2 }) + .addClass( + "highcharts-" + + this.coll.toLowerCase() + + " " + + (className || "") + ) + .add(axisParent); + axis.labelGroup = renderer + .g("axis-labels") + .attr({ zIndex: labelOptions.zIndex || 7 }) + .addClass( + "highcharts-" + + axis.coll.toLowerCase() + + "-labels " + + (className || "") + ) + .add(axisParent); + } + if (hasData || axis.isLinked) { + // Generate ticks + tickPositions.forEach(function (pos, i) { + // i is not used here, but may be used in overrides + axis.generateTick(pos, i); + }); + axis.renderUnsquish(); + // Left side must be align: right and right side must + // have align: left for labels + axis.reserveSpaceDefault = + side === 0 || + side === 2 || + { 1: "left", 3: "right" }[side] === axis.labelAlign; + if ( + pick( + labelOptions.reserveSpace, + axis.labelAlign === "center" ? true : null, + axis.reserveSpaceDefault + ) + ) { + tickPositions.forEach(function (pos) { + // get the highest offset + labelOffset = Math.max(ticks[pos].getLabelSize(), labelOffset); + }); + } + if (axis.staggerLines) { + labelOffset *= axis.staggerLines; + } + axis.labelOffset = labelOffset * (axis.opposite ? -1 : 1); + } else { + // doesn't have data + objectEach(ticks, function (tick, n) { + tick.destroy(); + delete ticks[n]; + }); + } + if ( + axisTitleOptions && + axisTitleOptions.text && + axisTitleOptions.enabled !== false + ) { + axis.addTitle(showAxis); + if (showAxis && axisTitleOptions.reserveSpace !== false) { + axis.titleOffset = titleOffset = + axis.axisTitle.getBBox()[horiz ? "height" : "width"]; + titleOffsetOption = axisTitleOptions.offset; + titleMargin = defined(titleOffsetOption) + ? 0 + : pick(axisTitleOptions.margin, horiz ? 5 : 10); + } + } + // Render the axis line + axis.renderLine(); + // handle automatic or user set offset + axis.offset = + directionFactor * + pick( + options.offset, + axisOffset[side] ? axisOffset[side] + (options.margin || 0) : 0 + ); + axis.tickRotCorr = axis.tickRotCorr || { x: 0, y: 0 }; // polar + if (side === 0) { + lineHeightCorrection = -axis.labelMetrics().h; + } else if (side === 2) { + lineHeightCorrection = axis.tickRotCorr.y; + } else { + lineHeightCorrection = 0; + } + // Find the padded label offset + labelOffsetPadded = Math.abs(labelOffset) + titleMargin; + if (labelOffset) { + labelOffsetPadded -= lineHeightCorrection; + labelOffsetPadded += + directionFactor * + (horiz + ? pick(labelOptions.y, axis.tickRotCorr.y + directionFactor * 8) + : labelOptions.x); + } + axis.axisTitleMargin = pick(titleOffsetOption, labelOffsetPadded); + if (axis.getMaxLabelDimensions) { + axis.maxLabelDimensions = axis.getMaxLabelDimensions( + ticks, + tickPositions + ); + } + // Due to GridAxis.tickSize, tickSize should be calculated after ticks + // has rendered. + var tickSize = this.tickSize("tick"); + axisOffset[side] = Math.max( + axisOffset[side], + axis.axisTitleMargin + titleOffset + directionFactor * axis.offset, + labelOffsetPadded, // #3027 + tickPositions && tickPositions.length && tickSize + ? tickSize[0] + directionFactor * axis.offset + : 0 // #4866 + ); + // Decide the clipping needed to keep the graph inside + // the plot area and axis lines + clip = options.offset + ? 0 + : // #4308, #4371: + Math.floor(axis.axisLine.strokeWidth() / 2) * 2; + clipOffset[invertedSide] = Math.max(clipOffset[invertedSide], clip); + fireEvent(this, "afterGetOffset"); + }; + /** + * Internal function to get the path for the axis line. Extended for polar + * charts. + * + * @function Highcharts.Axis#getLinePath + * + * @param {number} lineWidth + * The line width in pixels. + * + * @return {Highcharts.SVGPathArray} + * The SVG path definition in array form. + */ + Axis.prototype.getLinePath = function (lineWidth) { + var chart = this.chart, + opposite = this.opposite, + offset = this.offset, + horiz = this.horiz, + lineLeft = this.left + (opposite ? this.width : 0) + offset, + lineTop = + chart.chartHeight - + this.bottom - + (opposite ? this.height : 0) + + offset; + if (opposite) { + lineWidth *= -1; // crispify the other way - #1480, #1687 + } + return chart.renderer.crispLine( + [ + ["M", horiz ? this.left : lineLeft, horiz ? lineTop : this.top], + [ + "L", + horiz ? chart.chartWidth - this.right : lineLeft, + horiz ? lineTop : chart.chartHeight - this.bottom, + ], + ], + lineWidth + ); + }; + /** + * Render the axis line. Called internally when rendering and redrawing the + * axis. + * + * @function Highcharts.Axis#renderLine + */ + Axis.prototype.renderLine = function () { + if (!this.axisLine) { + this.axisLine = this.chart.renderer + .path() + .addClass("highcharts-axis-line") + .add(this.axisGroup); + if (!this.chart.styledMode) { + this.axisLine.attr({ + stroke: this.options.lineColor, + "stroke-width": this.options.lineWidth, + zIndex: 7, + }); + } + } + }; + /** + * Position the axis title. + * + * @private + * @function Highcharts.Axis#getTitlePosition + * + * @return {Highcharts.PositionObject} + * X and Y positions for the title. + */ + Axis.prototype.getTitlePosition = function () { + // compute anchor points for each of the title align options + var horiz = this.horiz, + axisLeft = this.left, + axisTop = this.top, + axisLength = this.len, + axisTitleOptions = this.options.title, + margin = horiz ? axisLeft : axisTop, + opposite = this.opposite, + offset = this.offset, + xOption = axisTitleOptions.x || 0, + yOption = axisTitleOptions.y || 0, + axisTitle = this.axisTitle, + fontMetrics = this.chart.renderer.fontMetrics( + axisTitleOptions.style && axisTitleOptions.style.fontSize, + axisTitle + ), + // The part of a multiline text that is below the baseline of the + // first line. Subtract 1 to preserve pixel-perfectness from the + // old behaviour (v5.0.12), where only one line was allowed. + textHeightOvershoot = Math.max( + axisTitle.getBBox(null, 0).height - fontMetrics.h - 1, + 0 + ), + // the position in the length direction of the axis + alongAxis = { + low: margin + (horiz ? 0 : axisLength), + middle: margin + axisLength / 2, + high: margin + (horiz ? axisLength : 0), + }[axisTitleOptions.align], + // the position in the perpendicular direction of the axis + offAxis = + (horiz ? axisTop + this.height : axisLeft) + + (horiz ? 1 : -1) * // horizontal axis reverses the margin + (opposite ? -1 : 1) * // so does opposite axes + this.axisTitleMargin + + [ + -textHeightOvershoot, + textHeightOvershoot, + fontMetrics.f, + -textHeightOvershoot, // left + ][this.side], + titlePosition = { + x: horiz + ? alongAxis + xOption + : offAxis + (opposite ? this.width : 0) + offset + xOption, + y: horiz + ? offAxis + yOption - (opposite ? this.height : 0) + offset + : alongAxis + yOption, + }; + fireEvent(this, "afterGetTitlePosition", { + titlePosition: titlePosition, + }); + return titlePosition; + }; + /** + * Render a minor tick into the given position. If a minor tick already + * exists in this position, move it. + * + * @function Highcharts.Axis#renderMinorTick + * + * @param {number} pos + * The position in axis values. + */ + Axis.prototype.renderMinorTick = function (pos) { + var axis = this; + var slideInTicks = axis.chart.hasRendered && isNumber(axis.oldMin); + var minorTicks = axis.minorTicks; + if (!minorTicks[pos]) { + minorTicks[pos] = new Tick(axis, pos, "minor"); + } + // Render new ticks in old position + if (slideInTicks && minorTicks[pos].isNew) { + minorTicks[pos].render(null, true); + } + minorTicks[pos].render(null, false, 1); + }; + /** + * Render a major tick into the given position. If a tick already exists + * in this position, move it. + * + * @function Highcharts.Axis#renderTick + * + * @param {number} pos + * The position in axis values. + * + * @param {number} i + * The tick index. + */ + Axis.prototype.renderTick = function (pos, i) { + var _a; + var axis = this; + var isLinked = axis.isLinked; + var ticks = axis.ticks; + var slideInTicks = axis.chart.hasRendered && isNumber(axis.oldMin); + // Linked axes need an extra check to find out if + if ( + !isLinked || + (pos >= axis.min && pos <= axis.max) || + ((_a = axis.grid) === null || _a === void 0 ? void 0 : _a.isColumn) + ) { + if (!ticks[pos]) { + ticks[pos] = new Tick(axis, pos); + } + // NOTE this seems like overkill. Could be handled in tick.render by + // setting old position in attr, then set new position in animate. + // render new ticks in old position + if (slideInTicks && ticks[pos].isNew) { + // Start with negative opacity so that it is visible from + // halfway into the animation + ticks[pos].render(i, true, -1); + } + ticks[pos].render(i); + } + }; + /** + * Render the axis. + * + * @private + * @function Highcharts.Axis#render + * + * @fires Highcharts.Axis#event:afterRender + */ + Axis.prototype.render = function () { + var axis = this, + chart = axis.chart, + log = axis.logarithmic, + renderer = chart.renderer, + options = axis.options, + isLinked = axis.isLinked, + tickPositions = axis.tickPositions, + axisTitle = axis.axisTitle, + ticks = axis.ticks, + minorTicks = axis.minorTicks, + alternateBands = axis.alternateBands, + stackLabelOptions = options.stackLabels, + alternateGridColor = options.alternateGridColor, + tickmarkOffset = axis.tickmarkOffset, + axisLine = axis.axisLine, + showAxis = axis.showAxis, + animation = animObject(renderer.globalAnimation), + from, + to; + // Reset + axis.labelEdge.length = 0; + axis.overlap = false; + // Mark all elements inActive before we go over and mark the active ones + [ticks, minorTicks, alternateBands].forEach(function (coll) { + objectEach(coll, function (tick) { + tick.isActive = false; + }); + }); + // If the series has data draw the ticks. Else only the line and title + if (axis.hasData() || isLinked) { + // minor ticks + if (axis.minorTickInterval && !axis.categories) { + axis.getMinorTickPositions().forEach(function (pos) { + axis.renderMinorTick(pos); + }); + } + // Major ticks. Pull out the first item and render it last so that + // we can get the position of the neighbour label. #808. + if (tickPositions.length) { + // #1300 + tickPositions.forEach(function (pos, i) { + axis.renderTick(pos, i); + }); + // In a categorized axis, the tick marks are displayed + // between labels. So we need to add a tick mark and + // grid line at the left edge of the X axis. + if (tickmarkOffset && (axis.min === 0 || axis.single)) { + if (!ticks[-1]) { + ticks[-1] = new Tick(axis, -1, null, true); + } + ticks[-1].render(-1); + } + } + // alternate grid color + if (alternateGridColor) { + tickPositions.forEach(function (pos, i) { + to = + typeof tickPositions[i + 1] !== "undefined" + ? tickPositions[i + 1] + tickmarkOffset + : axis.max - tickmarkOffset; + if ( + i % 2 === 0 && + pos < axis.max && + to <= + axis.max + (chart.polar ? -tickmarkOffset : tickmarkOffset) + ) { + // #2248, #4660 + if (!alternateBands[pos]) { + // Should be imported from PlotLineOrBand.js, but + // the dependency cycle with axis is a problem + alternateBands[pos] = new H.PlotLineOrBand(axis); + } + from = pos + tickmarkOffset; // #949 + alternateBands[pos].options = { + from: log ? log.lin2log(from) : from, + to: log ? log.lin2log(to) : to, + color: alternateGridColor, + className: "highcharts-alternate-grid", + }; + alternateBands[pos].render(); + alternateBands[pos].isActive = true; + } + }); + } + // custom plot lines and bands + if (!axis._addedPlotLB) { + // only first time + (options.plotLines || []) + .concat(options.plotBands || []) + .forEach(function (plotLineOptions) { + axis.addPlotBandOrLine(plotLineOptions); }); + axis._addedPlotLB = true; + } + } // end if hasData + // Remove inactive ticks + [ticks, minorTicks, alternateBands].forEach(function (coll) { + var i, + forDestruction = [], + delay = animation.duration, + destroyInactiveItems = function () { + i = forDestruction.length; while (i--) { - alignedObjects[i].align(); - } - }; - /** - * Create and return an svg group element. Child - * {@link Highcharts.SVGElement} objects are added to the group by using the - * group as the first parameter in {@link Highcharts.SVGElement#add|add()}. - * - * @function Highcharts.SVGRenderer#g - * - * @param {string} [name] - * The group will be given a class name of `highcharts-{name}`. This - * can be used for styling and scripting. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ - SVGRenderer.prototype.g = function (name) { - var elem = this.createElement('g'); - return name ? - elem.attr({ 'class': 'highcharts-' + name }) : - elem; - }; - /** - * Display an image. - * - * @sample highcharts/members/renderer-image-on-chart/ - * Add an image in a chart - * @sample highcharts/members/renderer-image/ - * Add an image independent of a chart - * - * @function Highcharts.SVGRenderer#image - * - * @param {string} src - * The image source. - * - * @param {number} [x] - * The X position. - * - * @param {number} [y] - * The Y position. - * - * @param {number} [width] - * The image width. If omitted, it defaults to the image file width. - * - * @param {number} [height] - * The image height. If omitted it defaults to the image file - * height. - * - * @param {Function} [onload] - * Event handler for image load. - * - * @return {Highcharts.SVGElement} - * The generated wrapper element. - */ - SVGRenderer.prototype.image = function (src, x, y, width, height, onload) { - var attribs = { preserveAspectRatio: 'none' }, elemWrapper, dummy, setSVGImageSource = function (el, src) { - // Set the href in the xlink namespace - if (el.setAttributeNS) { - el.setAttributeNS('http://www.w3.org/1999/xlink', 'href', src); - } - else { - // could be exporting in IE - // using href throws "not supported" in ie7 and under, - // requries regex shim to fix later - el.setAttribute('hc-svg-href', src); - } - }, onDummyLoad = function (e) { - setSVGImageSource(elemWrapper.element, src); - onload.call(elemWrapper, e); - }; - // optional properties - if (arguments.length > 1) { - extend(attribs, { - x: x, - y: y, - width: width, - height: height - }); - } - elemWrapper = this.createElement('image').attr(attribs); - // Add load event if supplied - if (onload) { - // We have to use a dummy HTML image since IE support for SVG image - // load events is very buggy. First set a transparent src, wait for - // dummy to load, and then add the real src to the SVG image. - setSVGImageSource(elemWrapper.element, '' /* eslint-disable-line */); - dummy = new win.Image(); - addEvent(dummy, 'load', onDummyLoad); - dummy.src = src; - if (dummy.complete) { - onDummyLoad({}); - } - } - else { - setSVGImageSource(elemWrapper.element, src); - } - return elemWrapper; - }; - /** - * Draw a symbol out of pre-defined shape paths from - * {@link SVGRenderer#symbols}. - * It is used in Highcharts for point makers, which cake a `symbol` option, - * and label and button backgrounds like in the tooltip and stock flags. - * - * @function Highcharts.SVGRenderer#symbol - * - * @param {string} symbol - * The symbol name. - * - * @param {number} [x] - * The X coordinate for the top left position. - * - * @param {number} [y] - * The Y coordinate for the top left position. - * - * @param {number} [width] - * The pixel width. - * - * @param {number} [height] - * The pixel height. - * - * @param {Highcharts.SymbolOptionsObject} [options] - * Additional options, depending on the actual symbol drawn. - * - * @return {Highcharts.SVGElement} - */ - SVGRenderer.prototype.symbol = function (symbol, x, y, width, height, options) { - var ren = this, - obj, - imageRegex = /^url\((.*?)\)$/, - isImage = imageRegex.test(symbol), - sym = (!isImage && (this.symbols[symbol] ? symbol : 'circle')), - // get the symbol definition function - symbolFn = (sym && this.symbols[sym]), - path, - imageSrc, - centerImage; - if (symbolFn) { - // Check if there's a path defined for this symbol - if (typeof x === 'number') { - path = symbolFn.call(this.symbols, Math.round(x || 0), Math.round(y || 0), width || 0, height || 0, options); - } - obj = this.path(path); - if (!ren.styledMode) { - obj.attr('fill', 'none'); - } - // expando properties for use in animate and attr - extend(obj, { - symbolName: sym, - x: x, - y: y, - width: width, - height: height - }); - if (options) { - extend(obj, options); - } - // Image symbols - } - else if (isImage) { - imageSrc = symbol.match(imageRegex)[1]; - // Create the image synchronously, add attribs async - obj = this.image(imageSrc); - // The image width is not always the same as the symbol width. The - // image may be centered within the symbol, as is the case when - // image shapes are used as label backgrounds, for example in flags. - obj.imgwidth = pick(symbolSizes[imageSrc] && symbolSizes[imageSrc].width, options && options.width); - obj.imgheight = pick(symbolSizes[imageSrc] && symbolSizes[imageSrc].height, options && options.height); - /** - * Set the size and position - */ - centerImage = function () { - obj.attr({ - width: obj.width, - height: obj.height - }); - }; - /** - * Width and height setters that take both the image's physical size - * and the label size into consideration, and translates the image - * to center within the label. - */ - ['width', 'height'].forEach(function (key) { - obj[key + 'Setter'] = function (value, key) { - var attribs = {}, imgSize = this['img' + key], trans = key === 'width' ? 'translateX' : 'translateY'; - this[key] = value; - if (defined(imgSize)) { - // Scale and center the image within its container. - // The name `backgroundSize` is taken from the CSS spec, - // but the value `within` is made up. Other possible - // values in the spec, `cover` and `contain`, can be - // implemented if needed. - if (options && - options.backgroundSize === 'within' && - this.width && - this.height) { - imgSize = Math.round(imgSize * Math.min(this.width / this.imgwidth, this.height / this.imgheight)); - } - if (this.element) { - this.element.setAttribute(key, imgSize); - } - if (!this.alignByTranslate) { - attribs[trans] = ((this[key] || 0) - imgSize) / 2; - this.attr(attribs); - } - } - }; - }); - if (defined(x)) { - obj.attr({ - x: x, - y: y - }); - } - obj.isImg = true; - if (defined(obj.imgwidth) && defined(obj.imgheight)) { - centerImage(); - } - else { - // Initialize image to be 0 size so export will still function - // if there's no cached sizes. - obj.attr({ width: 0, height: 0 }); - // Create a dummy JavaScript image to get the width and height. - createElement('img', { - onload: function () { - var chart = charts[ren.chartIndex]; - // Special case for SVGs on IE11, the width is not - // accessible until the image is part of the DOM - // (#2854). - if (this.width === 0) { - css(this, { - position: 'absolute', - top: '-999em' - }); - doc.body.appendChild(this); - } - // Center the image - symbolSizes[imageSrc] = { - width: this.width, - height: this.height - }; - obj.imgwidth = this.width; - obj.imgheight = this.height; - if (obj.element) { - centerImage(); - } - // Clean up after #2854 workaround. - if (this.parentNode) { - this.parentNode.removeChild(this); - } - // Fire the load event when all external images are - // loaded - ren.imgCount--; - if (!ren.imgCount && chart && !chart.hasLoaded) { - chart.onload(); - } - }, - src: imageSrc - }); - this.imgCount++; - } - } - return obj; - }; - /** - * Define a clipping rectangle. The clipping rectangle is later applied - * to {@link SVGElement} objects through the {@link SVGElement#clip} - * function. - * - * @example - * var circle = renderer.circle(100, 100, 100) - * .attr({ fill: 'red' }) - * .add(); - * var clipRect = renderer.clipRect(100, 100, 100, 100); - * - * // Leave only the lower right quarter visible - * circle.clip(clipRect); - * - * @function Highcharts.SVGRenderer#clipRect - * - * @param {number} [x] - * - * @param {number} [y] - * - * @param {number} [width] - * - * @param {number} [height] - * - * @return {Highcharts.ClipRectElement} - * A clipping rectangle. - */ - SVGRenderer.prototype.clipRect = function (x, y, width, height) { - var wrapper, - // Add a hyphen at the end to avoid confusion in testing indexes - // -1 and -10, -11 etc (#6550) - id = uniqueKey() + '-', clipPath = this.createElement('clipPath').attr({ - id: id - }).add(this.defs); - wrapper = this.rect(x, y, width, height, 0).add(clipPath); - wrapper.id = id; - wrapper.clipPath = clipPath; - wrapper.count = 0; - return wrapper; - }; - /** - * Draw text. The text can contain a subset of HTML, like spans and anchors - * and some basic text styling of these. For more advanced features like - * border and background, use {@link Highcharts.SVGRenderer#label} instead. - * To update the text after render, run `text.attr({ text: 'New text' })`. - * - * @sample highcharts/members/renderer-text-on-chart/ - * Annotate the chart freely - * @sample highcharts/members/renderer-on-chart/ - * Annotate with a border and in response to the data - * @sample highcharts/members/renderer-text/ - * Formatted text - * - * @function Highcharts.SVGRenderer#text - * - * @param {string} [str] - * The text of (subset) HTML to draw. - * - * @param {number} [x] - * The x position of the text's lower left corner. - * - * @param {number} [y] - * The y position of the text's lower left corner. - * - * @param {boolean} [useHTML=false] - * Use HTML to render the text. - * - * @return {Highcharts.SVGElement} - * The text object. - */ - SVGRenderer.prototype.text = function (str, x, y, useHTML) { - // declare variables - var renderer = this, - wrapper, - attribs = {}; - if (useHTML && (renderer.allowHTML || !renderer.forExport)) { - return renderer.html(str, x, y); - } - attribs.x = Math.round(x || 0); // X always needed for line-wrap logic - if (y) { - attribs.y = Math.round(y); - } - if (defined(str)) { - attribs.text = str; - } - wrapper = renderer.createElement('text') - .attr(attribs); - if (!useHTML) { - wrapper.xSetter = function (value, key, element) { - var tspans = element.getElementsByTagName('tspan'), - tspan, - parentVal = element.getAttribute(key), - i; - for (i = 0; i < tspans.length; i++) { - tspan = tspans[i]; - // If the x values are equal, the tspan represents a - // linebreak - if (tspan.getAttribute(key) === parentVal) { - tspan.setAttribute(key, value); - } - } - element.setAttribute(key, value); - }; - } - return wrapper; - }; - /** - * Utility to return the baseline offset and total line height from the font - * size. - * - * @function Highcharts.SVGRenderer#fontMetrics - * - * @param {number|string} [fontSize] - * The current font size to inspect. If not given, the font size - * will be found from the DOM element. - * - * @param {Highcharts.SVGElement|Highcharts.SVGDOMElement} [elem] - * The element to inspect for a current font size. - * - * @return {Highcharts.FontMetricsObject} - * The font metrics. - */ - SVGRenderer.prototype.fontMetrics = function (fontSize, elem) { - var lineHeight, - baseline; - if ((this.styledMode || !/px/.test(fontSize)) && - win.getComputedStyle // old IE doesn't support it - ) { - fontSize = elem && SVGElement.prototype.getStyle.call(elem, 'font-size'); - } - else { - fontSize = fontSize || - // When the elem is a DOM element (#5932) - (elem && elem.style && elem.style.fontSize) || - // Fall back on the renderer style default - (this.style && this.style.fontSize); - } - // Handle different units - if (/px/.test(fontSize)) { - fontSize = pInt(fontSize); - } - else { - fontSize = 12; - } - // Empirical values found by comparing font size and bounding box - // height. Applies to the default font family. - // https://jsfiddle.net/highcharts/7xvn7/ - lineHeight = fontSize < 24 ? fontSize + 3 : Math.round(fontSize * 1.2); - baseline = Math.round(lineHeight * 0.8); - return { - h: lineHeight, - b: baseline, - f: fontSize - }; - }; - /** - * Correct X and Y positioning of a label for rotation (#1764). - * - * @private - * @function Highcharts.SVGRenderer#rotCorr - * - * @param {number} baseline - * - * @param {number} rotation - * - * @param {boolean} [alterY] - * - * @param {Highcharts.PositionObject} - */ - SVGRenderer.prototype.rotCorr = function (baseline, rotation, alterY) { - var y = baseline; - if (rotation && alterY) { - y = Math.max(y * Math.cos(rotation * deg2rad), 4); - } - return { - x: (-baseline / 3) * Math.sin(rotation * deg2rad), - y: y - }; - }; - /** - * Compatibility function to convert the legacy one-dimensional path array - * into an array of segments. - * - * It is used in maps to parse the `path` option, and in SVGRenderer.dSetter - * to support legacy paths from demos. - * - * @private - * @function Highcharts.SVGRenderer#pathToSegments - */ - SVGRenderer.prototype.pathToSegments = function (path) { - var ret = []; - var segment = []; - var commandLength = { - A: 8, - C: 7, - H: 2, - L: 3, - M: 3, - Q: 5, - S: 5, - T: 3, - V: 2 - }; - // Short, non-typesafe parsing of the one-dimensional array. It splits - // the path on any string. This is not type checked against the tuple - // types, but is shorter, and doesn't require specific checks for any - // command type in SVG. - for (var i = 0; i < path.length; i++) { - // Command skipped, repeat previous or insert L/l for M/m - if (isString(segment[0]) && - isNumber(path[i]) && - segment.length === commandLength[(segment[0].toUpperCase())]) { - path.splice(i, 0, segment[0].replace('M', 'L').replace('m', 'l')); - } - // Split on string - if (typeof path[i] === 'string') { - if (segment.length) { - ret.push(segment.slice(0)); - } - segment.length = 0; - } - segment.push(path[i]); - } - ret.push(segment.slice(0)); - return ret; - /* - // Fully type-safe version where each tuple type is checked. The - // downside is filesize and a lack of flexibility for unsupported - // commands - const ret: SVGPath = [], - commands = { - A: 7, - C: 6, - H: 1, - L: 2, - M: 2, - Q: 4, - S: 4, - T: 2, - V: 1, - Z: 0 - }; - - let i = 0, - lastI = 0, - lastCommand; - - while (i < path.length) { - const item = path[i]; - - let command; - - if (typeof item === 'string') { - command = item; - i += 1; - } else { - command = lastCommand || 'M'; - } - - // Upper case - const commandUC = command.toUpperCase(); - - if (commandUC in commands) { - - // No numeric parameters - if (command === 'Z' || command === 'z') { - ret.push([command]); - - // One numeric parameter - } else { - const val0 = path[i]; - if (typeof val0 === 'number') { - - // Horizontal line to - if (command === 'H' || command === 'h') { - ret.push([command, val0]); - i += 1; - - // Vertical line to - } else if (command === 'V' || command === 'v') { - ret.push([command, val0]); - i += 1; - - // Two numeric parameters - } else { - const val1 = path[i + 1]; - if (typeof val1 === 'number') { - // lineTo - if (command === 'L' || command === 'l') { - ret.push([command, val0, val1]); - i += 2; - - // moveTo - } else if (command === 'M' || command === 'm') { - ret.push([command, val0, val1]); - i += 2; - - // Smooth quadratic bezier - } else if (command === 'T' || command === 't') { - ret.push([command, val0, val1]); - i += 2; - - // Four numeric parameters - } else { - const val2 = path[i + 2], - val3 = path[i + 3]; - if ( - typeof val2 === 'number' && - typeof val3 === 'number' - ) { - // Quadratic bezier to - if ( - command === 'Q' || - command === 'q' - ) { - ret.push([ - command, - val0, - val1, - val2, - val3 - ]); - i += 4; - - // Smooth cubic bezier to - } else if ( - command === 'S' || - command === 's' - ) { - ret.push([ - command, - val0, - val1, - val2, - val3 - ]); - i += 4; - - // Six numeric parameters - } else { - const val4 = path[i + 4], - val5 = path[i + 5]; - - if ( - typeof val4 === 'number' && - typeof val5 === 'number' - ) { - // Curve to - if ( - command === 'C' || - command === 'c' - ) { - ret.push([ - command, - val0, - val1, - val2, - val3, - val4, - val5 - ]); - i += 6; - - // Seven numeric parameters - } else { - const val6 = path[i + 6]; - - // Arc to - if ( - typeof val6 === - 'number' && - ( - command === 'A' || - command === 'a' - ) - ) { - ret.push([ - command, - val0, - val1, - val2, - val3, - val4, - val5, - val6 - ]); - i += 7; - - } - - } - } - } - } - } - } - - } - } - } - } - - // An unmarked command following a moveTo is a lineTo - lastCommand = command === 'M' ? 'L' : command; - - if (i === lastI) { - break; - } - lastI = i; - } - return ret; - */ - }; - /** - * Draw a label, which is an extended text element with support for border - * and background. Highcharts creates a `g` element with a text and a `path` - * or `rect` inside, to make it behave somewhat like a HTML div. Border and - * background are set through `stroke`, `stroke-width` and `fill` attributes - * using the {@link Highcharts.SVGElement#attr|attr} method. To update the - * text after render, run `label.attr({ text: 'New text' })`. - * - * @sample highcharts/members/renderer-label-on-chart/ - * A label on the chart - * - * @function Highcharts.SVGRenderer#label - * - * @param {string} str - * The initial text string or (subset) HTML to render. - * - * @param {number} x - * The x position of the label's left side. - * - * @param {number} [y] - * The y position of the label's top side or baseline, depending on - * the `baseline` parameter. - * - * @param {string} [shape='rect'] - * The shape of the label's border/background, if any. Defaults to - * `rect`. Other possible values are `callout` or other shapes - * defined in {@link Highcharts.SVGRenderer#symbols}. - * - * @param {number} [anchorX] - * In case the `shape` has a pointer, like a flag, this is the - * coordinates it should be pinned to. - * - * @param {number} [anchorY] - * In case the `shape` has a pointer, like a flag, this is the - * coordinates it should be pinned to. - * - * @param {boolean} [useHTML=false] - * Wether to use HTML to render the label. - * - * @param {boolean} [baseline=false] - * Whether to position the label relative to the text baseline, - * like {@link Highcharts.SVGRenderer#text|renderer.text}, or to the - * upper border of the rectangle. - * - * @param {string} [className] - * Class name for the group. - * - * @return {Highcharts.SVGElement} - * The generated label. - */ - SVGRenderer.prototype.label = function (str, x, y, shape, anchorX, anchorY, useHTML, baseline, className) { - return new SVGLabel(this, str, x, y, shape, anchorX, anchorY, useHTML, baseline, className); - }; - return SVGRenderer; - }()); + // When resizing rapidly, the same items + // may be destroyed in different timeouts, + // or the may be reactivated + if ( + coll[forDestruction[i]] && + !coll[forDestruction[i]].isActive + ) { + coll[forDestruction[i]].destroy(); + delete coll[forDestruction[i]]; + } + } + }; + objectEach(coll, function (tick, pos) { + if (!tick.isActive) { + // Render to zero opacity + tick.render(pos, false, 0); + tick.isActive = false; + forDestruction.push(pos); + } + }); + // When the objects are finished fading out, destroy them + syncTimeout( + destroyInactiveItems, + coll === alternateBands || !chart.hasRendered || !delay + ? 0 + : delay + ); + }); + // Set the axis line path + if (axisLine) { + axisLine[axisLine.isPlaced ? "animate" : "attr"]({ + d: this.getLinePath(axisLine.strokeWidth()), + }); + axisLine.isPlaced = true; + // Show or hide the line depending on options.showEmpty + axisLine[showAxis ? "show" : "hide"](showAxis); + } + if (axisTitle && showAxis) { + var titleXy = axis.getTitlePosition(); + if (isNumber(titleXy.y)) { + axisTitle[axisTitle.isNew ? "attr" : "animate"](titleXy); + axisTitle.isNew = false; + } else { + axisTitle.attr("y", -9999); + axisTitle.isNew = true; + } + } + // Stacked totals: + if (stackLabelOptions && stackLabelOptions.enabled && axis.stacking) { + axis.stacking.renderStackTotals(); + } + // End stacked totals + axis.isDirty = false; + fireEvent(this, "afterRender"); + }; /** - * A pointer to the renderer's associated Element class. The VMLRenderer - * will have a pointer to VMLElement here. + * Redraw the axis to reflect changes in the data or axis extremes. Called + * internally from Highcharts.Chart#redraw. * - * @name Highcharts.SVGRenderer#Element - * @type {Highcharts.SVGElement} - */ - SVGRenderer.prototype.Element = SVGElement; - /** * @private - */ - SVGRenderer.prototype.SVG_NS = SVG_NS; - /** - * Dummy function for plugins, called every time the renderer is updated. - * Prior to Highcharts 5, this was used for the canvg renderer. - * - * @deprecated - * @function Highcharts.SVGRenderer#draw - */ - SVGRenderer.prototype.draw = noop; + * @function Highcharts.Axis#redraw + */ + Axis.prototype.redraw = function () { + if (this.visible) { + // render the axis + this.render(); + // move plot lines and bands + this.plotLinesAndBands.forEach(function (plotLine) { + plotLine.render(); + }); + } + // mark associated series as dirty and ready for redraw + this.series.forEach(function (series) { + series.isDirty = true; + }); + }; /** - * A collection of characters mapped to HTML entities. When `useHTML` on an - * element is true, these entities will be rendered correctly by HTML. In - * the SVG pseudo-HTML, they need to be unescaped back to simple characters, - * so for example `<` will render as `<`. + * Returns an array of axis properties, that should be untouched during + * reinitialization. * - * @example - * // Add support for unescaping quotes - * Highcharts.SVGRenderer.prototype.escapes['"'] = '"'; + * @private + * @function Highcharts.Axis#getKeepProps * - * @name Highcharts.SVGRenderer#escapes - * @type {Highcharts.Dictionary<string>} + * @return {Array<string>} */ - SVGRenderer.prototype.escapes = { - '&': '&', - '<': '<', - '>': '>', - "'": ''', - '"': '"' + Axis.prototype.getKeepProps = function () { + return this.keepProps || Axis.keepProps; }; /** - * An extendable collection of functions for defining symbol paths. + * Destroys an Axis instance. See {@link Axis#remove} for the API endpoint + * to fully remove the axis. + * + * @private + * @function Highcharts.Axis#destroy * - * @name Highcharts.SVGRenderer#symbols - * @type {Highcharts.SymbolDictionary} + * @param {boolean} [keepEvents] + * Whether to preserve events, used internally in Axis.update. */ - SVGRenderer.prototype.symbols = { - circle: function (x, y, w, h) { - // Return a full arc - return this.arc(x + w / 2, y + h / 2, w / 2, h / 2, { - start: Math.PI * 0.5, - end: Math.PI * 2.5, - open: false - }); - }, - square: function (x, y, w, h) { - return [ - ['M', x, y], - ['L', x + w, y], - ['L', x + w, y + h], - ['L', x, y + h], - ['Z'] - ]; - }, - triangle: function (x, y, w, h) { - return [ - ['M', x + w / 2, y], - ['L', x + w, y + h], - ['L', x, y + h], - ['Z'] - ]; - }, - 'triangle-down': function (x, y, w, h) { - return [ - ['M', x, y], - ['L', x + w, y], - ['L', x + w / 2, y + h], - ['Z'] - ]; - }, - diamond: function (x, y, w, h) { - return [ - ['M', x + w / 2, y], - ['L', x + w, y + h / 2], - ['L', x + w / 2, y + h], - ['L', x, y + h / 2], - ['Z'] - ]; - }, - arc: function (x, y, w, h, options) { - var arc = []; - if (options) { - var start = options.start || 0, - end = options.end || 0, - rx = options.r || w, - ry = options.r || h || w, - proximity = 0.001, - fullCircle = Math.abs(end - start - 2 * Math.PI) < - proximity, - // Substract a small number to prevent cos and sin of start and - // end from becoming equal on 360 arcs (related: #1561) - end = end - proximity, - innerRadius = options.innerR, - open = pick(options.open, - fullCircle), - cosStart = Math.cos(start), - sinStart = Math.sin(start), - cosEnd = Math.cos(end), - sinEnd = Math.sin(end), - // Proximity takes care of rounding errors around PI (#6971) - longArc = pick(options.longArc, - end - start - Math.PI < proximity ? 0 : 1); - arc.push([ - 'M', - x + rx * cosStart, - y + ry * sinStart - ], [ - 'A', - rx, - ry, - 0, - longArc, - pick(options.clockwise, 1), - x + rx * cosEnd, - y + ry * sinEnd - ]); - if (defined(innerRadius)) { - arc.push(open ? - [ - 'M', - x + innerRadius * cosEnd, - y + innerRadius * sinEnd - ] : [ - 'L', - x + innerRadius * cosEnd, - y + innerRadius * sinEnd - ], [ - 'A', - innerRadius, - innerRadius, - 0, - longArc, - // Clockwise - opposite to the outer arc clockwise - defined(options.clockwise) ? 1 - options.clockwise : 0, - x + innerRadius * cosStart, - y + innerRadius * sinStart - ]); - } - if (!open) { - arc.push(['Z']); - } - } - return arc; - }, - /** - * Callout shape used for default tooltips, also used for rounded - * rectangles in VML - */ - callout: function (x, y, w, h, options) { - var arrowLength = 6, - halfDistance = 6, - r = Math.min((options && options.r) || 0, - w, - h), - safeDistance = r + halfDistance, - anchorX = options && options.anchorX || 0, - anchorY = options && options.anchorY || 0, - path; - path = [ - ['M', x + r, y], - ['L', x + w - r, y], - ['C', x + w, y, x + w, y, x + w, y + r], - ['L', x + w, y + h - r], - ['C', x + w, y + h, x + w, y + h, x + w - r, y + h], - ['L', x + r, y + h], - ['C', x, y + h, x, y + h, x, y + h - r], - ['L', x, y + r], - ['C', x, y, x, y, x + r, y] // top-left corner - ]; - // Anchor on right side - if (anchorX && anchorX > w) { - // Chevron - if (anchorY > y + safeDistance && - anchorY < y + h - safeDistance) { - path.splice(3, 1, ['L', x + w, anchorY - halfDistance], ['L', x + w + arrowLength, anchorY], ['L', x + w, anchorY + halfDistance], ['L', x + w, y + h - r]); - // Simple connector - } - else { - path.splice(3, 1, ['L', x + w, h / 2], ['L', anchorX, anchorY], ['L', x + w, h / 2], ['L', x + w, y + h - r]); - } - // Anchor on left side - } - else if (anchorX && anchorX < 0) { - // Chevron - if (anchorY > y + safeDistance && - anchorY < y + h - safeDistance) { - path.splice(7, 1, ['L', x, anchorY + halfDistance], ['L', x - arrowLength, anchorY], ['L', x, anchorY - halfDistance], ['L', x, y + r]); - // Simple connector - } - else { - path.splice(7, 1, ['L', x, h / 2], ['L', anchorX, anchorY], ['L', x, h / 2], ['L', x, y + r]); - } - } - else if ( // replace bottom - anchorY && - anchorY > h && - anchorX > x + safeDistance && - anchorX < x + w - safeDistance) { - path.splice(5, 1, ['L', anchorX + halfDistance, y + h], ['L', anchorX, y + h + arrowLength], ['L', anchorX - halfDistance, y + h], ['L', x + r, y + h]); - } - else if ( // replace top - anchorY && - anchorY < 0 && - anchorX > x + safeDistance && - anchorX < x + w - safeDistance) { - path.splice(1, 1, ['L', anchorX - halfDistance, y], ['L', anchorX, y - arrowLength], ['L', anchorX + halfDistance, y], ['L', w - r, y]); - } - return path; + Axis.prototype.destroy = function (keepEvents) { + var axis = this, + plotLinesAndBands = axis.plotLinesAndBands, + plotGroup, + i; + fireEvent(this, "destroy", { keepEvents: keepEvents }); + // Remove the events + if (!keepEvents) { + removeEvent(axis); + } + // Destroy collections + [axis.ticks, axis.minorTicks, axis.alternateBands].forEach(function ( + coll + ) { + destroyObjectProperties(coll); + }); + if (plotLinesAndBands) { + i = plotLinesAndBands.length; + while (i--) { + // #1975 + plotLinesAndBands[i].destroy(); + } + } + // Destroy elements + [ + "axisLine", + "axisTitle", + "axisGroup", + "gridGroup", + "labelGroup", + "cross", + "scrollbar", + ].forEach(function (prop) { + if (axis[prop]) { + axis[prop] = axis[prop].destroy(); } + }); + // Destroy each generated group for plotlines and plotbands + for (plotGroup in axis.plotLinesAndBandsGroups) { + // eslint-disable-line guard-for-in + axis.plotLinesAndBandsGroups[plotGroup] = + axis.plotLinesAndBandsGroups[plotGroup].destroy(); + } + // Delete all properties and fall back to the prototype. + objectEach(axis, function (val, key) { + if (axis.getKeepProps().indexOf(key) === -1) { + delete axis[key]; + } + }); }; - H.SVGRenderer = SVGRenderer; - H.Renderer = H.SVGRenderer; - - return H.Renderer; - }); - _registerModule(_modules, 'Core/Renderer/HTML/HTMLElement.js', [_modules['Core/Globals.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Utilities.js']], function (H, SVGElement, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var css = U.css, - defined = U.defined, - extend = U.extend, - pick = U.pick, - pInt = U.pInt; /** - * Element placebo - * @private - */ - var HTMLElement = SVGElement; - var isFirefox = H.isFirefox; - /* eslint-disable valid-jsdoc */ - // Extend SvgElement for useHTML option. - extend(HTMLElement.prototype, /** @lends SVGElement.prototype */ { - /** - * Apply CSS to HTML elements. This is used in text within SVG rendering and - * by the VML renderer - * - * @private - * @function Highcharts.SVGElement#htmlCss - * - * @param {Highcharts.CSSObject} styles - * - * @return {Highcharts.SVGElement} - */ - htmlCss: function (styles) { - var wrapper = this, - element = wrapper.element, - // When setting or unsetting the width style, we need to update - // transform (#8809) - isSettingWidth = (element.tagName === 'SPAN' && - styles && - 'width' in styles), - textWidth = pick(isSettingWidth && styles.width, - void 0), - doTransform; - if (isSettingWidth) { - delete styles.width; - wrapper.textWidth = textWidth; - doTransform = true; - } - if (styles && styles.textOverflow === 'ellipsis') { - styles.whiteSpace = 'nowrap'; - styles.overflow = 'hidden'; - } - wrapper.styles = extend(wrapper.styles, styles); - css(wrapper.element, styles); - // Now that all styles are applied, to the transform - if (doTransform) { - wrapper.htmlUpdateTransform(); - } - return wrapper; - }, - /** - * VML and useHTML method for calculating the bounding box based on offsets. - * - * @private - * @function Highcharts.SVGElement#htmlGetBBox - * - * @param {boolean} refresh - * Whether to force a fresh value from the DOM or to use the cached - * value. - * - * @return {Highcharts.BBoxObject} - * A hash containing values for x, y, width and height. - */ - htmlGetBBox: function () { - var wrapper = this, - element = wrapper.element; - return { - x: element.offsetLeft, - y: element.offsetTop, - width: element.offsetWidth, - height: element.offsetHeight - }; - }, - /** - * VML override private method to update elements based on internal - * properties based on SVG transform. - * - * @private - * @function Highcharts.SVGElement#htmlUpdateTransform - * @return {void} - */ - htmlUpdateTransform: function () { - // aligning non added elements is expensive - if (!this.added) { - this.alignOnAdd = true; - return; - } - var wrapper = this, - renderer = wrapper.renderer, - elem = wrapper.element, - translateX = wrapper.translateX || 0, - translateY = wrapper.translateY || 0, - x = wrapper.x || 0, - y = wrapper.y || 0, - align = wrapper.textAlign || 'left', - alignCorrection = { - left: 0, - center: 0.5, - right: 1 - }[align], - styles = wrapper.styles, - whiteSpace = styles && styles.whiteSpace; - /** - * @private - * @return {number} - */ - function getTextPxLength() { - // Reset multiline/ellipsis in order to read width (#4928, - // #5417) - css(elem, { - width: '', - whiteSpace: whiteSpace || 'nowrap' - }); - return elem.offsetWidth; - } - // apply translate - css(elem, { - marginLeft: translateX, - marginTop: translateY + * Internal function to draw a crosshair. + * + * @function Highcharts.Axis#drawCrosshair + * + * @param {Highcharts.PointerEventObject} [e] + * The event arguments from the modified pointer event, extended with + * `chartX` and `chartY` + * + * @param {Highcharts.Point} [point] + * The Point object if the crosshair snaps to points. + * + * @fires Highcharts.Axis#event:afterDrawCrosshair + * @fires Highcharts.Axis#event:drawCrosshair + */ + Axis.prototype.drawCrosshair = function (e, point) { + var path, + options = this.crosshair, + snap = pick(options.snap, true), + pos, + categorized, + graphic = this.cross, + crossOptions, + chart = this.chart; + fireEvent(this, "drawCrosshair", { e: e, point: point }); + // Use last available event when updating non-snapped crosshairs without + // mouse interaction (#5287) + if (!e) { + e = this.cross && this.cross.e; + } + if ( + // Disabled in options + !this.crosshair || + // Snap + (defined(point) || !snap) === false + ) { + this.hideCrosshair(); + } else { + // Get the path + if (!snap) { + pos = + e && + (this.horiz + ? e.chartX - this.pos + : this.len - e.chartY + this.pos); + } else if (defined(point)) { + // #3834 + pos = pick( + this.coll !== "colorAxis" + ? point.crosshairPos // 3D axis extension + : null, + this.isXAxis ? point.plotX : this.len - point.plotY + ); + } + if (defined(pos)) { + crossOptions = { + // value, only used on radial + value: + point && + (this.isXAxis ? point.x : pick(point.stackY, point.y)), + translatedValue: pos, + }; + if (chart.polar) { + // Additional information required for crosshairs in + // polar chart + extend(crossOptions, { + isCrosshair: true, + chartX: e && e.chartX, + chartY: e && e.chartY, + point: point, }); - if (!renderer.styledMode && wrapper.shadows) { // used in labels/tooltip - wrapper.shadows.forEach(function (shadow) { - css(shadow, { - marginLeft: translateX + 1, - marginTop: translateY + 1 - }); - }); - } - // apply inversion - if (wrapper.inverted) { // wrapper is a group - [].forEach.call(elem.childNodes, function (child) { - renderer.invertChild(child, elem); - }); - } - if (elem.tagName === 'SPAN') { - var rotation = wrapper.rotation, baseline, textWidth = wrapper.textWidth && pInt(wrapper.textWidth), currentTextTransform = [ - rotation, - align, - elem.innerHTML, - wrapper.textWidth, - wrapper.textAlign - ].join(','); - // Update textWidth. Use the memoized textPxLength if possible, to - // avoid the getTextPxLength function using elem.offsetWidth. - // Calling offsetWidth affects rendering time as it forces layout - // (#7656). - if (textWidth !== wrapper.oldTextWidth && - ((textWidth > wrapper.oldTextWidth) || - (wrapper.textPxLength || getTextPxLength()) > textWidth) && ( - // Only set the width if the text is able to word-wrap, or - // text-overflow is ellipsis (#9537) - /[ \-]/.test(elem.textContent || elem.innerText) || - elem.style.textOverflow === 'ellipsis')) { // #983, #1254 - css(elem, { - width: textWidth + 'px', - display: 'block', - whiteSpace: whiteSpace || 'normal' // #3331 - }); - wrapper.oldTextWidth = textWidth; - wrapper.hasBoxWidthChanged = true; // #8159 - } - else { - wrapper.hasBoxWidthChanged = false; // #8159 - } - // Do the calculations and DOM access only if properties changed - if (currentTextTransform !== wrapper.cTT) { - baseline = renderer.fontMetrics(elem.style.fontSize, elem).b; - // Renderer specific handling of span rotation, but only if we - // have something to update. - if (defined(rotation) && - ((rotation !== (wrapper.oldRotation || 0)) || - (align !== wrapper.oldAlign))) { - wrapper.setSpanRotation(rotation, alignCorrection, baseline); - } - wrapper.getSpanCorrection( - // Avoid elem.offsetWidth if we can, it affects rendering - // time heavily (#7656) - ((!defined(rotation) && wrapper.textPxLength) || // #7920 - elem.offsetWidth), baseline, alignCorrection, rotation, align); - } - // apply position with correction - css(elem, { - left: (x + (wrapper.xCorr || 0)) + 'px', - top: (y + (wrapper.yCorr || 0)) + 'px' - }); - // record current text transform - wrapper.cTT = currentTextTransform; - wrapper.oldRotation = rotation; - wrapper.oldAlign = align; - } - }, - /** - * Set the rotation of an individual HTML span. - * - * @private - * @function Highcharts.SVGElement#setSpanRotation - * @param {number} rotation - * @param {number} alignCorrection - * @param {number} baseline - * @return {void} - */ - setSpanRotation: function (rotation, alignCorrection, baseline) { - var rotationStyle = {}, - cssTransformKey = this.renderer.getTransformKey(); - rotationStyle[cssTransformKey] = rotationStyle.transform = - 'rotate(' + rotation + 'deg)'; - rotationStyle[cssTransformKey + (isFirefox ? 'Origin' : '-origin')] = - rotationStyle.transformOrigin = - (alignCorrection * 100) + '% ' + baseline + 'px'; - css(this.element, rotationStyle); - }, - /** - * Get the correction in X and Y positioning as the element is rotated. - * - * @private - * @function Highcharts.SVGElement#getSpanCorrection - * @param {number} width - * @param {number} baseline - * @param {number} alignCorrection - * @return {void} - */ - getSpanCorrection: function (width, baseline, alignCorrection) { - this.xCorr = -width * alignCorrection; - this.yCorr = -baseline; + } + path = this.getPlotLinePath(crossOptions) || null; // #3189 } - }); - - return HTMLElement; - }); - _registerModule(_modules, 'Core/Renderer/HTML/HTMLRenderer.js', [_modules['Core/Globals.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Renderer/SVG/SVGRenderer.js'], _modules['Core/Utilities.js']], function (H, SVGElement, SVGRenderer, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var isFirefox = H.isFirefox, - isMS = H.isMS, - isWebKit = H.isWebKit, - win = H.win; - var attr = U.attr, - createElement = U.createElement, - extend = U.extend, - pick = U.pick; - /** - * Renderer placebo - * @private - */ - var HTMLRenderer = SVGRenderer; - /* eslint-disable valid-jsdoc */ - // Extend SvgRenderer for useHTML option. - extend(SVGRenderer.prototype, /** @lends SVGRenderer.prototype */ { - /** - * @private - * @function Highcharts.SVGRenderer#getTransformKey - * - * @return {string} - */ - getTransformKey: function () { - return isMS && !/Edge/.test(win.navigator.userAgent) ? - '-ms-transform' : - isWebKit ? - '-webkit-transform' : - isFirefox ? - 'MozTransform' : - win.opera ? - '-o-transform' : - ''; - }, - /** - * Create HTML text node. This is used by the VML renderer as well as the - * SVG renderer through the useHTML option. - * - * @private - * @function Highcharts.SVGRenderer#html - * - * @param {string} str - * The text of (subset) HTML to draw. - * - * @param {number} x - * The x position of the text's lower left corner. - * - * @param {number} y - * The y position of the text's lower left corner. - * - * @return {Highcharts.HTMLDOMElement} - */ - html: function (str, x, y) { - var wrapper = this.createElement('span'), element = wrapper.element, renderer = wrapper.renderer, isSVG = renderer.isSVG, addSetters = function (gWrapper, style) { - // These properties are set as attributes on the SVG group, and - // as identical CSS properties on the div. (#3542) - ['opacity', 'visibility'].forEach(function (prop) { - gWrapper[prop + 'Setter'] = function (value, key, elem) { - var styleObject = gWrapper.div ? - gWrapper.div.style : - style; - SVGElement.prototype[prop + 'Setter'] - .call(this, value, key, elem); - if (styleObject) { - styleObject[key] = value; - } - }; - }); - gWrapper.addedSetters = true; - }; - // Text setter - wrapper.textSetter = function (value) { - if (value !== element.innerHTML) { - delete this.bBox; - delete this.oldTextWidth; - } - this.textStr = value; - element.innerHTML = pick(value, ''); - wrapper.doTransform = true; - }; - // Add setters for the element itself (#4938) - if (isSVG) { // #4938, only for HTML within SVG - addSetters(wrapper, wrapper.element.style); - } - // Various setters which rely on update transform - wrapper.xSetter = - wrapper.ySetter = - wrapper.alignSetter = - wrapper.rotationSetter = - function (value, key) { - if (key === 'align') { - // Do not overwrite the SVGElement.align method. Same as VML. - wrapper.alignValue = wrapper.textAlign = value; - } - else { - wrapper[key] = value; - } - wrapper.doTransform = true; - }; - // Runs at the end of .attr() - wrapper.afterSetters = function () { - // Update transform. Do this outside the loop to prevent redundant - // updating for batch setting of attributes. - if (this.doTransform) { - this.htmlUpdateTransform(); - this.doTransform = false; - } - }; - // Set the default attributes - wrapper - .attr({ - text: str, - x: Math.round(x), - y: Math.round(y) + if (!defined(path)) { + this.hideCrosshair(); + return; + } + categorized = this.categories && !this.isRadial; + // Draw the cross + if (!graphic) { + this.cross = graphic = chart.renderer + .path() + .addClass( + "highcharts-crosshair highcharts-crosshair-" + + (categorized ? "category " : "thin ") + + options.className + ) + .attr({ + zIndex: pick(options.zIndex, 2), }) - .css({ - position: 'absolute' - }); - if (!renderer.styledMode) { - wrapper.css({ - fontFamily: this.style.fontFamily, - fontSize: this.style.fontSize - }); - } - // Keep the whiteSpace style outside the wrapper.styles collection - element.style.whiteSpace = 'nowrap'; - // Use the HTML specific .css method - wrapper.css = wrapper.htmlCss; - // This is specific for HTML within SVG - if (isSVG) { - wrapper.add = function (svgGroupWrapper) { - var htmlGroup, - container = renderer.box.parentNode, - parentGroup, - parents = []; - this.parentGroup = svgGroupWrapper; - // Create a mock group to hold the HTML elements - if (svgGroupWrapper) { - htmlGroup = svgGroupWrapper.div; - if (!htmlGroup) { - // Read the parent chain into an array and read from top - // down - parentGroup = svgGroupWrapper; - while (parentGroup) { - parents.push(parentGroup); - // Move up to the next parent group - parentGroup = parentGroup.parentGroup; - } - // Ensure dynamically updating position when any parent - // is translated - parents.reverse().forEach(function (parentGroup) { - var htmlGroupStyle, - cls = attr(parentGroup.element, 'class'); - /** - * Common translate setter for X and Y on the HTML - * group. Reverted the fix for #6957 du to - * positioning problems and offline export (#7254, - * #7280, #7529) - * @private - * @param {*} value - * @param {string} key - * @return {void} - */ - function translateSetter(value, key) { - parentGroup[key] = value; - if (key === 'translateX') { - htmlGroupStyle.left = value + 'px'; - } - else { - htmlGroupStyle.top = value + 'px'; - } - parentGroup.doTransform = true; - } - // Create a HTML div and append it to the parent div - // to emulate the SVG group structure - htmlGroup = - parentGroup.div = - parentGroup.div || createElement('div', cls ? { className: cls } : void 0, { - position: 'absolute', - left: (parentGroup.translateX || 0) + 'px', - top: (parentGroup.translateY || 0) + 'px', - display: parentGroup.display, - opacity: parentGroup.opacity, - pointerEvents: (parentGroup.styles && - parentGroup.styles.pointerEvents) // #5595 - // the top group is appended to container - }, htmlGroup || container); - // Shortcut - htmlGroupStyle = htmlGroup.style; - // Set listeners to update the HTML div's position - // whenever the SVG group position is changed. - extend(parentGroup, { - // (#7287) Pass htmlGroup to use - // the related group - classSetter: (function (htmlGroup) { - return function (value) { - this.element.setAttribute('class', value); - htmlGroup.className = value; - }; - }(htmlGroup)), - on: function () { - if (parents[0].div) { // #6418 - wrapper.on.apply({ element: parents[0].div }, arguments); - } - return parentGroup; - }, - translateXSetter: translateSetter, - translateYSetter: translateSetter - }); - if (!parentGroup.addedSetters) { - addSetters(parentGroup); - } - }); - } - } - else { - htmlGroup = container; - } - htmlGroup.appendChild(element); - // Shared with VML: - wrapper.added = true; - if (wrapper.alignOnAdd) { - wrapper.htmlUpdateTransform(); - } - return wrapper; - }; - } - return wrapper; + .add(); + // Presentational attributes + if (!chart.styledMode) { + graphic + .attr({ + stroke: + options.color || + (categorized + ? Color.parse("#ccd6eb").setOpacity(0.25).get() + : "#cccccc"), + "stroke-width": pick(options.width, 1), + }) + .css({ + "pointer-events": "none", + }); + if (options.dashStyle) { + graphic.attr({ + dashstyle: options.dashStyle, + }); + } + } } - }); - - return HTMLRenderer; - }); - _registerModule(_modules, 'Core/Axis/Tick.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - /** - * Optional parameters for the tick. - * @private - * @interface Highcharts.TickParametersObject - */ /** - * Set category for the tick. - * @name Highcharts.TickParametersObject#category - * @type {string|undefined} - */ /** - * @name Highcharts.TickParametersObject#options - * @type {Highcharts.Dictionary<any>|undefined} - */ /** - * Set tickmarkOffset for the tick. - * @name Highcharts.TickParametersObject#tickmarkOffset - * @type {number|undefined} - */ - var clamp = U.clamp, - correctFloat = U.correctFloat, - defined = U.defined, - destroyObjectProperties = U.destroyObjectProperties, - extend = U.extend, - fireEvent = U.fireEvent, - isNumber = U.isNumber, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick; - var deg2rad = H.deg2rad; - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The Tick class. - * - * @class - * @name Highcharts.Tick - * - * @param {Highcharts.Axis} axis - * The axis of the tick. - * - * @param {number} pos - * The position of the tick on the axis in terms of axis values. - * - * @param {string} [type] - * The type of tick, either 'minor' or an empty string - * - * @param {boolean} [noLabel=false] - * Whether to disable the label or not. Defaults to false. - * - * @param {object} [parameters] - * Optional parameters for the tick. - */ - var Tick = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function Tick(axis, pos, type, noLabel, parameters) { - this.isNew = true; - this.isNewLabel = true; - /** - * The related axis of the tick. - * @name Highcharts.Tick#axis - * @type {Highcharts.Axis} - */ - this.axis = axis; - /** - * The logical position of the tick on the axis in terms of axis values. - * @name Highcharts.Tick#pos - * @type {number} - */ - this.pos = pos; - /** - * The tick type, which can be `"minor"`, or an empty string. - * @name Highcharts.Tick#type - * @type {string} - */ - this.type = type || ''; - this.parameters = parameters || {}; - /** - * The mark offset of the tick on the axis. Usually `undefined`, numeric - * for grid axes. - * @name Highcharts.Tick#tickmarkOffset - * @type {number|undefined} - */ - this.tickmarkOffset = this.parameters.tickmarkOffset; - this.options = this.parameters.options; - fireEvent(this, 'init'); - if (!type && !noLabel) { - this.addLabel(); - } + graphic.show().attr({ + d: path, + }); + if (categorized && !options.width) { + graphic.attr({ + "stroke-width": this.transA, + }); } - /* * - * - * Functions - * - * */ - /** - * Write the tick label. - * - * @private - * @function Highcharts.Tick#addLabel - * @return {void} - */ - Tick.prototype.addLabel = function () { - var tick = this, - axis = tick.axis, - options = axis.options, - chart = axis.chart, - categories = axis.categories, - log = axis.logarithmic, - names = axis.names, - pos = tick.pos, - labelOptions = pick(tick.options && tick.options.labels, - options.labels), - str, - tickPositions = axis.tickPositions, - isFirst = pos === tickPositions[0], - isLast = pos === tickPositions[tickPositions.length - 1], - value = this.parameters.category || (categories ? - pick(categories[pos], - names[pos], - pos) : - pos), - label = tick.label, - animateLabels = (!labelOptions.step || labelOptions.step === 1) && - axis.tickInterval === 1, - tickPositionInfo = tickPositions.info, - dateTimeLabelFormat, - dateTimeLabelFormats, - i, - list; - // Set the datetime label format. If a higher rank is set for this - // position, use that. If not, use the general format. - if (axis.dateTime && tickPositionInfo) { - dateTimeLabelFormats = chart.time.resolveDTLFormat(options.dateTimeLabelFormats[(!options.grid && - tickPositionInfo.higherRanks[pos]) || - tickPositionInfo.unitName]); - dateTimeLabelFormat = dateTimeLabelFormats.main; - } - // set properties for access in render method - /** - * True if the tick is the first one on the axis. - * @name Highcharts.Tick#isFirst - * @readonly - * @type {boolean|undefined} - */ - tick.isFirst = isFirst; - /** - * True if the tick is the last one on the axis. - * @name Highcharts.Tick#isLast - * @readonly - * @type {boolean|undefined} - */ - tick.isLast = isLast; - // Get the string - tick.formatCtx = { - axis: axis, - chart: chart, - isFirst: isFirst, - isLast: isLast, - dateTimeLabelFormat: dateTimeLabelFormat, - tickPositionInfo: tickPositionInfo, - value: log ? correctFloat(log.lin2log(value)) : value, - pos: pos - }; - str = axis.labelFormatter.call(tick.formatCtx, this.formatCtx); - // Set up conditional formatting based on the format list if existing. - list = dateTimeLabelFormats && dateTimeLabelFormats.list; - if (list) { - tick.shortenLabel = function () { - for (i = 0; i < list.length; i++) { - label.attr({ - text: axis.labelFormatter.call(extend(tick.formatCtx, { dateTimeLabelFormat: list[i] })) - }); - if (label.getBBox().width < - axis.getSlotWidth(tick) - 2 * - pick(labelOptions.padding, 5)) { - return; - } - } - label.attr({ - text: '' - }); - }; - } - // Call only after first render - if (animateLabels && axis._addedPlotLB) { - tick.moveLabel(str, labelOptions); - } - // First call - if (!defined(label) && !tick.movedLabel) { - /** - * The rendered text label of the tick. - * @name Highcharts.Tick#label - * @type {Highcharts.SVGElement|undefined} - */ - tick.label = label = tick.createLabel({ x: 0, y: 0 }, str, labelOptions); - // Base value to detect change for new calls to getBBox - tick.rotation = 0; - // update - } - else if (label && label.textStr !== str && !animateLabels) { - // When resetting text, also reset the width if dynamically set - // (#8809) - if (label.textWidth && - !(labelOptions.style && labelOptions.style.width) && - !label.styles.width) { - label.css({ width: null }); - } - label.attr({ text: str }); - label.textPxLength = label.getBBox().width; - } - }; - /** - * Render and return the label of the tick. - * - * @private - * @function Highcharts.Tick#createLabel - * @param {Highcharts.PositionObject} xy - * @param {string} str - * @param {Highcharts.XAxisLabelsOptions} labelOptions - * @return {Highcharts.SVGElement|undefined} - */ - Tick.prototype.createLabel = function (xy, str, labelOptions) { - var axis = this.axis, - chart = axis.chart, - label = defined(str) && labelOptions.enabled ? - chart.renderer - .text(str, - xy.x, - xy.y, - labelOptions.useHTML) - .add(axis.labelGroup) : - null; - // Un-rotated length - if (label) { - // Without position absolute, IE export sometimes is wrong - if (!chart.styledMode) { - label.css(merge(labelOptions.style)); - } - label.textPxLength = label.getBBox().width; - } - return label; - }; - /** - * Destructor for the tick prototype - * - * @private - * @function Highcharts.Tick#destroy - * @return {void} - */ - Tick.prototype.destroy = function () { - destroyObjectProperties(this, this.axis); - }; - /** - * Gets the x and y positions for ticks in terms of pixels. - * - * @private - * @function Highcharts.Tick#getPosition - * - * @param {boolean} horiz - * Whether the tick is on an horizontal axis or not. - * - * @param {number} tickPos - * Position of the tick. - * - * @param {number} tickmarkOffset - * Tickmark offset for all ticks. - * - * @param {boolean} [old] - * Whether the axis has changed or not. - * - * @return {Highcharts.PositionObject} - * The tick position. - * - * @fires Highcharts.Tick#event:afterGetPosition - */ - Tick.prototype.getPosition = function (horiz, tickPos, tickmarkOffset, old) { - var axis = this.axis, - chart = axis.chart, - cHeight = (old && chart.oldChartHeight) || chart.chartHeight, - pos; - pos = { - x: horiz ? - correctFloat(axis.translate(tickPos + tickmarkOffset, null, null, old) + - axis.transB) : - (axis.left + - axis.offset + - (axis.opposite ? - (((old && chart.oldChartWidth) || - chart.chartWidth) - - axis.right - - axis.left) : - 0)), - y: horiz ? - (cHeight - - axis.bottom + - axis.offset - - (axis.opposite ? axis.height : 0)) : - correctFloat(cHeight - - axis.translate(tickPos + tickmarkOffset, null, null, old) - - axis.transB) - }; - // Chrome workaround for #10516 - pos.y = clamp(pos.y, -1e5, 1e5); - fireEvent(this, 'afterGetPosition', { pos: pos }); - return pos; - }; - /** - * Get the x, y position of the tick label - * - * @private - * @return {Highcharts.PositionObject} - */ - Tick.prototype.getLabelPosition = function (x, y, label, horiz, labelOptions, tickmarkOffset, index, step) { - var axis = this.axis, - transA = axis.transA, - reversed = ( // #7911 - axis.isLinked && axis.linkedParent ? - axis.linkedParent.reversed : - axis.reversed), - staggerLines = axis.staggerLines, - rotCorr = axis.tickRotCorr || { x: 0, - y: 0 }, - yOffset = labelOptions.y, - // Adjust for label alignment if we use reserveSpace: true (#5286) - labelOffsetCorrection = (!horiz && !axis.reserveSpaceDefault ? - -axis.labelOffset * (axis.labelAlign === 'center' ? 0.5 : 1) : - 0), - line, - pos = {}; - if (!defined(yOffset)) { - if (axis.side === 0) { - yOffset = label.rotation ? -8 : -label.getBBox().height; - } - else if (axis.side === 2) { - yOffset = rotCorr.y + 8; - } - else { - // #3140, #3140 - yOffset = Math.cos(label.rotation * deg2rad) * - (rotCorr.y - label.getBBox(false, 0).height / 2); - } - } - x = x + - labelOptions.x + - labelOffsetCorrection + - rotCorr.x - - (tickmarkOffset && horiz ? - tickmarkOffset * transA * (reversed ? -1 : 1) : - 0); - y = y + yOffset - (tickmarkOffset && !horiz ? - tickmarkOffset * transA * (reversed ? 1 : -1) : 0); - // Correct for staggered labels - if (staggerLines) { - line = (index / (step || 1) % staggerLines); - if (axis.opposite) { - line = staggerLines - line - 1; - } - y += line * (axis.labelOffset / staggerLines); - } - pos.x = x; - pos.y = Math.round(y); - fireEvent(this, 'afterGetLabelPosition', { pos: pos, tickmarkOffset: tickmarkOffset, index: index }); - return pos; - }; - /** - * Get the offset height or width of the label - * - * @private - * @function Highcharts.Tick#getLabelSize - * @return {number} - */ - Tick.prototype.getLabelSize = function () { - return this.label ? - this.label.getBBox()[this.axis.horiz ? 'height' : 'width'] : - 0; - }; - /** - * Extendible method to return the path of the marker - * - * @private - * - */ - Tick.prototype.getMarkPath = function (x, y, tickLength, tickWidth, horiz, renderer) { - return renderer.crispLine([[ - 'M', - x, - y - ], [ - 'L', - x + (horiz ? 0 : -tickLength), - y + (horiz ? tickLength : 0) - ]], tickWidth); - }; - /** - * Handle the label overflow by adjusting the labels to the left and right - * edge, or hide them if they collide into the neighbour label. - * - * @private - * @function Highcharts.Tick#handleOverflow - * @param {Highcharts.PositionObject} xy - * @return {void} - */ - Tick.prototype.handleOverflow = function (xy) { - var tick = this, - axis = this.axis, - labelOptions = axis.options.labels, - pxPos = xy.x, - chartWidth = axis.chart.chartWidth, - spacing = axis.chart.spacing, - leftBound = pick(axis.labelLeft, - Math.min(axis.pos, - spacing[3])), - rightBound = pick(axis.labelRight, - Math.max(!axis.isRadial ? axis.pos + axis.len : 0, - chartWidth - spacing[1])), - label = this.label, - rotation = this.rotation, - factor = { - left: 0, - center: 0.5, - right: 1 - }[axis.labelAlign || label.attr('align')], - labelWidth = label.getBBox().width, - slotWidth = axis.getSlotWidth(tick), - modifiedSlotWidth = slotWidth, - xCorrection = factor, - goRight = 1, - leftPos, - rightPos, - textWidth, - css = {}; - // Check if the label overshoots the chart spacing box. If it does, move - // it. If it now overshoots the slotWidth, add ellipsis. - if (!rotation && - pick(labelOptions.overflow, 'justify') === 'justify') { - leftPos = pxPos - factor * labelWidth; - rightPos = pxPos + (1 - factor) * labelWidth; - if (leftPos < leftBound) { - modifiedSlotWidth = - xy.x + modifiedSlotWidth * (1 - factor) - leftBound; - } - else if (rightPos > rightBound) { - modifiedSlotWidth = - rightBound - xy.x + modifiedSlotWidth * factor; - goRight = -1; - } - modifiedSlotWidth = Math.min(slotWidth, modifiedSlotWidth); // #4177 - if (modifiedSlotWidth < slotWidth && axis.labelAlign === 'center') { - xy.x += (goRight * - (slotWidth - - modifiedSlotWidth - - xCorrection * (slotWidth - Math.min(labelWidth, modifiedSlotWidth)))); - } - // If the label width exceeds the available space, set a text width - // to be picked up below. Also, if a width has been set before, we - // need to set a new one because the reported labelWidth will be - // limited by the box (#3938). - if (labelWidth > modifiedSlotWidth || - (axis.autoRotation && (label.styles || {}).width)) { - textWidth = modifiedSlotWidth; - } - // Add ellipsis to prevent rotated labels to be clipped against the edge - // of the chart - } - else if (rotation < 0 && - pxPos - factor * labelWidth < leftBound) { - textWidth = Math.round(pxPos / Math.cos(rotation * deg2rad) - leftBound); - } - else if (rotation > 0 && - pxPos + factor * labelWidth > rightBound) { - textWidth = Math.round((chartWidth - pxPos) / - Math.cos(rotation * deg2rad)); - } - if (textWidth) { - if (tick.shortenLabel) { - tick.shortenLabel(); - } - else { - css.width = Math.floor(textWidth) + 'px'; - if (!(labelOptions.style || {}).textOverflow) { - css.textOverflow = 'ellipsis'; - } - label.css(css); - } - } - }; - /** - * Try to replace the label if the same one already exists. - * - * @private - * @function Highcharts.Tick#moveLabel - * @param {string} str - * @param {Highcharts.XAxisLabelsOptions} labelOptions - * - * @return {void} - */ - Tick.prototype.moveLabel = function (str, labelOptions) { - var tick = this, - label = tick.label, - moved = false, - axis = tick.axis, - labelPos, - reversed = axis.reversed, - xPos, - yPos; - if (label && label.textStr === str) { - tick.movedLabel = label; - moved = true; - delete tick.label; - } - else { // Find a label with the same string - objectEach(axis.ticks, function (currentTick) { - if (!moved && - !currentTick.isNew && - currentTick !== tick && - currentTick.label && - currentTick.label.textStr === str) { - tick.movedLabel = currentTick.label; - moved = true; - currentTick.labelPos = tick.movedLabel.xy; - delete currentTick.label; - } - }); - } - // Create new label if the actual one is moved - if (!moved && (tick.labelPos || label)) { - labelPos = tick.labelPos || label.xy; - xPos = axis.horiz ? - (reversed ? 0 : axis.width + axis.left) : labelPos.x; - yPos = axis.horiz ? - labelPos.y : (reversed ? (axis.width + axis.left) : 0); - tick.movedLabel = tick.createLabel({ x: xPos, y: yPos }, str, labelOptions); - if (tick.movedLabel) { - tick.movedLabel.attr({ opacity: 0 }); - } - } - }; - /** - * Put everything in place - * - * @private - * @param {number} index - * @param {boolean} [old] - * Use old coordinates to prepare an animation into new position - * @param {number} [opacity] - * @return {voids} - */ - Tick.prototype.render = function (index, old, opacity) { - var tick = this, - axis = tick.axis, - horiz = axis.horiz, - pos = tick.pos, - tickmarkOffset = pick(tick.tickmarkOffset, - axis.tickmarkOffset), - xy = tick.getPosition(horiz, - pos, - tickmarkOffset, - old), - x = xy.x, - y = xy.y, - reverseCrisp = ((horiz && x === axis.pos + axis.len) || - (!horiz && y === axis.pos)) ? -1 : 1; // #1480, #1687 - opacity = pick(opacity, 1); - this.isActive = true; - // Create the grid line - this.renderGridLine(old, opacity, reverseCrisp); - // create the tick mark - this.renderMark(xy, opacity, reverseCrisp); - // the label is created on init - now move it into place - this.renderLabel(xy, old, opacity, index); - tick.isNew = false; - fireEvent(this, 'afterRender'); - }; - /** - * Renders the gridLine. - * - * @private - * @param {boolean} old Whether or not the tick is old - * @param {number} opacity The opacity of the grid line - * @param {number} reverseCrisp Modifier for avoiding overlapping 1 or -1 - * @return {void} - */ - Tick.prototype.renderGridLine = function (old, opacity, reverseCrisp) { - var tick = this, axis = tick.axis, options = axis.options, gridLine = tick.gridLine, gridLinePath, attribs = {}, pos = tick.pos, type = tick.type, tickmarkOffset = pick(tick.tickmarkOffset, axis.tickmarkOffset), renderer = axis.chart.renderer, gridPrefix = type ? type + 'Grid' : 'grid', gridLineWidth = options[gridPrefix + 'LineWidth'], gridLineColor = options[gridPrefix + 'LineColor'], dashStyle = options[gridPrefix + 'LineDashStyle']; - if (!gridLine) { - if (!axis.chart.styledMode) { - attribs.stroke = gridLineColor; - attribs['stroke-width'] = gridLineWidth; - if (dashStyle) { - attribs.dashstyle = dashStyle; - } - } - if (!type) { - attribs.zIndex = 1; - } - if (old) { - opacity = 0; - } - /** - * The rendered grid line of the tick. - * @name Highcharts.Tick#gridLine - * @type {Highcharts.SVGElement|undefined} - */ - tick.gridLine = gridLine = renderer.path() - .attr(attribs) - .addClass('highcharts-' + (type ? type + '-' : '') + 'grid-line') - .add(axis.gridGroup); - } - if (gridLine) { - gridLinePath = axis.getPlotLinePath({ - value: pos + tickmarkOffset, - lineWidth: gridLine.strokeWidth() * reverseCrisp, - force: 'pass', - old: old - }); - // If the parameter 'old' is set, the current call will be followed - // by another call, therefore do not do any animations this time - if (gridLinePath) { - gridLine[old || tick.isNew ? 'attr' : 'animate']({ - d: gridLinePath, - opacity: opacity - }); - } - } - }; - /** - * Renders the tick mark. - * - * @private - * @param {Highcharts.PositionObject} xy The position vector of the mark - * @param {number} opacity The opacity of the mark - * @param {number} reverseCrisp Modifier for avoiding overlapping 1 or -1 - * @return {void} - */ - Tick.prototype.renderMark = function (xy, opacity, reverseCrisp) { - var tick = this, axis = tick.axis, options = axis.options, renderer = axis.chart.renderer, type = tick.type, tickPrefix = type ? type + 'Tick' : 'tick', tickSize = axis.tickSize(tickPrefix), mark = tick.mark, isNewMark = !mark, x = xy.x, y = xy.y, tickWidth = pick(options[tickPrefix + 'Width'], !type && axis.isXAxis ? 1 : 0), // X axis defaults to 1 - tickColor = options[tickPrefix + 'Color']; - if (tickSize) { - // negate the length - if (axis.opposite) { - tickSize[0] = -tickSize[0]; - } - // First time, create it - if (isNewMark) { - /** - * The rendered mark of the tick. - * @name Highcharts.Tick#mark - * @type {Highcharts.SVGElement|undefined} - */ - tick.mark = mark = renderer.path() - .addClass('highcharts-' + (type ? type + '-' : '') + 'tick') - .add(axis.axisGroup); - if (!axis.chart.styledMode) { - mark.attr({ - stroke: tickColor, - 'stroke-width': tickWidth - }); - } - } - mark[isNewMark ? 'attr' : 'animate']({ - d: tick.getMarkPath(x, y, tickSize[0], mark.strokeWidth() * reverseCrisp, axis.horiz, renderer), - opacity: opacity - }); - } - }; - /** - * Renders the tick label. - * Note: The label should already be created in init(), so it should only - * have to be moved into place. - * - * @private - * @param {Highcharts.PositionObject} xy The position vector of the label - * @param {boolean} old Whether or not the tick is old - * @param {number} opacity The opacity of the label - * @param {number} index The index of the tick - * @return {void} - */ - Tick.prototype.renderLabel = function (xy, old, opacity, index) { - var tick = this, - axis = tick.axis, - horiz = axis.horiz, - options = axis.options, - label = tick.label, - labelOptions = options.labels, - step = labelOptions.step, - tickmarkOffset = pick(tick.tickmarkOffset, - axis.tickmarkOffset), - show = true, - x = xy.x, - y = xy.y; - if (label && isNumber(x)) { - label.xy = xy = tick.getLabelPosition(x, y, label, horiz, labelOptions, tickmarkOffset, index, step); - // Apply show first and show last. If the tick is both first and - // last, it is a single centered tick, in which case we show the - // label anyway (#2100). - if ((tick.isFirst && - !tick.isLast && - !pick(options.showFirstLabel, 1)) || - (tick.isLast && - !tick.isFirst && - !pick(options.showLastLabel, 1))) { - show = false; - // Handle label overflow and show or hide accordingly - } - else if (horiz && - !labelOptions.step && - !labelOptions.rotation && - !old && - opacity !== 0) { - tick.handleOverflow(xy); - } - // apply step - if (step && index % step) { - // show those indices dividable by step - show = false; - } - // Set the new position, and show or hide - if (show && isNumber(xy.y)) { - xy.opacity = opacity; - label[tick.isNewLabel ? 'attr' : 'animate'](xy); - tick.isNewLabel = false; - } - else { - label.attr('y', -9999); // #1338 - tick.isNewLabel = true; - } - } - }; - /** - * Replace labels with the moved ones to perform animation. Additionally - * destroy unused labels. - * - * @private - * @function Highcharts.Tick#replaceMovedLabel - * @return {void} - */ - Tick.prototype.replaceMovedLabel = function () { - var tick = this, - label = tick.label, - axis = tick.axis, - reversed = axis.reversed, - x, - y; - // Animate and destroy - if (label && !tick.isNew) { - x = axis.horiz ? (reversed ? axis.left : axis.width + axis.left) : label.xy.x; - y = axis.horiz ? - label.xy.y : - (reversed ? axis.width + axis.top : axis.top); - label.animate({ x: x, y: y, opacity: 0 }, void 0, label.destroy); - delete tick.label; - } - axis.isDirty = true; - tick.label = tick.movedLabel; - delete tick.movedLabel; - }; - return Tick; - }()); - H.Tick = Tick; - - return H.Tick; - }); - _registerModule(_modules, 'Core/Time.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Highcharts, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - /** - * Normalized interval. - * - * @interface Highcharts.TimeNormalizedObject - */ /** - * The count. - * - * @name Highcharts.TimeNormalizedObject#count - * @type {number} - */ /** - * The interval in axis values (ms). - * - * @name Highcharts.TimeNormalizedObject#unitRange - * @type {number} - */ - /** - * Function of an additional date format specifier. - * - * @callback Highcharts.TimeFormatCallbackFunction - * - * @param {number} timestamp - * The time to format. - * - * @return {string} - * The formatted portion of the date. - */ - /** - * Additonal time tick information. - * - * @interface Highcharts.TimeTicksInfoObject - * @extends Highcharts.TimeNormalizedObject - */ /** - * @name Highcharts.TimeTicksInfoObject#higherRanks - * @type {Array<string>} - */ /** - * @name Highcharts.TimeTicksInfoObject#totalRange - * @type {number} - */ - /** - * Time ticks. - * - * @interface Highcharts.AxisTickPositionsArray - * @extends global.Array<number> - */ /** - * @name Highcharts.AxisTickPositionsArray#info - * @type {Highcharts.TimeTicksInfoObject|undefined} - */ - /** - * A callback to return the time zone offset for a given datetime. It - * takes the timestamp in terms of milliseconds since January 1 1970, - * and returns the timezone offset in minutes. This provides a hook - * for drawing time based charts in specific time zones using their - * local DST crossover dates, with the help of external libraries. - * - * @callback Highcharts.TimezoneOffsetCallbackFunction - * - * @param {number} timestamp - * Timestamp in terms of milliseconds since January 1 1970. - * - * @return {number} - * Timezone offset in minutes. - */ - /** - * Allows to manually load the `moment.js` library from Highcharts options - * instead of the `window`. - * In case of loading the library from a `script` tag, - * this option is not needed, it will be loaded from there by default. - * - * @type {function} - * @since 8.2.0 - * @apioption time.moment - */ - var defined = U.defined, - error = U.error, - extend = U.extend, - isObject = U.isObject, - merge = U.merge, - objectEach = U.objectEach, - pad = U.pad, - pick = U.pick, - splat = U.splat, - timeUnits = U.timeUnits; - var H = Highcharts, - win = H.win; - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The Time class. Time settings are applied in general for each page using - * `Highcharts.setOptions`, or individually for each Chart item through the - * [time](https://api.highcharts.com/highcharts/time) options set. - * - * The Time object is available from {@link Highcharts.Chart#time}, - * which refers to `Highcharts.time` if no individual time settings are - * applied. - * - * @example - * // Apply time settings globally - * Highcharts.setOptions({ - * time: { - * timezone: 'Europe/London' - * } - * }); - * - * // Apply time settings by instance - * var chart = Highcharts.chart('container', { - * time: { - * timezone: 'America/New_York' - * }, - * series: [{ - * data: [1, 4, 3, 5] - * }] - * }); - * - * // Use the Time object - * console.log( - * 'Current time in New York', - * chart.time.dateFormat('%Y-%m-%d %H:%M:%S', Date.now()) - * ); - * - * @since 6.0.5 - * - * @class - * @name Highcharts.Time - * - * @param {Highcharts.TimeOptions} options - * Time options as defined in [chart.options.time](/highcharts/time). - */ - var Time = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function Time(options) { - /* * - * - * Properties - * - * */ - this.options = {}; - this.useUTC = false; - this.variableTimezone = false; - this.Date = win.Date; - /** - * Get the time zone offset based on the current timezone information as - * set in the global options. - * - * @function Highcharts.Time#getTimezoneOffset - * - * @param {number} timestamp - * The JavaScript timestamp to inspect. - * - * @return {number} - * The timezone offset in minutes compared to UTC. - */ - this.getTimezoneOffset = this.timezoneOffsetFunction(); - this.update(options); - } - /* * - * - * Functions - * - * */ - /** - * Time units used in `Time.get` and `Time.set` - * - * @typedef {"Date"|"Day"|"FullYear"|"Hours"|"Milliseconds"|"Minutes"|"Month"|"Seconds"} Highcharts.TimeUnitValue - */ - /** - * Get the value of a date object in given units, and subject to the Time - * object's current timezone settings. This function corresponds directly to - * JavaScripts `Date.getXXX / Date.getUTCXXX`, so instead of calling - * `date.getHours()` or `date.getUTCHours()` we will call - * `time.get('Hours')`. - * - * @function Highcharts.Time#get - * - * @param {Highcharts.TimeUnitValue} unit - * @param {Date} date - * - * @return {number} - * The given time unit - */ - Time.prototype.get = function (unit, date) { - if (this.variableTimezone || this.timezoneOffset) { - var realMs = date.getTime(); - var ms = realMs - this.getTimezoneOffset(date); - date.setTime(ms); // Temporary adjust to timezone - var ret = date['getUTC' + unit](); - date.setTime(realMs); // Reset - return ret; - } - // UTC time with no timezone handling - if (this.useUTC) { - return date['getUTC' + unit](); - } - // Else, local time - return date['get' + unit](); - }; - /** - * Set the value of a date object in given units, and subject to the Time - * object's current timezone settings. This function corresponds directly to - * JavaScripts `Date.setXXX / Date.setUTCXXX`, so instead of calling - * `date.setHours(0)` or `date.setUTCHours(0)` we will call - * `time.set('Hours', 0)`. - * - * @function Highcharts.Time#set - * - * @param {Highcharts.TimeUnitValue} unit - * @param {Date} date - * @param {number} value - * - * @return {number} - * The epoch milliseconds of the updated date - */ - Time.prototype.set = function (unit, date, value) { - // UTC time with timezone handling - if (this.variableTimezone || this.timezoneOffset) { - // For lower order time units, just set it directly using UTC - // time - if (unit === 'Milliseconds' || - unit === 'Seconds' || - unit === 'Minutes') { - return date['setUTC' + unit](value); - } - // Higher order time units need to take the time zone into - // account - // Adjust by timezone - var offset = this.getTimezoneOffset(date); - var ms = date.getTime() - offset; - date.setTime(ms); - date['setUTC' + unit](value); - var newOffset = this.getTimezoneOffset(date); - ms = date.getTime() + newOffset; - return date.setTime(ms); - } - // UTC time with no timezone handling - if (this.useUTC) { - return date['setUTC' + unit](value); - } - // Else, local time - return date['set' + unit](value); - }; - /** - * Update the Time object with current options. It is called internally on - * initializing Highcharts, after running `Highcharts.setOptions` and on - * `Chart.update`. - * - * @private - * @function Highcharts.Time#update - * - * @param {Highcharts.TimeOptions} options - * - * @return {void} - */ - Time.prototype.update = function (options) { - var useUTC = pick(options && options.useUTC, - true), - time = this; - this.options = options = merge(true, this.options || {}, options); - // Allow using a different Date class - this.Date = options.Date || win.Date || Date; - this.useUTC = useUTC; - this.timezoneOffset = (useUTC && options.timezoneOffset); - this.getTimezoneOffset = this.timezoneOffsetFunction(); - /* - * The time object has options allowing for variable time zones, meaning - * the axis ticks or series data needs to consider this. - */ - this.variableTimezone = !!(!useUTC || - options.getTimezoneOffset || - options.timezone); - }; - /** - * Make a time and returns milliseconds. Interprets the inputs as UTC time, - * local time or a specific timezone time depending on the current time - * settings. - * - * @function Highcharts.Time#makeTime - * - * @param {number} year - * The year - * - * @param {number} month - * The month. Zero-based, so January is 0. - * - * @param {number} [date=1] - * The day of the month - * - * @param {number} [hours=0] - * The hour of the day, 0-23. - * - * @param {number} [minutes=0] - * The minutes - * - * @param {number} [seconds=0] - * The seconds - * - * @return {number} - * The time in milliseconds since January 1st 1970. - */ - Time.prototype.makeTime = function (year, month, date, hours, minutes, seconds) { - var d, - offset, - newOffset; - if (this.useUTC) { - d = this.Date.UTC.apply(0, arguments); - offset = this.getTimezoneOffset(d); - d += offset; - newOffset = this.getTimezoneOffset(d); - if (offset !== newOffset) { - d += newOffset - offset; - // A special case for transitioning from summer time to winter time. - // When the clock is set back, the same time is repeated twice, i.e. - // 02:30 am is repeated since the clock is set back from 3 am to - // 2 am. We need to make the same time as local Date does. - } - else if (offset - 36e5 === this.getTimezoneOffset(d - 36e5) && - !H.isSafari) { - d -= 36e5; - } - } - else { - d = new this.Date(year, month, pick(date, 1), pick(hours, 0), pick(minutes, 0), pick(seconds, 0)).getTime(); - } - return d; - }; - /** - * Sets the getTimezoneOffset function. If the `timezone` option is set, a - * default getTimezoneOffset function with that timezone is returned. If - * a `getTimezoneOffset` option is defined, it is returned. If neither are - * specified, the function using the `timezoneOffset` option or 0 offset is - * returned. - * - * @private - * @function Highcharts.Time#timezoneOffsetFunction - * - * @return {Function} - * A getTimezoneOffset function - */ - Time.prototype.timezoneOffsetFunction = function () { - var time = this, - options = this.options, - moment = options.moment || win.moment; - if (!this.useUTC) { - return function (timestamp) { - return new Date(timestamp.toString()).getTimezoneOffset() * 60000; - }; - } - if (options.timezone) { - if (!moment) { - // getTimezoneOffset-function stays undefined because it depends - // on Moment.js - error(25); - } - else { - return function (timestamp) { - return -moment.tz(timestamp, options.timezone).utcOffset() * 60000; - }; - } - } - // If not timezone is set, look for the getTimezoneOffset callback - if (this.useUTC && options.getTimezoneOffset) { - return function (timestamp) { - return options.getTimezoneOffset(timestamp.valueOf()) * 60000; - }; - } - // Last, use the `timezoneOffset` option if set - return function () { - return (time.timezoneOffset || 0) * 60000; - }; - }; - /** - * Formats a JavaScript date timestamp (milliseconds since Jan 1st 1970) - * into a human readable date string. The available format keys are listed - * below. Additional formats can be given in the - * {@link Highcharts.dateFormats} hook. - * - * Supported format keys: - * - `%a`: Short weekday, like 'Mon' - * - `%A`: Long weekday, like 'Monday' - * - `%d`: Two digit day of the month, 01 to 31 - * - `%e`: Day of the month, 1 through 31 - * - `%w`: Day of the week, 0 through 6 - * - `%b`: Short month, like 'Jan' - * - `%B`: Long month, like 'January' - * - `%m`: Two digit month number, 01 through 12 - * - `%y`: Two digits year, like 09 for 2009 - * - `%Y`: Four digits year, like 2009 - * - `%H`: Two digits hours in 24h format, 00 through 23 - * - `%k`: Hours in 24h format, 0 through 23 - * - `%I`: Two digits hours in 12h format, 00 through 11 - * - `%l`: Hours in 12h format, 1 through 12 - * - `%M`: Two digits minutes, 00 through 59 - * - `%p`: Upper case AM or PM - * - `%P`: Lower case AM or PM - * - `%S`: Two digits seconds, 00 through 59 - * - `%L`: Milliseconds (naming from Ruby) - * - * @example - * const time = new Highcharts.Time(); - * const s = time.dateFormat('%Y-%m-%d %H:%M:%S', Date.UTC(2020, 0, 1)); - * console.log(s); // => 2020-01-01 00:00:00 - * - * @function Highcharts.Time#dateFormat - * - * @param {string} format - * The desired format where various time representations are - * prefixed with %. - * - * @param {number} timestamp - * The JavaScript timestamp. - * - * @param {boolean} [capitalize=false] - * Upper case first letter in the return. - * - * @return {string} - * The formatted date. - */ - Time.prototype.dateFormat = function (format, timestamp, capitalize) { - var _a; - if (!defined(timestamp) || isNaN(timestamp)) { - return ((_a = H.defaultOptions.lang) === null || _a === void 0 ? void 0 : _a.invalidDate) || ''; - } - format = pick(format, '%Y-%m-%d %H:%M:%S'); - var time = this, date = new this.Date(timestamp), - // get the basic time values - hours = this.get('Hours', date), day = this.get('Day', date), dayOfMonth = this.get('Date', date), month = this.get('Month', date), fullYear = this.get('FullYear', date), lang = H.defaultOptions.lang, langWeekdays = lang === null || lang === void 0 ? void 0 : lang.weekdays, shortWeekdays = lang === null || lang === void 0 ? void 0 : lang.shortWeekdays, - // List all format keys. Custom formats can be added from the - // outside. - replacements = extend({ - // Day - // Short weekday, like 'Mon' - a: shortWeekdays ? - shortWeekdays[day] : - langWeekdays[day].substr(0, 3), - // Long weekday, like 'Monday' - A: langWeekdays[day], - // Two digit day of the month, 01 to 31 - d: pad(dayOfMonth), - // Day of the month, 1 through 31 - e: pad(dayOfMonth, 2, ' '), - // Day of the week, 0 through 6 - w: day, - // Week (none implemented) - // 'W': weekNumber(), - // Month - // Short month, like 'Jan' - b: lang.shortMonths[month], - // Long month, like 'January' - B: lang.months[month], - // Two digit month number, 01 through 12 - m: pad(month + 1), - // Month number, 1 through 12 (#8150) - o: month + 1, - // Year - // Two digits year, like 09 for 2009 - y: fullYear.toString().substr(2, 2), - // Four digits year, like 2009 - Y: fullYear, - // Time - // Two digits hours in 24h format, 00 through 23 - H: pad(hours), - // Hours in 24h format, 0 through 23 - k: hours, - // Two digits hours in 12h format, 00 through 11 - I: pad((hours % 12) || 12), - // Hours in 12h format, 1 through 12 - l: (hours % 12) || 12, - // Two digits minutes, 00 through 59 - M: pad(this.get('Minutes', date)), - // Upper case AM or PM - p: hours < 12 ? 'AM' : 'PM', - // Lower case AM or PM - P: hours < 12 ? 'am' : 'pm', - // Two digits seconds, 00 through 59 - S: pad(date.getSeconds()), - // Milliseconds (naming from Ruby) - L: pad(Math.floor(timestamp % 1000), 3) - }, H.dateFormats); - // Do the replaces - objectEach(replacements, function (val, key) { - // Regex would do it in one line, but this is faster - while (format.indexOf('%' + key) !== -1) { - format = format.replace('%' + key, typeof val === 'function' ? val.call(time, timestamp) : val); - } - }); - // Optionally capitalize the string and return - return capitalize ? - (format.substr(0, 1).toUpperCase() + - format.substr(1)) : - format; - }; - /** - * Resolve legacy formats of dateTimeLabelFormats (strings and arrays) into - * an object. - * @private - * @param {string|Array<T>|Highcharts.Dictionary<T>} f - General format description - * @return {Highcharts.Dictionary<T>} - The object definition - */ - Time.prototype.resolveDTLFormat = function (f) { - if (!isObject(f, true)) { // check for string or array - f = splat(f); - return { - main: f[0], - from: f[1], - to: f[2] - }; - } - return f; - }; - /** - * Return an array with time positions distributed on round time values - * right and right after min and max. Used in datetime axes as well as for - * grouping data on a datetime axis. - * - * @function Highcharts.Time#getTimeTicks - * - * @param {Highcharts.TimeNormalizedObject} normalizedInterval - * The interval in axis values (ms) and the count - * - * @param {number} [min] - * The minimum in axis values - * - * @param {number} [max] - * The maximum in axis values - * - * @param {number} [startOfWeek=1] - * - * @return {Highcharts.AxisTickPositionsArray} - */ - Time.prototype.getTimeTicks = function (normalizedInterval, min, max, startOfWeek) { - var time = this, - Date = time.Date, - tickPositions = [], - i, - higherRanks = {}, - minYear, // used in months and years as a basis for Date.UTC() - // When crossing DST, use the max. Resolves #6278. - minDate = new Date(min), - interval = normalizedInterval.unitRange, - count = normalizedInterval.count || 1, - variableDayLength, - minDay; - startOfWeek = pick(startOfWeek, 1); - if (defined(min)) { // #1300 - time.set('Milliseconds', minDate, interval >= timeUnits.second ? - 0 : // #3935 - count * Math.floor(time.get('Milliseconds', minDate) / count)); // #3652, #3654 - if (interval >= timeUnits.second) { // second - time.set('Seconds', minDate, interval >= timeUnits.minute ? - 0 : // #3935 - count * Math.floor(time.get('Seconds', minDate) / count)); - } - if (interval >= timeUnits.minute) { // minute - time.set('Minutes', minDate, interval >= timeUnits.hour ? - 0 : - count * Math.floor(time.get('Minutes', minDate) / count)); - } - if (interval >= timeUnits.hour) { // hour - time.set('Hours', minDate, interval >= timeUnits.day ? - 0 : - count * Math.floor(time.get('Hours', minDate) / count)); - } - if (interval >= timeUnits.day) { // day - time.set('Date', minDate, interval >= timeUnits.month ? - 1 : - Math.max(1, count * Math.floor(time.get('Date', minDate) / count))); - } - if (interval >= timeUnits.month) { // month - time.set('Month', minDate, interval >= timeUnits.year ? 0 : - count * Math.floor(time.get('Month', minDate) / count)); - minYear = time.get('FullYear', minDate); - } - if (interval >= timeUnits.year) { // year - minYear -= minYear % count; - time.set('FullYear', minDate, minYear); - } - // week is a special case that runs outside the hierarchy - if (interval === timeUnits.week) { - // get start of current week, independent of count - minDay = time.get('Day', minDate); - time.set('Date', minDate, (time.get('Date', minDate) - - minDay + startOfWeek + - // We don't want to skip days that are before - // startOfWeek (#7051) - (minDay < startOfWeek ? -7 : 0))); - } - // Get basics for variable time spans - minYear = time.get('FullYear', minDate); - var minMonth = time.get('Month', minDate), minDateDate = time.get('Date', minDate), minHours = time.get('Hours', minDate); - // Redefine min to the floored/rounded minimum time (#7432) - min = minDate.getTime(); - // Handle local timezone offset - if (time.variableTimezone) { - // Detect whether we need to take the DST crossover into - // consideration. If we're crossing over DST, the day length may - // be 23h or 25h and we need to compute the exact clock time for - // each tick instead of just adding hours. This comes at a cost, - // so first we find out if it is needed (#4951). - variableDayLength = ( - // Long range, assume we're crossing over. - max - min > 4 * timeUnits.month || - // Short range, check if min and max are in different time - // zones. - time.getTimezoneOffset(min) !== - time.getTimezoneOffset(max)); - } - // Iterate and add tick positions at appropriate values - var t = minDate.getTime(); - i = 1; - while (t < max) { - tickPositions.push(t); - // if the interval is years, use Date.UTC to increase years - if (interval === timeUnits.year) { - t = time.makeTime(minYear + i * count, 0); - // if the interval is months, use Date.UTC to increase months - } - else if (interval === timeUnits.month) { - t = time.makeTime(minYear, minMonth + i * count); - // if we're using global time, the interval is not fixed as it - // jumps one hour at the DST crossover - } - else if (variableDayLength && - (interval === timeUnits.day || interval === timeUnits.week)) { - t = time.makeTime(minYear, minMonth, minDateDate + - i * count * (interval === timeUnits.day ? 1 : 7)); - } - else if (variableDayLength && - interval === timeUnits.hour && - count > 1) { - // make sure higher ranks are preserved across DST (#6797, - // #7621) - t = time.makeTime(minYear, minMonth, minDateDate, minHours + i * count); - // else, the interval is fixed and we use simple addition - } - else { - t += interval * count; - } - i++; - } - // push the last time - tickPositions.push(t); - // Handle higher ranks. Mark new days if the time is on midnight - // (#950, #1649, #1760, #3349). Use a reasonable dropout threshold - // to prevent looping over dense data grouping (#6156). - if (interval <= timeUnits.hour && tickPositions.length < 10000) { - tickPositions.forEach(function (t) { - if ( - // Speed optimization, no need to run dateFormat unless - // we're on a full or half hour - t % 1800000 === 0 && - // Check for local or global midnight - time.dateFormat('%H%M%S%L', t) === '000000000') { - higherRanks[t] = 'day'; - } - }); - } - } - // record information on the chosen unit - for dynamic label formatter - tickPositions.info = extend(normalizedInterval, { - higherRanks: higherRanks, - totalRange: interval * count - }); - return tickPositions; - }; - return Time; - }()); - H.Time = Time; - - return H.Time; - }); - _registerModule(_modules, 'Core/Options.js', [_modules['Core/Globals.js'], _modules['Core/Color/Color.js'], _modules['Core/Time.js'], _modules['Core/Utilities.js']], function (H, Color, Time, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var isTouchDevice = H.isTouchDevice, - svg = H.svg; - var color = Color.parse; - var merge = U.merge; - /** - * @typedef {"plotBox"|"spacingBox"} Highcharts.ButtonRelativeToValue - */ - /** - * Gets fired when a series is added to the chart after load time, using the - * `addSeries` method. Returning `false` prevents the series from being added. - * - * @callback Highcharts.ChartAddSeriesCallbackFunction - * - * @param {Highcharts.Chart} this - * The chart on which the event occured. - * - * @param {Highcharts.ChartAddSeriesEventObject} event - * The event that occured. - */ - /** - * Contains common event information. Through the `options` property you can - * access the series options that were passed to the `addSeries` method. - * - * @interface Highcharts.ChartAddSeriesEventObject - */ /** - * The series options that were passed to the `addSeries` method. - * @name Highcharts.ChartAddSeriesEventObject#options - * @type {Highcharts.SeriesOptionsType} - */ /** - * Prevents the default behaviour of the event. - * @name Highcharts.ChartAddSeriesEventObject#preventDefault - * @type {Function} - */ /** - * The event target. - * @name Highcharts.ChartAddSeriesEventObject#target - * @type {Highcharts.Chart} - */ /** - * The event type. - * @name Highcharts.ChartAddSeriesEventObject#type - * @type {"addSeries"} - */ - /** - * Gets fired when clicking on the plot background. - * - * @callback Highcharts.ChartClickCallbackFunction - * - * @param {Highcharts.Chart} this - * The chart on which the event occured. - * - * @param {Highcharts.PointerEventObject} event - * The event that occured. - */ - /** - * Contains an axes of the clicked spot. - * - * @interface Highcharts.ChartClickEventAxisObject - */ /** - * Axis at the clicked spot. - * @name Highcharts.ChartClickEventAxisObject#axis - * @type {Highcharts.Axis} - */ /** - * Axis value at the clicked spot. - * @name Highcharts.ChartClickEventAxisObject#value - * @type {number} - */ - /** - * Contains information about the clicked spot on the chart. Remember the unit - * of a datetime axis is milliseconds since 1970-01-01 00:00:00. - * - * @interface Highcharts.ChartClickEventObject - * @extends Highcharts.PointerEventObject - */ /** - * Information about the x-axis on the clicked spot. - * @name Highcharts.ChartClickEventObject#xAxis - * @type {Array<Highcharts.ChartClickEventAxisObject>} - */ /** - * Information about the y-axis on the clicked spot. - * @name Highcharts.ChartClickEventObject#yAxis - * @type {Array<Highcharts.ChartClickEventAxisObject>} - */ /** - * Information about the z-axis on the clicked spot. - * @name Highcharts.ChartClickEventObject#zAxis - * @type {Array<Highcharts.ChartClickEventAxisObject>|undefined} - */ - /** - * Gets fired when the chart is finished loading. - * - * @callback Highcharts.ChartLoadCallbackFunction - * - * @param {Highcharts.Chart} this - * The chart on which the event occured. - * - * @param {global.Event} event - * The event that occured. - */ - /** - * Fires when the chart is redrawn, either after a call to `chart.redraw()` or - * after an axis, series or point is modified with the `redraw` option set to - * `true`. - * - * @callback Highcharts.ChartRedrawCallbackFunction - * - * @param {Highcharts.Chart} this - * The chart on which the event occured. - * - * @param {global.Event} event - * The event that occured. - */ - /** - * Gets fired after initial load of the chart (directly after the `load` event), - * and after each redraw (directly after the `redraw` event). - * - * @callback Highcharts.ChartRenderCallbackFunction - * - * @param {Highcharts.Chart} this - * The chart on which the event occured. - * - * @param {global.Event} event - * The event that occured. - */ - /** - * Gets fired when an area of the chart has been selected. The default action - * for the selection event is to zoom the chart to the selected area. It can be - * prevented by calling `event.preventDefault()` or return false. - * - * @callback Highcharts.ChartSelectionCallbackFunction - * - * @param {Highcharts.Chart} this - * The chart on which the event occured. - * - * @param {global.ChartSelectionContextObject} event - * Event informations - * - * @return {boolean|undefined} - * Return false to prevent the default action, usually zoom. - */ - /** - * The primary axes are `xAxis[0]` and `yAxis[0]`. Remember the unit of a - * datetime axis is milliseconds since 1970-01-01 00:00:00. - * - * @interface Highcharts.ChartSelectionContextObject - * @extends global.Event - */ /** - * Arrays containing the axes of each dimension and each axis' min and max - * values. - * @name Highcharts.ChartSelectionContextObject#xAxis - * @type {Array<Highcharts.ChartSelectionAxisContextObject>} - */ /** - * Arrays containing the axes of each dimension and each axis' min and max - * values. - * @name Highcharts.ChartSelectionContextObject#yAxis - * @type {Array<Highcharts.ChartSelectionAxisContextObject>} - */ - /** - * Axis context of the selection. - * - * @interface Highcharts.ChartSelectionAxisContextObject - */ /** - * The selected Axis. - * @name Highcharts.ChartSelectionAxisContextObject#axis - * @type {Highcharts.Axis} - */ /** - * The maximum axis value, either automatic or set manually. - * @name Highcharts.ChartSelectionAxisContextObject#max - * @type {number} - */ /** - * The minimum axis value, either automatic or set manually. - * @name Highcharts.ChartSelectionAxisContextObject#min - * @type {number} - */ - ''; // detach doclets above - /* ************************************************************************** * - * Handle the options * - * ************************************************************************** */ - /** - * Global default settings. - * - * @name Highcharts.defaultOptions - * @type {Highcharts.Options} - */ /** - * @optionparent - */ - H.defaultOptions = { - /** - * An array containing the default colors for the chart's series. When - * all colors are used, new colors are pulled from the start again. - * - * Default colors can also be set on a series or series.type basis, - * see [column.colors](#plotOptions.column.colors), - * [pie.colors](#plotOptions.pie.colors). - * - * In styled mode, the colors option doesn't exist. Instead, colors - * are defined in CSS and applied either through series or point class - * names, or through the [chart.colorCount](#chart.colorCount) option. - * - * - * ### Legacy - * - * In Highcharts 3.x, the default colors were: - * ```js - * colors: ['#2f7ed8', '#0d233a', '#8bbc21', '#910000', '#1aadce', - * '#492970', '#f28f43', '#77a1e5', '#c42525', '#a6c96a'] - * ``` - * - * In Highcharts 2.x, the default colors were: - * ```js - * colors: ['#4572A7', '#AA4643', '#89A54E', '#80699B', '#3D96AE', - * '#DB843D', '#92A8CD', '#A47D7C', '#B5CA92'] - * ``` - * - * @sample {highcharts} highcharts/chart/colors/ - * Assign a global color theme - * - * @type {Array<Highcharts.ColorString>} - * @default ["#7cb5ec", "#434348", "#90ed7d", "#f7a35c", "#8085e9", - * "#f15c80", "#e4d354", "#2b908f", "#f45b5b", "#91e8e1"] - */ - colors: '#7cb5ec #434348 #90ed7d #f7a35c #8085e9 #f15c80 #e4d354 #2b908f #f45b5b #91e8e1'.split(' '), - /** - * Styled mode only. Configuration object for adding SVG definitions for - * reusable elements. See [gradients, shadows and - * patterns](https://www.highcharts.com/docs/chart-design-and-style/gradients-shadows-and-patterns) - * for more information and code examples. - * - * @type {*} - * @since 5.0.0 - * @apioption defs - */ - /** - * @ignore-option - */ - symbols: ['circle', 'diamond', 'square', 'triangle', 'triangle-down'], - /** - * The language object is global and it can't be set on each chart - * initialization. Instead, use `Highcharts.setOptions` to set it before any - * chart is initialized. - * - * ```js - * Highcharts.setOptions({ - * lang: { - * months: [ - * 'Janvier', 'Février', 'Mars', 'Avril', - * 'Mai', 'Juin', 'Juillet', 'Août', - * 'Septembre', 'Octobre', 'Novembre', 'Décembre' - * ], - * weekdays: [ - * 'Dimanche', 'Lundi', 'Mardi', 'Mercredi', - * 'Jeudi', 'Vendredi', 'Samedi' - * ] - * } - * }); - * ``` - */ - lang: { - /** - * The loading text that appears when the chart is set into the loading - * state following a call to `chart.showLoading`. - */ - loading: 'Loading...', - /** - * An array containing the months names. Corresponds to the `%B` format - * in `Highcharts.dateFormat()`. - * - * @type {Array<string>} - * @default ["January", "February", "March", "April", "May", "June", - * "July", "August", "September", "October", "November", - * "December"] - */ - months: [ - 'January', 'February', 'March', 'April', 'May', 'June', 'July', - 'August', 'September', 'October', 'November', 'December' - ], - /** - * An array containing the months names in abbreviated form. Corresponds - * to the `%b` format in `Highcharts.dateFormat()`. - * - * @type {Array<string>} - * @default ["Jan", "Feb", "Mar", "Apr", "May", "Jun", - * "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"] - */ - shortMonths: [ - 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', - 'Aug', 'Sep', 'Oct', 'Nov', 'Dec' - ], - /** - * An array containing the weekday names. - * - * @type {Array<string>} - * @default ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", - * "Friday", "Saturday"] - */ - weekdays: [ - 'Sunday', 'Monday', 'Tuesday', 'Wednesday', - 'Thursday', 'Friday', 'Saturday' - ], - /** - * Short week days, starting Sunday. If not specified, Highcharts uses - * the first three letters of the `lang.weekdays` option. - * - * @sample highcharts/lang/shortweekdays/ - * Finnish two-letter abbreviations - * - * @type {Array<string>} - * @since 4.2.4 - * @apioption lang.shortWeekdays - */ - /** - * What to show in a date field for invalid dates. Defaults to an empty - * string. - * - * @type {string} - * @since 4.1.8 - * @product highcharts highstock - * @apioption lang.invalidDate - */ - /** - * The title appearing on hovering the zoom in button. The text itself - * defaults to "+" and can be changed in the button options. - * - * @type {string} - * @default Zoom in - * @product highmaps - * @apioption lang.zoomIn - */ - /** - * The title appearing on hovering the zoom out button. The text itself - * defaults to "-" and can be changed in the button options. - * - * @type {string} - * @default Zoom out - * @product highmaps - * @apioption lang.zoomOut - */ - /** - * The default decimal point used in the `Highcharts.numberFormat` - * method unless otherwise specified in the function arguments. - * - * @since 1.2.2 - */ - decimalPoint: '.', - /** - * [Metric prefixes](https://en.wikipedia.org/wiki/Metric_prefix) used - * to shorten high numbers in axis labels. Replacing any of the - * positions with `null` causes the full number to be written. Setting - * `numericSymbols` to `null` disables shortening altogether. - * - * @sample {highcharts} highcharts/lang/numericsymbols/ - * Replacing the symbols with text - * @sample {highstock} highcharts/lang/numericsymbols/ - * Replacing the symbols with text - * - * @type {Array<string>} - * @default ["k", "M", "G", "T", "P", "E"] - * @since 2.3.0 - */ - numericSymbols: ['k', 'M', 'G', 'T', 'P', 'E'], - /** - * The magnitude of [numericSymbols](#lang.numericSymbol) replacements. - * Use 10000 for Japanese, Korean and various Chinese locales, which - * use symbols for 10^4, 10^8 and 10^12. - * - * @sample highcharts/lang/numericsymbolmagnitude/ - * 10000 magnitude for Japanese - * - * @type {number} - * @default 1000 - * @since 5.0.3 - * @apioption lang.numericSymbolMagnitude - */ - /** - * The text for the label appearing when a chart is zoomed. - * - * @since 1.2.4 - */ - resetZoom: 'Reset zoom', - /** - * The tooltip title for the label appearing when a chart is zoomed. - * - * @since 1.2.4 - */ - resetZoomTitle: 'Reset zoom level 1:1', - /** - * The default thousands separator used in the `Highcharts.numberFormat` - * method unless otherwise specified in the function arguments. Defaults - * to a single space character, which is recommended in - * [ISO 31-0](https://en.wikipedia.org/wiki/ISO_31-0#Numbers) and works - * across Anglo-American and continental European languages. - * - * @default \u0020 - * @since 1.2.2 - */ - thousandsSep: ' ' - }, - /** - * Global options that don't apply to each chart. These options, like - * the `lang` options, must be set using the `Highcharts.setOptions` - * method. - * - * ```js - * Highcharts.setOptions({ - * global: { - * useUTC: false - * } - * }); - * ``` - */ - /** - * _Canvg rendering for Android 2.x is removed as of Highcharts 5.0\. - * Use the [libURL](#exporting.libURL) option to configure exporting._ - * - * The URL to the additional file to lazy load for Android 2.x devices. - * These devices don't support SVG, so we download a helper file that - * contains [canvg](https://github.com/canvg/canvg), its dependency - * rbcolor, and our own CanVG Renderer class. To avoid hotlinking to - * our site, you can install canvas-tools.js on your own server and - * change this option accordingly. - * - * @deprecated - * - * @type {string} - * @default https://code.highcharts.com/{version}/modules/canvas-tools.js - * @product highcharts highmaps - * @apioption global.canvasToolsURL - */ - /** - * This option is deprecated since v6.0.5. Instead, use - * [time.useUTC](#time.useUTC) that supports individual time settings - * per chart. - * - * @deprecated - * - * @type {boolean} - * @apioption global.useUTC - */ - /** - * This option is deprecated since v6.0.5. Instead, use - * [time.Date](#time.Date) that supports individual time settings - * per chart. - * - * @deprecated - * - * @type {Function} - * @product highcharts highstock - * @apioption global.Date - */ - /** - * This option is deprecated since v6.0.5. Instead, use - * [time.getTimezoneOffset](#time.getTimezoneOffset) that supports - * individual time settings per chart. - * - * @deprecated - * - * @type {Function} - * @product highcharts highstock - * @apioption global.getTimezoneOffset - */ - /** - * This option is deprecated since v6.0.5. Instead, use - * [time.timezone](#time.timezone) that supports individual time - * settings per chart. - * - * @deprecated - * - * @type {string} - * @product highcharts highstock - * @apioption global.timezone - */ - /** - * This option is deprecated since v6.0.5. Instead, use - * [time.timezoneOffset](#time.timezoneOffset) that supports individual - * time settings per chart. - * - * @deprecated - * - * @type {number} - * @product highcharts highstock - * @apioption global.timezoneOffset - */ - global: {}, - /** - * Time options that can apply globally or to individual charts. These - * settings affect how `datetime` axes are laid out, how tooltips are - * formatted, how series - * [pointIntervalUnit](#plotOptions.series.pointIntervalUnit) works and how - * the Highstock range selector handles time. - * - * The common use case is that all charts in the same Highcharts object - * share the same time settings, in which case the global settings are set - * using `setOptions`. - * - * ```js - * // Apply time settings globally - * Highcharts.setOptions({ - * time: { - * timezone: 'Europe/London' - * } - * }); - * // Apply time settings by instance - * var chart = Highcharts.chart('container', { - * time: { - * timezone: 'America/New_York' - * }, - * series: [{ - * data: [1, 4, 3, 5] - * }] - * }); - * - * // Use the Time object - * console.log( - * 'Current time in New York', - * chart.time.dateFormat('%Y-%m-%d %H:%M:%S', Date.now()) - * ); - * ``` - * - * Since v6.0.5, the time options were moved from the `global` obect to the - * `time` object, and time options can be set on each individual chart. - * - * @sample {highcharts|highstock} - * highcharts/time/timezone/ - * Set the timezone globally - * @sample {highcharts} - * highcharts/time/individual/ - * Set the timezone per chart instance - * @sample {highstock} - * stock/time/individual/ - * Set the timezone per chart instance - * - * @since 6.0.5 - * @optionparent time - */ - time: { - /** - * A custom `Date` class for advanced date handling. For example, - * [JDate](https://github.com/tahajahangir/jdate) can be hooked in to - * handle Jalali dates. - * - * @type {*} - * @since 4.0.4 - * @product highcharts highstock gantt - */ - Date: void 0, - /** - * A callback to return the time zone offset for a given datetime. It - * takes the timestamp in terms of milliseconds since January 1 1970, - * and returns the timezone offset in minutes. This provides a hook - * for drawing time based charts in specific time zones using their - * local DST crossover dates, with the help of external libraries. - * - * @see [global.timezoneOffset](#global.timezoneOffset) - * - * @sample {highcharts|highstock} highcharts/time/gettimezoneoffset/ - * Use moment.js to draw Oslo time regardless of browser locale - * - * @type {Highcharts.TimezoneOffsetCallbackFunction} - * @since 4.1.0 - * @product highcharts highstock gantt - */ - getTimezoneOffset: void 0, - /** - * Requires [moment.js](https://momentjs.com/). If the timezone option - * is specified, it creates a default - * [getTimezoneOffset](#time.getTimezoneOffset) function that looks - * up the specified timezone in moment.js. If moment.js is not included, - * this throws a Highcharts error in the console, but does not crash the - * chart. - * - * @see [getTimezoneOffset](#time.getTimezoneOffset) - * - * @sample {highcharts|highstock} highcharts/time/timezone/ - * Europe/Oslo - * - * @type {string} - * @since 5.0.7 - * @product highcharts highstock gantt - */ - timezone: void 0, - /** - * The timezone offset in minutes. Positive values are west, negative - * values are east of UTC, as in the ECMAScript - * [getTimezoneOffset](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/getTimezoneOffset) - * method. Use this to display UTC based data in a predefined time zone. - * - * @see [time.getTimezoneOffset](#time.getTimezoneOffset) - * - * @sample {highcharts|highstock} highcharts/time/timezoneoffset/ - * Timezone offset - * - * @since 3.0.8 - * @product highcharts highstock gantt - */ - timezoneOffset: 0, - /** - * Whether to use UTC time for axis scaling, tickmark placement and - * time display in `Highcharts.dateFormat`. Advantages of using UTC - * is that the time displays equally regardless of the user agent's - * time zone settings. Local time can be used when the data is loaded - * in real time or when correct Daylight Saving Time transitions are - * required. - * - * @sample {highcharts} highcharts/time/useutc-true/ - * True by default - * @sample {highcharts} highcharts/time/useutc-false/ - * False - */ - useUTC: true - }, - /** - * General options for the chart. - */ - chart: { - /** - * Default `mapData` for all series. If set to a string, it functions - * as an index into the `Highcharts.maps` array. Otherwise it is - * interpreted as map data. - * - * @see [mapData](#series.map.mapData) - * - * @sample maps/demo/geojson - * Loading geoJSON data - * @sample maps/chart/topojson - * Loading topoJSON converted to geoJSON - * - * @type {string|Array<*>|Highcharts.GeoJSON} - * @since 5.0.0 - * @product highmaps - * @apioption chart.map - */ - /** - * Set lat/lon transformation definitions for the chart. If not defined, - * these are extracted from the map data. - * - * @type {*} - * @since 5.0.0 - * @product highmaps - * @apioption chart.mapTransforms - */ - /** - * When using multiple axis, the ticks of two or more opposite axes - * will automatically be aligned by adding ticks to the axis or axes - * with the least ticks, as if `tickAmount` were specified. - * - * This can be prevented by setting `alignTicks` to false. If the grid - * lines look messy, it's a good idea to hide them for the secondary - * axis by setting `gridLineWidth` to 0. - * - * If `startOnTick` or `endOnTick` in an Axis options are set to false, - * then the `alignTicks ` will be disabled for the Axis. - * - * Disabled for logarithmic axes. - * - * @sample {highcharts} highcharts/chart/alignticks-true/ - * True by default - * @sample {highcharts} highcharts/chart/alignticks-false/ - * False - * @sample {highstock} stock/chart/alignticks-true/ - * True by default - * @sample {highstock} stock/chart/alignticks-false/ - * False - * - * @type {boolean} - * @default true - * @product highcharts highstock gantt - * @apioption chart.alignTicks - */ - /** - * Set the overall animation for all chart updating. Animation can be - * disabled throughout the chart by setting it to false here. It can - * be overridden for each individual API method as a function parameter. - * The only animation not affected by this option is the initial series - * animation, see [plotOptions.series.animation]( - * #plotOptions.series.animation). - * - * The animation can either be set as a boolean or a configuration - * object. If `true`, it will use the 'swing' jQuery easing and a - * duration of 500 ms. If used as a configuration object, the following - * properties are supported: - * - * - `defer`: The animation delay time in milliseconds. - * - * - `duration`: The duration of the animation in milliseconds. - * - * - `easing`: A string reference to an easing function set on the - * `Math` object. See - * [the easing demo](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-animation-easing/). - * - * When zooming on a series with less than 100 points, the chart redraw - * will be done with animation, but in case of more data points, it is - * necessary to set this option to ensure animation on zoom. - * - * @sample {highcharts} highcharts/chart/animation-none/ - * Updating with no animation - * @sample {highcharts} highcharts/chart/animation-duration/ - * With a longer duration - * @sample {highcharts} highcharts/chart/animation-easing/ - * With a jQuery UI easing - * @sample {highmaps} maps/chart/animation-none/ - * Updating with no animation - * @sample {highmaps} maps/chart/animation-duration/ - * With a longer duration - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - * @default undefined - * @apioption chart.animation - */ - /** - * A CSS class name to apply to the charts container `div`, allowing - * unique CSS styling for each chart. - * - * @type {string} - * @apioption chart.className - */ - /** - * Event listeners for the chart. - * - * @apioption chart.events - */ - /** - * Fires when a series is added to the chart after load time, using the - * `addSeries` method. One parameter, `event`, is passed to the - * function, containing common event information. Through - * `event.options` you can access the series options that were passed to - * the `addSeries` method. Returning false prevents the series from - * being added. - * - * @sample {highcharts} highcharts/chart/events-addseries/ - * Alert on add series - * @sample {highstock} stock/chart/events-addseries/ - * Alert on add series - * - * @type {Highcharts.ChartAddSeriesCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Chart - * @apioption chart.events.addSeries - */ - /** - * Fires when clicking on the plot background. One parameter, `event`, - * is passed to the function, containing common event information. - * - * Information on the clicked spot can be found through `event.xAxis` - * and `event.yAxis`, which are arrays containing the axes of each - * dimension and each axis' value at the clicked spot. The primary axes - * are `event.xAxis[0]` and `event.yAxis[0]`. Remember the unit of a - * datetime axis is milliseconds since 1970-01-01 00:00:00. - * - * ```js - * click: function(e) { - * console.log( - * Highcharts.dateFormat('%Y-%m-%d %H:%M:%S', e.xAxis[0].value), - * e.yAxis[0].value - * ) - * } - * ``` - * - * @sample {highcharts} highcharts/chart/events-click/ - * Alert coordinates on click - * @sample {highcharts} highcharts/chart/events-container/ - * Alternatively, attach event to container - * @sample {highstock} stock/chart/events-click/ - * Alert coordinates on click - * @sample {highstock} highcharts/chart/events-container/ - * Alternatively, attach event to container - * @sample {highmaps} maps/chart/events-click/ - * Record coordinates on click - * @sample {highmaps} highcharts/chart/events-container/ - * Alternatively, attach event to container - * - * @type {Highcharts.ChartClickCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Chart - * @apioption chart.events.click - */ - /** - * Fires when the chart is finished loading. Since v4.2.2, it also waits - * for images to be loaded, for example from point markers. One - * parameter, `event`, is passed to the function, containing common - * event information. - * - * There is also a second parameter to the chart constructor where a - * callback function can be passed to be executed on chart.load. - * - * @sample {highcharts} highcharts/chart/events-load/ - * Alert on chart load - * @sample {highstock} stock/chart/events-load/ - * Alert on chart load - * @sample {highmaps} maps/chart/events-load/ - * Add series on chart load - * - * @type {Highcharts.ChartLoadCallbackFunction} - * @context Highcharts.Chart - * @apioption chart.events.load - */ - /** - * Fires when the chart is redrawn, either after a call to - * `chart.redraw()` or after an axis, series or point is modified with - * the `redraw` option set to `true`. One parameter, `event`, is passed - * to the function, containing common event information. - * - * @sample {highcharts} highcharts/chart/events-redraw/ - * Alert on chart redraw - * @sample {highstock} stock/chart/events-redraw/ - * Alert on chart redraw when adding a series or moving the - * zoomed range - * @sample {highmaps} maps/chart/events-redraw/ - * Set subtitle on chart redraw - * - * @type {Highcharts.ChartRedrawCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Chart - * @apioption chart.events.redraw - */ - /** - * Fires after initial load of the chart (directly after the `load` - * event), and after each redraw (directly after the `redraw` event). - * - * @type {Highcharts.ChartRenderCallbackFunction} - * @since 5.0.7 - * @context Highcharts.Chart - * @apioption chart.events.render - */ - /** - * Fires when an area of the chart has been selected. Selection is - * enabled by setting the chart's zoomType. One parameter, `event`, is - * passed to the function, containing common event information. The - * default action for the selection event is to zoom the chart to the - * selected area. It can be prevented by calling - * `event.preventDefault()` or return false. - * - * Information on the selected area can be found through `event.xAxis` - * and `event.yAxis`, which are arrays containing the axes of each - * dimension and each axis' min and max values. The primary axes are - * `event.xAxis[0]` and `event.yAxis[0]`. Remember the unit of a - * datetime axis is milliseconds since 1970-01-01 00:00:00. - * - * ```js - * selection: function(event) { - * // log the min and max of the primary, datetime x-axis - * console.log( - * Highcharts.dateFormat( - * '%Y-%m-%d %H:%M:%S', - * event.xAxis[0].min - * ), - * Highcharts.dateFormat( - * '%Y-%m-%d %H:%M:%S', - * event.xAxis[0].max - * ) - * ); - * // log the min and max of the y axis - * console.log(event.yAxis[0].min, event.yAxis[0].max); - * } - * ``` - * - * @sample {highcharts} highcharts/chart/events-selection/ - * Report on selection and reset - * @sample {highcharts} highcharts/chart/events-selection-points/ - * Select a range of points through a drag selection - * @sample {highstock} stock/chart/events-selection/ - * Report on selection and reset - * @sample {highstock} highcharts/chart/events-selection-points/ - * Select a range of points through a drag selection - * (Highcharts) - * - * @type {Highcharts.ChartSelectionCallbackFunction} - * @apioption chart.events.selection - */ - /** - * The margin between the outer edge of the chart and the plot area. - * The numbers in the array designate top, right, bottom and left - * respectively. Use the options `marginTop`, `marginRight`, - * `marginBottom` and `marginLeft` for shorthand setting of one option. - * - * By default there is no margin. The actual space is dynamically - * calculated from the offset of axis labels, axis title, title, - * subtitle and legend in addition to the `spacingTop`, `spacingRight`, - * `spacingBottom` and `spacingLeft` options. - * - * @sample {highcharts} highcharts/chart/margins-zero/ - * Zero margins - * @sample {highstock} stock/chart/margin-zero/ - * Zero margins - * - * @type {number|Array<number>} - * @apioption chart.margin - */ - /** - * The margin between the bottom outer edge of the chart and the plot - * area. Use this to set a fixed pixel value for the margin as opposed - * to the default dynamic margin. See also `spacingBottom`. - * - * @sample {highcharts} highcharts/chart/marginbottom/ - * 100px bottom margin - * @sample {highstock} stock/chart/marginbottom/ - * 100px bottom margin - * @sample {highmaps} maps/chart/margin/ - * 100px margins - * - * @type {number} - * @since 2.0 - * @apioption chart.marginBottom - */ - /** - * The margin between the left outer edge of the chart and the plot - * area. Use this to set a fixed pixel value for the margin as opposed - * to the default dynamic margin. See also `spacingLeft`. - * - * @sample {highcharts} highcharts/chart/marginleft/ - * 150px left margin - * @sample {highstock} stock/chart/marginleft/ - * 150px left margin - * @sample {highmaps} maps/chart/margin/ - * 100px margins - * - * @type {number} - * @since 2.0 - * @apioption chart.marginLeft - */ - /** - * The margin between the right outer edge of the chart and the plot - * area. Use this to set a fixed pixel value for the margin as opposed - * to the default dynamic margin. See also `spacingRight`. - * - * @sample {highcharts} highcharts/chart/marginright/ - * 100px right margin - * @sample {highstock} stock/chart/marginright/ - * 100px right margin - * @sample {highmaps} maps/chart/margin/ - * 100px margins - * - * @type {number} - * @since 2.0 - * @apioption chart.marginRight - */ - /** - * The margin between the top outer edge of the chart and the plot area. - * Use this to set a fixed pixel value for the margin as opposed to - * the default dynamic margin. See also `spacingTop`. - * - * @sample {highcharts} highcharts/chart/margintop/ 100px top margin - * @sample {highstock} stock/chart/margintop/ - * 100px top margin - * @sample {highmaps} maps/chart/margin/ - * 100px margins - * - * @type {number} - * @since 2.0 - * @apioption chart.marginTop - */ - /** - * Callback function to override the default function that formats all - * the numbers in the chart. Returns a string with the formatted number. - * - * @sample highcharts/members/highcharts-numberformat - * Arabic digits in Highcharts - * @type {Highcharts.NumberFormatterCallbackFunction} - * @since 8.0.0 - * @apioption chart.numberFormatter - */ - /** - * Allows setting a key to switch between zooming and panning. Can be - * one of `alt`, `ctrl`, `meta` (the command key on Mac and Windows - * key on Windows) or `shift`. The keys are mapped directly to the key - * properties of the click event argument (`event.altKey`, - * `event.ctrlKey`, `event.metaKey` and `event.shiftKey`). - * - * @type {string} - * @since 4.0.3 - * @product highcharts gantt - * @validvalue ["alt", "ctrl", "meta", "shift"] - * @apioption chart.panKey - */ - /** - * Allow panning in a chart. Best used with [panKey](#chart.panKey) - * to combine zooming and panning. - * - * On touch devices, when the [tooltip.followTouchMove]( - * #tooltip.followTouchMove) option is `true` (default), panning - * requires two fingers. To allow panning with one finger, set - * `followTouchMove` to `false`. - * - * @sample {highcharts} highcharts/chart/pankey/ Zooming and panning - * @sample {highstock} stock/chart/panning/ Zooming and xy panning - * - * @product highcharts highstock gantt - * @apioption chart.panning - */ - /** - * Enable or disable chart panning. - * - * @type {boolean} - * @default {highcharts} false - * @default {highstock} true - * @apioption chart.panning.enabled - */ - /** - * Decides in what dimensions the user can pan the chart. Can be - * one of `x`, `y`, or `xy`. - * - * @sample {highcharts} highcharts/chart/panning-type - * Zooming and xy panning - * - * @type {string} - * @validvalue ["x", "y", "xy"] - * @default x - * @apioption chart.panning.type - */ - /** - * Equivalent to [zoomType](#chart.zoomType), but for multitouch - * gestures only. By default, the `pinchType` is the same as the - * `zoomType` setting. However, pinching can be enabled separately in - * some cases, for example in stock charts where a mouse drag pans the - * chart, while pinching is enabled. When [tooltip.followTouchMove]( - * #tooltip.followTouchMove) is true, pinchType only applies to - * two-finger touches. - * - * @type {string} - * @default {highcharts} undefined - * @default {highstock} x - * @since 3.0 - * @product highcharts highstock gantt - * @validvalue ["x", "y", "xy"] - * @apioption chart.pinchType - */ - /** - * Whether to apply styled mode. When in styled mode, no presentational - * attributes or CSS are applied to the chart SVG. Instead, CSS rules - * are required to style the chart. The default style sheet is - * available from `https://code.highcharts.com/css/highcharts.css`. - * - * @type {boolean} - * @default false - * @since 7.0 - * @apioption chart.styledMode - */ - styledMode: false, - /** - * The corner radius of the outer chart border. - * - * @sample {highcharts} highcharts/chart/borderradius/ - * 20px radius - * @sample {highstock} stock/chart/border/ - * 10px radius - * @sample {highmaps} maps/chart/border/ - * Border options - * - */ - borderRadius: 0, - /** - * In styled mode, this sets how many colors the class names - * should rotate between. With ten colors, series (or points) are - * given class names like `highcharts-color-0`, `highcharts-color-0` - * [...] `highcharts-color-9`. The equivalent in non-styled mode - * is to set colors using the [colors](#colors) setting. - * - * @since 5.0.0 - */ - colorCount: 10, - /** - * Alias of `type`. - * - * @sample {highcharts} highcharts/chart/defaultseriestype/ - * Bar - * - * @deprecated - * - * @product highcharts - */ - defaultSeriesType: 'line', - /** - * If true, the axes will scale to the remaining visible series once - * one series is hidden. If false, hiding and showing a series will - * not affect the axes or the other series. For stacks, once one series - * within the stack is hidden, the rest of the stack will close in - * around it even if the axis is not affected. - * - * @sample {highcharts} highcharts/chart/ignorehiddenseries-true/ - * True by default - * @sample {highcharts} highcharts/chart/ignorehiddenseries-false/ - * False - * @sample {highcharts} highcharts/chart/ignorehiddenseries-true-stacked/ - * True with stack - * @sample {highstock} stock/chart/ignorehiddenseries-true/ - * True by default - * @sample {highstock} stock/chart/ignorehiddenseries-false/ - * False - * - * @since 1.2.0 - * @product highcharts highstock gantt - */ - ignoreHiddenSeries: true, - /** - * Whether to invert the axes so that the x axis is vertical and y axis - * is horizontal. When `true`, the x axis is [reversed](#xAxis.reversed) - * by default. - * - * @productdesc {highcharts} - * If a bar series is present in the chart, it will be inverted - * automatically. Inverting the chart doesn't have an effect if there - * are no cartesian series in the chart, or if the chart is - * [polar](#chart.polar). - * - * @sample {highcharts} highcharts/chart/inverted/ - * Inverted line - * @sample {highstock} stock/navigator/inverted/ - * Inverted stock chart - * - * @type {boolean} - * @default false - * @product highcharts highstock gantt - * @apioption chart.inverted - */ - /** - * The distance between the outer edge of the chart and the content, - * like title or legend, or axis title and labels if present. The - * numbers in the array designate top, right, bottom and left - * respectively. Use the options spacingTop, spacingRight, spacingBottom - * and spacingLeft options for shorthand setting of one option. - * - * @type {Array<number>} - * @see [chart.margin](#chart.margin) - * @default [10, 10, 15, 10] - * @since 3.0.6 - */ - spacing: [10, 10, 15, 10], - /** - * The button that appears after a selection zoom, allowing the user - * to reset zoom. - */ - resetZoomButton: { - /** - * What frame the button placement should be related to. Can be - * either `plotBox` or `spacingBox`. - * - * @sample {highcharts} highcharts/chart/resetzoombutton-relativeto/ - * Relative to the chart - * @sample {highstock} highcharts/chart/resetzoombutton-relativeto/ - * Relative to the chart - * - * @type {Highcharts.ButtonRelativeToValue} - * @default plot - * @since 2.2 - * @apioption chart.resetZoomButton.relativeTo - */ - /** - * A collection of attributes for the button. The object takes SVG - * attributes like `fill`, `stroke`, `stroke-width` or `r`, the - * border radius. The theme also supports `style`, a collection of - * CSS properties for the text. Equivalent attributes for the hover - * state are given in `theme.states.hover`. - * - * @sample {highcharts} highcharts/chart/resetzoombutton-theme/ - * Theming the button - * @sample {highstock} highcharts/chart/resetzoombutton-theme/ - * Theming the button - * - * @type {Highcharts.SVGAttributes} - * @since 2.2 - */ - theme: { - /** @internal */ - zIndex: 6 - }, - /** - * The position of the button. - * - * @sample {highcharts} highcharts/chart/resetzoombutton-position/ - * Above the plot area - * @sample {highstock} highcharts/chart/resetzoombutton-position/ - * Above the plot area - * @sample {highmaps} highcharts/chart/resetzoombutton-position/ - * Above the plot area - * - * @type {Highcharts.AlignObject} - * @since 2.2 - */ - position: { - /** - * The horizontal alignment of the button. - */ - align: 'right', - /** - * The horizontal offset of the button. - */ - x: -10, - /** - * The vertical alignment of the button. - * - * @type {Highcharts.VerticalAlignValue} - * @default top - * @apioption chart.resetZoomButton.position.verticalAlign - */ - /** - * The vertical offset of the button. - */ - y: 10 - } - }, - /** - * The pixel width of the plot area border. - * - * @sample {highcharts} highcharts/chart/plotborderwidth/ - * 1px border - * @sample {highstock} stock/chart/plotborder/ - * 2px border - * @sample {highmaps} maps/chart/plotborder/ - * Plot border options - * - * @type {number} - * @default 0 - * @apioption chart.plotBorderWidth - */ - /** - * Whether to apply a drop shadow to the plot area. Requires that - * plotBackgroundColor be set. The shadow can be an object configuration - * containing `color`, `offsetX`, `offsetY`, `opacity` and `width`. - * - * @sample {highcharts} highcharts/chart/plotshadow/ - * Plot shadow - * @sample {highstock} stock/chart/plotshadow/ - * Plot shadow - * @sample {highmaps} maps/chart/plotborder/ - * Plot border options - * - * @type {boolean|Highcharts.CSSObject} - * @default false - * @apioption chart.plotShadow - */ - /** - * When true, cartesian charts like line, spline, area and column are - * transformed into the polar coordinate system. This produces _polar - * charts_, also known as _radar charts_. - * - * @sample {highcharts} highcharts/demo/polar/ - * Polar chart - * @sample {highcharts} highcharts/demo/polar-wind-rose/ - * Wind rose, stacked polar column chart - * @sample {highcharts} highcharts/demo/polar-spider/ - * Spider web chart - * @sample {highcharts} highcharts/parallel-coordinates/polar/ - * Star plot, multivariate data in a polar chart - * - * @type {boolean} - * @default false - * @since 2.3.0 - * @product highcharts - * @requires highcharts-more - * @apioption chart.polar - */ - /** - * Whether to reflow the chart to fit the width of the container div - * on resizing the window. - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * True by default - * @sample {highcharts} highcharts/chart/reflow-false/ - * False - * @sample {highstock} stock/chart/reflow-true/ - * True by default - * @sample {highstock} stock/chart/reflow-false/ - * False - * @sample {highmaps} maps/chart/reflow-true/ - * True by default - * @sample {highmaps} maps/chart/reflow-false/ - * False - * - * @type {boolean} - * @default true - * @since 2.1 - * @apioption chart.reflow - */ - /** - * The HTML element where the chart will be rendered. If it is a string, - * the element by that id is used. The HTML element can also be passed - * by direct reference, or as the first argument of the chart - * constructor, in which case the option is not needed. - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * String - * @sample {highcharts} highcharts/chart/renderto-object/ - * Object reference - * @sample {highcharts} highcharts/chart/renderto-jquery/ - * Object reference through jQuery - * @sample {highstock} stock/chart/renderto-string/ - * String - * @sample {highstock} stock/chart/renderto-object/ - * Object reference - * @sample {highstock} stock/chart/renderto-jquery/ - * Object reference through jQuery - * - * @type {string|Highcharts.HTMLDOMElement} - * @apioption chart.renderTo - */ - /** - * The background color of the marker square when selecting (zooming - * in on) an area of the chart. - * - * @see In styled mode, the selection marker fill is set with the - * `.highcharts-selection-marker` class. - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @default rgba(51,92,173,0.25) - * @since 2.1.7 - * @apioption chart.selectionMarkerFill - */ - /** - * Whether to apply a drop shadow to the outer chart area. Requires - * that backgroundColor be set. The shadow can be an object - * configuration containing `color`, `offsetX`, `offsetY`, `opacity` and - * `width`. - * - * @sample {highcharts} highcharts/chart/shadow/ - * Shadow - * @sample {highstock} stock/chart/shadow/ - * Shadow - * @sample {highmaps} maps/chart/border/ - * Chart border and shadow - * - * @type {boolean|Highcharts.CSSObject} - * @default false - * @apioption chart.shadow - */ - /** - * Whether to show the axes initially. This only applies to empty charts - * where series are added dynamically, as axes are automatically added - * to cartesian series. - * - * @sample {highcharts} highcharts/chart/showaxes-false/ - * False by default - * @sample {highcharts} highcharts/chart/showaxes-true/ - * True - * - * @type {boolean} - * @since 1.2.5 - * @product highcharts gantt - * @apioption chart.showAxes - */ - /** - * The space between the bottom edge of the chart and the content (plot - * area, axis title and labels, title, subtitle or legend in top - * position). - * - * @sample {highcharts} highcharts/chart/spacingbottom/ - * Spacing bottom set to 100 - * @sample {highstock} stock/chart/spacingbottom/ - * Spacing bottom set to 100 - * @sample {highmaps} maps/chart/spacing/ - * Spacing 100 all around - * - * @type {number} - * @default 15 - * @since 2.1 - * @apioption chart.spacingBottom - */ - /** - * The space between the left edge of the chart and the content (plot - * area, axis title and labels, title, subtitle or legend in top - * position). - * - * @sample {highcharts} highcharts/chart/spacingleft/ - * Spacing left set to 100 - * @sample {highstock} stock/chart/spacingleft/ - * Spacing left set to 100 - * @sample {highmaps} maps/chart/spacing/ - * Spacing 100 all around - * - * @type {number} - * @default 10 - * @since 2.1 - * @apioption chart.spacingLeft - */ - /** - * The space between the right edge of the chart and the content (plot - * area, axis title and labels, title, subtitle or legend in top - * position). - * - * @sample {highcharts} highcharts/chart/spacingright-100/ - * Spacing set to 100 - * @sample {highcharts} highcharts/chart/spacingright-legend/ - * Legend in right position with default spacing - * @sample {highstock} stock/chart/spacingright/ - * Spacing set to 100 - * @sample {highmaps} maps/chart/spacing/ - * Spacing 100 all around - * - * @type {number} - * @default 10 - * @since 2.1 - * @apioption chart.spacingRight - */ - /** - * The space between the top edge of the chart and the content (plot - * area, axis title and labels, title, subtitle or legend in top - * position). - * - * @sample {highcharts} highcharts/chart/spacingtop-100/ - * A top spacing of 100 - * @sample {highcharts} highcharts/chart/spacingtop-10/ - * Floating chart title makes the plot area align to the default - * spacingTop of 10. - * @sample {highstock} stock/chart/spacingtop/ - * A top spacing of 100 - * @sample {highmaps} maps/chart/spacing/ - * Spacing 100 all around - * - * @type {number} - * @default 10 - * @since 2.1 - * @apioption chart.spacingTop - */ - /** - * Additional CSS styles to apply inline to the container `div`. Note - * that since the default font styles are applied in the renderer, it - * is ignorant of the individual chart options and must be set globally. - * - * @see In styled mode, general chart styles can be set with the - * `.highcharts-root` class. - * @sample {highcharts} highcharts/chart/style-serif-font/ - * Using a serif type font - * @sample {highcharts} highcharts/css/em/ - * Styled mode with relative font sizes - * @sample {highstock} stock/chart/style/ - * Using a serif type font - * @sample {highmaps} maps/chart/style-serif-font/ - * Using a serif type font - * - * @type {Highcharts.CSSObject} - * @default {"fontFamily": "\"Lucida Grande\", \"Lucida Sans Unicode\", Verdana, Arial, Helvetica, sans-serif","fontSize":"12px"} - * @apioption chart.style - */ - /** - * The default series type for the chart. Can be any of the chart types - * listed under [plotOptions](#plotOptions) and [series](#series) or can - * be a series provided by an additional module. - * - * In TypeScript this option has no effect in sense of typing and - * instead the `type` option must always be set in the series. - * - * @sample {highcharts} highcharts/chart/type-bar/ - * Bar - * @sample {highstock} stock/chart/type/ - * Areaspline - * @sample {highmaps} maps/chart/type-mapline/ - * Mapline - * - * @type {string} - * @default {highcharts} line - * @default {highstock} line - * @default {highmaps} map - * @since 2.1.0 - * @apioption chart.type - */ - /** - * Decides in what dimensions the user can zoom by dragging the mouse. - * Can be one of `x`, `y` or `xy`. - * - * @see [panKey](#chart.panKey) - * - * @sample {highcharts} highcharts/chart/zoomtype-none/ - * None by default - * @sample {highcharts} highcharts/chart/zoomtype-x/ - * X - * @sample {highcharts} highcharts/chart/zoomtype-y/ - * Y - * @sample {highcharts} highcharts/chart/zoomtype-xy/ - * Xy - * @sample {highstock} stock/demo/basic-line/ - * None by default - * @sample {highstock} stock/chart/zoomtype-x/ - * X - * @sample {highstock} stock/chart/zoomtype-y/ - * Y - * @sample {highstock} stock/chart/zoomtype-xy/ - * Xy - * - * @type {string} - * @product highcharts highstock gantt - * @validvalue ["x", "y", "xy"] - * @apioption chart.zoomType - */ - /** - * An explicit width for the chart. By default (when `null`) the width - * is calculated from the offset width of the containing element. - * - * @sample {highcharts} highcharts/chart/width/ - * 800px wide - * @sample {highstock} stock/chart/width/ - * 800px wide - * @sample {highmaps} maps/chart/size/ - * Chart with explicit size - * - * @type {null|number|string} - */ - width: null, - /** - * An explicit height for the chart. If a _number_, the height is - * given in pixels. If given a _percentage string_ (for example - * `'56%'`), the height is given as the percentage of the actual chart - * width. This allows for preserving the aspect ratio across responsive - * sizes. - * - * By default (when `null`) the height is calculated from the offset - * height of the containing element, or 400 pixels if the containing - * element's height is 0. - * - * @sample {highcharts} highcharts/chart/height/ - * 500px height - * @sample {highstock} stock/chart/height/ - * 300px height - * @sample {highmaps} maps/chart/size/ - * Chart with explicit size - * @sample highcharts/chart/height-percent/ - * Highcharts with percentage height - * - * @type {null|number|string} - */ - height: null, - /** - * The color of the outer chart border. - * - * @see In styled mode, the stroke is set with the - * `.highcharts-background` class. - * - * @sample {highcharts} highcharts/chart/bordercolor/ - * Brown border - * @sample {highstock} stock/chart/border/ - * Brown border - * @sample {highmaps} maps/chart/border/ - * Border options - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - borderColor: '#335cad', - /** - * The pixel width of the outer chart border. - * - * @see In styled mode, the stroke is set with the - * `.highcharts-background` class. - * - * @sample {highcharts} highcharts/chart/borderwidth/ - * 5px border - * @sample {highstock} stock/chart/border/ - * 2px border - * @sample {highmaps} maps/chart/border/ - * Border options - * - * @type {number} - * @default 0 - * @apioption chart.borderWidth - */ - /** - * The background color or gradient for the outer chart area. - * - * @see In styled mode, the background is set with the - * `.highcharts-background` class. - * - * @sample {highcharts} highcharts/chart/backgroundcolor-color/ - * Color - * @sample {highcharts} highcharts/chart/backgroundcolor-gradient/ - * Gradient - * @sample {highstock} stock/chart/backgroundcolor-color/ - * Color - * @sample {highstock} stock/chart/backgroundcolor-gradient/ - * Gradient - * @sample {highmaps} maps/chart/backgroundcolor-color/ - * Color - * @sample {highmaps} maps/chart/backgroundcolor-gradient/ - * Gradient - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - backgroundColor: '#ffffff', - /** - * The background color or gradient for the plot area. - * - * @see In styled mode, the plot background is set with the - * `.highcharts-plot-background` class. - * - * @sample {highcharts} highcharts/chart/plotbackgroundcolor-color/ - * Color - * @sample {highcharts} highcharts/chart/plotbackgroundcolor-gradient/ - * Gradient - * @sample {highstock} stock/chart/plotbackgroundcolor-color/ - * Color - * @sample {highstock} stock/chart/plotbackgroundcolor-gradient/ - * Gradient - * @sample {highmaps} maps/chart/plotbackgroundcolor-color/ - * Color - * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/ - * Gradient - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption chart.plotBackgroundColor - */ - /** - * The URL for an image to use as the plot background. To set an image - * as the background for the entire chart, set a CSS background image - * to the container element. Note that for the image to be applied to - * exported charts, its URL needs to be accessible by the export server. - * - * @see In styled mode, a plot background image can be set with the - * `.highcharts-plot-background` class and a [custom pattern]( - * https://www.highcharts.com/docs/chart-design-and-style/ - * gradients-shadows-and-patterns). - * - * @sample {highcharts} highcharts/chart/plotbackgroundimage/ - * Skies - * @sample {highstock} stock/chart/plotbackgroundimage/ - * Skies - * - * @type {string} - * @apioption chart.plotBackgroundImage - */ - /** - * The color of the inner chart or plot area border. - * - * @see In styled mode, a plot border stroke can be set with the - * `.highcharts-plot-border` class. - * - * @sample {highcharts} highcharts/chart/plotbordercolor/ - * Blue border - * @sample {highstock} stock/chart/plotborder/ - * Blue border - * @sample {highmaps} maps/chart/plotborder/ - * Plot border options - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - plotBorderColor: '#cccccc' - }, - /** - * The chart's main title. - * - * @sample {highmaps} maps/title/title/ - * Title options demonstrated - */ - title: { - /** - * When the title is floating, the plot area will not move to make space - * for it. - * - * @sample {highcharts} highcharts/chart/zoomtype-none/ - * False by default - * @sample {highcharts} highcharts/title/floating/ - * True - title on top of the plot area - * @sample {highstock} stock/chart/title-floating/ - * True - title on top of the plot area - * - * @type {boolean} - * @default false - * @since 2.1 - * @apioption title.floating - */ - /** - * CSS styles for the title. Use this for font styling, but use `align`, - * `x` and `y` for text alignment. - * - * In styled mode, the title style is given in the `.highcharts-title` - * class. - * - * @sample {highcharts} highcharts/title/style/ - * Custom color and weight - * @sample {highstock} stock/chart/title-style/ - * Custom color and weight - * @sample highcharts/css/titles/ - * Styled mode - * - * @type {Highcharts.CSSObject} - * @default {highcharts|highmaps} { "color": "#333333", "fontSize": "18px" } - * @default {highstock} { "color": "#333333", "fontSize": "16px" } - * @apioption title.style - */ - /** - * Whether to - * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the text. - * - * @type {boolean} - * @default false - * @apioption title.useHTML - */ - /** - * The vertical alignment of the title. Can be one of `"top"`, - * `"middle"` and `"bottom"`. When a value is given, the title behaves - * as if [floating](#title.floating) were `true`. - * - * @sample {highcharts} highcharts/title/verticalalign/ - * Chart title in bottom right corner - * @sample {highstock} stock/chart/title-verticalalign/ - * Chart title in bottom right corner - * - * @type {Highcharts.VerticalAlignValue} - * @since 2.1 - * @apioption title.verticalAlign - */ - /** - * The x position of the title relative to the alignment within - * `chart.spacingLeft` and `chart.spacingRight`. - * - * @sample {highcharts} highcharts/title/align/ - * Aligned to the plot area (x = 70px = margin left - spacing - * left) - * @sample {highstock} stock/chart/title-align/ - * Aligned to the plot area (x = 50px = margin left - spacing - * left) - * - * @type {number} - * @default 0 - * @since 2.0 - * @apioption title.x - */ - /** - * The y position of the title relative to the alignment within - * [chart.spacingTop](#chart.spacingTop) and [chart.spacingBottom]( - * #chart.spacingBottom). By default it depends on the font size. - * - * @sample {highcharts} highcharts/title/y/ - * Title inside the plot area - * @sample {highstock} stock/chart/title-verticalalign/ - * Chart title in bottom right corner - * - * @type {number} - * @since 2.0 - * @apioption title.y - */ - /** - * The title of the chart. To disable the title, set the `text` to - * `undefined`. - * - * @sample {highcharts} highcharts/title/text/ - * Custom title - * @sample {highstock} stock/chart/title-text/ - * Custom title - * - * @default {highcharts|highmaps} Chart title - * @default {highstock} undefined - */ - text: 'Chart title', - /** - * The horizontal alignment of the title. Can be one of "left", "center" - * and "right". - * - * @sample {highcharts} highcharts/title/align/ - * Aligned to the plot area (x = 70px = margin left - spacing - * left) - * @sample {highstock} stock/chart/title-align/ - * Aligned to the plot area (x = 50px = margin left - spacing - * left) - * - * @type {Highcharts.AlignValue} - * @since 2.0 - */ - align: 'center', - /** - * The margin between the title and the plot area, or if a subtitle - * is present, the margin between the subtitle and the plot area. - * - * @sample {highcharts} highcharts/title/margin-50/ - * A chart title margin of 50 - * @sample {highcharts} highcharts/title/margin-subtitle/ - * The same margin applied with a subtitle - * @sample {highstock} stock/chart/title-margin/ - * A chart title margin of 50 - * - * @since 2.1 - */ - margin: 15, - /** - * Adjustment made to the title width, normally to reserve space for - * the exporting burger menu. - * - * @sample highcharts/title/widthadjust/ - * Wider menu, greater padding - * - * @since 4.2.5 - */ - widthAdjust: -44 - }, - /** - * The chart's subtitle. This can be used both to display a subtitle below - * the main title, and to display random text anywhere in the chart. The - * subtitle can be updated after chart initialization through the - * `Chart.setTitle` method. - * - * @sample {highmaps} maps/title/subtitle/ - * Subtitle options demonstrated - */ - subtitle: { - /** - * When the subtitle is floating, the plot area will not move to make - * space for it. - * - * @sample {highcharts} highcharts/subtitle/floating/ - * Floating title and subtitle - * @sample {highstock} stock/chart/subtitle-footnote - * Footnote floating at bottom right of plot area - * - * @type {boolean} - * @default false - * @since 2.1 - * @apioption subtitle.floating - */ - /** - * CSS styles for the title. - * - * In styled mode, the subtitle style is given in the - * `.highcharts-subtitle` class. - * - * @sample {highcharts} highcharts/subtitle/style/ - * Custom color and weight - * @sample {highcharts} highcharts/css/titles/ - * Styled mode - * @sample {highstock} stock/chart/subtitle-style - * Custom color and weight - * @sample {highstock} highcharts/css/titles/ - * Styled mode - * @sample {highmaps} highcharts/css/titles/ - * Styled mode - * - * @type {Highcharts.CSSObject} - * @default {"color": "#666666"} - * @apioption subtitle.style - */ - /** - * Whether to - * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the text. - * - * @type {boolean} - * @default false - * @apioption subtitle.useHTML - */ - /** - * The vertical alignment of the title. Can be one of `"top"`, - * `"middle"` and `"bottom"`. When middle, the subtitle behaves as - * floating. - * - * @sample {highcharts} highcharts/subtitle/verticalalign/ - * Footnote at the bottom right of plot area - * @sample {highstock} stock/chart/subtitle-footnote - * Footnote at the bottom right of plot area - * - * @type {Highcharts.VerticalAlignValue} - * @since 2.1 - * @apioption subtitle.verticalAlign - */ - /** - * The x position of the subtitle relative to the alignment within - * `chart.spacingLeft` and `chart.spacingRight`. - * - * @sample {highcharts} highcharts/subtitle/align/ - * Footnote at right of plot area - * @sample {highstock} stock/chart/subtitle-footnote - * Footnote at the bottom right of plot area - * - * @type {number} - * @default 0 - * @since 2.0 - * @apioption subtitle.x - */ - /** - * The y position of the subtitle relative to the alignment within - * `chart.spacingTop` and `chart.spacingBottom`. By default the subtitle - * is laid out below the title unless the title is floating. - * - * @sample {highcharts} highcharts/subtitle/verticalalign/ - * Footnote at the bottom right of plot area - * @sample {highstock} stock/chart/subtitle-footnote - * Footnote at the bottom right of plot area - * - * @type {number} - * @since 2.0 - * @apioption subtitle.y - */ - /** - * The subtitle of the chart. - * - * @sample {highcharts|highstock} highcharts/subtitle/text/ - * Custom subtitle - * @sample {highcharts|highstock} highcharts/subtitle/text-formatted/ - * Formatted and linked text. - */ - text: '', - /** - * The horizontal alignment of the subtitle. Can be one of "left", - * "center" and "right". - * - * @sample {highcharts} highcharts/subtitle/align/ - * Footnote at right of plot area - * @sample {highstock} stock/chart/subtitle-footnote - * Footnote at bottom right of plot area - * - * @type {Highcharts.AlignValue} - * @since 2.0 - */ - align: 'center', - /** - * Adjustment made to the subtitle width, normally to reserve space - * for the exporting burger menu. - * - * @see [title.widthAdjust](#title.widthAdjust) - * - * @sample highcharts/title/widthadjust/ - * Wider menu, greater padding - * - * @since 4.2.5 - */ - widthAdjust: -44 - }, - /** - * The chart's caption, which will render below the chart and will be part - * of exported charts. The caption can be updated after chart initialization - * through the `Chart.update` or `Chart.caption.update` methods. - * - * @sample highcharts/caption/text/ - * A chart with a caption - * @since 7.2.0 - */ - caption: { - /** - * When the caption is floating, the plot area will not move to make - * space for it. - * - * @type {boolean} - * @default false - * @apioption caption.floating - */ - /** - * The margin between the caption and the plot area. - */ - margin: 15, - /** - * CSS styles for the caption. - * - * In styled mode, the caption style is given in the - * `.highcharts-caption` class. - * - * @sample {highcharts} highcharts/css/titles/ - * Styled mode - * - * @type {Highcharts.CSSObject} - * @default {"color": "#666666"} - * @apioption caption.style - */ - /** - * Whether to - * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the text. - * - * @type {boolean} - * @default false - * @apioption caption.useHTML - */ - /** - * The x position of the caption relative to the alignment within - * `chart.spacingLeft` and `chart.spacingRight`. - * - * @type {number} - * @default 0 - * @apioption caption.x - */ - /** - * The y position of the caption relative to the alignment within - * `chart.spacingTop` and `chart.spacingBottom`. - * - * @type {number} - * @apioption caption.y - */ - /** - * The caption text of the chart. - * - * @sample {highcharts} highcharts/caption/text/ - * Custom caption - */ - text: '', - /** - * The horizontal alignment of the caption. Can be one of "left", - * "center" and "right". - * - * @type {Highcharts.AlignValue} - */ - align: 'left', - /** - * The vertical alignment of the caption. Can be one of `"top"`, - * `"middle"` and `"bottom"`. When middle, the caption behaves as - * floating. - * - * @type {Highcharts.VerticalAlignValue} - */ - verticalAlign: 'bottom' - }, - /** - * The plotOptions is a wrapper object for config objects for each series - * type. The config objects for each series can also be overridden for - * each series item as given in the series array. - * - * Configuration options for the series are given in three levels. Options - * for all series in a chart are given in the [plotOptions.series]( - * #plotOptions.series) object. Then options for all series of a specific - * type are given in the plotOptions of that type, for example - * `plotOptions.line`. Next, options for one single series are given in - * [the series array](#series). - */ - plotOptions: {}, - /** - * HTML labels that can be positioned anywhere in the chart area. - * - * This option is deprecated since v7.1.2. Instead, use - * [annotations](#annotations) that support labels. - * - * @deprecated - * @product highcharts highstock - */ - labels: { - /** - * An HTML label that can be positioned anywhere in the chart area. - * - * @deprecated - * @type {Array<*>} - * @apioption labels.items - */ - /** - * Inner HTML or text for the label. - * - * @deprecated - * @type {string} - * @apioption labels.items.html - */ - /** - * CSS styles for each label. To position the label, use left and top - * like this: - * ```js - * style: { - * left: '100px', - * top: '100px' - * } - * ``` - * - * @deprecated - * @type {Highcharts.CSSObject} - * @apioption labels.items.style - */ - /** - * Shared CSS styles for all labels. - * - * @deprecated - * @type {Highcharts.CSSObject} - * @default {"color": "#333333", "position": "absolute"} - */ - style: { - /** - * @ignore-option - */ - position: 'absolute', - /** - * @ignore-option - */ - color: '#333333' - } - }, - /** - * The legend is a box containing a symbol and name for each series - * item or point item in the chart. Each series (or points in case - * of pie charts) is represented by a symbol and its name in the legend. - * - * It is possible to override the symbol creator function and create - * [custom legend symbols](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/studies/legend-custom-symbol/). - * - * @productdesc {highmaps} - * A Highmaps legend by default contains one legend item per series, but if - * a `colorAxis` is defined, the axis will be displayed in the legend. - * Either as a gradient, or as multiple legend items for `dataClasses`. - */ - legend: { - /** - * The background color of the legend. - * - * @see In styled mode, the legend background fill can be applied with - * the `.highcharts-legend-box` class. - * - * @sample {highcharts} highcharts/legend/backgroundcolor/ - * Yellowish background - * @sample {highstock} stock/legend/align/ - * Various legend options - * @sample {highmaps} maps/legend/border-background/ - * Border and background options - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption legend.backgroundColor - */ - /** - * The width of the drawn border around the legend. - * - * @see In styled mode, the legend border stroke width can be applied - * with the `.highcharts-legend-box` class. - * - * @sample {highcharts} highcharts/legend/borderwidth/ - * 2px border width - * @sample {highstock} stock/legend/align/ - * Various legend options - * @sample {highmaps} maps/legend/border-background/ - * Border and background options - * - * @type {number} - * @default 0 - * @apioption legend.borderWidth - */ - /** - * Enable or disable the legend. There is also a series-specific option, - * [showInLegend](#plotOptions.series.showInLegend), that can hide the - * series from the legend. In some series types this is `false` by - * default, so it must set to `true` in order to show the legend for the - * series. - * - * @sample {highcharts} highcharts/legend/enabled-false/ Legend disabled - * @sample {highstock} stock/legend/align/ Various legend options - * @sample {highmaps} maps/legend/enabled-false/ Legend disabled - * - * @default {highstock} false - * @default {highmaps} true - * @default {gantt} false - */ - enabled: true, - /** - * The horizontal alignment of the legend box within the chart area. - * Valid values are `left`, `center` and `right`. - * - * In the case that the legend is aligned in a corner position, the - * `layout` option will determine whether to place it above/below - * or on the side of the plot area. - * - * @sample {highcharts} highcharts/legend/align/ - * Legend at the right of the chart - * @sample {highstock} stock/legend/align/ - * Various legend options - * @sample {highmaps} maps/legend/alignment/ - * Legend alignment - * - * @type {Highcharts.AlignValue} - * @since 2.0 - */ - align: 'center', - /** - * If the [layout](legend.layout) is `horizontal` and the legend items - * span over two lines or more, whether to align the items into vertical - * columns. Setting this to `false` makes room for more items, but will - * look more messy. - * - * @since 6.1.0 - */ - alignColumns: true, - /** - * When the legend is floating, the plot area ignores it and is allowed - * to be placed below it. - * - * @sample {highcharts} highcharts/legend/floating-false/ - * False by default - * @sample {highcharts} highcharts/legend/floating-true/ - * True - * @sample {highmaps} maps/legend/alignment/ - * Floating legend - * - * @type {boolean} - * @default false - * @since 2.1 - * @apioption legend.floating - */ - /** - * The layout of the legend items. Can be one of `horizontal` or - * `vertical` or `proximate`. When `proximate`, the legend items will be - * placed as close as possible to the graphs they're representing, - * except in inverted charts or when the legend position doesn't allow - * it. - * - * @sample {highcharts} highcharts/legend/layout-horizontal/ - * Horizontal by default - * @sample {highcharts} highcharts/legend/layout-vertical/ - * Vertical - * @sample highcharts/legend/layout-proximate - * Labels proximate to the data - * @sample {highstock} stock/legend/layout-horizontal/ - * Horizontal by default - * @sample {highmaps} maps/legend/padding-itemmargin/ - * Vertical with data classes - * @sample {highmaps} maps/legend/layout-vertical/ - * Vertical with color axis gradient - * - * @validvalue ["horizontal", "vertical", "proximate"] - */ - layout: 'horizontal', - /** - * In a legend with horizontal layout, the itemDistance defines the - * pixel distance between each item. - * - * @sample {highcharts} highcharts/legend/layout-horizontal/ - * 50px item distance - * @sample {highstock} highcharts/legend/layout-horizontal/ - * 50px item distance - * - * @type {number} - * @default {highcharts} 20 - * @default {highstock} 20 - * @default {highmaps} 8 - * @since 3.0.3 - * @apioption legend.itemDistance - */ - /** - * The pixel bottom margin for each legend item. - * - * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/ - * Padding and item margins demonstrated - * @sample {highmaps} maps/legend/padding-itemmargin/ - * Padding and item margins demonstrated - * - * @type {number} - * @default 0 - * @since 2.2.0 - * @apioption legend.itemMarginBottom - */ - /** - * The pixel top margin for each legend item. - * - * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/ - * Padding and item margins demonstrated - * @sample {highmaps} maps/legend/padding-itemmargin/ - * Padding and item margins demonstrated - * - * @type {number} - * @default 0 - * @since 2.2.0 - * @apioption legend.itemMarginTop - */ - /** - * The width for each legend item. By default the items are laid out - * successively. In a [horizontal layout](legend.layout), if the items - * are laid out across two rows or more, they will be vertically aligned - * depending on the [legend.alignColumns](legend.alignColumns) option. - * - * @sample {highcharts} highcharts/legend/itemwidth-default/ - * Undefined by default - * @sample {highcharts} highcharts/legend/itemwidth-80/ - * 80 for aligned legend items - * - * @type {number} - * @since 2.0 - * @apioption legend.itemWidth - */ - /** - * A [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) - * for each legend label. Available variables relates to properties on - * the series, or the point in case of pies. - * - * @type {string} - * @default {name} - * @since 1.3 - * @apioption legend.labelFormat - */ - /* eslint-disable valid-jsdoc */ - /** - * Callback function to format each of the series' labels. The `this` - * keyword refers to the series object, or the point object in case of - * pie charts. By default the series or point name is printed. - * - * @productdesc {highmaps} - * In Highmaps the context can also be a data class in case of a - * `colorAxis`. - * - * @sample {highcharts} highcharts/legend/labelformatter/ - * Add text - * @sample {highmaps} maps/legend/labelformatter/ - * Data classes with label formatter - * - * @type {Highcharts.FormatterCallbackFunction<Point|Series>} - */ - labelFormatter: function () { - /** eslint-enable valid-jsdoc */ - return this.name; - }, - /** - * Line height for the legend items. Deprecated as of 2.1\. Instead, - * the line height for each item can be set using - * `itemStyle.lineHeight`, and the padding between items using - * `itemMarginTop` and `itemMarginBottom`. - * - * @sample {highcharts} highcharts/legend/lineheight/ - * Setting padding - * - * @deprecated - * - * @type {number} - * @default 16 - * @since 2.0 - * @product highcharts gantt - * @apioption legend.lineHeight - */ - /** - * If the plot area sized is calculated automatically and the legend is - * not floating, the legend margin is the space between the legend and - * the axis labels or plot area. - * - * @sample {highcharts} highcharts/legend/margin-default/ - * 12 pixels by default - * @sample {highcharts} highcharts/legend/margin-30/ - * 30 pixels - * - * @type {number} - * @default 12 - * @since 2.1 - * @apioption legend.margin - */ - /** - * Maximum pixel height for the legend. When the maximum height is - * extended, navigation will show. - * - * @type {number} - * @since 2.3.0 - * @apioption legend.maxHeight - */ - /** - * The color of the drawn border around the legend. - * - * @see In styled mode, the legend border stroke can be applied with the - * `.highcharts-legend-box` class. - * - * @sample {highcharts} highcharts/legend/bordercolor/ - * Brown border - * @sample {highstock} stock/legend/align/ - * Various legend options - * @sample {highmaps} maps/legend/border-background/ - * Border and background options - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - borderColor: '#999999', - /** - * The border corner radius of the legend. - * - * @sample {highcharts} highcharts/legend/borderradius-default/ - * Square by default - * @sample {highcharts} highcharts/legend/borderradius-round/ - * 5px rounded - * @sample {highmaps} maps/legend/border-background/ - * Border and background options - */ - borderRadius: 0, - /** - * Options for the paging or navigation appearing when the legend is - * overflown. Navigation works well on screen, but not in static - * exported images. One way of working around that is to - * [increase the chart height in - * export](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/legend/navigation-enabled-false/). - */ - navigation: { - /** - * How to animate the pages when navigating up or down. A value of - * `true` applies the default navigation given in the - * `chart.animation` option. Additional options can be given as an - * object containing values for easing and duration. - * - * @sample {highcharts} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * @sample {highstock} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - * @default true - * @since 2.2.4 - * @apioption legend.navigation.animation - */ - /** - * The pixel size of the up and down arrows in the legend paging - * navigation. - * - * @sample {highcharts} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * @sample {highstock} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * - * @type {number} - * @default 12 - * @since 2.2.4 - * @apioption legend.navigation.arrowSize - */ - /** - * Whether to enable the legend navigation. In most cases, disabling - * the navigation results in an unwanted overflow. - * - * See also the [adapt chart to legend]( - * https://www.highcharts.com/products/plugin-registry/single/8/Adapt-Chart-To-Legend) - * plugin for a solution to extend the chart height to make room for - * the legend, optionally in exported charts only. - * - * @type {boolean} - * @default true - * @since 4.2.4 - * @apioption legend.navigation.enabled - */ - /** - * Text styles for the legend page navigation. - * - * @see In styled mode, the navigation items are styled with the - * `.highcharts-legend-navigation` class. - * - * @sample {highcharts} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * @sample {highstock} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * - * @type {Highcharts.CSSObject} - * @since 2.2.4 - * @apioption legend.navigation.style - */ - /** - * The color for the active up or down arrow in the legend page - * navigation. - * - * @see In styled mode, the active arrow be styled with the - * `.highcharts-legend-nav-active` class. - * - * @sample {highcharts} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * @sample {highstock} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 2.2.4 - */ - activeColor: '#003399', - /** - * The color of the inactive up or down arrow in the legend page - * navigation. . - * - * @see In styled mode, the inactive arrow be styled with the - * `.highcharts-legend-nav-inactive` class. - * - * @sample {highcharts} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * @sample {highstock} highcharts/legend/navigation/ - * Legend page navigation demonstrated - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 2.2.4 - */ - inactiveColor: '#cccccc' - }, - /** - * The inner padding of the legend box. - * - * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/ - * Padding and item margins demonstrated - * @sample {highmaps} maps/legend/padding-itemmargin/ - * Padding and item margins demonstrated - * - * @type {number} - * @default 8 - * @since 2.2.0 - * @apioption legend.padding - */ - /** - * Whether to reverse the order of the legend items compared to the - * order of the series or points as defined in the configuration object. - * - * @see [yAxis.reversedStacks](#yAxis.reversedStacks), - * [series.legendIndex](#series.legendIndex) - * - * @sample {highcharts} highcharts/legend/reversed/ - * Stacked bar with reversed legend - * - * @type {boolean} - * @default false - * @since 1.2.5 - * @apioption legend.reversed - */ - /** - * Whether to show the symbol on the right side of the text rather than - * the left side. This is common in Arabic and Hebraic. - * - * @sample {highcharts} highcharts/legend/rtl/ - * Symbol to the right - * - * @type {boolean} - * @default false - * @since 2.2 - * @apioption legend.rtl - */ - /** - * CSS styles for the legend area. In the 1.x versions the position - * of the legend area was determined by CSS. In 2.x, the position is - * determined by properties like `align`, `verticalAlign`, `x` and `y`, - * but the styles are still parsed for backwards compatibility. - * - * @deprecated - * - * @type {Highcharts.CSSObject} - * @product highcharts highstock - * @apioption legend.style - */ - /** - * CSS styles for each legend item. Only a subset of CSS is supported, - * notably those options related to text. The default `textOverflow` - * property makes long texts truncate. Set it to `undefined` to wrap - * text instead. A `width` property can be added to control the text - * width. - * - * @see In styled mode, the legend items can be styled with the - * `.highcharts-legend-item` class. - * - * @sample {highcharts} highcharts/legend/itemstyle/ - * Bold black text - * @sample {highmaps} maps/legend/itemstyle/ - * Item text styles - * - * @type {Highcharts.CSSObject} - * @default {"color": "#333333", "cursor": "pointer", "fontSize": "12px", "fontWeight": "bold", "textOverflow": "ellipsis"} - */ - itemStyle: { - /** - * @ignore - */ - color: '#333333', - /** - * @ignore - */ - cursor: 'pointer', - /** - * @ignore - */ - fontSize: '12px', - /** - * @ignore - */ - fontWeight: 'bold', - /** - * @ignore - */ - textOverflow: 'ellipsis' - }, - /** - * CSS styles for each legend item in hover mode. Only a subset of - * CSS is supported, notably those options related to text. Properties - * are inherited from `style` unless overridden here. - * - * @see In styled mode, the hovered legend items can be styled with - * the `.highcharts-legend-item:hover` pesudo-class. - * - * @sample {highcharts} highcharts/legend/itemhoverstyle/ - * Red on hover - * @sample {highmaps} maps/legend/itemstyle/ - * Item text styles - * - * @type {Highcharts.CSSObject} - * @default {"color": "#000000"} - */ - itemHoverStyle: { - /** - * @ignore - */ - color: '#000000' - }, - /** - * CSS styles for each legend item when the corresponding series or - * point is hidden. Only a subset of CSS is supported, notably those - * options related to text. Properties are inherited from `style` - * unless overridden here. - * - * @see In styled mode, the hidden legend items can be styled with - * the `.highcharts-legend-item-hidden` class. - * - * @sample {highcharts} highcharts/legend/itemhiddenstyle/ - * Darker gray color - * - * @type {Highcharts.CSSObject} - * @default {"color": "#cccccc"} - */ - itemHiddenStyle: { - /** - * @ignore - */ - color: '#cccccc' - }, - /** - * Whether to apply a drop shadow to the legend. A `backgroundColor` - * also needs to be applied for this to take effect. The shadow can be - * an object configuration containing `color`, `offsetX`, `offsetY`, - * `opacity` and `width`. - * - * @sample {highcharts} highcharts/legend/shadow/ - * White background and drop shadow - * @sample {highstock} stock/legend/align/ - * Various legend options - * @sample {highmaps} maps/legend/border-background/ - * Border and background options - * - * @type {boolean|Highcharts.CSSObject} - */ - shadow: false, - /** - * Default styling for the checkbox next to a legend item when - * `showCheckbox` is true. - * - * @type {Highcharts.CSSObject} - * @default {"width": "13px", "height": "13px", "position":"absolute"} - */ - itemCheckboxStyle: { - /** - * @ignore - */ - position: 'absolute', - /** - * @ignore - */ - width: '13px', - /** - * @ignore - */ - height: '13px' - }, - // itemWidth: undefined, - /** - * When this is true, the legend symbol width will be the same as - * the symbol height, which in turn defaults to the font size of the - * legend items. - * - * @since 5.0.0 - */ - squareSymbol: true, - /** - * The pixel height of the symbol for series types that use a rectangle - * in the legend. Defaults to the font size of legend items. - * - * @productdesc {highmaps} - * In Highmaps, when the symbol is the gradient of a vertical color - * axis, the height defaults to 200. - * - * @sample {highmaps} maps/legend/layout-vertical-sized/ - * Sized vertical gradient - * @sample {highmaps} maps/legend/padding-itemmargin/ - * No distance between data classes - * - * @type {number} - * @since 3.0.8 - * @apioption legend.symbolHeight - */ - /** - * The border radius of the symbol for series types that use a rectangle - * in the legend. Defaults to half the `symbolHeight`. - * - * @sample {highcharts} highcharts/legend/symbolradius/ - * Round symbols - * @sample {highstock} highcharts/legend/symbolradius/ - * Round symbols - * @sample {highmaps} highcharts/legend/symbolradius/ - * Round symbols - * - * @type {number} - * @since 3.0.8 - * @apioption legend.symbolRadius - */ - /** - * The pixel width of the legend item symbol. When the `squareSymbol` - * option is set, this defaults to the `symbolHeight`, otherwise 16. - * - * @productdesc {highmaps} - * In Highmaps, when the symbol is the gradient of a horizontal color - * axis, the width defaults to 200. - * - * @sample {highcharts} highcharts/legend/symbolwidth/ - * Greater symbol width and padding - * @sample {highmaps} maps/legend/padding-itemmargin/ - * Padding and item margins demonstrated - * @sample {highmaps} maps/legend/layout-vertical-sized/ - * Sized vertical gradient - * - * @type {number} - * @apioption legend.symbolWidth - */ - /** - * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the legend item texts. - * - * Prior to 4.1.7, when using HTML, [legend.navigation]( - * #legend.navigation) was disabled. - * - * @type {boolean} - * @default false - * @apioption legend.useHTML - */ - /** - * The width of the legend box. If a number is set, it translates to - * pixels. Since v7.0.2 it allows setting a percent string of the full - * chart width, for example `40%`. - * - * Defaults to the full chart width for legends below or above the - * chart, half the chart width for legends to the left and right. - * - * @sample {highcharts} highcharts/legend/width/ - * Aligned to the plot area - * @sample {highcharts} highcharts/legend/width-percent/ - * A percent of the chart width - * - * @type {number|string} - * @since 2.0 - * @apioption legend.width - */ - /** - * The pixel padding between the legend item symbol and the legend - * item text. - * - * @sample {highcharts} highcharts/legend/symbolpadding/ - * Greater symbol width and padding - */ - symbolPadding: 5, - /** - * The vertical alignment of the legend box. Can be one of `top`, - * `middle` or `bottom`. Vertical position can be further determined - * by the `y` option. - * - * In the case that the legend is aligned in a corner position, the - * `layout` option will determine whether to place it above/below - * or on the side of the plot area. - * - * When the [layout](#legend.layout) option is `proximate`, the - * `verticalAlign` option doesn't apply. - * - * @sample {highcharts} highcharts/legend/verticalalign/ - * Legend 100px from the top of the chart - * @sample {highstock} stock/legend/align/ - * Various legend options - * @sample {highmaps} maps/legend/alignment/ - * Legend alignment - * - * @type {Highcharts.VerticalAlignValue} - * @since 2.0 - */ - verticalAlign: 'bottom', - // width: undefined, - /** - * The x offset of the legend relative to its horizontal alignment - * `align` within chart.spacingLeft and chart.spacingRight. Negative - * x moves it to the left, positive x moves it to the right. - * - * @sample {highcharts} highcharts/legend/width/ - * Aligned to the plot area - * - * @since 2.0 - */ - x: 0, - /** - * The vertical offset of the legend relative to it's vertical alignment - * `verticalAlign` within chart.spacingTop and chart.spacingBottom. - * Negative y moves it up, positive y moves it down. - * - * @sample {highcharts} highcharts/legend/verticalalign/ - * Legend 100px from the top of the chart - * @sample {highstock} stock/legend/align/ - * Various legend options - * @sample {highmaps} maps/legend/alignment/ - * Legend alignment - * - * @since 2.0 - */ - y: 0, - /** - * A title to be added on top of the legend. - * - * @sample {highcharts} highcharts/legend/title/ - * Legend title - * @sample {highmaps} maps/legend/alignment/ - * Legend with title - * - * @since 3.0 - */ - title: { - /** - * A text or HTML string for the title. - * - * @type {string} - * @since 3.0 - * @apioption legend.title.text - */ - /** - * Generic CSS styles for the legend title. - * - * @see In styled mode, the legend title is styled with the - * `.highcharts-legend-title` class. - * - * @type {Highcharts.CSSObject} - * @default {"fontWeight": "bold"} - * @since 3.0 - */ - style: { - /** - * @ignore - */ - fontWeight: 'bold' - } - } - }, - /** - * The loading options control the appearance of the loading screen - * that covers the plot area on chart operations. This screen only - * appears after an explicit call to `chart.showLoading()`. It is a - * utility for developers to communicate to the end user that something - * is going on, for example while retrieving new data via an XHR connection. - * The "Loading..." text itself is not part of this configuration - * object, but part of the `lang` object. - */ - loading: { - /** - * The duration in milliseconds of the fade out effect. - * - * @sample highcharts/loading/hideduration/ - * Fade in and out over a second - * - * @type {number} - * @default 100 - * @since 1.2.0 - * @apioption loading.hideDuration - */ - /** - * The duration in milliseconds of the fade in effect. - * - * @sample highcharts/loading/hideduration/ - * Fade in and out over a second - * - * @type {number} - * @default 100 - * @since 1.2.0 - * @apioption loading.showDuration - */ - /** - * CSS styles for the loading label `span`. - * - * @see In styled mode, the loading label is styled with the - * `.highcharts-loading-inner` class. - * - * @sample {highcharts|highmaps} highcharts/loading/labelstyle/ - * Vertically centered - * @sample {highstock} stock/loading/general/ - * Label styles - * - * @type {Highcharts.CSSObject} - * @default {"fontWeight": "bold", "position": "relative", "top": "45%"} - * @since 1.2.0 - */ - labelStyle: { - /** - * @ignore - */ - fontWeight: 'bold', - /** - * @ignore - */ - position: 'relative', - /** - * @ignore - */ - top: '45%' - }, - /** - * CSS styles for the loading screen that covers the plot area. - * - * In styled mode, the loading label is styled with the - * `.highcharts-loading` class. - * - * @sample {highcharts|highmaps} highcharts/loading/style/ - * Gray plot area, white text - * @sample {highstock} stock/loading/general/ - * Gray plot area, white text - * - * @type {Highcharts.CSSObject} - * @default {"position": "absolute", "backgroundColor": "#ffffff", "opacity": 0.5, "textAlign": "center"} - * @since 1.2.0 - */ - style: { - /** - * @ignore - */ - position: 'absolute', - /** - * @ignore - */ - backgroundColor: '#ffffff', - /** - * @ignore - */ - opacity: 0.5, - /** - * @ignore - */ - textAlign: 'center' - } - }, - /** - * Options for the tooltip that appears when the user hovers over a - * series or point. - * - * @declare Highcharts.TooltipOptions - */ - tooltip: { - /** - * The color of the tooltip border. When `undefined`, the border takes - * the color of the corresponding series or point. - * - * @sample {highcharts} highcharts/tooltip/bordercolor-default/ - * Follow series by default - * @sample {highcharts} highcharts/tooltip/bordercolor-black/ - * Black border - * @sample {highstock} stock/tooltip/general/ - * Styled tooltip - * @sample {highmaps} maps/tooltip/background-border/ - * Background and border demo - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption tooltip.borderColor - */ - /** - * A CSS class name to apply to the tooltip's container div, - * allowing unique CSS styling for each chart. - * - * @type {string} - * @apioption tooltip.className - */ - /** - * Since 4.1, the crosshair definitions are moved to the Axis object - * in order for a better separation from the tooltip. See - * [xAxis.crosshair](#xAxis.crosshair). - * - * @sample {highcharts} highcharts/tooltip/crosshairs-x/ - * Enable a crosshair for the x value - * - * @deprecated - * - * @type {*} - * @default true - * @apioption tooltip.crosshairs - */ - /** - * Distance from point to tooltip in pixels. - * - * @type {number} - * @default 16 - * @apioption tooltip.distance - */ - /** - * Whether the tooltip should follow the mouse as it moves across - * columns, pie slices and other point types with an extent. - * By default it behaves this way for pie, polygon, map, sankey - * and wordcloud series by override in the `plotOptions` - * for those series types. - * - * For touch moves to behave the same way, [followTouchMove]( - * #tooltip.followTouchMove) must be `true` also. - * - * @type {boolean} - * @default {highcharts} false - * @default {highstock} false - * @default {highmaps} true - * @since 3.0 - * @apioption tooltip.followPointer - */ - /** - * Whether the tooltip should update as the finger moves on a touch - * device. If this is `true` and [chart.panning](#chart.panning) is - * set,`followTouchMove` will take over one-finger touches, so the user - * needs to use two fingers for zooming and panning. - * - * Note the difference to [followPointer](#tooltip.followPointer) that - * only defines the _position_ of the tooltip. If `followPointer` is - * false in for example a column series, the tooltip will show above or - * below the column, but as `followTouchMove` is true, the tooltip will - * jump from column to column as the user swipes across the plot area. - * - * @type {boolean} - * @default {highcharts} true - * @default {highstock} true - * @default {highmaps} false - * @since 3.0.1 - * @apioption tooltip.followTouchMove - */ - /** - * Callback function to format the text of the tooltip from scratch. In - * case of single or [shared](#tooltip.shared) tooltips, a string should - * be returned. In case of [split](#tooltip.split) tooltips, it should - * return an array where the first item is the header, and subsequent - * items are mapped to the points. Return `false` to disable tooltip for - * a specific point on series. - * - * A subset of HTML is supported. Unless `useHTML` is true, the HTML of - * the tooltip is parsed and converted to SVG, therefore this isn't a - * complete HTML renderer. The following HTML tags are supported: `b`, - * `br`, `em`, `i`, `span`, `strong`. Spans can be styled with a `style` - * attribute, but only text-related CSS, that is shared with SVG, is - * handled. - * - * The available data in the formatter differ a bit depending on whether - * the tooltip is shared or split, or belongs to a single point. In a - * shared/split tooltip, all properties except `x`, which is common for - * all points, are kept in an array, `this.points`. - * - * Available data are: - * - * - **this.percentage (not shared) /** - * **this.points[i].percentage (shared)**: - * Stacked series and pies only. The point's percentage of the total. - * - * - **this.point (not shared) / this.points[i].point (shared)**: - * The point object. The point name, if defined, is available through - * `this.point.name`. - * - * - **this.points**: - * In a shared tooltip, this is an array containing all other - * properties for each point. - * - * - **this.series (not shared) / this.points[i].series (shared)**: - * The series object. The series name is available through - * `this.series.name`. - * - * - **this.total (not shared) / this.points[i].total (shared)**: - * Stacked series only. The total value at this point's x value. - * - * - **this.x**: - * The x value. This property is the same regardless of the tooltip - * being shared or not. - * - * - **this.y (not shared) / this.points[i].y (shared)**: - * The y value. - * - * @sample {highcharts} highcharts/tooltip/formatter-simple/ - * Simple string formatting - * @sample {highcharts} highcharts/tooltip/formatter-shared/ - * Formatting with shared tooltip - * @sample {highcharts|highstock} highcharts/tooltip/formatter-split/ - * Formatting with split tooltip - * @sample highcharts/tooltip/formatter-conditional-default/ - * Extending default formatter - * @sample {highstock} stock/tooltip/formatter/ - * Formatting with shared tooltip - * @sample {highmaps} maps/tooltip/formatter/ - * String formatting - * - * @type {Highcharts.TooltipFormatterCallbackFunction} - * @apioption tooltip.formatter - */ - /** - * Callback function to format the text of the tooltip for - * visible null points. - * Works analogously to [formatter](#tooltip.formatter). - * - * @sample highcharts/plotoptions/series-nullformat - * Format data label and tooltip for null point. - * - * @type {Highcharts.TooltipFormatterCallbackFunction} - * @apioption tooltip.nullFormatter - */ - /** - * The number of milliseconds to wait until the tooltip is hidden when - * mouse out from a point or chart. - * - * @type {number} - * @default 500 - * @since 3.0 - * @apioption tooltip.hideDelay - */ - /** - * Whether to allow the tooltip to render outside the chart's SVG - * element box. By default (`false`), the tooltip is rendered within the - * chart's SVG element, which results in the tooltip being aligned - * inside the chart area. For small charts, this may result in clipping - * or overlapping. When `true`, a separate SVG element is created and - * overlaid on the page, allowing the tooltip to be aligned inside the - * page itself. - * - * Defaults to `true` if `chart.scrollablePlotArea` is activated, - * otherwise `false`. - * - * @sample highcharts/tooltip/outside - * Small charts with tooltips outside - * - * @type {boolean|undefined} - * @default undefined - * @since 6.1.1 - * @apioption tooltip.outside - */ - /** - * A callback function for formatting the HTML output for a single point - * in the tooltip. Like the `pointFormat` string, but with more - * flexibility. - * - * @type {Highcharts.FormatterCallbackFunction<Highcharts.Point>} - * @since 4.1.0 - * @context Highcharts.Point - * @apioption tooltip.pointFormatter - */ - /** - * A callback function to place the tooltip in a default position. The - * callback receives three parameters: `labelWidth`, `labelHeight` and - * `point`, where point contains values for `plotX` and `plotY` telling - * where the reference point is in the plot area. Add `chart.plotLeft` - * and `chart.plotTop` to get the full coordinates. - * - * Since v7, when [tooltip.split](#tooltip.split) option is enabled, - * positioner is called for each of the boxes separately, including - * xAxis header. xAxis header is not a point, instead `point` argument - * contains info: - * `{ plotX: Number, plotY: Number, isHeader: Boolean }` - * - * - * The return should be an object containing x and y values, for example - * `{ x: 100, y: 100 }`. - * - * @sample {highcharts} highcharts/tooltip/positioner/ - * A fixed tooltip position - * @sample {highstock} stock/tooltip/positioner/ - * A fixed tooltip position on top of the chart - * @sample {highmaps} maps/tooltip/positioner/ - * A fixed tooltip position - * @sample {highstock} stock/tooltip/split-positioner/ - * Split tooltip with fixed positions - * @sample {highstock} stock/tooltip/positioner-scrollable-plotarea/ - * Scrollable plot area combined with tooltip positioner - * - * @type {Highcharts.TooltipPositionerCallbackFunction} - * @since 2.2.4 - * @apioption tooltip.positioner - */ - /** - * The name of a symbol to use for the border around the tooltip. Can - * be one of: `"callout"`, `"circle"`, or `"square"`. When - * [tooltip.split](#tooltip.split) - * option is enabled, shape is applied to all boxes except header, which - * is controlled by - * [tooltip.headerShape](#tooltip.headerShape). - * - * Custom callbacks for symbol path generation can also be added to - * `Highcharts.SVGRenderer.prototype.symbols` the same way as for - * [series.marker.symbol](plotOptions.line.marker.symbol). - * - * @type {Highcharts.TooltipShapeValue} - * @default callout - * @since 4.0 - * @apioption tooltip.shape - */ - /** - * The name of a symbol to use for the border around the tooltip - * header. Applies only when [tooltip.split](#tooltip.split) is - * enabled. - * - * Custom callbacks for symbol path generation can also be added to - * `Highcharts.SVGRenderer.prototype.symbols` the same way as for - * [series.marker.symbol](plotOptions.line.marker.symbol). - * - * @see [tooltip.shape](#tooltip.shape) - * - * @sample {highstock} stock/tooltip/split-positioner/ - * Different shapes for header and split boxes - * - * @type {Highcharts.TooltipShapeValue} - * @default callout - * @validvalue ["callout", "square"] - * @since 7.0 - * @apioption tooltip.headerShape - */ - /** - * When the tooltip is shared, the entire plot area will capture mouse - * movement or touch events. Tooltip texts for series types with ordered - * data (not pie, scatter, flags etc) will be shown in a single bubble. - * This is recommended for single series charts and for tablet/mobile - * optimized charts. - * - * See also [tooltip.split](#tooltip.split), that is better suited for - * charts with many series, especially line-type series. The - * `tooltip.split` option takes precedence over `tooltip.shared`. - * - * @sample {highcharts} highcharts/tooltip/shared-false/ - * False by default - * @sample {highcharts} highcharts/tooltip/shared-true/ - * True - * @sample {highcharts} highcharts/tooltip/shared-x-crosshair/ - * True with x axis crosshair - * @sample {highcharts} highcharts/tooltip/shared-true-mixed-types/ - * True with mixed series types - * - * @type {boolean} - * @default false - * @since 2.1 - * @product highcharts highstock - * @apioption tooltip.shared - */ - /** - * Split the tooltip into one label per series, with the header close - * to the axis. This is recommended over [shared](#tooltip.shared) - * tooltips for charts with multiple line series, generally making them - * easier to read. This option takes precedence over `tooltip.shared`. - * - * @productdesc {highstock} In Highstock, tooltips are split by default - * since v6.0.0. Stock charts typically contain multi-dimension points - * and multiple panes, making split tooltips the preferred layout over - * the previous `shared` tooltip. - * - * @sample highcharts/tooltip/split/ - * Split tooltip - * @sample {highcharts|highstock} highcharts/tooltip/formatter-split/ - * Split tooltip and custom formatter callback - * - * @type {boolean} - * @default {highcharts} false - * @default {highstock} true - * @since 5.0.0 - * @product highcharts highstock - * @apioption tooltip.split - */ - /** - * Prevents the tooltip from switching or closing, when touched or - * pointed. - * - * @sample highcharts/tooltip/stickoncontact/ - * Tooltip sticks on pointer contact - * - * @type {boolean} - * @since 8.0.1 - * @apioption tooltip.stickOnContact - */ - /** - * Use HTML to render the contents of the tooltip instead of SVG. Using - * HTML allows advanced formatting like tables and images in the - * tooltip. It is also recommended for rtl languages as it works around - * rtl bugs in early Firefox. - * - * @sample {highcharts|highstock} highcharts/tooltip/footerformat/ - * A table for value alignment - * @sample {highcharts|highstock} highcharts/tooltip/fullhtml/ - * Full HTML tooltip - * @sample {highmaps} maps/tooltip/usehtml/ - * Pure HTML tooltip - * - * @type {boolean} - * @default false - * @since 2.2 - * @apioption tooltip.useHTML - */ - /** - * How many decimals to show in each series' y value. This is - * overridable in each series' tooltip options object. The default is to - * preserve all decimals. - * - * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/ - * Set decimals, prefix and suffix for the value - * @sample {highmaps} maps/tooltip/valuedecimals/ - * Set decimals, prefix and suffix for the value - * - * @type {number} - * @since 2.2 - * @apioption tooltip.valueDecimals - */ - /** - * A string to prepend to each series' y value. Overridable in each - * series' tooltip options object. - * - * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/ - * Set decimals, prefix and suffix for the value - * @sample {highmaps} maps/tooltip/valuedecimals/ - * Set decimals, prefix and suffix for the value - * - * @type {string} - * @since 2.2 - * @apioption tooltip.valuePrefix - */ - /** - * A string to append to each series' y value. Overridable in each - * series' tooltip options object. - * - * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/ - * Set decimals, prefix and suffix for the value - * @sample {highmaps} maps/tooltip/valuedecimals/ - * Set decimals, prefix and suffix for the value - * - * @type {string} - * @since 2.2 - * @apioption tooltip.valueSuffix - */ - /** - * The format for the date in the tooltip header if the X axis is a - * datetime axis. The default is a best guess based on the smallest - * distance between points in the chart. - * - * @sample {highcharts} highcharts/tooltip/xdateformat/ - * A different format - * - * @type {string} - * @product highcharts highstock gantt - * @apioption tooltip.xDateFormat - */ - /** - * How many decimals to show for the `point.change` value when the - * `series.compare` option is set. This is overridable in each series' - * tooltip options object. The default is to preserve all decimals. - * - * @type {number} - * @since 1.0.1 - * @product highstock - * @apioption tooltip.changeDecimals - */ - /** - * Enable or disable the tooltip. - * - * @sample {highcharts} highcharts/tooltip/enabled/ - * Disabled - * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/ - * Disable tooltip and show values on chart instead - */ - enabled: true, - /** - * Enable or disable animation of the tooltip. - * - * @type {boolean} - * @default true - * @since 2.3.0 - */ - animation: svg, - /** - * The radius of the rounded border corners. - * - * @sample {highcharts} highcharts/tooltip/bordercolor-default/ - * 5px by default - * @sample {highcharts} highcharts/tooltip/borderradius-0/ - * Square borders - * @sample {highmaps} maps/tooltip/background-border/ - * Background and border demo - */ - borderRadius: 3, - /** - * For series on a datetime axes, the date format in the tooltip's - * header will by default be guessed based on the closest data points. - * This member gives the default string representations used for - * each unit. For an overview of the replacement codes, see - * [dateFormat](/class-reference/Highcharts#dateFormat). - * - * @see [xAxis.dateTimeLabelFormats](#xAxis.dateTimeLabelFormats) - * - * @type {Highcharts.Dictionary<string>} - * @product highcharts highstock gantt - */ - dateTimeLabelFormats: { - /** @internal */ - millisecond: '%A, %b %e, %H:%M:%S.%L', - /** @internal */ - second: '%A, %b %e, %H:%M:%S', - /** @internal */ - minute: '%A, %b %e, %H:%M', - /** @internal */ - hour: '%A, %b %e, %H:%M', - /** @internal */ - day: '%A, %b %e, %Y', - /** @internal */ - week: 'Week from %A, %b %e, %Y', - /** @internal */ - month: '%B %Y', - /** @internal */ - year: '%Y' - }, - /** - * A string to append to the tooltip format. - * - * @sample {highcharts} highcharts/tooltip/footerformat/ - * A table for value alignment - * @sample {highmaps} maps/tooltip/format/ - * Format demo - * - * @since 2.2 - */ - footerFormat: '', - /** - * Padding inside the tooltip, in pixels. - * - * @since 5.0.0 - */ - padding: 8, - /** - * Proximity snap for graphs or single points. It defaults to 10 for - * mouse-powered devices and 25 for touch devices. - * - * Note that in most cases the whole plot area captures the mouse - * movement, and in these cases `tooltip.snap` doesn't make sense. This - * applies when [stickyTracking](#plotOptions.series.stickyTracking) - * is `true` (default) and when the tooltip is [shared](#tooltip.shared) - * or [split](#tooltip.split). - * - * @sample {highcharts} highcharts/tooltip/bordercolor-default/ - * 10 px by default - * @sample {highcharts} highcharts/tooltip/snap-50/ - * 50 px on graph - * - * @type {number} - * @default 10/25 - * @since 1.2.0 - * @product highcharts highstock - */ - snap: isTouchDevice ? 25 : 10, - /** - * The HTML of the tooltip header line. Variables are enclosed by - * curly brackets. Available variables are `point.key`, `series.name`, - * `series.color` and other members from the `point` and `series` - * objects. The `point.key` variable contains the category name, x - * value or datetime string depending on the type of axis. For datetime - * axes, the `point.key` date format can be set using - * `tooltip.xDateFormat`. - * - * @sample {highcharts} highcharts/tooltip/footerformat/ - * An HTML table in the tooltip - * @sample {highstock} highcharts/tooltip/footerformat/ - * An HTML table in the tooltip - * @sample {highmaps} maps/tooltip/format/ - * Format demo - * - * @type {string} - * @apioption tooltip.headerFormat - */ - headerFormat: '<span style="font-size: 10px">{point.key}</span><br/>', - /** - * The HTML of the null point's line in the tooltip. Works analogously - * to [pointFormat](#tooltip.pointFormat). - * - * @sample {highcharts} highcharts/plotoptions/series-nullformat - * Format data label and tooltip for null point. - * - * @type {string} - * @apioption tooltip.nullFormat - */ - /** - * The HTML of the point's line in the tooltip. Variables are enclosed - * by curly brackets. Available variables are `point.x`, `point.y`, - * `series.name` and `series.color` and other properties on the same - * form. Furthermore, `point.y` can be extended by the - * `tooltip.valuePrefix` and `tooltip.valueSuffix` variables. This can - * also be overridden for each series, which makes it a good hook for - * displaying units. - * - * In styled mode, the dot is colored by a class name rather - * than the point color. - * - * @sample {highcharts} highcharts/tooltip/pointformat/ - * A different point format with value suffix - * @sample {highmaps} maps/tooltip/format/ - * Format demo - * - * @type {string} - * @since 2.2 - * @apioption tooltip.pointFormat - */ - pointFormat: '<span style="color:{point.color}">\u25CF</span> {series.name}: <b>{point.y}</b><br/>', - /** - * The background color or gradient for the tooltip. - * - * In styled mode, the stroke width is set in the - * `.highcharts-tooltip-box` class. - * - * @sample {highcharts} highcharts/tooltip/backgroundcolor-solid/ - * Yellowish background - * @sample {highcharts} highcharts/tooltip/backgroundcolor-gradient/ - * Gradient - * @sample {highcharts} highcharts/css/tooltip-border-background/ - * Tooltip in styled mode - * @sample {highstock} stock/tooltip/general/ - * Custom tooltip - * @sample {highstock} highcharts/css/tooltip-border-background/ - * Tooltip in styled mode - * @sample {highmaps} maps/tooltip/background-border/ - * Background and border demo - * @sample {highmaps} highcharts/css/tooltip-border-background/ - * Tooltip in styled mode - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - backgroundColor: color('#f7f7f7') - .setOpacity(0.85).get(), - /** - * The pixel width of the tooltip border. - * - * In styled mode, the stroke width is set in the - * `.highcharts-tooltip-box` class. - * - * @sample {highcharts} highcharts/tooltip/bordercolor-default/ - * 2px by default - * @sample {highcharts} highcharts/tooltip/borderwidth/ - * No border (shadow only) - * @sample {highcharts} highcharts/css/tooltip-border-background/ - * Tooltip in styled mode - * @sample {highstock} stock/tooltip/general/ - * Custom tooltip - * @sample {highstock} highcharts/css/tooltip-border-background/ - * Tooltip in styled mode - * @sample {highmaps} maps/tooltip/background-border/ - * Background and border demo - * @sample {highmaps} highcharts/css/tooltip-border-background/ - * Tooltip in styled mode - */ - borderWidth: 1, - /** - * Whether to apply a drop shadow to the tooltip. - * - * @sample {highcharts} highcharts/tooltip/bordercolor-default/ - * True by default - * @sample {highcharts} highcharts/tooltip/shadow/ - * False - * @sample {highmaps} maps/tooltip/positioner/ - * Fixed tooltip position, border and shadow disabled - * - * @type {boolean|Highcharts.ShadowOptionsObject} - */ - shadow: true, - /** - * CSS styles for the tooltip. The tooltip can also be styled through - * the CSS class `.highcharts-tooltip`. - * - * Note that the default `pointerEvents` style makes the tooltip ignore - * mouse events, so in order to use clickable tooltips, this value must - * be set to `auto`. - * - * @sample {highcharts} highcharts/tooltip/style/ - * Greater padding, bold text - * - * @type {Highcharts.CSSObject} - */ - style: { - /** @internal */ - color: '#333333', - /** @internal */ - cursor: 'default', - /** @internal */ - fontSize: '12px', - /** @internal */ - whiteSpace: 'nowrap' - } - }, - /** - * Highchart by default puts a credits label in the lower right corner - * of the chart. This can be changed using these options. - */ - credits: { - /** - * Credits for map source to be concatenated with conventional credit - * text. By default this is a format string that collects copyright - * information from the map if available. - * - * @see [mapTextFull](#credits.mapTextFull) - * @see [text](#credits.text) - * - * @type {string} - * @default \u00a9 <a href="{geojson.copyrightUrl}">{geojson.copyrightShort}</a> - * @since 4.2.2 - * @product highmaps - * @apioption credits.mapText - */ - /** - * Detailed credits for map source to be displayed on hover of credits - * text. By default this is a format string that collects copyright - * information from the map if available. - * - * @see [mapText](#credits.mapText) - * @see [text](#credits.text) - * - * @type {string} - * @default {geojson.copyright} - * @since 4.2.2 - * @product highmaps - * @apioption credits.mapTextFull - */ - /** - * Whether to show the credits text. - * - * @sample {highcharts} highcharts/credits/enabled-false/ - * Credits disabled - * @sample {highstock} stock/credits/enabled/ - * Credits disabled - * @sample {highmaps} maps/credits/enabled-false/ - * Credits disabled - */ - enabled: true, - /** - * The URL for the credits label. - * - * @sample {highcharts} highcharts/credits/href/ - * Custom URL and text - * @sample {highmaps} maps/credits/customized/ - * Custom URL and text - */ - href: 'https://www.highcharts.com?credits', - /** - * Position configuration for the credits label. - * - * @sample {highcharts} highcharts/credits/position-left/ - * Left aligned - * @sample {highcharts} highcharts/credits/position-left/ - * Left aligned - * @sample {highmaps} maps/credits/customized/ - * Left aligned - * @sample {highmaps} maps/credits/customized/ - * Left aligned - * - * @type {Highcharts.AlignObject} - * @since 2.1 - */ - position: { - /** @internal */ - align: 'right', - /** @internal */ - x: -10, - /** @internal */ - verticalAlign: 'bottom', - /** @internal */ - y: -5 - }, - /** - * CSS styles for the credits label. - * - * @see In styled mode, credits styles can be set with the - * `.highcharts-credits` class. - * - * @type {Highcharts.CSSObject} - */ - style: { - /** @internal */ - cursor: 'pointer', - /** @internal */ - color: '#999999', - /** @internal */ - fontSize: '9px' - }, - /** - * The text for the credits label. - * - * @productdesc {highmaps} - * If a map is loaded as GeoJSON, the text defaults to - * `Highcharts @ {map-credits}`. Otherwise, it defaults to - * `Highcharts.com`. - * - * @sample {highcharts} highcharts/credits/href/ - * Custom URL and text - * @sample {highmaps} maps/credits/customized/ - * Custom URL and text - */ - text: 'Highcharts.com' - } - }; - /* eslint-disable spaced-comment */ - - ''; - /** - * Global `Time` object with default options. Since v6.0.5, time settings can be - * applied individually for each chart. If no individual settings apply, this - * `Time` object is shared by all instances. - * - * @name Highcharts.time - * @type {Highcharts.Time} - */ - H.time = new Time(merge(H.defaultOptions.global, H.defaultOptions.time)); - /** - * Formats a JavaScript date timestamp (milliseconds since Jan 1st 1970) into a - * human readable date string. The format is a subset of the formats for PHP's - * [strftime](https://www.php.net/manual/en/function.strftime.php) function. - * Additional formats can be given in the {@link Highcharts.dateFormats} hook. - * - * Since v6.0.5, all internal dates are formatted through the - * {@link Highcharts.Chart#time} instance to respect chart-level time settings. - * The `Highcharts.dateFormat` function only reflects global time settings set - * with `setOptions`. - * - * Supported format keys: - * - `%a`: Short weekday, like 'Mon' - * - `%A`: Long weekday, like 'Monday' - * - `%d`: Two digit day of the month, 01 to 31 - * - `%e`: Day of the month, 1 through 31 - * - `%w`: Day of the week, 0 through 6 - * - `%b`: Short month, like 'Jan' - * - `%B`: Long month, like 'January' - * - `%m`: Two digit month number, 01 through 12 - * - `%y`: Two digits year, like 09 for 2009 - * - `%Y`: Four digits year, like 2009 - * - `%H`: Two digits hours in 24h format, 00 through 23 - * - `%k`: Hours in 24h format, 0 through 23 - * - `%I`: Two digits hours in 12h format, 00 through 11 - * - `%l`: Hours in 12h format, 1 through 12 - * - `%M`: Two digits minutes, 00 through 59 - * - `%p`: Upper case AM or PM - * - `%P`: Lower case AM or PM - * - `%S`: Two digits seconds, 00 through 59 - * - `%L`: Milliseconds (naming from Ruby) - * - * @function Highcharts.dateFormat - * - * @param {string} format - * The desired format where various time representations are prefixed - * with `%`. - * - * @param {number} timestamp - * The JavaScript timestamp. - * - * @param {boolean} [capitalize=false] - * Upper case first letter in the return. - * - * @return {string} - * The formatted date. - */ - H.dateFormat = function (format, timestamp, capitalize) { - return H.time.dateFormat(format, timestamp, capitalize); - }; - var optionsModule = { - dateFormat: H.dateFormat, - defaultOptions: H.defaultOptions, - time: H.time - }; - - return optionsModule; - }); - _registerModule(_modules, 'Core/Axis/Axis.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Color/Color.js'], _modules['Core/Globals.js'], _modules['Core/Axis/Tick.js'], _modules['Core/Utilities.js'], _modules['Core/Options.js']], function (A, Color, H, Tick, U, O) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var animObject = A.animObject; - var addEvent = U.addEvent, - arrayMax = U.arrayMax, - arrayMin = U.arrayMin, - clamp = U.clamp, - correctFloat = U.correctFloat, - defined = U.defined, - destroyObjectProperties = U.destroyObjectProperties, - error = U.error, - extend = U.extend, - fireEvent = U.fireEvent, - format = U.format, - getMagnitude = U.getMagnitude, - isArray = U.isArray, - isFunction = U.isFunction, - isNumber = U.isNumber, - isString = U.isString, - merge = U.merge, - normalizeTickInterval = U.normalizeTickInterval, - objectEach = U.objectEach, - pick = U.pick, - relativeLength = U.relativeLength, - removeEvent = U.removeEvent, - splat = U.splat, - syncTimeout = U.syncTimeout; - /** - * Options for the path on the Axis to be calculated. - * @interface Highcharts.AxisPlotLinePathOptionsObject - */ /** - * Axis value. - * @name Highcharts.AxisPlotLinePathOptionsObject#value - * @type {number|undefined} - */ /** - * Line width used for calculation crisp line coordinates. Defaults to 1. - * @name Highcharts.AxisPlotLinePathOptionsObject#lineWidth - * @type {number|undefined} - */ /** - * If `false`, the function will return null when it falls outside the axis - * bounds. If `true`, the function will return a path aligned to the plot area - * sides if it falls outside. If `pass`, it will return a path outside. - * @name Highcharts.AxisPlotLinePathOptionsObject#force - * @type {string|boolean|undefined} - */ /** - * Used in Highstock. When `true`, plot paths (crosshair, plotLines, gridLines) - * will be rendered on all axes when defined on the first axis. - * @name Highcharts.AxisPlotLinePathOptionsObject#acrossPanes - * @type {boolean|undefined} - */ /** - * Use old coordinates (for resizing and rescaling). - * If not set, defaults to `false`. - * @name Highcharts.AxisPlotLinePathOptionsObject#old - * @type {boolean|undefined} - */ /** - * If given, return the plot line path of a pixel position on the axis. - * @name Highcharts.AxisPlotLinePathOptionsObject#translatedValue - * @type {number|undefined} - */ /** - * Used in Polar axes. Reverse the positions for concatenation of polygonal - * plot bands - * @name Highcharts.AxisPlotLinePathOptionsObject#reverse - * @type {boolean|undefined} - */ - /** - * Options for crosshairs on axes. - * - * @product highstock - * - * @typedef {Highcharts.XAxisCrosshairOptions|Highcharts.YAxisCrosshairOptions} Highcharts.AxisCrosshairOptions - */ - /** - * @typedef {"navigator"|"pan"|"rangeSelectorButton"|"rangeSelectorInput"|"scrollbar"|"traverseUpButton"|"zoom"} Highcharts.AxisExtremesTriggerValue - */ - /** - * @callback Highcharts.AxisEventCallbackFunction - * - * @param {Highcharts.Axis} this - */ - /** - * @callback Highcharts.AxisLabelsFormatterCallbackFunction - * - * @param {Highcharts.AxisLabelsFormatterContextObject<number>} this - * - * @param {Highcharts.AxisLabelsFormatterContextObject<string>} that - * - * @return {string} - */ - /** - * @interface Highcharts.AxisLabelsFormatterContextObject<T> - */ /** - * @name Highcharts.AxisLabelsFormatterContextObject<T>#axis - * @type {Highcharts.Axis} - */ /** - * @name Highcharts.AxisLabelsFormatterContextObject<T>#chart - * @type {Highcharts.Chart} - */ /** - * @name Highcharts.AxisLabelsFormatterContextObject<T>#isFirst - * @type {boolean} - */ /** - * @name Highcharts.AxisLabelsFormatterContextObject<T>#isLast - * @type {boolean} - */ /** - * @name Highcharts.AxisLabelsFormatterContextObject<T>#pos - * @type {number} - */ /** - * This can be either a numeric value or a category string. - * @name Highcharts.AxisLabelsFormatterContextObject<T>#value - * @type {T} - */ - /** - * Options for axes. - * - * @typedef {Highcharts.XAxisOptions|Highcharts.YAxisOptions|Highcharts.ZAxisOptions} Highcharts.AxisOptions - */ - /** - * @callback Highcharts.AxisPointBreakEventCallbackFunction - * - * @param {Highcharts.Axis} this - * - * @param {Highcharts.AxisPointBreakEventObject} evt - */ - /** - * @interface Highcharts.AxisPointBreakEventObject - */ /** - * @name Highcharts.AxisPointBreakEventObject#brk - * @type {Highcharts.Dictionary<number>} - */ /** - * @name Highcharts.AxisPointBreakEventObject#point - * @type {Highcharts.Point} - */ /** - * @name Highcharts.AxisPointBreakEventObject#preventDefault - * @type {Function} - */ /** - * @name Highcharts.AxisPointBreakEventObject#target - * @type {Highcharts.SVGElement} - */ /** - * @name Highcharts.AxisPointBreakEventObject#type - * @type {"pointBreak"|"pointInBreak"} - */ - /** - * @callback Highcharts.AxisSetExtremesEventCallbackFunction - * - * @param {Highcharts.Axis} this - * - * @param {Highcharts.AxisSetExtremesEventObject} evt - */ - /** - * @interface Highcharts.AxisSetExtremesEventObject - * @extends Highcharts.ExtremesObject - */ /** - * @name Highcharts.AxisSetExtremesEventObject#preventDefault - * @type {Function} - */ /** - * @name Highcharts.AxisSetExtremesEventObject#target - * @type {Highcharts.SVGElement} - */ /** - * @name Highcharts.AxisSetExtremesEventObject#trigger - * @type {Highcharts.AxisExtremesTriggerValue|string} - */ /** - * @name Highcharts.AxisSetExtremesEventObject#type - * @type {"setExtremes"} - */ - /** - * @callback Highcharts.AxisTickPositionerCallbackFunction - * - * @param {Highcharts.Axis} this - * - * @return {Highcharts.AxisTickPositionsArray} - */ - /** - * @interface Highcharts.AxisTickPositionsArray - * @augments Array<number> - */ - /** - * @typedef {"high"|"low"|"middle"} Highcharts.AxisTitleAlignValue - */ - /** - * @typedef {Highcharts.XAxisTitleOptions|Highcharts.YAxisTitleOptions|Highcharts.ZAxisTitleOptions} Highcharts.AxisTitleOptions - */ - /** - * @typedef {"linear"|"logarithmic"|"datetime"|"category"|"treegrid"} Highcharts.AxisTypeValue - */ - /** - * The returned object literal from the {@link Highcharts.Axis#getExtremes} - * function. - * - * @interface Highcharts.ExtremesObject - */ /** - * The maximum value of the axis' associated series. - * @name Highcharts.ExtremesObject#dataMax - * @type {number} - */ /** - * The minimum value of the axis' associated series. - * @name Highcharts.ExtremesObject#dataMin - * @type {number} - */ /** - * The maximum axis value, either automatic or set manually. If the `max` option - * is not set, `maxPadding` is 0 and `endOnTick` is false, this value will be - * the same as `dataMax`. - * @name Highcharts.ExtremesObject#max - * @type {number} - */ /** - * The minimum axis value, either automatic or set manually. If the `min` option - * is not set, `minPadding` is 0 and `startOnTick` is false, this value will be - * the same as `dataMin`. - * @name Highcharts.ExtremesObject#min - * @type {number} - */ /** - * The user defined maximum, either from the `max` option or from a zoom or - * `setExtremes` action. - * @name Highcharts.ExtremesObject#userMax - * @type {number} - */ /** - * The user defined minimum, either from the `min` option or from a zoom or - * `setExtremes` action. - * @name Highcharts.ExtremesObject#userMin - * @type {number} - */ - /** - * Formatter function for the text of a crosshair label. - * - * @callback Highcharts.XAxisCrosshairLabelFormatterCallbackFunction - * - * @param {Highcharts.Axis} this - * Axis context - * - * @param {number} value - * Y value of the data point - * - * @return {string} - */ - var defaultOptions = O.defaultOptions; - var deg2rad = H.deg2rad; - /** - * Create a new axis object. Called internally when instanciating a new chart or - * adding axes by {@link Highcharts.Chart#addAxis}. - * - * A chart can have from 0 axes (pie chart) to multiples. In a normal, single - * series cartesian chart, there is one X axis and one Y axis. - * - * The X axis or axes are referenced by {@link Highcharts.Chart.xAxis}, which is - * an array of Axis objects. If there is only one axis, it can be referenced - * through `chart.xAxis[0]`, and multiple axes have increasing indices. The same - * pattern goes for Y axes. - * - * If you need to get the axes from a series object, use the `series.xAxis` and - * `series.yAxis` properties. These are not arrays, as one series can only be - * associated to one X and one Y axis. - * - * A third way to reference the axis programmatically is by `id`. Add an `id` in - * the axis configuration options, and get the axis by - * {@link Highcharts.Chart#get}. - * - * Configuration options for the axes are given in options.xAxis and - * options.yAxis. - * - * @class - * @name Highcharts.Axis - * - * @param {Highcharts.Chart} chart - * The Chart instance to apply the axis on. - * - * @param {Highcharts.AxisOptions} userOptions - * Axis options. - */ - var Axis = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function Axis(chart, userOptions) { - this.alternateBands = void 0; - this.bottom = void 0; - this.categories = void 0; - this.chart = void 0; - this.closestPointRange = void 0; - this.coll = void 0; - this.hasNames = void 0; - this.hasVisibleSeries = void 0; - this.height = void 0; - this.isLinked = void 0; - this.labelEdge = void 0; // @todo - this.labelFormatter = void 0; - this.left = void 0; - this.len = void 0; - this.max = void 0; - this.maxLabelLength = void 0; - this.min = void 0; - this.minorTickInterval = void 0; - this.minorTicks = void 0; - this.minPixelPadding = void 0; - this.names = void 0; - this.offset = void 0; - this.oldMax = void 0; - this.oldMin = void 0; - this.options = void 0; - this.overlap = void 0; - this.paddedTicks = void 0; - this.plotLinesAndBands = void 0; - this.plotLinesAndBandsGroups = void 0; - this.pointRange = void 0; - this.pointRangePadding = void 0; - this.pos = void 0; - this.positiveValuesOnly = void 0; - this.right = void 0; - this.series = void 0; - this.side = void 0; - this.tickAmount = void 0; - this.tickInterval = void 0; - this.tickmarkOffset = void 0; - this.tickPositions = void 0; - this.tickRotCorr = void 0; - this.ticks = void 0; - this.top = void 0; - this.transA = void 0; - this.transB = void 0; - this.translationSlope = void 0; - this.userOptions = void 0; - this.visible = void 0; - this.width = void 0; - this.zoomEnabled = void 0; - this.init(chart, userOptions); - } - /* * - * - * Functions - * - * */ - /** - * Overrideable function to initialize the axis. - * - * @see {@link Axis} - * - * @function Highcharts.Axis#init - * - * @param {Highcharts.Chart} chart - * The Chart instance to apply the axis on. - * - * @param {Highcharts.AxisOptions} userOptions - * Axis options. - * - * @fires Highcharts.Axis#event:afterInit - * @fires Highcharts.Axis#event:init - */ - Axis.prototype.init = function (chart, userOptions) { - var isXAxis = userOptions.isX, - axis = this; - /** - * The Chart that the axis belongs to. - * - * @name Highcharts.Axis#chart - * @type {Highcharts.Chart} - */ - axis.chart = chart; - /** - * Whether the axis is horizontal. - * - * @name Highcharts.Axis#horiz - * @type {boolean|undefined} - */ - axis.horiz = chart.inverted && !axis.isZAxis ? !isXAxis : isXAxis; - /** - * Whether the axis is the x-axis. - * - * @name Highcharts.Axis#isXAxis - * @type {boolean|undefined} - */ - axis.isXAxis = isXAxis; - /** - * The collection where the axis belongs, for example `xAxis`, `yAxis` - * or `colorAxis`. Corresponds to properties on Chart, for example - * {@link Chart.xAxis}. - * - * @name Highcharts.Axis#coll - * @type {string} - */ - axis.coll = axis.coll || (isXAxis ? 'xAxis' : 'yAxis'); - fireEvent(this, 'init', { userOptions: userOptions }); - axis.opposite = userOptions.opposite; // needed in setOptions - /** - * The side on which the axis is rendered. 0 is top, 1 is right, 2 - * is bottom and 3 is left. - * - * @name Highcharts.Axis#side - * @type {number} - */ - axis.side = userOptions.side || (axis.horiz ? - (axis.opposite ? 0 : 2) : // top : bottom - (axis.opposite ? 1 : 3)); // right : left - /** - * Current options for the axis after merge of defaults and user's - * options. - * - * @name Highcharts.Axis#options - * @type {Highcharts.AxisOptions} - */ - axis.setOptions(userOptions); - var options = this.options, - type = options.type; - axis.labelFormatter = (options.labels.formatter || - // can be overwritten by dynamic format - axis.defaultLabelFormatter); - /** - * User's options for this axis without defaults. - * - * @name Highcharts.Axis#userOptions - * @type {Highcharts.AxisOptions} - */ - axis.userOptions = userOptions; - axis.minPixelPadding = 0; - /** - * Whether the axis is reversed. Based on the `axis.reversed`, - * option, but inverted charts have reversed xAxis by default. - * - * @name Highcharts.Axis#reversed - * @type {boolean} - */ - axis.reversed = options.reversed; - axis.visible = options.visible !== false; - axis.zoomEnabled = options.zoomEnabled !== false; - // Initial categories - axis.hasNames = - type === 'category' || options.categories === true; - /** - * If categories are present for the axis, names are used instead of - * numbers for that axis. - * - * Since Highcharts 3.0, categories can also be extracted by giving each - * point a name and setting axis type to `category`. However, if you - * have multiple series, best practice remains defining the `categories` - * array. - * - * @see [xAxis.categories](/highcharts/xAxis.categories) - * - * @name Highcharts.Axis#categories - * @type {Array<string>} - * @readonly - */ - axis.categories = options.categories || axis.hasNames; - if (!axis.names) { // Preserve on update (#3830) - axis.names = []; - axis.names.keys = {}; - } - // Placeholder for plotlines and plotbands groups - axis.plotLinesAndBandsGroups = {}; - // Shorthand types - axis.positiveValuesOnly = !!axis.logarithmic; - // Flag, if axis is linked to another axis - axis.isLinked = defined(options.linkedTo); - /** - * List of major ticks mapped by postition on axis. - * - * @see {@link Highcharts.Tick} - * - * @name Highcharts.Axis#ticks - * @type {Highcharts.Dictionary<Highcharts.Tick>} - */ - axis.ticks = {}; - axis.labelEdge = []; - /** - * List of minor ticks mapped by position on the axis. - * - * @see {@link Highcharts.Tick} - * - * @name Highcharts.Axis#minorTicks - * @type {Highcharts.Dictionary<Highcharts.Tick>} - */ - axis.minorTicks = {}; - // List of plotLines/Bands - axis.plotLinesAndBands = []; - // Alternate bands - axis.alternateBands = {}; - // Axis metrics - axis.len = 0; - axis.minRange = axis.userMinRange = options.minRange || options.maxZoom; - axis.range = options.range; - axis.offset = options.offset || 0; - /** - * The maximum value of the axis. In a logarithmic axis, this is the - * logarithm of the real value, and the real value can be obtained from - * {@link Axis#getExtremes}. - * - * @name Highcharts.Axis#max - * @type {number|null} - */ - axis.max = null; - /** - * The minimum value of the axis. In a logarithmic axis, this is the - * logarithm of the real value, and the real value can be obtained from - * {@link Axis#getExtremes}. - * - * @name Highcharts.Axis#min - * @type {number|null} - */ - axis.min = null; - /** - * The processed crosshair options. - * - * @name Highcharts.Axis#crosshair - * @type {boolean|Highcharts.AxisCrosshairOptions} - */ - axis.crosshair = pick(options.crosshair, splat(chart.options.tooltip.crosshairs)[isXAxis ? 0 : 1], false); - var events = axis.options.events; - // Register. Don't add it again on Axis.update(). - if (chart.axes.indexOf(axis) === -1) { // - if (isXAxis) { // #2713 - chart.axes.splice(chart.xAxis.length, 0, axis); - } - else { - chart.axes.push(axis); - } - chart[axis.coll].push(axis); - } - /** - * All series associated to the axis. - * - * @name Highcharts.Axis#series - * @type {Array<Highcharts.Series>} - */ - axis.series = axis.series || []; // populated by Series - // Reversed axis - if (chart.inverted && - !axis.isZAxis && - isXAxis && - typeof axis.reversed === 'undefined') { - axis.reversed = true; - } - axis.labelRotation = axis.options.labels.rotation; - // register event listeners - objectEach(events, function (event, eventType) { - if (isFunction(event)) { - addEvent(axis, eventType, event); - } - }); - fireEvent(this, 'afterInit'); - }; - /** - * Merge and set options. - * - * @private - * @function Highcharts.Axis#setOptions - * - * @param {Highcharts.AxisOptions} userOptions - * Axis options. - * - * @fires Highcharts.Axis#event:afterSetOptions - */ - Axis.prototype.setOptions = function (userOptions) { - this.options = merge(Axis.defaultOptions, (this.coll === 'yAxis') && Axis.defaultYAxisOptions, [ - Axis.defaultTopAxisOptions, - Axis.defaultRightAxisOptions, - Axis.defaultBottomAxisOptions, - Axis.defaultLeftAxisOptions - ][this.side], merge( - // if set in setOptions (#1053): - defaultOptions[this.coll], userOptions)); - fireEvent(this, 'afterSetOptions', { userOptions: userOptions }); - }; - /** - * The default label formatter. The context is a special config object for - * the label. In apps, use the - * [labels.formatter](https://api.highcharts.com/highcharts/xAxis.labels.formatter) - * instead, except when a modification is needed. - * - * @function Highcharts.Axis#defaultLabelFormatter - * - * @param {Highcharts.AxisLabelsFormatterContextObject<number>|Highcharts.AxisLabelsFormatterContextObject<string>} this - * Formatter context of axis label. - * - * @return {string} - * The formatted label content. - */ - Axis.prototype.defaultLabelFormatter = function () { - var axis = this.axis, - value = isNumber(this.value) ? this.value : NaN, - time = axis.chart.time, - categories = axis.categories, - dateTimeLabelFormat = this.dateTimeLabelFormat, - lang = defaultOptions.lang, - numericSymbols = lang.numericSymbols, - numSymMagnitude = lang.numericSymbolMagnitude || 1000, - i = numericSymbols && numericSymbols.length, - multi, - ret, - formatOption = axis.options.labels.format, - // make sure the same symbol is added for all labels on a linear - // axis - numericSymbolDetector = axis.logarithmic ? - Math.abs(value) : - axis.tickInterval; - var chart = this.chart; - var numberFormatter = chart.numberFormatter; - if (formatOption) { - ret = format(formatOption, this, chart); - } - else if (categories) { - ret = "" + this.value; - } - else if (dateTimeLabelFormat) { // datetime axis - ret = time.dateFormat(dateTimeLabelFormat, value); - } - else if (i && numericSymbolDetector >= 1000) { - // Decide whether we should add a numeric symbol like k (thousands) - // or M (millions). If we are to enable this in tooltip or other - // places as well, we can move this logic to the numberFormatter and - // enable it by a parameter. - while (i-- && typeof ret === 'undefined') { - multi = Math.pow(numSymMagnitude, i + 1); - if ( - // Only accept a numeric symbol when the distance is more - // than a full unit. So for example if the symbol is k, we - // don't accept numbers like 0.5k. - numericSymbolDetector >= multi && - // Accept one decimal before the symbol. Accepts 0.5k but - // not 0.25k. How does this work with the previous? - (value * 10) % multi === 0 && - numericSymbols[i] !== null && - value !== 0) { // #5480 - ret = numberFormatter(value / multi, -1) + numericSymbols[i]; - } - } - } - if (typeof ret === 'undefined') { - if (Math.abs(value) >= 10000) { // add thousands separators - ret = numberFormatter(value, -1); - } - else { // small numbers - ret = numberFormatter(value, -1, void 0, ''); // #2466 - } - } - return ret; - }; - /** - * Get the minimum and maximum for the series of each axis. The function - * analyzes the axis series and updates `this.dataMin` and `this.dataMax`. - * - * @private - * @function Highcharts.Axis#getSeriesExtremes - * - * @fires Highcharts.Axis#event:afterGetSeriesExtremes - * @fires Highcharts.Axis#event:getSeriesExtremes - */ - Axis.prototype.getSeriesExtremes = function () { - var axis = this, - chart = axis.chart, - xExtremes; - fireEvent(this, 'getSeriesExtremes', null, function () { - axis.hasVisibleSeries = false; - // Reset properties in case we're redrawing (#3353) - axis.dataMin = axis.dataMax = axis.threshold = null; - axis.softThreshold = !axis.isXAxis; - if (axis.stacking) { - axis.stacking.buildStacks(); - } - // loop through this axis' series - axis.series.forEach(function (series) { - if (series.visible || - !chart.options.chart.ignoreHiddenSeries) { - var seriesOptions = series.options, - xData, - threshold = seriesOptions.threshold, - seriesDataMin, - seriesDataMax; - axis.hasVisibleSeries = true; - // Validate threshold in logarithmic axes - if (axis.positiveValuesOnly && threshold <= 0) { - threshold = null; - } - // Get dataMin and dataMax for X axes - if (axis.isXAxis) { - xData = series.xData; - if (xData.length) { - var isPositive = function (number) { return number > 0; }; - xData = axis.logarithmic ? - xData.filter(axis.validatePositiveValue) : - xData; - xExtremes = series.getXExtremes(xData); - // If xData contains values which is not numbers, - // then filter them out. To prevent performance hit, - // we only do this after we have already found - // seriesDataMin because in most cases all data is - // valid. #5234. - seriesDataMin = xExtremes.min; - seriesDataMax = xExtremes.max; - if (!isNumber(seriesDataMin) && - // #5010: - !(seriesDataMin instanceof Date)) { - xData = xData.filter(isNumber); - xExtremes = series.getXExtremes(xData); - // Do it again with valid data - seriesDataMin = xExtremes.min; - seriesDataMax = xExtremes.max; - } - if (xData.length) { - axis.dataMin = Math.min(pick(axis.dataMin, seriesDataMin), seriesDataMin); - axis.dataMax = Math.max(pick(axis.dataMax, seriesDataMax), seriesDataMax); - } - } - // Get dataMin and dataMax for Y axes, as well as handle - // stacking and processed data - } - else { - // Get this particular series extremes - var dataExtremes = series.applyExtremes(); - // Get the dataMin and dataMax so far. If percentage is - // used, the min and max are always 0 and 100. If - // seriesDataMin and seriesDataMax is null, then series - // doesn't have active y data, we continue with nulls - if (isNumber(dataExtremes.dataMin)) { - seriesDataMin = dataExtremes.dataMin; - axis.dataMin = Math.min(pick(axis.dataMin, seriesDataMin), seriesDataMin); - } - if (isNumber(dataExtremes.dataMax)) { - seriesDataMax = dataExtremes.dataMax; - axis.dataMax = Math.max(pick(axis.dataMax, seriesDataMax), seriesDataMax); - } - // Adjust to threshold - if (defined(threshold)) { - axis.threshold = threshold; - } - // If any series has a hard threshold, it takes - // precedence - if (!seriesOptions.softThreshold || - axis.positiveValuesOnly) { - axis.softThreshold = false; - } - } - } - }); - }); - fireEvent(this, 'afterGetSeriesExtremes'); - }; - /** - * Translate from axis value to pixel position on the chart, or back. Use - * the `toPixels` and `toValue` functions in applications. - * - * @private - * @function Highcharts.Axis#translate - * - * @param {number} val - * TO-DO: parameter description - * - * @param {boolean|null} [backwards] - * TO-DO: parameter description - * - * @param {boolean|null} [cvsCoord] - * TO-DO: parameter description - * - * @param {boolean|null} [old] - * TO-DO: parameter description - * - * @param {boolean} [handleLog] - * TO-DO: parameter description - * - * @param {number} [pointPlacement] - * TO-DO: parameter description - * - * @return {number|undefined} - */ - Axis.prototype.translate = function (val, backwards, cvsCoord, old, handleLog, pointPlacement) { - var axis = this.linkedParent || this, // #1417 - sign = 1, - cvsOffset = 0, - localA = old ? axis.oldTransA : axis.transA, - localMin = old ? axis.oldMin : axis.min, - returnValue = 0, - minPixelPadding = axis.minPixelPadding, - doPostTranslate = (axis.isOrdinal || - axis.brokenAxis && axis.brokenAxis.hasBreaks || - (axis.logarithmic && handleLog)) && axis.lin2val; - if (!localA) { - localA = axis.transA; - } - // In vertical axes, the canvas coordinates start from 0 at the top like - // in SVG. - if (cvsCoord) { - sign *= -1; // canvas coordinates inverts the value - cvsOffset = axis.len; - } - // Handle reversed axis - if (axis.reversed) { - sign *= -1; - cvsOffset -= sign * (axis.sector || axis.len); - } - // From pixels to value - if (backwards) { // reverse translation - val = val * sign + cvsOffset; - val -= minPixelPadding; - // from chart pixel to value: - returnValue = val / localA + localMin; - if (doPostTranslate) { // log and ordinal axes - returnValue = axis.lin2val(returnValue); - } - // From value to pixels - } - else { - if (doPostTranslate) { // log and ordinal axes - val = axis.val2lin(val); - } - returnValue = isNumber(localMin) ? - (sign * (val - localMin) * localA + - cvsOffset + - (sign * minPixelPadding) + - (isNumber(pointPlacement) ? - localA * pointPlacement : - 0)) : - void 0; - } - return returnValue; - }; - /** - * Translate a value in terms of axis units into pixels within the chart. - * - * @function Highcharts.Axis#toPixels - * - * @param {number} value - * A value in terms of axis units. - * - * @param {boolean} paneCoordinates - * Whether to return the pixel coordinate relative to the chart or just the - * axis/pane itself. - * - * @return {number} - * Pixel position of the value on the chart or axis. - */ - Axis.prototype.toPixels = function (value, paneCoordinates) { - return this.translate(value, false, !this.horiz, null, true) + - (paneCoordinates ? 0 : this.pos); - }; - /** - * Translate a pixel position along the axis to a value in terms of axis - * units. - * - * @function Highcharts.Axis#toValue - * - * @param {number} pixel - * The pixel value coordinate. - * - * @param {boolean} [paneCoordinates=false] - * Whether the input pixel is relative to the chart or just the axis/pane - * itself. - * - * @return {number} - * The axis value. - */ - Axis.prototype.toValue = function (pixel, paneCoordinates) { - return this.translate(pixel - (paneCoordinates ? 0 : this.pos), true, !this.horiz, null, true); - }; - /** - * Create the path for a plot line that goes from the given value on - * this axis, across the plot to the opposite side. Also used internally for - * grid lines and crosshairs. - * - * @function Highcharts.Axis#getPlotLinePath - * - * @param {Highcharts.AxisPlotLinePathOptionsObject} options - * Options for the path. - * - * @return {Highcharts.SVGPathArray|null} - * The SVG path definition for the plot line. - */ - Axis.prototype.getPlotLinePath = function (options) { - var axis = this, - chart = axis.chart, - axisLeft = axis.left, - axisTop = axis.top, - old = options.old, - value = options.value, - translatedValue = options.translatedValue, - lineWidth = options.lineWidth, - force = options.force, - x1, - y1, - x2, - y2, - cHeight = (old && chart.oldChartHeight) || chart.chartHeight, - cWidth = (old && chart.oldChartWidth) || chart.chartWidth, - skip, - transB = axis.transB, - evt; - // eslint-disable-next-line valid-jsdoc - /** - * Check if x is between a and b. If not, either move to a/b - * or skip, depending on the force parameter. - * @private - */ - function between(x, a, b) { - if (force !== 'pass' && x < a || x > b) { - if (force) { - x = clamp(x, a, b); - } - else { - skip = true; - } - } - return x; - } - evt = { - value: value, - lineWidth: lineWidth, - old: old, - force: force, - acrossPanes: options.acrossPanes, - translatedValue: translatedValue - }; - fireEvent(this, 'getPlotLinePath', evt, function (e) { - translatedValue = pick(translatedValue, axis.translate(value, null, null, old)); - // Keep the translated value within sane bounds, and avoid Infinity - // to fail the isNumber test (#7709). - translatedValue = clamp(translatedValue, -1e5, 1e5); - x1 = x2 = Math.round(translatedValue + transB); - y1 = y2 = Math.round(cHeight - translatedValue - transB); - if (!isNumber(translatedValue)) { // no min or max - skip = true; - force = false; // #7175, don't force it when path is invalid - } - else if (axis.horiz) { - y1 = axisTop; - y2 = cHeight - axis.bottom; - x1 = x2 = between(x1, axisLeft, axisLeft + axis.width); - } - else { - x1 = axisLeft; - x2 = cWidth - axis.right; - y1 = y2 = between(y1, axisTop, axisTop + axis.height); - } - e.path = skip && !force ? - null : - chart.renderer.crispLine([['M', x1, y1], ['L', x2, y2]], lineWidth || 1); - }); - return evt.path; - }; - /** - * Internal function to et the tick positions of a linear axis to round - * values like whole tens or every five. - * - * @function Highcharts.Axis#getLinearTickPositions - * - * @param {number} tickInterval - * The normalized tick interval. - * - * @param {number} min - * Axis minimum. - * - * @param {number} max - * Axis maximum. - * - * @return {Array<number>} - * An array of axis values where ticks should be placed. - */ - Axis.prototype.getLinearTickPositions = function (tickInterval, min, max) { - var pos, - lastPos, - roundedMin = correctFloat(Math.floor(min / tickInterval) * tickInterval), - roundedMax = correctFloat(Math.ceil(max / tickInterval) * tickInterval), - tickPositions = [], - precision; - // When the precision is higher than what we filter out in - // correctFloat, skip it (#6183). - if (correctFloat(roundedMin + tickInterval) === roundedMin) { - precision = 20; - } - // For single points, add a tick regardless of the relative position - // (#2662, #6274) - if (this.single) { - return [min]; - } - // Populate the intermediate values - pos = roundedMin; - while (pos <= roundedMax) { - // Place the tick on the rounded value - tickPositions.push(pos); - // Always add the raw tickInterval, not the corrected one. - pos = correctFloat(pos + tickInterval, precision); - // If the interval is not big enough in the current min - max range - // to actually increase the loop variable, we need to break out to - // prevent endless loop. Issue #619 - if (pos === lastPos) { - break; - } - // Record the last value - lastPos = pos; - } - return tickPositions; - }; - /** - * Resolve the new minorTicks/minorTickInterval options into the legacy - * loosely typed minorTickInterval option. - * - * @function Highcharts.Axis#getMinorTickInterval - * - * @return {number|"auto"|null} - */ - Axis.prototype.getMinorTickInterval = function () { - var options = this.options; - if (options.minorTicks === true) { - return pick(options.minorTickInterval, 'auto'); - } - if (options.minorTicks === false) { - return null; - } - return options.minorTickInterval; - }; - /** - * Internal function to return the minor tick positions. For logarithmic - * axes, the same logic as for major ticks is reused. - * - * @function Highcharts.Axis#getMinorTickPositions - * - * @return {Array<number>} - * An array of axis values where ticks should be placed. - */ - Axis.prototype.getMinorTickPositions = function () { - var axis = this, - options = axis.options, - tickPositions = axis.tickPositions, - minorTickInterval = axis.minorTickInterval, - minorTickPositions = [], - pos, - pointRangePadding = axis.pointRangePadding || 0, - min = axis.min - pointRangePadding, // #1498 - max = axis.max + pointRangePadding, // #1498 - range = max - min; - // If minor ticks get too dense, they are hard to read, and may cause - // long running script. So we don't draw them. - if (range && range / minorTickInterval < axis.len / 3) { // #3875 - var logarithmic_1 = axis.logarithmic; - if (logarithmic_1) { - // For each interval in the major ticks, compute the minor ticks - // separately. - this.paddedTicks.forEach(function (_pos, i, paddedTicks) { - if (i) { - minorTickPositions.push.apply(minorTickPositions, logarithmic_1.getLogTickPositions(minorTickInterval, paddedTicks[i - 1], paddedTicks[i], true)); - } - }); - } - else if (axis.dateTime && - this.getMinorTickInterval() === 'auto') { // #1314 - minorTickPositions = minorTickPositions.concat(axis.getTimeTicks(axis.dateTime.normalizeTimeTickInterval(minorTickInterval), min, max, options.startOfWeek)); - } - else { - for (pos = min + (tickPositions[0] - min) % minorTickInterval; pos <= max; pos += minorTickInterval) { - // Very, very, tight grid lines (#5771) - if (pos === minorTickPositions[0]) { - break; - } - minorTickPositions.push(pos); - } - } - } - if (minorTickPositions.length !== 0) { - axis.trimTicks(minorTickPositions); // #3652 #3743 #1498 #6330 - } - return minorTickPositions; - }; - /** - * Adjust the min and max for the minimum range. Keep in mind that the - * series data is not yet processed, so we don't have information on data - * cropping and grouping, or updated `axis.pointRange` or - * `series.pointRange`. The data can't be processed until we have finally - * established min and max. - * - * @private - * @function Highcharts.Axis#adjustForMinRange - */ - Axis.prototype.adjustForMinRange = function () { - var axis = this, - options = axis.options, - min = axis.min, - max = axis.max, - log = axis.logarithmic, - zoomOffset, - spaceAvailable, - closestDataRange, - i, - distance, - xData, - loopLength, - minArgs, - maxArgs, - minRange; - // Set the automatic minimum range based on the closest point distance - if (axis.isXAxis && - typeof axis.minRange === 'undefined' && - !log) { - if (defined(options.min) || defined(options.max)) { - axis.minRange = null; // don't do this again - } - else { - // Find the closest distance between raw data points, as opposed - // to closestPointRange that applies to processed points - // (cropped and grouped) - axis.series.forEach(function (series) { - xData = series.xData; - loopLength = series.xIncrement ? 1 : xData.length - 1; - for (i = loopLength; i > 0; i--) { - distance = xData[i] - xData[i - 1]; - if (typeof closestDataRange === 'undefined' || - distance < closestDataRange) { - closestDataRange = distance; - } - } - }); - axis.minRange = Math.min(closestDataRange * 5, axis.dataMax - axis.dataMin); - } - } - // if minRange is exceeded, adjust - if (max - min < axis.minRange) { - spaceAvailable = - axis.dataMax - axis.dataMin >= - axis.minRange; - minRange = axis.minRange; - zoomOffset = (minRange - max + min) / 2; - // if min and max options have been set, don't go beyond it - minArgs = [ - min - zoomOffset, - pick(options.min, min - zoomOffset) - ]; - // If space is available, stay within the data range - if (spaceAvailable) { - minArgs[2] = axis.logarithmic ? - axis.logarithmic.log2lin(axis.dataMin) : - axis.dataMin; - } - min = arrayMax(minArgs); - maxArgs = [ - min + minRange, - pick(options.max, min + minRange) - ]; - // If space is availabe, stay within the data range - if (spaceAvailable) { - maxArgs[2] = log ? - log.log2lin(axis.dataMax) : - axis.dataMax; - } - max = arrayMin(maxArgs); - // now if the max is adjusted, adjust the min back - if (max - min < minRange) { - minArgs[0] = max - minRange; - minArgs[1] = pick(options.min, max - minRange); - min = arrayMax(minArgs); - } - } - // Record modified extremes - axis.min = min; - axis.max = max; - }; - // eslint-disable-next-line valid-jsdoc - /** - * Find the closestPointRange across all series. - * - * @private - * @function Highcharts.Axis#getClosest - */ - Axis.prototype.getClosest = function () { - var ret; - if (this.categories) { - ret = 1; - } - else { - this.series.forEach(function (series) { - var seriesClosest = series.closestPointRange, - visible = series.visible || - !series.chart.options.chart.ignoreHiddenSeries; - if (!series.noSharedTooltip && - defined(seriesClosest) && - visible) { - ret = defined(ret) ? - Math.min(ret, seriesClosest) : - seriesClosest; - } - }); - } - return ret; - }; - /** - * When a point name is given and no x, search for the name in the existing - * categories, or if categories aren't provided, search names or create a - * new category (#2522). - * @private - * @function Highcharts.Axis#nameToX - * - * @param {Highcharts.Point} point - * The point to inspect. - * - * @return {number} - * The X value that the point is given. - */ - Axis.prototype.nameToX = function (point) { - var explicitCategories = isArray(this.categories), - names = explicitCategories ? this.categories : this.names, - nameX = point.options.x, - x; - point.series.requireSorting = false; - if (!defined(nameX)) { - nameX = this.options.uniqueNames === false ? - point.series.autoIncrement() : - (explicitCategories ? - names.indexOf(point.name) : - pick(names.keys[point.name], -1)); - } - if (nameX === -1) { // Not found in currenct categories - if (!explicitCategories) { - x = names.length; - } - } - else { - x = nameX; - } - // Write the last point's name to the names array - if (typeof x !== 'undefined') { - this.names[x] = point.name; - // Backwards mapping is much faster than array searching (#7725) - this.names.keys[point.name] = x; - } - return x; - }; - /** - * When changes have been done to series data, update the axis.names. - * - * @private - * @function Highcharts.Axis#updateNames - */ - Axis.prototype.updateNames = function () { - var axis = this, - names = this.names, - i = names.length; - if (i > 0) { - Object.keys(names.keys).forEach(function (key) { - delete (names.keys)[key]; - }); - names.length = 0; - this.minRange = this.userMinRange; // Reset - (this.series || []).forEach(function (series) { - // Reset incrementer (#5928) - series.xIncrement = null; - // When adding a series, points are not yet generated - if (!series.points || series.isDirtyData) { - // When we're updating the series with data that is longer - // than it was, and cropThreshold is passed, we need to make - // sure that the axis.max is increased _before_ running the - // premature processData. Otherwise this early iteration of - // processData will crop the points to axis.max, and the - // names array will be too short (#5857). - axis.max = Math.max(axis.max, series.xData.length - 1); - series.processData(); - series.generatePoints(); - } - series.data.forEach(function (point, i) { - var x; - if (point && - point.options && - typeof point.name !== 'undefined' // #9562 - ) { - x = axis.nameToX(point); - if (typeof x !== 'undefined' && x !== point.x) { - point.x = x; - series.xData[i] = x; - } - } - }); - }); - } - }; - /** - * Update translation information. - * - * @private - * @function Highcharts.Axis#setAxisTranslation - * - * @param {boolean} [saveOld] - * TO-DO: parameter description - * - * @fires Highcharts.Axis#event:afterSetAxisTranslation - */ - Axis.prototype.setAxisTranslation = function (saveOld) { - var axis = this, - range = axis.max - axis.min, - pointRange = axis.axisPointRange || 0, - closestPointRange, - minPointOffset = 0, - pointRangePadding = 0, - linkedParent = axis.linkedParent, - ordinalCorrection, - hasCategories = !!axis.categories, - transA = axis.transA, - isXAxis = axis.isXAxis; - // Adjust translation for padding. Y axis with categories need to go - // through the same (#1784). - if (isXAxis || hasCategories || pointRange) { - // Get the closest points - closestPointRange = axis.getClosest(); - if (linkedParent) { - minPointOffset = linkedParent.minPointOffset; - pointRangePadding = linkedParent.pointRangePadding; - } - else { - axis.series.forEach(function (series) { - var seriesPointRange = hasCategories ? - 1 : - (isXAxis ? - pick(series.options.pointRange, - closestPointRange, 0) : - (axis.axisPointRange || 0)), // #2806 - pointPlacement = series.options.pointPlacement; - pointRange = Math.max(pointRange, seriesPointRange); - if (!axis.single || hasCategories) { - // TODO: series should internally set x- and y- - // pointPlacement to simplify this logic. - var isPointPlacementAxis = series.is('xrange') ? !isXAxis : isXAxis; - // minPointOffset is the value padding to the left of - // the axis in order to make room for points with a - // pointRange, typically columns. When the - // pointPlacement option is 'between' or 'on', this - // padding does not apply. - minPointOffset = Math.max(minPointOffset, isPointPlacementAxis && isString(pointPlacement) ? - 0 : - seriesPointRange / 2); - // Determine the total padding needed to the length of - // the axis to make room for the pointRange. If the - // series' pointPlacement is 'on', no padding is added. - pointRangePadding = Math.max(pointRangePadding, isPointPlacementAxis && pointPlacement === 'on' ? - 0 : - seriesPointRange); - } - }); - } - // Record minPointOffset and pointRangePadding - ordinalCorrection = axis.ordinal && axis.ordinal.slope && closestPointRange ? - axis.ordinal.slope / closestPointRange : - 1; // #988, #1853 - axis.minPointOffset = minPointOffset = - minPointOffset * ordinalCorrection; - axis.pointRangePadding = - pointRangePadding = pointRangePadding * ordinalCorrection; - // pointRange means the width reserved for each point, like in a - // column chart - axis.pointRange = Math.min(pointRange, axis.single && hasCategories ? 1 : range); - // closestPointRange means the closest distance between points. In - // columns it is mostly equal to pointRange, but in lines pointRange - // is 0 while closestPointRange is some other value - if (isXAxis) { - axis.closestPointRange = closestPointRange; - } - } - // Secondary values - if (saveOld) { - axis.oldTransA = transA; - } - axis.translationSlope = axis.transA = transA = - axis.staticScale || - axis.len / ((range + pointRangePadding) || 1); - // Translation addend - axis.transB = axis.horiz ? axis.left : axis.bottom; - axis.minPixelPadding = transA * minPointOffset; - fireEvent(this, 'afterSetAxisTranslation'); - }; - /** - * @private - * @function Highcharts.Axis#minFromRange - * - * @return {number} - */ - Axis.prototype.minFromRange = function () { - var axis = this; - return axis.max - axis.range; - }; - /** - * Set the tick positions to round values and optionally extend the extremes - * to the nearest tick. - * - * @private - * @function Highcharts.Axis#setTickInterval - * - * @param {boolean} secondPass - * TO-DO: parameter description - * - * @fires Highcharts.Axis#event:foundExtremes - */ - Axis.prototype.setTickInterval = function (secondPass) { - var axis = this, - chart = axis.chart, - log = axis.logarithmic, - options = axis.options, - isXAxis = axis.isXAxis, - isLinked = axis.isLinked, - maxPadding = options.maxPadding, - minPadding = options.minPadding, - length, - linkedParentExtremes, - tickIntervalOption = options.tickInterval, - minTickInterval, - tickPixelIntervalOption = options.tickPixelInterval, - categories = axis.categories, - threshold = isNumber(axis.threshold) ? axis.threshold : null, - softThreshold = axis.softThreshold, - thresholdMin, - thresholdMax, - hardMin, - hardMax; - if (!axis.dateTime && !categories && !isLinked) { - this.getTickAmount(); - } - // Min or max set either by zooming/setExtremes or initial options - hardMin = pick(axis.userMin, options.min); - hardMax = pick(axis.userMax, options.max); - // Linked axis gets the extremes from the parent axis - if (isLinked) { - axis.linkedParent = chart[axis.coll][options.linkedTo]; - linkedParentExtremes = axis.linkedParent.getExtremes(); - axis.min = pick(linkedParentExtremes.min, linkedParentExtremes.dataMin); - axis.max = pick(linkedParentExtremes.max, linkedParentExtremes.dataMax); - if (options.type !== axis.linkedParent.options.type) { - // Can't link axes of different type - error(11, 1, chart); - } - // Initial min and max from the extreme data values - } - else { - // Adjust to hard threshold - if (softThreshold && defined(threshold)) { - if (axis.dataMin >= threshold) { - thresholdMin = threshold; - minPadding = 0; - } - else if (axis.dataMax <= threshold) { - thresholdMax = threshold; - maxPadding = 0; - } - } - axis.min = pick(hardMin, thresholdMin, axis.dataMin); - axis.max = pick(hardMax, thresholdMax, axis.dataMax); - } - if (log) { - if (axis.positiveValuesOnly && - !secondPass && - Math.min(axis.min, pick(axis.dataMin, axis.min)) <= 0) { // #978 - // Can't plot negative values on log axis - error(10, 1, chart); - } - // The correctFloat cures #934, float errors on full tens. But it - // was too aggressive for #4360 because of conversion back to lin, - // therefore use precision 15. - axis.min = correctFloat(log.log2lin(axis.min), 16); - axis.max = correctFloat(log.log2lin(axis.max), 16); - } - // handle zoomed range - if (axis.range && defined(axis.max)) { - // #618, #6773: - axis.userMin = axis.min = hardMin = - Math.max(axis.dataMin, axis.minFromRange()); - axis.userMax = hardMax = axis.max; - axis.range = null; // don't use it when running setExtremes - } - // Hook for Highstock Scroller. Consider combining with beforePadding. - fireEvent(axis, 'foundExtremes'); - // Hook for adjusting this.min and this.max. Used by bubble series. - if (axis.beforePadding) { - axis.beforePadding(); - } - // adjust min and max for the minimum range - axis.adjustForMinRange(); - // Pad the values to get clear of the chart's edges. To avoid - // tickInterval taking the padding into account, we do this after - // computing tick interval (#1337). - if (!categories && - !axis.axisPointRange && - !(axis.stacking && axis.stacking.usePercentage) && - !isLinked && - defined(axis.min) && - defined(axis.max)) { - length = axis.max - axis.min; - if (length) { - if (!defined(hardMin) && minPadding) { - axis.min -= length * minPadding; - } - if (!defined(hardMax) && maxPadding) { - axis.max += length * maxPadding; - } - } - } - // Handle options for floor, ceiling, softMin and softMax (#6359) - if (!isNumber(axis.userMin)) { - if (isNumber(options.softMin) && options.softMin < axis.min) { - axis.min = hardMin = options.softMin; // #6894 - } - if (isNumber(options.floor)) { - axis.min = Math.max(axis.min, options.floor); - } - } - if (!isNumber(axis.userMax)) { - if (isNumber(options.softMax) && options.softMax > axis.max) { - axis.max = hardMax = options.softMax; // #6894 - } - if (isNumber(options.ceiling)) { - axis.max = Math.min(axis.max, options.ceiling); - } - } - // When the threshold is soft, adjust the extreme value only if the data - // extreme and the padded extreme land on either side of the threshold. - // For example, a series of [0, 1, 2, 3] would make the yAxis add a tick - // for -1 because of the default minPadding and startOnTick options. - // This is prevented by the softThreshold option. - if (softThreshold && defined(axis.dataMin)) { - threshold = threshold || 0; - if (!defined(hardMin) && - axis.min < threshold && - axis.dataMin >= threshold) { - axis.min = axis.options.minRange ? - Math.min(threshold, axis.max - - axis.minRange) : - threshold; - } - else if (!defined(hardMax) && - axis.max > threshold && - axis.dataMax <= threshold) { - axis.max = axis.options.minRange ? - Math.max(threshold, axis.min + - axis.minRange) : - threshold; - } - } - // get tickInterval - if (axis.min === axis.max || - typeof axis.min === 'undefined' || - typeof axis.max === 'undefined') { - axis.tickInterval = 1; - } - else if (isLinked && - !tickIntervalOption && - tickPixelIntervalOption === - axis.linkedParent.options.tickPixelInterval) { - axis.tickInterval = tickIntervalOption = - axis.linkedParent.tickInterval; - } - else { - axis.tickInterval = pick(tickIntervalOption, this.tickAmount ? - ((axis.max - axis.min) / - Math.max(this.tickAmount - 1, 1)) : - void 0, - // For categoried axis, 1 is default, for linear axis use - // tickPix - categories ? - 1 : - // don't let it be more than the data range - (axis.max - axis.min) * - tickPixelIntervalOption / - Math.max(axis.len, tickPixelIntervalOption)); - } - // Now we're finished detecting min and max, crop and group series data. - // This is in turn needed in order to find tick positions in ordinal - // axes. - if (isXAxis && !secondPass) { - axis.series.forEach(function (series) { - series.processData(axis.min !== axis.oldMin || axis.max !== axis.oldMax); - }); - } - // set the translation factor used in translate function - axis.setAxisTranslation(true); - // hook for ordinal axes and radial axes - fireEvent(this, 'initialAxisTranslation'); - // In column-like charts, don't cramp in more ticks than there are - // points (#1943, #4184) - if (axis.pointRange && !tickIntervalOption) { - axis.tickInterval = Math.max(axis.pointRange, axis.tickInterval); - } - // Before normalizing the tick interval, handle minimum tick interval. - // This applies only if tickInterval is not defined. - minTickInterval = pick(options.minTickInterval, - // In datetime axes, don't go below the data interval, except when - // there are scatter-like series involved (#13369). - axis.dateTime && - !axis.series.some(function (s) { return s.noSharedTooltip; }) ? - axis.closestPointRange : 0); - if (!tickIntervalOption && axis.tickInterval < minTickInterval) { - axis.tickInterval = minTickInterval; - } - // for linear axes, get magnitude and normalize the interval - if (!axis.dateTime && !axis.logarithmic && !tickIntervalOption) { - axis.tickInterval = normalizeTickInterval(axis.tickInterval, void 0, getMagnitude(axis.tickInterval), pick(options.allowDecimals, - // If the tick interval is greather than 0.5, avoid - // decimals, as linear axes are often used to render - // discrete values. #3363. If a tick amount is set, allow - // decimals by default, as it increases the chances for a - // good fit. - axis.tickInterval < 0.5 || this.tickAmount !== void 0), !!this.tickAmount); - } - // Prevent ticks from getting so close that we can't draw the labels - if (!this.tickAmount) { - axis.tickInterval = axis.unsquish(); - } - this.setTickPositions(); - }; - /** - * Now we have computed the normalized tickInterval, get the tick positions. - * - * @private - * @function Highcharts.Axis#setTickPositions - * - * @fires Highcharts.Axis#event:afterSetTickPositions - */ - Axis.prototype.setTickPositions = function () { - var axis = this, - options = this.options, - tickPositions, - tickPositionsOption = options.tickPositions, - minorTickIntervalOption = this.getMinorTickInterval(), - tickPositioner = options.tickPositioner, - hasVerticalPanning = this.hasVerticalPanning(), - isColorAxis = this.coll === 'colorAxis', - startOnTick = (isColorAxis || !hasVerticalPanning) && options.startOnTick, - endOnTick = (isColorAxis || !hasVerticalPanning) && options.endOnTick; - // Set the tickmarkOffset - this.tickmarkOffset = (this.categories && - options.tickmarkPlacement === 'between' && - this.tickInterval === 1) ? 0.5 : 0; // #3202 - // get minorTickInterval - this.minorTickInterval = - minorTickIntervalOption === 'auto' && - this.tickInterval ? - this.tickInterval / 5 : - minorTickIntervalOption; - // When there is only one point, or all points have the same value on - // this axis, then min and max are equal and tickPositions.length is 0 - // or 1. In this case, add some padding in order to center the point, - // but leave it with one tick. #1337. - this.single = - this.min === this.max && - defined(this.min) && - !this.tickAmount && - ( - // Data is on integer (#6563) - parseInt(this.min, 10) === this.min || - // Between integers and decimals are not allowed (#6274) - options.allowDecimals !== false); - /** - * Contains the current positions that are laid out on the axis. The - * positions are numbers in terms of axis values. In a category axis - * they are integers, in a datetime axis they are also integers, but - * designating milliseconds. - * - * This property is read only - for modifying the tick positions, use - * the `tickPositioner` callback or [axis.tickPositions( - * https://api.highcharts.com/highcharts/xAxis.tickPositions) option - * instead. - * - * @name Highcharts.Axis#tickPositions - * @type {Highcharts.AxisTickPositionsArray|undefined} - */ - this.tickPositions = - // Find the tick positions. Work on a copy (#1565) - tickPositions = - (tickPositionsOption && tickPositionsOption.slice()); - if (!tickPositions) { - // Too many ticks (#6405). Create a friendly warning and provide two - // ticks so at least we can show the data series. - if ((!axis.ordinal || !axis.ordinal.positions) && - ((this.max - this.min) / - this.tickInterval > - Math.max(2 * this.len, 200))) { - tickPositions = [this.min, this.max]; - error(19, false, this.chart); - } - else if (axis.dateTime) { - tickPositions = axis.getTimeTicks(axis.dateTime.normalizeTimeTickInterval(this.tickInterval, options.units), this.min, this.max, options.startOfWeek, axis.ordinal && axis.ordinal.positions, this.closestPointRange, true); - } - else if (axis.logarithmic) { - tickPositions = axis.logarithmic.getLogTickPositions(this.tickInterval, this.min, this.max); - } - else { - tickPositions = this.getLinearTickPositions(this.tickInterval, this.min, this.max); - } - // Too dense ticks, keep only the first and last (#4477) - if (tickPositions.length > this.len) { - tickPositions = [tickPositions[0], tickPositions.pop()]; - // Reduce doubled value (#7339) - if (tickPositions[0] === tickPositions[1]) { - tickPositions.length = 1; - } - } - this.tickPositions = tickPositions; - // Run the tick positioner callback, that allows modifying auto tick - // positions. - if (tickPositioner) { - tickPositioner = tickPositioner.apply(axis, [this.min, this.max]); - if (tickPositioner) { - this.tickPositions = tickPositions = tickPositioner; - } - } - } - // Reset min/max or remove extremes based on start/end on tick - this.paddedTicks = tickPositions.slice(0); // Used for logarithmic minor - this.trimTicks(tickPositions, startOnTick, endOnTick); - if (!this.isLinked) { - // Substract half a unit (#2619, #2846, #2515, #3390), - // but not in case of multiple ticks (#6897) - if (this.single && - tickPositions.length < 2 && - !this.categories && - !this.series.some(function (s) { - return (s.is('heatmap') && s.options.pointPlacement === 'between'); - })) { - this.min -= 0.5; - this.max += 0.5; - } - if (!tickPositionsOption && !tickPositioner) { - this.adjustTickAmount(); - } - } - fireEvent(this, 'afterSetTickPositions'); - }; - /** - * Handle startOnTick and endOnTick by either adapting to padding min/max or - * rounded min/max. Also handle single data points. - * - * @private - * @function Highcharts.Axis#trimTicks - * - * @param {Array<number>} tickPositions - * TO-DO: parameter description - * - * @param {boolean} [startOnTick] - * TO-DO: parameter description - * - * @param {boolean} [endOnTick] - * TO-DO: parameter description - */ - Axis.prototype.trimTicks = function (tickPositions, startOnTick, endOnTick) { - var roundedMin = tickPositions[0], - roundedMax = tickPositions[tickPositions.length - 1], - minPointOffset = (!this.isOrdinal && this.minPointOffset) || 0; // (#12716) - fireEvent(this, 'trimTicks'); - if (!this.isLinked) { - if (startOnTick && roundedMin !== -Infinity) { // #6502 - this.min = roundedMin; - } - else { - while (this.min - minPointOffset > tickPositions[0]) { - tickPositions.shift(); - } - } - if (endOnTick) { - this.max = roundedMax; - } - else { - while (this.max + minPointOffset < - tickPositions[tickPositions.length - 1]) { - tickPositions.pop(); - } - } - // If no tick are left, set one tick in the middle (#3195) - if (tickPositions.length === 0 && - defined(roundedMin) && - !this.options.tickPositions) { - tickPositions.push((roundedMax + roundedMin) / 2); - } - } - }; - /** - * Check if there are multiple axes in the same pane. - * - * @private - * @function Highcharts.Axis#alignToOthers - * - * @return {boolean|undefined} - * True if there are other axes. - */ - Axis.prototype.alignToOthers = function () { - var axis = this, - others = // Whether there is another axis to pair with this one - {}, - hasOther, - options = axis.options; - if ( - // Only if alignTicks is true - this.chart.options.chart.alignTicks !== false && - options.alignTicks !== false && - // Disabled when startOnTick or endOnTick are false (#7604) - options.startOnTick !== false && - options.endOnTick !== false && - // Don't try to align ticks on a log axis, they are not evenly - // spaced (#6021) - !axis.logarithmic) { - this.chart[this.coll].forEach(function (axis) { - var otherOptions = axis.options, horiz = axis.horiz, key = [ - horiz ? otherOptions.left : otherOptions.top, - otherOptions.width, - otherOptions.height, - otherOptions.pane - ].join(','); - if (axis.series.length) { // #4442 - if (others[key]) { - hasOther = true; // #4201 - } - else { - others[key] = 1; - } - } - }); - } - return hasOther; - }; - /** - * Find the max ticks of either the x and y axis collection, and record it - * in `this.tickAmount`. - * - * @private - * @function Highcharts.Axis#getTickAmount - */ - Axis.prototype.getTickAmount = function () { - var axis = this, - options = this.options, - tickAmount = options.tickAmount, - tickPixelInterval = options.tickPixelInterval; - if (!defined(options.tickInterval) && - !tickAmount && this.len < tickPixelInterval && - !this.isRadial && - !axis.logarithmic && - options.startOnTick && - options.endOnTick) { - tickAmount = 2; - } - if (!tickAmount && this.alignToOthers()) { - // Add 1 because 4 tick intervals require 5 ticks (including first - // and last) - tickAmount = Math.ceil(this.len / tickPixelInterval) + 1; - } - // For tick amounts of 2 and 3, compute five ticks and remove the - // intermediate ones. This prevents the axis from adding ticks that are - // too far away from the data extremes. - if (tickAmount < 4) { - this.finalTickAmt = tickAmount; - tickAmount = 5; - } - this.tickAmount = tickAmount; - }; - /** - * When using multiple axes, adjust the number of ticks to match the highest - * number of ticks in that group. - * - * @private - * @function Highcharts.Axis#adjustTickAmount - */ - Axis.prototype.adjustTickAmount = function () { - var axis = this, - axisOptions = axis.options, - tickInterval = axis.tickInterval, - tickPositions = axis.tickPositions, - tickAmount = axis.tickAmount, - finalTickAmt = axis.finalTickAmt, - currentTickAmount = tickPositions && tickPositions.length, - threshold = pick(axis.threshold, - axis.softThreshold ? 0 : null), - min, - len, - i; - if (axis.hasData()) { - if (currentTickAmount < tickAmount) { - min = axis.min; - while (tickPositions.length < tickAmount) { - // Extend evenly for both sides unless we're on the - // threshold (#3965) - if (tickPositions.length % 2 || - min === threshold) { - // to the end - tickPositions.push(correctFloat(tickPositions[tickPositions.length - 1] + - tickInterval)); - } - else { - // to the start - tickPositions.unshift(correctFloat(tickPositions[0] - tickInterval)); - } - } - axis.transA *= (currentTickAmount - 1) / (tickAmount - 1); - // Do not crop when ticks are not extremes (#9841) - axis.min = axisOptions.startOnTick ? - tickPositions[0] : - Math.min(axis.min, tickPositions[0]); - axis.max = axisOptions.endOnTick ? - tickPositions[tickPositions.length - 1] : - Math.max(axis.max, tickPositions[tickPositions.length - 1]); - // We have too many ticks, run second pass to try to reduce ticks - } - else if (currentTickAmount > tickAmount) { - axis.tickInterval *= 2; - axis.setTickPositions(); - } - // The finalTickAmt property is set in getTickAmount - if (defined(finalTickAmt)) { - i = len = tickPositions.length; - while (i--) { - if ( - // Remove every other tick - (finalTickAmt === 3 && i % 2 === 1) || - // Remove all but first and last - (finalTickAmt <= 2 && i > 0 && i < len - 1)) { - tickPositions.splice(i, 1); - } - } - axis.finalTickAmt = void 0; - } - } - }; - /** - * Set the scale based on data min and max, user set min and max or options. - * - * @private - * @function Highcharts.Axis#setScale - * - * @fires Highcharts.Axis#event:afterSetScale - */ - Axis.prototype.setScale = function () { - var axis = this, - isDirtyAxisLength, - isDirtyData = false, - isXAxisDirty = false; - axis.series.forEach(function (series) { - var _a; - isDirtyData = isDirtyData || series.isDirtyData || series.isDirty; - // When x axis is dirty, we need new data extremes for y as - // well: - isXAxisDirty = isXAxisDirty || ((_a = series.xAxis) === null || _a === void 0 ? void 0 : _a.isDirty) || false; - }); - axis.oldMin = axis.min; - axis.oldMax = axis.max; - axis.oldAxisLength = axis.len; - // set the new axisLength - axis.setAxisSize(); - isDirtyAxisLength = axis.len !== axis.oldAxisLength; - // do we really need to go through all this? - if (isDirtyAxisLength || - isDirtyData || - isXAxisDirty || - axis.isLinked || - axis.forceRedraw || - axis.userMin !== axis.oldUserMin || - axis.userMax !== axis.oldUserMax || - axis.alignToOthers()) { - if (axis.stacking) { - axis.stacking.resetStacks(); - } - axis.forceRedraw = false; - // get data extremes if needed - axis.getSeriesExtremes(); - // get fixed positions based on tickInterval - axis.setTickInterval(); - // record old values to decide whether a rescale is necessary later - // on (#540) - axis.oldUserMin = axis.userMin; - axis.oldUserMax = axis.userMax; - // Mark as dirty if it is not already set to dirty and extremes have - // changed. #595. - if (!axis.isDirty) { - axis.isDirty = - isDirtyAxisLength || - axis.min !== axis.oldMin || - axis.max !== axis.oldMax; - } - } - else if (axis.stacking) { - axis.stacking.cleanStacks(); - } - // Recalculate panning state object, when the data - // has changed. It is required when vertical panning is enabled. - if (isDirtyData && axis.panningState) { - axis.panningState.isDirty = true; - } - fireEvent(this, 'afterSetScale'); - }; - /** - * Set the minimum and maximum of the axes after render time. If the - * `startOnTick` and `endOnTick` options are true, the minimum and maximum - * values are rounded off to the nearest tick. To prevent this, these - * options can be set to false before calling setExtremes. Also, setExtremes - * will not allow a range lower than the `minRange` option, which by default - * is the range of five points. - * - * @sample highcharts/members/axis-setextremes/ - * Set extremes from a button - * @sample highcharts/members/axis-setextremes-datetime/ - * Set extremes on a datetime axis - * @sample highcharts/members/axis-setextremes-off-ticks/ - * Set extremes off ticks - * @sample stock/members/axis-setextremes/ - * Set extremes in Highstock - * @sample maps/members/axis-setextremes/ - * Set extremes in Highmaps - * - * @function Highcharts.Axis#setExtremes - * - * @param {number} [newMin] - * The new minimum value. - * - * @param {number} [newMax] - * The new maximum value. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart or wait for an explicit call to - * {@link Highcharts.Chart#redraw} - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] - * Enable or modify animations. - * - * @param {*} [eventArguments] - * Arguments to be accessed in event handler. - * - * @fires Highcharts.Axis#event:setExtremes - */ - Axis.prototype.setExtremes = function (newMin, newMax, redraw, animation, eventArguments) { - var axis = this, - chart = axis.chart; - redraw = pick(redraw, true); // defaults to true - axis.series.forEach(function (serie) { - delete serie.kdTree; - }); - // Extend the arguments with min and max - eventArguments = extend(eventArguments, { - min: newMin, - max: newMax - }); - // Fire the event - fireEvent(axis, 'setExtremes', eventArguments, function () { - axis.userMin = newMin; - axis.userMax = newMax; - axis.eventArgs = eventArguments; - if (redraw) { - chart.redraw(animation); - } - }); - }; - /** - * Overridable method for zooming chart. Pulled out in a separate method to - * allow overriding in stock charts. - * @private - * @function Highcharts.Axis#zoom - * - * @param {number} newMin - * TO-DO: parameter description - * - * @param {number} newMax - * TO-DO: parameter description - * - * @return {boolean} - */ - Axis.prototype.zoom = function (newMin, newMax) { - var axis = this, - dataMin = this.dataMin, - dataMax = this.dataMax, - options = this.options, - min = Math.min(dataMin, - pick(options.min, - dataMin)), - max = Math.max(dataMax, - pick(options.max, - dataMax)), - evt = { - newMin: newMin, - newMax: newMax - }; - fireEvent(this, 'zoom', evt, function (e) { - // Use e.newMin and e.newMax - event handlers may have altered them - var newMin = e.newMin, - newMax = e.newMax; - if (newMin !== axis.min || newMax !== axis.max) { // #5790 - // Prevent pinch zooming out of range. Check for defined is for - // #1946. #1734. - if (!axis.allowZoomOutside) { - // #6014, sometimes newMax will be smaller than min (or - // newMin will be larger than max). - if (defined(dataMin)) { - if (newMin < min) { - newMin = min; - } - if (newMin > max) { - newMin = max; - } - } - if (defined(dataMax)) { - if (newMax < min) { - newMax = min; - } - if (newMax > max) { - newMax = max; - } - } - } - // In full view, displaying the reset zoom button is not - // required - axis.displayBtn = (typeof newMin !== 'undefined' || - typeof newMax !== 'undefined'); - // Do it - axis.setExtremes(newMin, newMax, false, void 0, { trigger: 'zoom' }); - } - e.zoomed = true; - }); - return evt.zoomed; - }; - /** - * Update the axis metrics. - * - * @private - * @function Highcharts.Axis#setAxisSize - */ - Axis.prototype.setAxisSize = function () { - var chart = this.chart, - options = this.options, - // [top, right, bottom, left] - offsets = options.offsets || [0, 0, 0, 0], - horiz = this.horiz, - // Check for percentage based input values. Rounding fixes problems - // with column overflow and plot line filtering (#4898, #4899) - width = this.width = Math.round(relativeLength(pick(options.width, - chart.plotWidth - offsets[3] + offsets[1]), - chart.plotWidth)), - height = this.height = Math.round(relativeLength(pick(options.height, - chart.plotHeight - offsets[0] + offsets[2]), - chart.plotHeight)), - top = this.top = Math.round(relativeLength(pick(options.top, - chart.plotTop + offsets[0]), - chart.plotHeight, - chart.plotTop)), - left = this.left = Math.round(relativeLength(pick(options.left, - chart.plotLeft + offsets[3]), - chart.plotWidth, - chart.plotLeft)); - // Expose basic values to use in Series object and navigator - this.bottom = chart.chartHeight - height - top; - this.right = chart.chartWidth - width - left; - // Direction agnostic properties - this.len = Math.max(horiz ? width : height, 0); // Math.max fixes #905 - this.pos = horiz ? left : top; // distance from SVG origin - }; - /** - * Get the current extremes for the axis. - * - * @sample highcharts/members/axis-getextremes/ - * Report extremes by click on a button - * @sample maps/members/axis-getextremes/ - * Get extremes in Highmaps - * - * @function Highcharts.Axis#getExtremes - * - * @return {Highcharts.ExtremesObject} - * An object containing extremes information. - */ - Axis.prototype.getExtremes = function () { - var axis = this; - var log = axis.logarithmic; - return { - min: log ? - correctFloat(log.lin2log(axis.min)) : - axis.min, - max: log ? - correctFloat(log.lin2log(axis.max)) : - axis.max, - dataMin: axis.dataMin, - dataMax: axis.dataMax, - userMin: axis.userMin, - userMax: axis.userMax - }; - }; - /** - * Get the zero plane either based on zero or on the min or max value. - * Used in bar and area plots. - * - * @function Highcharts.Axis#getThreshold - * - * @param {number} threshold - * The threshold in axis values. - * - * @return {number|undefined} - * The translated threshold position in terms of pixels, and corrected to - * stay within the axis bounds. - */ - Axis.prototype.getThreshold = function (threshold) { - var axis = this, - log = axis.logarithmic, - realMin = log ? log.lin2log(axis.min) : axis.min, - realMax = log ? log.lin2log(axis.max) : axis.max; - if (threshold === null || threshold === -Infinity) { - threshold = realMin; - } - else if (threshold === Infinity) { - threshold = realMax; - } - else if (realMin > threshold) { - threshold = realMin; - } - else if (realMax < threshold) { - threshold = realMax; - } - return axis.translate(threshold, 0, 1, 0, 1); - }; - /** - * Compute auto alignment for the axis label based on which side the axis is - * on and the given rotation for the label. - * - * @private - * @function Highcharts.Axis#autoLabelAlign - * - * @param {number} rotation - * The rotation in degrees as set by either the `rotation` or `autoRotation` - * options. - * - * @return {Highcharts.AlignValue} - * Can be `"center"`, `"left"` or `"right"`. - */ - Axis.prototype.autoLabelAlign = function (rotation) { - var angle = (pick(rotation, 0) - (this.side * 90) + 720) % 360, - evt = { align: 'center' }; - fireEvent(this, 'autoLabelAlign', evt, function (e) { - if (angle > 15 && angle < 165) { - e.align = 'right'; - } - else if (angle > 195 && angle < 345) { - e.align = 'left'; - } - }); - return evt.align; - }; - /** - * Get the tick length and width for the axis based on axis options. - * @private - * @function Highcharts.Axis#tickSize - * - * @param {string} [prefix] - * 'tick' or 'minorTick' - * - * @return {Array<number,number>|undefined} - * An array of tickLength and tickWidth - */ - Axis.prototype.tickSize = function (prefix) { - var options = this.options, tickLength = options[prefix === 'tick' ? 'tickLength' : 'minorTickLength'], tickWidth = pick(options[prefix === 'tick' ? 'tickWidth' : 'minorTickWidth'], - // Default to 1 on linear and datetime X axes - prefix === 'tick' && this.isXAxis && !this.categories ? 1 : 0), e, tickSize; - if (tickWidth && tickLength) { - // Negate the length - if (options[prefix + 'Position'] === 'inside') { - tickLength = -tickLength; - } - tickSize = [tickLength, tickWidth]; - } - e = { tickSize: tickSize }; - fireEvent(this, 'afterTickSize', e); - return e.tickSize; - }; - /** - * Return the size of the labels. - * - * @private - * @function Highcharts.Axis#labelMetrics - * - * @return {Highcharts.FontMetricsObject} - */ - Axis.prototype.labelMetrics = function () { - var index = this.tickPositions && this.tickPositions[0] || 0; - return this.chart.renderer.fontMetrics(this.options.labels.style && - this.options.labels.style.fontSize, this.ticks[index] && this.ticks[index].label); - }; - /** - * Prevent the ticks from getting so close we can't draw the labels. On a - * horizontal axis, this is handled by rotating the labels, removing ticks - * and adding ellipsis. On a vertical axis remove ticks and add ellipsis. - * - * @private - * @function Highcharts.Axis#unsquish - * - * @return {number} - */ - Axis.prototype.unsquish = function () { - var labelOptions = this.options.labels, - horiz = this.horiz, - tickInterval = this.tickInterval, - newTickInterval = tickInterval, - slotSize = this.len / (((this.categories ? 1 : 0) + - this.max - - this.min) / - tickInterval), - rotation, - rotationOption = labelOptions.rotation, - labelMetrics = this.labelMetrics(), - step, - bestScore = Number.MAX_VALUE, - autoRotation, - range = this.max - this.min, - // Return the multiple of tickInterval that is needed to avoid - // collision - getStep = function (spaceNeeded) { - var step = spaceNeeded / (slotSize || 1); - step = step > 1 ? Math.ceil(step) : 1; - // Guard for very small or negative angles (#9835) - if (step * tickInterval > range && - spaceNeeded !== Infinity && - slotSize !== Infinity && - range) { - step = Math.ceil(range / tickInterval); - } - return correctFloat(step * tickInterval); - }; - if (horiz) { - autoRotation = !labelOptions.staggerLines && - !labelOptions.step && - ( // #3971 - defined(rotationOption) ? - [rotationOption] : - slotSize < pick(labelOptions.autoRotationLimit, 80) && labelOptions.autoRotation); - if (autoRotation) { - // Loop over the given autoRotation options, and determine - // which gives the best score. The best score is that with - // the lowest number of steps and a rotation closest - // to horizontal. - autoRotation.forEach(function (rot) { - var score; - if (rot === rotationOption || - (rot && rot >= -90 && rot <= 90)) { // #3891 - step = getStep(Math.abs(labelMetrics.h / Math.sin(deg2rad * rot))); - score = step + Math.abs(rot / 360); - if (score < bestScore) { - bestScore = score; - rotation = rot; - newTickInterval = step; - } - } - }); - } - } - else if (!labelOptions.step) { // #4411 - newTickInterval = getStep(labelMetrics.h); - } - this.autoRotation = autoRotation; - this.labelRotation = pick(rotation, rotationOption); - return newTickInterval; - }; - /** - * Get the general slot width for labels/categories on this axis. This may - * change between the pre-render (from Axis.getOffset) and the final tick - * rendering and placement. - * - * @private - * @function Highcharts.Axis#getSlotWidth - * - * @param {Highcharts.Tick} [tick] Optionally, calculate the slot width - * basing on tick label. It is used in highcharts-3d module, where the slots - * has different widths depending on perspective angles. - * - * @return {number} - * The pixel width allocated to each axis label. - */ - Axis.prototype.getSlotWidth = function (tick) { - var _a; - // #5086, #1580, #1931 - var chart = this.chart, - horiz = this.horiz, - labelOptions = this.options.labels, - slotCount = Math.max(this.tickPositions.length - (this.categories ? 0 : 1), 1), - marginLeft = chart.margin[3]; - // Used by grid axis - if (tick && isNumber(tick.slotWidth)) { // #13221, can be 0 - return tick.slotWidth; - } - if (horiz && - labelOptions && - (labelOptions.step || 0) < 2) { - if (labelOptions.rotation) { // #4415 - return 0; - } - return ((this.staggerLines || 1) * this.len) / slotCount; - } - if (!horiz) { - // #7028 - var cssWidth = (_a = labelOptions === null || labelOptions === void 0 ? void 0 : labelOptions.style) === null || _a === void 0 ? void 0 : _a.width; - if (cssWidth !== void 0) { - return parseInt(cssWidth, 10); - } - if (marginLeft) { - return marginLeft - chart.spacing[3]; - } - } - // Last resort, a fraction of the available size - return chart.chartWidth * 0.33; - }; - /** - * Render the axis labels and determine whether ellipsis or rotation need to - * be applied. - * - * @private - * @function Highcharts.Axis#renderUnsquish - */ - Axis.prototype.renderUnsquish = function () { - var chart = this.chart, - renderer = chart.renderer, - tickPositions = this.tickPositions, - ticks = this.ticks, - labelOptions = this.options.labels, - labelStyleOptions = (labelOptions && labelOptions.style || {}), - horiz = this.horiz, - slotWidth = this.getSlotWidth(), - innerWidth = Math.max(1, - Math.round(slotWidth - 2 * (labelOptions.padding || 5))), - attr = {}, - labelMetrics = this.labelMetrics(), - textOverflowOption = (labelOptions.style && - labelOptions.style.textOverflow), - commonWidth, - commonTextOverflow, - maxLabelLength = 0, - label, - i, - pos; - // Set rotation option unless it is "auto", like in gauges - if (!isString(labelOptions.rotation)) { - // #4443: - attr.rotation = labelOptions.rotation || 0; - } - // Get the longest label length - tickPositions.forEach(function (tick) { - tick = ticks[tick]; - // Replace label - sorting animation - if (tick.movedLabel) { - tick.replaceMovedLabel(); - } - if (tick && - tick.label && - tick.label.textPxLength > maxLabelLength) { - maxLabelLength = tick.label.textPxLength; - } - }); - this.maxLabelLength = maxLabelLength; - // Handle auto rotation on horizontal axis - if (this.autoRotation) { - // Apply rotation only if the label is too wide for the slot, and - // the label is wider than its height. - if (maxLabelLength > innerWidth && - maxLabelLength > labelMetrics.h) { - attr.rotation = this.labelRotation; - } - else { - this.labelRotation = 0; - } - // Handle word-wrap or ellipsis on vertical axis - } - else if (slotWidth) { - // For word-wrap or ellipsis - commonWidth = innerWidth; - if (!textOverflowOption) { - commonTextOverflow = 'clip'; - // On vertical axis, only allow word wrap if there is room - // for more lines. - i = tickPositions.length; - while (!horiz && i--) { - pos = tickPositions[i]; - label = ticks[pos].label; - if (label) { - // Reset ellipsis in order to get the correct - // bounding box (#4070) - if (label.styles && - label.styles.textOverflow === 'ellipsis') { - label.css({ textOverflow: 'clip' }); - // Set the correct width in order to read - // the bounding box height (#4678, #5034) - } - else if (label.textPxLength > slotWidth) { - label.css({ width: slotWidth + 'px' }); - } - if (label.getBBox().height > (this.len / tickPositions.length - - (labelMetrics.h - labelMetrics.f))) { - label.specificTextOverflow = 'ellipsis'; - } - } - } - } - } - // Add ellipsis if the label length is significantly longer than ideal - if (attr.rotation) { - commonWidth = (maxLabelLength > chart.chartHeight * 0.5 ? - chart.chartHeight * 0.33 : - maxLabelLength); - if (!textOverflowOption) { - commonTextOverflow = 'ellipsis'; - } - } - // Set the explicit or automatic label alignment - this.labelAlign = labelOptions.align || - this.autoLabelAlign(this.labelRotation); - if (this.labelAlign) { - attr.align = this.labelAlign; - } - // Apply general and specific CSS - tickPositions.forEach(function (pos) { - var tick = ticks[pos], - label = tick && tick.label, - widthOption = labelStyleOptions.width, - css = {}; - if (label) { - // This needs to go before the CSS in old IE (#4502) - label.attr(attr); - if (tick.shortenLabel) { - tick.shortenLabel(); - } - else if (commonWidth && - !widthOption && - // Setting width in this case messes with the bounding box - // (#7975) - labelStyleOptions.whiteSpace !== 'nowrap' && - ( - // Speed optimizing, #7656 - commonWidth < label.textPxLength || - // Resetting CSS, #4928 - label.element.tagName === 'SPAN')) { - css.width = commonWidth + 'px'; - if (!textOverflowOption) { - css.textOverflow = (label.specificTextOverflow || - commonTextOverflow); - } - label.css(css); - // Reset previously shortened label (#8210) - } - else if (label.styles && - label.styles.width && - !css.width && - !widthOption) { - label.css({ width: null }); - } - delete label.specificTextOverflow; - tick.rotation = attr.rotation; - } - }, this); - // Note: Why is this not part of getLabelPosition? - this.tickRotCorr = renderer.rotCorr(labelMetrics.b, this.labelRotation || 0, this.side !== 0); - }; - /** - * Return true if the axis has associated data. - * - * @function Highcharts.Axis#hasData - * - * @return {boolean} - * True if the axis has associated visible series and those series have - * either valid data points or explicit `min` and `max` settings. - */ - Axis.prototype.hasData = function () { - return this.series.some(function (s) { - return s.hasData(); - }) || - (this.options.showEmpty && - defined(this.min) && - defined(this.max)); - }; - /** - * Adds the title defined in axis.options.title. - * - * @function Highcharts.Axis#addTitle - * - * @param {boolean} [display] - * Whether or not to display the title. - */ - Axis.prototype.addTitle = function (display) { - var axis = this, - renderer = axis.chart.renderer, - horiz = axis.horiz, - opposite = axis.opposite, - options = axis.options, - axisTitleOptions = options.title, - textAlign, - styledMode = axis.chart.styledMode; - if (!axis.axisTitle) { - textAlign = axisTitleOptions.textAlign; - if (!textAlign) { - textAlign = (horiz ? { - low: 'left', - middle: 'center', - high: 'right' - } : { - low: opposite ? 'right' : 'left', - middle: 'center', - high: opposite ? 'left' : 'right' - })[axisTitleOptions.align]; - } - axis.axisTitle = renderer - .text(axisTitleOptions.text, 0, 0, axisTitleOptions.useHTML) - .attr({ - zIndex: 7, - rotation: axisTitleOptions.rotation || 0, - align: textAlign - }) - .addClass('highcharts-axis-title'); - // #7814, don't mutate style option - if (!styledMode) { - axis.axisTitle.css(merge(axisTitleOptions.style)); - } - axis.axisTitle.add(axis.axisGroup); - axis.axisTitle.isNew = true; - } - // Max width defaults to the length of the axis - if (!styledMode && - !axisTitleOptions.style.width && - !axis.isRadial) { - axis.axisTitle.css({ - width: axis.len + 'px' - }); - } - // hide or show the title depending on whether showEmpty is set - axis.axisTitle[display ? 'show' : 'hide'](display); - }; - /** - * Generates a tick for initial positioning. - * - * @private - * @function Highcharts.Axis#generateTick - * - * @param {number} pos - * The tick position in axis values. - * - * @param {number} [i] - * The index of the tick in {@link Axis.tickPositions}. - */ - Axis.prototype.generateTick = function (pos) { - var axis = this; - var ticks = axis.ticks; - if (!ticks[pos]) { - ticks[pos] = new Tick(axis, pos); - } - else { - ticks[pos].addLabel(); // update labels depending on tick interval - } - }; - /** - * Render the tick labels to a preliminary position to get their sizes - * - * @private - * @function Highcharts.Axis#getOffset - * - * @fires Highcharts.Axis#event:afterGetOffset - */ - Axis.prototype.getOffset = function () { - var axis = this, - chart = axis.chart, - renderer = chart.renderer, - options = axis.options, - tickPositions = axis.tickPositions, - ticks = axis.ticks, - horiz = axis.horiz, - side = axis.side, - invertedSide = chart.inverted && - !axis.isZAxis ? [1, 0, 3, 2][side] : side, - hasData, - showAxis, - titleOffset = 0, - titleOffsetOption, - titleMargin = 0, - axisTitleOptions = options.title, - labelOptions = options.labels, - labelOffset = 0, // reset - labelOffsetPadded, - axisOffset = chart.axisOffset, - clipOffset = chart.clipOffset, - clip, - directionFactor = [-1, 1, 1, -1][side], - className = options.className, - axisParent = axis.axisParent, // Used in color axis - lineHeightCorrection; - // For reuse in Axis.render - hasData = axis.hasData(); - axis.showAxis = showAxis = hasData || pick(options.showEmpty, true); - // Set/reset staggerLines - axis.staggerLines = axis.horiz && labelOptions.staggerLines; - // Create the axisGroup and gridGroup elements on first iteration - if (!axis.axisGroup) { - axis.gridGroup = renderer.g('grid') - .attr({ zIndex: options.gridZIndex || 1 }) - .addClass('highcharts-' + this.coll.toLowerCase() + '-grid ' + - (className || '')) - .add(axisParent); - axis.axisGroup = renderer.g('axis') - .attr({ zIndex: options.zIndex || 2 }) - .addClass('highcharts-' + this.coll.toLowerCase() + ' ' + - (className || '')) - .add(axisParent); - axis.labelGroup = renderer.g('axis-labels') - .attr({ zIndex: labelOptions.zIndex || 7 }) - .addClass('highcharts-' + axis.coll.toLowerCase() + '-labels ' + - (className || '')) - .add(axisParent); - } - if (hasData || axis.isLinked) { - // Generate ticks - tickPositions.forEach(function (pos, i) { - // i is not used here, but may be used in overrides - axis.generateTick(pos, i); - }); - axis.renderUnsquish(); - // Left side must be align: right and right side must - // have align: left for labels - axis.reserveSpaceDefault = (side === 0 || - side === 2 || - { 1: 'left', 3: 'right' }[side] === axis.labelAlign); - if (pick(labelOptions.reserveSpace, axis.labelAlign === 'center' ? true : null, axis.reserveSpaceDefault)) { - tickPositions.forEach(function (pos) { - // get the highest offset - labelOffset = Math.max(ticks[pos].getLabelSize(), labelOffset); - }); - } - if (axis.staggerLines) { - labelOffset *= axis.staggerLines; - } - axis.labelOffset = labelOffset * (axis.opposite ? -1 : 1); - } - else { // doesn't have data - objectEach(ticks, function (tick, n) { - tick.destroy(); - delete ticks[n]; - }); - } - if (axisTitleOptions && - axisTitleOptions.text && - axisTitleOptions.enabled !== false) { - axis.addTitle(showAxis); - if (showAxis && axisTitleOptions.reserveSpace !== false) { - axis.titleOffset = titleOffset = - axis.axisTitle.getBBox()[horiz ? 'height' : 'width']; - titleOffsetOption = axisTitleOptions.offset; - titleMargin = defined(titleOffsetOption) ? - 0 : - pick(axisTitleOptions.margin, horiz ? 5 : 10); - } - } - // Render the axis line - axis.renderLine(); - // handle automatic or user set offset - axis.offset = directionFactor * pick(options.offset, axisOffset[side] ? axisOffset[side] + (options.margin || 0) : 0); - axis.tickRotCorr = axis.tickRotCorr || { x: 0, y: 0 }; // polar - if (side === 0) { - lineHeightCorrection = -axis.labelMetrics().h; - } - else if (side === 2) { - lineHeightCorrection = axis.tickRotCorr.y; - } - else { - lineHeightCorrection = 0; - } - // Find the padded label offset - labelOffsetPadded = Math.abs(labelOffset) + titleMargin; - if (labelOffset) { - labelOffsetPadded -= lineHeightCorrection; - labelOffsetPadded += directionFactor * (horiz ? - pick(labelOptions.y, axis.tickRotCorr.y + directionFactor * 8) : - labelOptions.x); - } - axis.axisTitleMargin = pick(titleOffsetOption, labelOffsetPadded); - if (axis.getMaxLabelDimensions) { - axis.maxLabelDimensions = axis.getMaxLabelDimensions(ticks, tickPositions); - } - // Due to GridAxis.tickSize, tickSize should be calculated after ticks - // has rendered. - var tickSize = this.tickSize('tick'); - axisOffset[side] = Math.max(axisOffset[side], axis.axisTitleMargin + titleOffset + - directionFactor * axis.offset, labelOffsetPadded, // #3027 - tickPositions && tickPositions.length && tickSize ? - tickSize[0] + directionFactor * axis.offset : - 0 // #4866 - ); - // Decide the clipping needed to keep the graph inside - // the plot area and axis lines - clip = options.offset ? - 0 : - // #4308, #4371: - Math.floor(axis.axisLine.strokeWidth() / 2) * 2; - clipOffset[invertedSide] = - Math.max(clipOffset[invertedSide], clip); - fireEvent(this, 'afterGetOffset'); - }; - /** - * Internal function to get the path for the axis line. Extended for polar - * charts. - * - * @function Highcharts.Axis#getLinePath - * - * @param {number} lineWidth - * The line width in pixels. - * - * @return {Highcharts.SVGPathArray} - * The SVG path definition in array form. - */ - Axis.prototype.getLinePath = function (lineWidth) { - var chart = this.chart, - opposite = this.opposite, - offset = this.offset, - horiz = this.horiz, - lineLeft = this.left + (opposite ? this.width : 0) + offset, - lineTop = chart.chartHeight - this.bottom - - (opposite ? this.height : 0) + offset; - if (opposite) { - lineWidth *= -1; // crispify the other way - #1480, #1687 - } - return chart.renderer - .crispLine([ - [ - 'M', - horiz ? - this.left : - lineLeft, - horiz ? - lineTop : - this.top - ], - [ - 'L', - horiz ? - chart.chartWidth - this.right : - lineLeft, - horiz ? - lineTop : - chart.chartHeight - this.bottom - ] - ], lineWidth); - }; - /** - * Render the axis line. Called internally when rendering and redrawing the - * axis. - * - * @function Highcharts.Axis#renderLine - */ - Axis.prototype.renderLine = function () { - if (!this.axisLine) { - this.axisLine = this.chart.renderer.path() - .addClass('highcharts-axis-line') - .add(this.axisGroup); - if (!this.chart.styledMode) { - this.axisLine.attr({ - stroke: this.options.lineColor, - 'stroke-width': this.options.lineWidth, - zIndex: 7 - }); - } - } - }; - /** - * Position the axis title. - * - * @private - * @function Highcharts.Axis#getTitlePosition - * - * @return {Highcharts.PositionObject} - * X and Y positions for the title. - */ - Axis.prototype.getTitlePosition = function () { - // compute anchor points for each of the title align options - var horiz = this.horiz, - axisLeft = this.left, - axisTop = this.top, - axisLength = this.len, - axisTitleOptions = this.options.title, - margin = horiz ? axisLeft : axisTop, - opposite = this.opposite, - offset = this.offset, - xOption = axisTitleOptions.x || 0, - yOption = axisTitleOptions.y || 0, - axisTitle = this.axisTitle, - fontMetrics = this.chart.renderer.fontMetrics(axisTitleOptions.style && - axisTitleOptions.style.fontSize, - axisTitle), - // The part of a multiline text that is below the baseline of the - // first line. Subtract 1 to preserve pixel-perfectness from the - // old behaviour (v5.0.12), where only one line was allowed. - textHeightOvershoot = Math.max(axisTitle.getBBox(null, 0).height - fontMetrics.h - 1, 0), - // the position in the length direction of the axis - alongAxis = { - low: margin + (horiz ? 0 : axisLength), - middle: margin + axisLength / 2, - high: margin + (horiz ? axisLength : 0) - }[axisTitleOptions.align], - // the position in the perpendicular direction of the axis - offAxis = (horiz ? axisTop + this.height : axisLeft) + - (horiz ? 1 : -1) * // horizontal axis reverses the margin - (opposite ? -1 : 1) * // so does opposite axes - this.axisTitleMargin + - [ - -textHeightOvershoot, - textHeightOvershoot, - fontMetrics.f, - -textHeightOvershoot // left - ][this.side], - titlePosition = { - x: horiz ? - alongAxis + xOption : - offAxis + (opposite ? this.width : 0) + offset + xOption, - y: horiz ? - offAxis + yOption - (opposite ? this.height : 0) + offset : - alongAxis + yOption - }; - fireEvent(this, 'afterGetTitlePosition', { titlePosition: titlePosition }); - return titlePosition; - }; - /** - * Render a minor tick into the given position. If a minor tick already - * exists in this position, move it. - * - * @function Highcharts.Axis#renderMinorTick - * - * @param {number} pos - * The position in axis values. - */ - Axis.prototype.renderMinorTick = function (pos) { - var axis = this; - var slideInTicks = axis.chart.hasRendered && isNumber(axis.oldMin); - var minorTicks = axis.minorTicks; - if (!minorTicks[pos]) { - minorTicks[pos] = new Tick(axis, pos, 'minor'); - } - // Render new ticks in old position - if (slideInTicks && minorTicks[pos].isNew) { - minorTicks[pos].render(null, true); - } - minorTicks[pos].render(null, false, 1); - }; - /** - * Render a major tick into the given position. If a tick already exists - * in this position, move it. - * - * @function Highcharts.Axis#renderTick - * - * @param {number} pos - * The position in axis values. - * - * @param {number} i - * The tick index. - */ - Axis.prototype.renderTick = function (pos, i) { - var _a; - var axis = this; - var isLinked = axis.isLinked; - var ticks = axis.ticks; - var slideInTicks = axis.chart.hasRendered && isNumber(axis.oldMin); - // Linked axes need an extra check to find out if - if (!isLinked || - (pos >= axis.min && pos <= axis.max) || ((_a = axis.grid) === null || _a === void 0 ? void 0 : _a.isColumn)) { - if (!ticks[pos]) { - ticks[pos] = new Tick(axis, pos); - } - // NOTE this seems like overkill. Could be handled in tick.render by - // setting old position in attr, then set new position in animate. - // render new ticks in old position - if (slideInTicks && ticks[pos].isNew) { - // Start with negative opacity so that it is visible from - // halfway into the animation - ticks[pos].render(i, true, -1); - } - ticks[pos].render(i); - } - }; - /** - * Render the axis. - * - * @private - * @function Highcharts.Axis#render - * - * @fires Highcharts.Axis#event:afterRender - */ - Axis.prototype.render = function () { - var axis = this, - chart = axis.chart, - log = axis.logarithmic, - renderer = chart.renderer, - options = axis.options, - isLinked = axis.isLinked, - tickPositions = axis.tickPositions, - axisTitle = axis.axisTitle, - ticks = axis.ticks, - minorTicks = axis.minorTicks, - alternateBands = axis.alternateBands, - stackLabelOptions = options.stackLabels, - alternateGridColor = options.alternateGridColor, - tickmarkOffset = axis.tickmarkOffset, - axisLine = axis.axisLine, - showAxis = axis.showAxis, - animation = animObject(renderer.globalAnimation), - from, - to; - // Reset - axis.labelEdge.length = 0; - axis.overlap = false; - // Mark all elements inActive before we go over and mark the active ones - [ticks, minorTicks, alternateBands].forEach(function (coll) { - objectEach(coll, function (tick) { - tick.isActive = false; - }); - }); - // If the series has data draw the ticks. Else only the line and title - if (axis.hasData() || isLinked) { - // minor ticks - if (axis.minorTickInterval && !axis.categories) { - axis.getMinorTickPositions().forEach(function (pos) { - axis.renderMinorTick(pos); - }); - } - // Major ticks. Pull out the first item and render it last so that - // we can get the position of the neighbour label. #808. - if (tickPositions.length) { // #1300 - tickPositions.forEach(function (pos, i) { - axis.renderTick(pos, i); - }); - // In a categorized axis, the tick marks are displayed - // between labels. So we need to add a tick mark and - // grid line at the left edge of the X axis. - if (tickmarkOffset && (axis.min === 0 || axis.single)) { - if (!ticks[-1]) { - ticks[-1] = new Tick(axis, -1, null, true); - } - ticks[-1].render(-1); - } - } - // alternate grid color - if (alternateGridColor) { - tickPositions.forEach(function (pos, i) { - to = typeof tickPositions[i + 1] !== 'undefined' ? - tickPositions[i + 1] + tickmarkOffset : - axis.max - tickmarkOffset; - if (i % 2 === 0 && - pos < axis.max && - to <= axis.max + (chart.polar ? - -tickmarkOffset : - tickmarkOffset)) { // #2248, #4660 - if (!alternateBands[pos]) { - // Should be imported from PlotLineOrBand.js, but - // the dependency cycle with axis is a problem - alternateBands[pos] = new H.PlotLineOrBand(axis); - } - from = pos + tickmarkOffset; // #949 - alternateBands[pos].options = { - from: log ? log.lin2log(from) : from, - to: log ? log.lin2log(to) : to, - color: alternateGridColor, - className: 'highcharts-alternate-grid' - }; - alternateBands[pos].render(); - alternateBands[pos].isActive = true; - } - }); - } - // custom plot lines and bands - if (!axis._addedPlotLB) { // only first time - (options.plotLines || []) - .concat(options.plotBands || []) - .forEach(function (plotLineOptions) { - axis.addPlotBandOrLine(plotLineOptions); - }); - axis._addedPlotLB = true; - } - } // end if hasData - // Remove inactive ticks - [ticks, minorTicks, alternateBands].forEach(function (coll) { - var i, - forDestruction = [], - delay = animation.duration, - destroyInactiveItems = function () { - i = forDestruction.length; - while (i--) { - // When resizing rapidly, the same items - // may be destroyed in different timeouts, - // or the may be reactivated - if (coll[forDestruction[i]] && - !coll[forDestruction[i]].isActive) { - coll[forDestruction[i]].destroy(); - delete coll[forDestruction[i]]; - } - } - }; - objectEach(coll, function (tick, pos) { - if (!tick.isActive) { - // Render to zero opacity - tick.render(pos, false, 0); - tick.isActive = false; - forDestruction.push(pos); - } - }); - // When the objects are finished fading out, destroy them - syncTimeout(destroyInactiveItems, coll === alternateBands || - !chart.hasRendered || - !delay ? - 0 : - delay); - }); - // Set the axis line path - if (axisLine) { - axisLine[axisLine.isPlaced ? 'animate' : 'attr']({ - d: this.getLinePath(axisLine.strokeWidth()) - }); - axisLine.isPlaced = true; - // Show or hide the line depending on options.showEmpty - axisLine[showAxis ? 'show' : 'hide'](showAxis); - } - if (axisTitle && showAxis) { - var titleXy = axis.getTitlePosition(); - if (isNumber(titleXy.y)) { - axisTitle[axisTitle.isNew ? 'attr' : 'animate'](titleXy); - axisTitle.isNew = false; - } - else { - axisTitle.attr('y', -9999); - axisTitle.isNew = true; - } - } - // Stacked totals: - if (stackLabelOptions && stackLabelOptions.enabled && axis.stacking) { - axis.stacking.renderStackTotals(); - } - // End stacked totals - axis.isDirty = false; - fireEvent(this, 'afterRender'); - }; - /** - * Redraw the axis to reflect changes in the data or axis extremes. Called - * internally from Highcharts.Chart#redraw. - * - * @private - * @function Highcharts.Axis#redraw - */ - Axis.prototype.redraw = function () { - if (this.visible) { - // render the axis - this.render(); - // move plot lines and bands - this.plotLinesAndBands.forEach(function (plotLine) { - plotLine.render(); - }); - } - // mark associated series as dirty and ready for redraw - this.series.forEach(function (series) { - series.isDirty = true; - }); - }; - /** - * Returns an array of axis properties, that should be untouched during - * reinitialization. - * - * @private - * @function Highcharts.Axis#getKeepProps - * - * @return {Array<string>} - */ - Axis.prototype.getKeepProps = function () { - return (this.keepProps || Axis.keepProps); - }; - /** - * Destroys an Axis instance. See {@link Axis#remove} for the API endpoint - * to fully remove the axis. - * - * @private - * @function Highcharts.Axis#destroy - * - * @param {boolean} [keepEvents] - * Whether to preserve events, used internally in Axis.update. - */ - Axis.prototype.destroy = function (keepEvents) { - var axis = this, - plotLinesAndBands = axis.plotLinesAndBands, - plotGroup, - i; - fireEvent(this, 'destroy', { keepEvents: keepEvents }); - // Remove the events - if (!keepEvents) { - removeEvent(axis); - } - // Destroy collections - [axis.ticks, axis.minorTicks, axis.alternateBands].forEach(function (coll) { - destroyObjectProperties(coll); - }); - if (plotLinesAndBands) { - i = plotLinesAndBands.length; - while (i--) { // #1975 - plotLinesAndBands[i].destroy(); - } - } - // Destroy elements - ['axisLine', 'axisTitle', 'axisGroup', - 'gridGroup', 'labelGroup', 'cross', 'scrollbar'].forEach(function (prop) { - if (axis[prop]) { - axis[prop] = axis[prop].destroy(); - } - }); - // Destroy each generated group for plotlines and plotbands - for (plotGroup in axis.plotLinesAndBandsGroups) { // eslint-disable-line guard-for-in - axis.plotLinesAndBandsGroups[plotGroup] = - axis.plotLinesAndBandsGroups[plotGroup].destroy(); - } - // Delete all properties and fall back to the prototype. - objectEach(axis, function (val, key) { - if (axis.getKeepProps().indexOf(key) === -1) { - delete axis[key]; - } - }); - }; - /** - * Internal function to draw a crosshair. - * - * @function Highcharts.Axis#drawCrosshair - * - * @param {Highcharts.PointerEventObject} [e] - * The event arguments from the modified pointer event, extended with - * `chartX` and `chartY` - * - * @param {Highcharts.Point} [point] - * The Point object if the crosshair snaps to points. - * - * @fires Highcharts.Axis#event:afterDrawCrosshair - * @fires Highcharts.Axis#event:drawCrosshair - */ - Axis.prototype.drawCrosshair = function (e, point) { - var path, - options = this.crosshair, - snap = pick(options.snap, - true), - pos, - categorized, - graphic = this.cross, - crossOptions, - chart = this.chart; - fireEvent(this, 'drawCrosshair', { e: e, point: point }); - // Use last available event when updating non-snapped crosshairs without - // mouse interaction (#5287) - if (!e) { - e = this.cross && this.cross.e; - } - if ( - // Disabled in options - !this.crosshair || - // Snap - ((defined(point) || !snap) === false)) { - this.hideCrosshair(); - } - else { - // Get the path - if (!snap) { - pos = e && - (this.horiz ? - e.chartX - this.pos : - this.len - e.chartY + this.pos); - } - else if (defined(point)) { - // #3834 - pos = pick(this.coll !== 'colorAxis' ? - point.crosshairPos : // 3D axis extension - null, this.isXAxis ? - point.plotX : - this.len - point.plotY); - } - if (defined(pos)) { - crossOptions = { - // value, only used on radial - value: point && (this.isXAxis ? - point.x : - pick(point.stackY, point.y)), - translatedValue: pos - }; - if (chart.polar) { - // Additional information required for crosshairs in - // polar chart - extend(crossOptions, { - isCrosshair: true, - chartX: e && e.chartX, - chartY: e && e.chartY, - point: point - }); - } - path = this.getPlotLinePath(crossOptions) || - null; // #3189 - } - if (!defined(path)) { - this.hideCrosshair(); - return; - } - categorized = this.categories && !this.isRadial; - // Draw the cross - if (!graphic) { - this.cross = graphic = chart.renderer - .path() - .addClass('highcharts-crosshair highcharts-crosshair-' + - (categorized ? 'category ' : 'thin ') + - options.className) - .attr({ - zIndex: pick(options.zIndex, 2) - }) - .add(); - // Presentational attributes - if (!chart.styledMode) { - graphic.attr({ - stroke: options.color || - (categorized ? - Color - .parse('#ccd6eb') - .setOpacity(0.25) - .get() : - '#cccccc'), - 'stroke-width': pick(options.width, 1) - }).css({ - 'pointer-events': 'none' - }); - if (options.dashStyle) { - graphic.attr({ - dashstyle: options.dashStyle - }); - } - } - } - graphic.show().attr({ - d: path - }); - if (categorized && !options.width) { - graphic.attr({ - 'stroke-width': this.transA - }); - } - this.cross.e = e; - } - fireEvent(this, 'afterDrawCrosshair', { e: e, point: point }); - }; - /** - * Hide the crosshair if visible. - * - * @function Highcharts.Axis#hideCrosshair - */ - Axis.prototype.hideCrosshair = function () { - if (this.cross) { - this.cross.hide(); - } - fireEvent(this, 'afterHideCrosshair'); - }; - /** - * Check whether the chart has vertical panning ('y' or 'xy' type). - * - * @private - * @function Highcharts.Axis#hasVerticalPanning - * @return {boolean} - * - */ - Axis.prototype.hasVerticalPanning = function () { - var _a, - _b; - return /y/.test(((_b = (_a = this.chart.options.chart) === null || _a === void 0 ? void 0 : _a.panning) === null || _b === void 0 ? void 0 : _b.type) || ''); - }; - /** - * Check whether the given value is a positive valid axis value. - * - * @private - * @function Highcharts.Axis#validatePositiveValue - * - * @param {unknown} value - * The axis value - * @return {boolean} - * - */ - Axis.prototype.validatePositiveValue = function (value) { - return isNumber(value) && value > 0; - }; - /* * - * - * Static Properties - * - * */ - /** - * The X axis or category axis. Normally this is the horizontal axis, - * though if the chart is inverted this is the vertical axis. In case of - * multiple axes, the xAxis node is an array of configuration objects. - * - * See the [Axis class](/class-reference/Highcharts.Axis) for programmatic - * access to the axis. - * - * @productdesc {highmaps} - * In Highmaps, the axis is hidden, but it is used behind the scenes to - * control features like zooming and panning. Zooming is in effect the same - * as setting the extremes of one of the exes. - * - * @type {*|Array<*>} - * @optionparent xAxis - * - * @private - */ - Axis.defaultOptions = { - /** - * When using multiple axis, the ticks of two or more opposite axes - * will automatically be aligned by adding ticks to the axis or axes - * with the least ticks, as if `tickAmount` were specified. - * - * This can be prevented by setting `alignTicks` to false. If the grid - * lines look messy, it's a good idea to hide them for the secondary - * axis by setting `gridLineWidth` to 0. - * - * If `startOnTick` or `endOnTick` in an Axis options are set to false, - * then the `alignTicks ` will be disabled for the Axis. - * - * Disabled for logarithmic axes. - * - * @type {boolean} - * @default true - * @product highcharts highstock gantt - * @apioption xAxis.alignTicks - */ - /** - * Whether to allow decimals in this axis' ticks. When counting - * integers, like persons or hits on a web page, decimals should - * be avoided in the labels. - * - * @see [minTickInterval](#xAxis.minTickInterval) - * - * @sample {highcharts|highstock} highcharts/yaxis/allowdecimals-true/ - * True by default - * @sample {highcharts|highstock} highcharts/yaxis/allowdecimals-false/ - * False - * - * @type {boolean} - * @default true - * @since 2.0 - * @apioption xAxis.allowDecimals - */ - /** - * When using an alternate grid color, a band is painted across the - * plot area between every other grid line. - * - * @sample {highcharts} highcharts/yaxis/alternategridcolor/ - * Alternate grid color on the Y axis - * @sample {highstock} stock/xaxis/alternategridcolor/ - * Alternate grid color on the Y axis - * - * @type {Highcharts.ColorType} - * @apioption xAxis.alternateGridColor - */ - /** - * An array defining breaks in the axis, the sections defined will be - * left out and all the points shifted closer to each other. - * - * @productdesc {highcharts} - * Requires that the broken-axis.js module is loaded. - * - * @sample {highcharts} highcharts/axisbreak/break-simple/ - * Simple break - * @sample {highcharts|highstock} highcharts/axisbreak/break-visualized/ - * Advanced with callback - * @sample {highstock} stock/demo/intraday-breaks/ - * Break on nights and weekends - * - * @type {Array<*>} - * @since 4.1.0 - * @product highcharts highstock gantt - * @apioption xAxis.breaks - */ - /** - * A number indicating how much space should be left between the start - * and the end of the break. The break size is given in axis units, - * so for instance on a `datetime` axis, a break size of 3600000 would - * indicate the equivalent of an hour. - * - * @type {number} - * @default 0 - * @since 4.1.0 - * @product highcharts highstock gantt - * @apioption xAxis.breaks.breakSize - */ - /** - * The point where the break starts. - * - * @type {number} - * @since 4.1.0 - * @product highcharts highstock gantt - * @apioption xAxis.breaks.from - */ - /** - * Defines an interval after which the break appears again. By default - * the breaks do not repeat. - * - * @type {number} - * @default 0 - * @since 4.1.0 - * @product highcharts highstock gantt - * @apioption xAxis.breaks.repeat - */ - /** - * The point where the break ends. - * - * @type {number} - * @since 4.1.0 - * @product highcharts highstock gantt - * @apioption xAxis.breaks.to - */ - /** - * If categories are present for the xAxis, names are used instead of - * numbers for that axis. - * - * Since Highcharts 3.0, categories can also - * be extracted by giving each point a [name](#series.data) and setting - * axis [type](#xAxis.type) to `category`. However, if you have multiple - * series, best practice remains defining the `categories` array. - * - * Example: `categories: ['Apples', 'Bananas', 'Oranges']` - * - * @sample {highcharts} highcharts/demo/line-labels/ - * With - * @sample {highcharts} highcharts/xaxis/categories/ - * Without - * - * @type {Array<string>} - * @product highcharts gantt - * @apioption xAxis.categories - */ - /** - * The highest allowed value for automatically computed axis extremes. - * - * @see [floor](#xAxis.floor) - * - * @sample {highcharts|highstock} highcharts/yaxis/floor-ceiling/ - * Floor and ceiling - * - * @type {number} - * @since 4.0 - * @product highcharts highstock gantt - * @apioption xAxis.ceiling - */ - /** - * A class name that opens for styling the axis by CSS, especially in - * Highcharts styled mode. The class name is applied to group elements - * for the grid, axis elements and labels. - * - * @sample {highcharts|highstock|highmaps} highcharts/css/axis/ - * Multiple axes with separate styling - * - * @type {string} - * @since 5.0.0 - * @apioption xAxis.className - */ - /** - * Configure a crosshair that follows either the mouse pointer or the - * hovered point. - * - * In styled mode, the crosshairs are styled in the - * `.highcharts-crosshair`, `.highcharts-crosshair-thin` or - * `.highcharts-xaxis-category` classes. - * - * @productdesc {highstock} - * In Highstock, by default, the crosshair is enabled on the X axis and - * disabled on the Y axis. - * - * @sample {highcharts} highcharts/xaxis/crosshair-both/ - * Crosshair on both axes - * @sample {highstock} stock/xaxis/crosshairs-xy/ - * Crosshair on both axes - * @sample {highmaps} highcharts/xaxis/crosshair-both/ - * Crosshair on both axes - * - * @declare Highcharts.AxisCrosshairOptions - * @type {boolean|*} - * @default false - * @since 4.1 - * @apioption xAxis.crosshair - */ - /** - * A class name for the crosshair, especially as a hook for styling. - * - * @type {string} - * @since 5.0.0 - * @apioption xAxis.crosshair.className - */ - /** - * The color of the crosshair. Defaults to `#cccccc` for numeric and - * datetime axes, and `rgba(204,214,235,0.25)` for category axes, where - * the crosshair by default highlights the whole category. - * - * @sample {highcharts|highstock|highmaps} highcharts/xaxis/crosshair-customized/ - * Customized crosshairs - * - * @type {Highcharts.ColorType} - * @default #cccccc - * @since 4.1 - * @apioption xAxis.crosshair.color - */ - /** - * The dash style for the crosshair. See - * [plotOptions.series.dashStyle](#plotOptions.series.dashStyle) - * for possible values. - * - * @sample {highcharts|highmaps} highcharts/xaxis/crosshair-dotted/ - * Dotted crosshair - * @sample {highstock} stock/xaxis/crosshair-dashed/ - * Dashed X axis crosshair - * - * @type {Highcharts.DashStyleValue} - * @default Solid - * @since 4.1 - * @apioption xAxis.crosshair.dashStyle - */ - /** - * A label on the axis next to the crosshair. - * - * In styled mode, the label is styled with the - * `.highcharts-crosshair-label` class. - * - * @sample {highstock} stock/xaxis/crosshair-label/ - * Crosshair labels - * @sample {highstock} highcharts/css/crosshair-label/ - * Style mode - * - * @declare Highcharts.AxisCrosshairLabelOptions - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label - */ - /** - * Alignment of the label compared to the axis. Defaults to `"left"` for - * right-side axes, `"right"` for left-side axes and `"center"` for - * horizontal axes. - * - * @type {Highcharts.AlignValue} - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.align - */ - /** - * The background color for the label. Defaults to the related series - * color, or `#666666` if that is not available. - * - * @type {Highcharts.ColorType} - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.backgroundColor - */ - /** - * The border color for the crosshair label - * - * @type {Highcharts.ColorType} - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.borderColor - */ - /** - * The border corner radius of the crosshair label. - * - * @type {number} - * @default 3 - * @since 2.1.10 - * @product highstock - * @apioption xAxis.crosshair.label.borderRadius - */ - /** - * The border width for the crosshair label. - * - * @type {number} - * @default 0 - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.borderWidth - */ - /** - * Flag to enable crosshair's label. - * - * @sample {highstock} stock/xaxis/crosshairs-xy/ - * Enabled label for yAxis' crosshair - * - * @type {boolean} - * @default false - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.enabled - */ - /** - * A format string for the crosshair label. Defaults to `{value}` for - * numeric axes and `{value:%b %d, %Y}` for datetime axes. - * - * @type {string} - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.format - */ - /** - * Formatter function for the label text. - * - * @type {Highcharts.XAxisCrosshairLabelFormatterCallbackFunction} - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.formatter - */ - /** - * Padding inside the crosshair label. - * - * @type {number} - * @default 8 - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.padding - */ - /** - * The shape to use for the label box. - * - * @type {string} - * @default callout - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.shape - */ - /** - * Text styles for the crosshair label. - * - * @type {Highcharts.CSSObject} - * @default {"color": "white", "fontWeight": "normal", "fontSize": "11px", "textAlign": "center"} - * @since 2.1 - * @product highstock - * @apioption xAxis.crosshair.label.style - */ - /** - * Whether the crosshair should snap to the point or follow the pointer - * independent of points. - * - * @sample {highcharts|highstock} highcharts/xaxis/crosshair-snap-false/ - * True by default - * @sample {highmaps} maps/demo/latlon-advanced/ - * Snap is false - * - * @type {boolean} - * @default true - * @since 4.1 - * @apioption xAxis.crosshair.snap - */ - /** - * The pixel width of the crosshair. Defaults to 1 for numeric or - * datetime axes, and for one category width for category axes. - * - * @sample {highcharts} highcharts/xaxis/crosshair-customized/ - * Customized crosshairs - * @sample {highstock} highcharts/xaxis/crosshair-customized/ - * Customized crosshairs - * @sample {highmaps} highcharts/xaxis/crosshair-customized/ - * Customized crosshairs - * - * @type {number} - * @default 1 - * @since 4.1 - * @apioption xAxis.crosshair.width - */ - /** - * The Z index of the crosshair. Higher Z indices allow drawing the - * crosshair on top of the series or behind the grid lines. - * - * @type {number} - * @default 2 - * @since 4.1 - * @apioption xAxis.crosshair.zIndex - */ - /** - * Whether to zoom axis. If `chart.zoomType` is set, the option allows - * to disable zooming on an individual axis. - * - * @sample {highcharts} highcharts/xaxis/zoomenabled/ - * Zoom enabled is false - * - * - * @type {boolean} - * @default enabled - * @apioption xAxis.zoomEnabled - */ - /** - * For a datetime axis, the scale will automatically adjust to the - * appropriate unit. This member gives the default string - * representations used for each unit. For intermediate values, - * different units may be used, for example the `day` unit can be used - * on midnight and `hour` unit be used for intermediate values on the - * same axis. - * - * For an overview of the replacement codes, see - * [dateFormat](/class-reference/Highcharts#dateFormat). - * - * Defaults to: - * ```js - * { - * millisecond: '%H:%M:%S.%L', - * second: '%H:%M:%S', - * minute: '%H:%M', - * hour: '%H:%M', - * day: '%e. %b', - * week: '%e. %b', - * month: '%b \'%y', - * year: '%Y' - * } - * ``` - * - * @sample {highcharts} highcharts/xaxis/datetimelabelformats/ - * Different day format on X axis - * @sample {highstock} stock/xaxis/datetimelabelformats/ - * More information in x axis labels - * - * @declare Highcharts.AxisDateTimeLabelFormatsOptions - * @product highcharts highstock gantt - */ - dateTimeLabelFormats: { - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - millisecond: { - main: '%H:%M:%S.%L', - range: false - }, - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - second: { - main: '%H:%M:%S', - range: false - }, - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - minute: { - main: '%H:%M', - range: false - }, - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - hour: { - main: '%H:%M', - range: false - }, - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - day: { - main: '%e. %b' - }, - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - week: { - main: '%e. %b' - }, - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - month: { - main: '%b \'%y' - }, - /** - * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject - * @type {string|*} - */ - year: { - main: '%Y' - } - }, - /** - * Whether to force the axis to end on a tick. Use this option with - * the `maxPadding` option to control the axis end. - * - * @productdesc {highstock} - * In Highstock, `endOnTick` is always `false` when the navigator - * is enabled, to prevent jumpy scrolling. - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * True by default - * @sample {highcharts} highcharts/yaxis/endontick/ - * False - * @sample {highstock} stock/demo/basic-line/ - * True by default - * @sample {highstock} stock/xaxis/endontick/ - * False - * - * @since 1.2.0 - */ - endOnTick: false, - /** - * Event handlers for the axis. - * - * @type {*} - * @apioption xAxis.events - */ - /** - * An event fired after the breaks have rendered. - * - * @see [breaks](#xAxis.breaks) - * - * @sample {highcharts} highcharts/axisbreak/break-event/ - * AfterBreak Event - * - * @type {Highcharts.AxisEventCallbackFunction} - * @since 4.1.0 - * @product highcharts gantt - * @apioption xAxis.events.afterBreaks - */ - /** - * As opposed to the `setExtremes` event, this event fires after the - * final min and max values are computed and corrected for `minRange`. - * - * Fires when the minimum and maximum is set for the axis, either by - * calling the `.setExtremes()` method or by selecting an area in the - * chart. One parameter, `event`, is passed to the function, containing - * common event information. - * - * The new user set minimum and maximum values can be found by - * `event.min` and `event.max`. These reflect the axis minimum and - * maximum in axis values. The actual data extremes are found in - * `event.dataMin` and `event.dataMax`. - * - * @type {Highcharts.AxisSetExtremesEventCallbackFunction} - * @since 2.3 - * @context Highcharts.Axis - * @apioption xAxis.events.afterSetExtremes - */ - /** - * An event fired when a break from this axis occurs on a point. - * - * @see [breaks](#xAxis.breaks) - * - * @sample {highcharts} highcharts/axisbreak/break-visualized/ - * Visualization of a Break - * - * @type {Highcharts.AxisPointBreakEventCallbackFunction} - * @since 4.1.0 - * @product highcharts gantt - * @context Highcharts.Axis - * @apioption xAxis.events.pointBreak - */ - /** - * An event fired when a point falls inside a break from this axis. - * - * @type {Highcharts.AxisPointBreakEventCallbackFunction} - * @product highcharts highstock gantt - * @context Highcharts.Axis - * @apioption xAxis.events.pointInBreak - */ - /** - * Fires when the minimum and maximum is set for the axis, either by - * calling the `.setExtremes()` method or by selecting an area in the - * chart. One parameter, `event`, is passed to the function, - * containing common event information. - * - * The new user set minimum and maximum values can be found by - * `event.min` and `event.max`. These reflect the axis minimum and - * maximum in data values. When an axis is zoomed all the way out from - * the "Reset zoom" button, `event.min` and `event.max` are null, and - * the new extremes are set based on `this.dataMin` and `this.dataMax`. - * - * @sample {highstock} stock/xaxis/events-setextremes/ - * Log new extremes on x axis - * - * @type {Highcharts.AxisSetExtremesEventCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Axis - * @apioption xAxis.events.setExtremes - */ - /** - * The lowest allowed value for automatically computed axis extremes. - * - * @see [ceiling](#yAxis.ceiling) - * - * @sample {highcharts} highcharts/yaxis/floor-ceiling/ - * Floor and ceiling - * @sample {highstock} stock/demo/lazy-loading/ - * Prevent negative stock price on Y axis - * - * @type {number} - * @since 4.0 - * @product highcharts highstock gantt - * @apioption xAxis.floor - */ - /** - * The dash or dot style of the grid lines. For possible values, see - * [this demonstration](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/). - * - * @sample {highcharts} highcharts/yaxis/gridlinedashstyle/ - * Long dashes - * @sample {highstock} stock/xaxis/gridlinedashstyle/ - * Long dashes - * - * @type {Highcharts.DashStyleValue} - * @default Solid - * @since 1.2 - * @apioption xAxis.gridLineDashStyle - */ - /** - * The Z index of the grid lines. - * - * @sample {highcharts|highstock} highcharts/xaxis/gridzindex/ - * A Z index of 4 renders the grid above the graph - * - * @type {number} - * @default 1 - * @product highcharts highstock gantt - * @apioption xAxis.gridZIndex - */ - /** - * An id for the axis. This can be used after render time to get - * a pointer to the axis object through `chart.get()`. - * - * @sample {highcharts} highcharts/xaxis/id/ - * Get the object - * @sample {highstock} stock/xaxis/id/ - * Get the object - * - * @type {string} - * @since 1.2.0 - * @apioption xAxis.id - */ - /** - * The axis labels show the number or category for each tick. - * - * Since v8.0.0: Labels are animated in categorized x-axis with - * updating data if `tickInterval` and `step` is set to 1. - * - * @productdesc {highmaps} - * X and Y axis labels are by default disabled in Highmaps, but the - * functionality is inherited from Highcharts and used on `colorAxis`, - * and can be enabled on X and Y axes too. - */ - labels: { - /** - * What part of the string the given position is anchored to. - * If `left`, the left side of the string is at the axis position. - * Can be one of `"left"`, `"center"` or `"right"`. Defaults to - * an intelligent guess based on which side of the chart the axis - * is on and the rotation of the label. - * - * @see [reserveSpace](#xAxis.labels.reserveSpace) - * - * @sample {highcharts} highcharts/xaxis/labels-align-left/ - * Left - * @sample {highcharts} highcharts/xaxis/labels-align-right/ - * Right - * @sample {highcharts} highcharts/xaxis/labels-reservespace-true/ - * Left-aligned labels on a vertical category axis - * - * @type {Highcharts.AlignValue} - * @apioption xAxis.labels.align - */ - /** - * For horizontal axes, the allowed degrees of label rotation - * to prevent overlapping labels. If there is enough space, - * labels are not rotated. As the chart gets narrower, it - * will start rotating the labels -45 degrees, then remove - * every second label and try again with rotations 0 and -45 etc. - * Set it to `false` to disable rotation, which will - * cause the labels to word-wrap if possible. - * - * @sample {highcharts|highstock} highcharts/xaxis/labels-autorotation-default/ - * Default auto rotation of 0 or -45 - * @sample {highcharts|highstock} highcharts/xaxis/labels-autorotation-0-90/ - * Custom graded auto rotation - * - * @type {Array<number>|false} - * @default [-45] - * @since 4.1.0 - * @product highcharts highstock gantt - * @apioption xAxis.labels.autoRotation - */ - /** - * When each category width is more than this many pixels, we don't - * apply auto rotation. Instead, we lay out the axis label with word - * wrap. A lower limit makes sense when the label contains multiple - * short words that don't extend the available horizontal space for - * each label. - * - * @sample {highcharts} highcharts/xaxis/labels-autorotationlimit/ - * Lower limit - * - * @type {number} - * @default 80 - * @since 4.1.5 - * @product highcharts gantt - * @apioption xAxis.labels.autoRotationLimit - */ - /** - * Polar charts only. The label's pixel distance from the perimeter - * of the plot area. - * - * @type {number} - * @default 15 - * @product highcharts gantt - * @apioption xAxis.labels.distance - */ - /** - * Enable or disable the axis labels. - * - * @sample {highcharts} highcharts/xaxis/labels-enabled/ - * X axis labels disabled - * @sample {highstock} stock/xaxis/labels-enabled/ - * X axis labels disabled - * - * @default {highcharts|highstock|gantt} true - * @default {highmaps} false - */ - enabled: true, - /** - * A [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) - * for the axis label. - * - * @sample {highcharts|highstock} highcharts/yaxis/labels-format/ - * Add units to Y axis label - * - * @type {string} - * @default {value} - * @since 3.0 - * @apioption xAxis.labels.format - */ - /** - * Callback JavaScript function to format the label. The value - * is given by `this.value`. Additional properties for `this` are - * `axis`, `chart`, `isFirst` and `isLast`. The value of the default - * label formatter can be retrieved by calling - * `this.axis.defaultLabelFormatter.call(this)` within the function. - * - * Defaults to: - * ```js - * function() { - * return this.value; - * } - * ``` - * - * @sample {highcharts} highcharts/xaxis/labels-formatter-linked/ - * Linked category names - * @sample {highcharts} highcharts/xaxis/labels-formatter-extended/ - * Modified numeric labels - * @sample {highstock} stock/xaxis/labels-formatter/ - * Added units on Y axis - * - * @type {Highcharts.AxisLabelsFormatterCallbackFunction} - * @apioption xAxis.labels.formatter - */ - /** - * The number of pixels to indent the labels per level in a treegrid - * axis. - * - * @sample gantt/treegrid-axis/demo - * Indentation 10px by default. - * @sample gantt/treegrid-axis/indentation-0px - * Indentation set to 0px. - * - * @product gantt - */ - indentation: 10, - /** - * Horizontal axis only. When `staggerLines` is not set, - * `maxStaggerLines` defines how many lines the axis is allowed to - * add to automatically avoid overlapping X labels. Set to `1` to - * disable overlap detection. - * - * @deprecated - * @type {number} - * @default 5 - * @since 1.3.3 - * @apioption xAxis.labels.maxStaggerLines - */ - /** - * How to handle overflowing labels on horizontal axis. If set to - * `"allow"`, it will not be aligned at all. By default it - * `"justify"` labels inside the chart area. If there is room to - * move it, it will be aligned to the edge, else it will be removed. - * - * @type {string} - * @default justify - * @since 2.2.5 - * @validvalue ["allow", "justify"] - * @apioption xAxis.labels.overflow - */ - /** - * The pixel padding for axis labels, to ensure white space between - * them. - * - * @type {number} - * @default 5 - * @product highcharts gantt - * @apioption xAxis.labels.padding - */ - /** - * Whether to reserve space for the labels. By default, space is - * reserved for the labels in these cases: - * - * * On all horizontal axes. - * * On vertical axes if `label.align` is `right` on a left-side - * axis or `left` on a right-side axis. - * * On vertical axes if `label.align` is `center`. - * - * This can be turned off when for example the labels are rendered - * inside the plot area instead of outside. - * - * @see [labels.align](#xAxis.labels.align) - * - * @sample {highcharts} highcharts/xaxis/labels-reservespace/ - * No reserved space, labels inside plot - * @sample {highcharts} highcharts/xaxis/labels-reservespace-true/ - * Left-aligned labels on a vertical category axis - * - * @type {boolean} - * @since 4.1.10 - * @product highcharts gantt - * @apioption xAxis.labels.reserveSpace - */ - /** - * Rotation of the labels in degrees. - * - * @sample {highcharts} highcharts/xaxis/labels-rotation/ - * X axis labels rotated 90° - * - * @type {number} - * @default 0 - * @apioption xAxis.labels.rotation - */ - /** - * Horizontal axes only. The number of lines to spread the labels - * over to make room or tighter labels. - * - * @sample {highcharts} highcharts/xaxis/labels-staggerlines/ - * Show labels over two lines - * @sample {highstock} stock/xaxis/labels-staggerlines/ - * Show labels over two lines - * - * @type {number} - * @since 2.1 - * @apioption xAxis.labels.staggerLines - */ - /** - * To show only every _n_'th label on the axis, set the step to _n_. - * Setting the step to 2 shows every other label. - * - * By default, the step is calculated automatically to avoid - * overlap. To prevent this, set it to 1\. This usually only - * happens on a category axis, and is often a sign that you have - * chosen the wrong axis type. - * - * Read more at - * [Axis docs](https://www.highcharts.com/docs/chart-concepts/axes) - * => What axis should I use? - * - * @sample {highcharts} highcharts/xaxis/labels-step/ - * Showing only every other axis label on a categorized - * x-axis - * @sample {highcharts} highcharts/xaxis/labels-step-auto/ - * Auto steps on a category axis - * - * @type {number} - * @since 2.1 - * @apioption xAxis.labels.step - */ - /** - * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the labels. - * - * @type {boolean} - * @default false - * @apioption xAxis.labels.useHTML - */ - /** - * The x position offset of all labels relative to the tick - * positions on the axis. - * - * @sample {highcharts} highcharts/xaxis/labels-x/ - * Y axis labels placed on grid lines - */ - x: 0, - /** - * The y position offset of all labels relative to the tick - * positions on the axis. The default makes it adapt to the font - * size of the bottom axis. - * - * @sample {highcharts} highcharts/xaxis/labels-x/ - * Y axis labels placed on grid lines - * - * @type {number} - * @apioption xAxis.labels.y - */ - /** - * The Z index for the axis labels. - * - * @type {number} - * @default 7 - * @apioption xAxis.labels.zIndex - */ - /** - * CSS styles for the label. Use `whiteSpace: 'nowrap'` to prevent - * wrapping of category labels. Use `textOverflow: 'none'` to - * prevent ellipsis (dots). - * - * In styled mode, the labels are styled with the - * `.highcharts-axis-labels` class. - * - * @sample {highcharts} highcharts/xaxis/labels-style/ - * Red X axis labels - * - * @type {Highcharts.CSSObject} - */ - style: { - /** @internal */ - color: '#666666', - /** @internal */ - cursor: 'default', - /** @internal */ - fontSize: '11px' - } - }, - /** - * The left position as the horizontal axis. If it's a number, it is - * interpreted as pixel position relative to the chart. - * - * Since Highcharts v5.0.13: If it's a percentage string, it is - * interpreted as percentages of the plot width, offset from plot area - * left. - * - * @type {number|string} - * @product highcharts highstock - * @apioption xAxis.left - */ - /** - * The top position as the vertical axis. If it's a number, it is - * interpreted as pixel position relative to the chart. - * - * Since Highcharts 2: If it's a percentage string, it is interpreted - * as percentages of the plot height, offset from plot area top. - * - * @type {number|string} - * @product highcharts highstock - * @apioption xAxis.top - */ - /** - * Index of another axis that this axis is linked to. When an axis is - * linked to a master axis, it will take the same extremes as - * the master, but as assigned by min or max or by setExtremes. - * It can be used to show additional info, or to ease reading the - * chart by duplicating the scales. - * - * @sample {highcharts} highcharts/xaxis/linkedto/ - * Different string formats of the same date - * @sample {highcharts} highcharts/yaxis/linkedto/ - * Y values on both sides - * - * @type {number} - * @since 2.0.2 - * @product highcharts highstock gantt - * @apioption xAxis.linkedTo - */ - /** - * The maximum value of the axis. If `null`, the max value is - * automatically calculated. - * - * If the [endOnTick](#yAxis.endOnTick) option is true, the `max` value - * might be rounded up. - * - * If a [tickAmount](#yAxis.tickAmount) is set, the axis may be extended - * beyond the set max in order to reach the given number of ticks. The - * same may happen in a chart with multiple axes, determined by [chart. - * alignTicks](#chart), where a `tickAmount` is applied internally. - * - * @sample {highcharts} highcharts/yaxis/max-200/ - * Y axis max of 200 - * @sample {highcharts} highcharts/yaxis/max-logarithmic/ - * Y axis max on logarithmic axis - * @sample {highstock} stock/xaxis/min-max/ - * Fixed min and max on X axis - * @sample {highmaps} maps/axis/min-max/ - * Pre-zoomed to a specific area - * - * @type {number|null} - * @apioption xAxis.max - */ - /** - * Padding of the max value relative to the length of the axis. A - * padding of 0.05 will make a 100px axis 5px longer. This is useful - * when you don't want the highest data value to appear on the edge - * of the plot area. When the axis' `max` option is set or a max extreme - * is set using `axis.setExtremes()`, the maxPadding will be ignored. - * - * @sample {highcharts} highcharts/yaxis/maxpadding/ - * Max padding of 0.25 on y axis - * @sample {highstock} stock/xaxis/minpadding-maxpadding/ - * Greater min- and maxPadding - * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/ - * Add some padding - * - * @default {highcharts} 0.01 - * @default {highstock|highmaps} 0 - * @since 1.2.0 - */ - maxPadding: 0.01, - /** - * Deprecated. Use `minRange` instead. - * - * @deprecated - * @type {number} - * @product highcharts highstock - * @apioption xAxis.maxZoom - */ - /** - * The minimum value of the axis. If `null` the min value is - * automatically calculated. - * - * If the [startOnTick](#yAxis.startOnTick) option is true (default), - * the `min` value might be rounded down. - * - * The automatically calculated minimum value is also affected by - * [floor](#yAxis.floor), [softMin](#yAxis.softMin), - * [minPadding](#yAxis.minPadding), [minRange](#yAxis.minRange) - * as well as [series.threshold](#plotOptions.series.threshold) - * and [series.softThreshold](#plotOptions.series.softThreshold). - * - * @sample {highcharts} highcharts/yaxis/min-startontick-false/ - * -50 with startOnTick to false - * @sample {highcharts} highcharts/yaxis/min-startontick-true/ - * -50 with startOnTick true by default - * @sample {highstock} stock/xaxis/min-max/ - * Set min and max on X axis - * @sample {highmaps} maps/axis/min-max/ - * Pre-zoomed to a specific area - * - * @type {number|null} - * @apioption xAxis.min - */ - /** - * The dash or dot style of the minor grid lines. For possible values, - * see [this demonstration](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/). - * - * @sample {highcharts} highcharts/yaxis/minorgridlinedashstyle/ - * Long dashes on minor grid lines - * @sample {highstock} stock/xaxis/minorgridlinedashstyle/ - * Long dashes on minor grid lines - * - * @type {Highcharts.DashStyleValue} - * @default Solid - * @since 1.2 - * @apioption xAxis.minorGridLineDashStyle - */ - /** - * Specific tick interval in axis units for the minor ticks. On a linear - * axis, if `"auto"`, the minor tick interval is calculated as a fifth - * of the tickInterval. If `null` or `undefined`, minor ticks are not - * shown. - * - * On logarithmic axes, the unit is the power of the value. For example, - * setting the minorTickInterval to 1 puts one tick on each of 0.1, 1, - * 10, 100 etc. Setting the minorTickInterval to 0.1 produces 9 ticks - * between 1 and 10, 10 and 100 etc. - * - * If user settings dictate minor ticks to become too dense, they don't - * make sense, and will be ignored to prevent performance problems. - * - * @sample {highcharts} highcharts/yaxis/minortickinterval-null/ - * Null by default - * @sample {highcharts} highcharts/yaxis/minortickinterval-5/ - * 5 units - * @sample {highcharts} highcharts/yaxis/minortickinterval-log-auto/ - * "auto" - * @sample {highcharts} highcharts/yaxis/minortickinterval-log/ - * 0.1 - * @sample {highstock} stock/demo/basic-line/ - * Null by default - * @sample {highstock} stock/xaxis/minortickinterval-auto/ - * "auto" - * - * @type {number|string|null} - * @apioption xAxis.minorTickInterval - */ - /** - * The pixel length of the minor tick marks. - * - * @sample {highcharts} highcharts/yaxis/minorticklength/ - * 10px on Y axis - * @sample {highstock} stock/xaxis/minorticks/ - * 10px on Y axis - */ - minorTickLength: 2, - /** - * The position of the minor tick marks relative to the axis line. - * Can be one of `inside` and `outside`. - * - * @sample {highcharts} highcharts/yaxis/minortickposition-outside/ - * Outside by default - * @sample {highcharts} highcharts/yaxis/minortickposition-inside/ - * Inside - * @sample {highstock} stock/xaxis/minorticks/ - * Inside - * - * @validvalue ["inside", "outside"] - */ - minorTickPosition: 'outside', - /** - * Enable or disable minor ticks. Unless - * [minorTickInterval](#xAxis.minorTickInterval) is set, the tick - * interval is calculated as a fifth of the `tickInterval`. - * - * On a logarithmic axis, minor ticks are laid out based on a best - * guess, attempting to enter approximately 5 minor ticks between - * each major tick. - * - * Prior to v6.0.0, ticks were unabled in auto layout by setting - * `minorTickInterval` to `"auto"`. - * - * @productdesc {highcharts} - * On axes using [categories](#xAxis.categories), minor ticks are not - * supported. - * - * @sample {highcharts} highcharts/yaxis/minorticks-true/ - * Enabled on linear Y axis - * - * @type {boolean} - * @default false - * @since 6.0.0 - * @apioption xAxis.minorTicks - */ - /** - * The pixel width of the minor tick mark. - * - * @sample {highcharts} highcharts/yaxis/minortickwidth/ - * 3px width - * @sample {highstock} stock/xaxis/minorticks/ - * 1px width - * - * @type {number} - * @default 0 - * @apioption xAxis.minorTickWidth - */ - /** - * Padding of the min value relative to the length of the axis. A - * padding of 0.05 will make a 100px axis 5px longer. This is useful - * when you don't want the lowest data value to appear on the edge - * of the plot area. When the axis' `min` option is set or a min extreme - * is set using `axis.setExtremes()`, the minPadding will be ignored. - * - * @sample {highcharts} highcharts/yaxis/minpadding/ - * Min padding of 0.2 - * @sample {highstock} stock/xaxis/minpadding-maxpadding/ - * Greater min- and maxPadding - * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/ - * Add some padding - * - * @default {highcharts} 0.01 - * @default {highstock|highmaps} 0 - * @since 1.2.0 - * @product highcharts highstock gantt - */ - minPadding: 0.01, - /** - * The minimum range to display on this axis. The entire axis will not - * be allowed to span over a smaller interval than this. For example, - * for a datetime axis the main unit is milliseconds. If minRange is - * set to 3600000, you can't zoom in more than to one hour. - * - * The default minRange for the x axis is five times the smallest - * interval between any of the data points. - * - * On a logarithmic axis, the unit for the minimum range is the power. - * So a minRange of 1 means that the axis can be zoomed to 10-100, - * 100-1000, 1000-10000 etc. - * - * **Note**: The `minPadding`, `maxPadding`, `startOnTick` and - * `endOnTick` settings also affect how the extremes of the axis - * are computed. - * - * @sample {highcharts} highcharts/xaxis/minrange/ - * Minimum range of 5 - * @sample {highstock} stock/xaxis/minrange/ - * Max zoom of 6 months overrides user selections - * @sample {highmaps} maps/axis/minrange/ - * Minimum range of 1000 - * - * @type {number} - * @apioption xAxis.minRange - */ - /** - * The minimum tick interval allowed in axis values. For example on - * zooming in on an axis with daily data, this can be used to prevent - * the axis from showing hours. Defaults to the closest distance between - * two points on the axis. - * - * @type {number} - * @since 2.3.0 - * @apioption xAxis.minTickInterval - */ - /** - * The distance in pixels from the plot area to the axis line. - * A positive offset moves the axis with it's line, labels and ticks - * away from the plot area. This is typically used when two or more - * axes are displayed on the same side of the plot. With multiple - * axes the offset is dynamically adjusted to avoid collision, this - * can be overridden by setting offset explicitly. - * - * @sample {highcharts} highcharts/yaxis/offset/ - * Y axis offset of 70 - * @sample {highcharts} highcharts/yaxis/offset-centered/ - * Axes positioned in the center of the plot - * @sample {highstock} stock/xaxis/offset/ - * Y axis offset by 70 px - * - * @type {number} - * @default 0 - * @apioption xAxis.offset - */ - /** - * Whether to display the axis on the opposite side of the normal. The - * normal is on the left side for vertical axes and bottom for - * horizontal, so the opposite sides will be right and top respectively. - * This is typically used with dual or multiple axes. - * - * @sample {highcharts} highcharts/yaxis/opposite/ - * Secondary Y axis opposite - * @sample {highstock} stock/xaxis/opposite/ - * Y axis on left side - * - * @type {boolean} - * @default {highcharts|highstock|highmaps} false - * @default {gantt} true - * @apioption xAxis.opposite - */ - /** - * In an ordinal axis, the points are equally spaced in the chart - * regardless of the actual time or x distance between them. This means - * that missing data periods (e.g. nights or weekends for a stock chart) - * will not take up space in the chart. - * Having `ordinal: false` will show any gaps created by the `gapSize` - * setting proportionate to their duration. - * - * In stock charts the X axis is ordinal by default, unless - * the boost module is used and at least one of the series' data length - * exceeds the [boostThreshold](#series.line.boostThreshold). - * - * @sample {highstock} stock/xaxis/ordinal-true/ - * True by default - * @sample {highstock} stock/xaxis/ordinal-false/ - * False - * - * @type {boolean} - * @default true - * @since 1.1 - * @product highstock - * @apioption xAxis.ordinal - */ - /** - * Additional range on the right side of the xAxis. Works similar to - * `xAxis.maxPadding`, but value is set in milliseconds. Can be set for - * both main `xAxis` and the navigator's `xAxis`. - * - * @sample {highstock} stock/xaxis/overscroll/ - * One minute overscroll with live data - * - * @type {number} - * @default 0 - * @since 6.0.0 - * @product highstock - * @apioption xAxis.overscroll - */ - /** - * Refers to the index in the [panes](#panes) array. Used for circular - * gauges and polar charts. When the option is not set then first pane - * will be used. - * - * @sample highcharts/demo/gauge-vu-meter - * Two gauges with different center - * - * @type {number} - * @product highcharts - * @apioption xAxis.pane - */ - /** - * The zoomed range to display when only defining one or none of `min` - * or `max`. For example, to show the latest month, a range of one month - * can be set. - * - * @sample {highstock} stock/xaxis/range/ - * Setting a zoomed range when the rangeSelector is disabled - * - * @type {number} - * @product highstock - * @apioption xAxis.range - */ - /** - * Whether to reverse the axis so that the highest number is closest - * to the origin. If the chart is inverted, the x axis is reversed by - * default. - * - * @sample {highcharts} highcharts/yaxis/reversed/ - * Reversed Y axis - * @sample {highstock} stock/xaxis/reversed/ - * Reversed Y axis - * - * @type {boolean} - * @default false - * @apioption xAxis.reversed - */ - // reversed: false, - /** - * This option determines how stacks should be ordered within a group. - * For example reversed xAxis also reverses stacks, so first series - * comes last in a group. To keep order like for non-reversed xAxis - * enable this option. - * - * @sample {highcharts} highcharts/xaxis/reversedstacks/ - * Reversed stacks comparison - * @sample {highstock} highcharts/xaxis/reversedstacks/ - * Reversed stacks comparison - * - * @type {boolean} - * @default false - * @since 6.1.1 - * @product highcharts highstock - * @apioption xAxis.reversedStacks - */ - /** - * An optional scrollbar to display on the X axis in response to - * limiting the minimum and maximum of the axis values. - * - * In styled mode, all the presentational options for the scrollbar are - * replaced by the classes `.highcharts-scrollbar-thumb`, - * `.highcharts-scrollbar-arrow`, `.highcharts-scrollbar-button`, - * `.highcharts-scrollbar-rifles` and `.highcharts-scrollbar-track`. - * - * @sample {highstock} stock/yaxis/heatmap-scrollbars/ - * Heatmap with both scrollbars - * - * @extends scrollbar - * @since 4.2.6 - * @product highstock - * @apioption xAxis.scrollbar - */ - /** - * Whether to show the axis line and title when the axis has no data. - * - * @sample {highcharts} highcharts/yaxis/showempty/ - * When clicking the legend to hide series, one axis preserves - * line and title, the other doesn't - * @sample {highstock} highcharts/yaxis/showempty/ - * When clicking the legend to hide series, one axis preserves - * line and title, the other doesn't - * - * @since 1.1 - */ - showEmpty: true, - /** - * Whether to show the first tick label. - * - * @sample {highcharts} highcharts/xaxis/showfirstlabel-false/ - * Set to false on X axis - * @sample {highstock} stock/xaxis/showfirstlabel/ - * Labels below plot lines on Y axis - * - * @type {boolean} - * @default true - * @apioption xAxis.showFirstLabel - */ - /** - * Whether to show the last tick label. Defaults to `true` on cartesian - * charts, and `false` on polar charts. - * - * @sample {highcharts} highcharts/xaxis/showlastlabel-true/ - * Set to true on X axis - * @sample {highstock} stock/xaxis/showfirstlabel/ - * Labels below plot lines on Y axis - * - * @type {boolean} - * @default true - * @product highcharts highstock gantt - * @apioption xAxis.showLastLabel - */ - /** - * A soft maximum for the axis. If the series data maximum is less than - * this, the axis will stay at this maximum, but if the series data - * maximum is higher, the axis will flex to show all data. - * - * @sample highcharts/yaxis/softmin-softmax/ - * Soft min and max - * - * @type {number} - * @since 5.0.1 - * @product highcharts highstock gantt - * @apioption xAxis.softMax - */ - /** - * A soft minimum for the axis. If the series data minimum is greater - * than this, the axis will stay at this minimum, but if the series - * data minimum is lower, the axis will flex to show all data. - * - * @sample highcharts/yaxis/softmin-softmax/ - * Soft min and max - * - * @type {number} - * @since 5.0.1 - * @product highcharts highstock gantt - * @apioption xAxis.softMin - */ - /** - * For datetime axes, this decides where to put the tick between weeks. - * 0 = Sunday, 1 = Monday. - * - * @sample {highcharts} highcharts/xaxis/startofweek-monday/ - * Monday by default - * @sample {highcharts} highcharts/xaxis/startofweek-sunday/ - * Sunday - * @sample {highstock} stock/xaxis/startofweek-1 - * Monday by default - * @sample {highstock} stock/xaxis/startofweek-0 - * Sunday - * - * @product highcharts highstock gantt - */ - startOfWeek: 1, - /** - * Whether to force the axis to start on a tick. Use this option with - * the `minPadding` option to control the axis start. - * - * @productdesc {highstock} - * In Highstock, `startOnTick` is always `false` when the navigator - * is enabled, to prevent jumpy scrolling. - * - * @sample {highcharts} highcharts/xaxis/startontick-false/ - * False by default - * @sample {highcharts} highcharts/xaxis/startontick-true/ - * True - * - * @since 1.2.0 - */ - startOnTick: false, - /** - * The amount of ticks to draw on the axis. This opens up for aligning - * the ticks of multiple charts or panes within a chart. This option - * overrides the `tickPixelInterval` option. - * - * This option only has an effect on linear axes. Datetime, logarithmic - * or category axes are not affected. - * - * @sample {highcharts} highcharts/yaxis/tickamount/ - * 8 ticks on Y axis - * @sample {highstock} highcharts/yaxis/tickamount/ - * 8 ticks on Y axis - * - * @type {number} - * @since 4.1.0 - * @product highcharts highstock gantt - * @apioption xAxis.tickAmount - */ - /** - * The interval of the tick marks in axis units. When `undefined`, the - * tick interval is computed to approximately follow the - * [tickPixelInterval](#xAxis.tickPixelInterval) on linear and datetime - * axes. On categorized axes, a `undefined` tickInterval will default to - * 1, one category. Note that datetime axes are based on milliseconds, - * so for example an interval of one day is expressed as - * `24 * 3600 * 1000`. - * - * On logarithmic axes, the tickInterval is based on powers, so a - * tickInterval of 1 means one tick on each of 0.1, 1, 10, 100 etc. A - * tickInterval of 2 means a tick of 0.1, 10, 1000 etc. A tickInterval - * of 0.2 puts a tick on 0.1, 0.2, 0.4, 0.6, 0.8, 1, 2, 4, 6, 8, 10, 20, - * 40 etc. - * - * - * If the tickInterval is too dense for labels to be drawn, Highcharts - * may remove ticks. - * - * If the chart has multiple axes, the [alignTicks](#chart.alignTicks) - * option may interfere with the `tickInterval` setting. - * - * @see [tickPixelInterval](#xAxis.tickPixelInterval) - * @see [tickPositions](#xAxis.tickPositions) - * @see [tickPositioner](#xAxis.tickPositioner) - * - * @sample {highcharts} highcharts/xaxis/tickinterval-5/ - * Tick interval of 5 on a linear axis - * @sample {highstock} stock/xaxis/tickinterval/ - * Tick interval of 0.01 on Y axis - * - * @type {number} - * @apioption xAxis.tickInterval - */ - /** - * The pixel length of the main tick marks. - * - * @sample {highcharts} highcharts/xaxis/ticklength/ - * 20 px tick length on the X axis - * @sample {highstock} stock/xaxis/ticks/ - * Formatted ticks on X axis - */ - tickLength: 10, - /** - * If tickInterval is `null` this option sets the approximate pixel - * interval of the tick marks. Not applicable to categorized axis. - * - * The tick interval is also influenced by the [minTickInterval]( - * #xAxis.minTickInterval) option, that, by default prevents ticks from - * being denser than the data points. - * - * @see [tickInterval](#xAxis.tickInterval) - * @see [tickPositioner](#xAxis.tickPositioner) - * @see [tickPositions](#xAxis.tickPositions) - * - * @sample {highcharts} highcharts/xaxis/tickpixelinterval-50/ - * 50 px on X axis - * @sample {highstock} stock/xaxis/tickpixelinterval/ - * 200 px on X axis - */ - tickPixelInterval: 100, - /** - * For categorized axes only. If `on` the tick mark is placed in the - * center of the category, if `between` the tick mark is placed between - * categories. The default is `between` if the `tickInterval` is 1, else - * `on`. - * - * @sample {highcharts} highcharts/xaxis/tickmarkplacement-between/ - * "between" by default - * @sample {highcharts} highcharts/xaxis/tickmarkplacement-on/ - * "on" - * - * @product highcharts gantt - * @validvalue ["on", "between"] - */ - tickmarkPlacement: 'between', - /** - * The position of the major tick marks relative to the axis line. - * Can be one of `inside` and `outside`. - * - * @sample {highcharts} highcharts/xaxis/tickposition-outside/ - * "outside" by default - * @sample {highcharts} highcharts/xaxis/tickposition-inside/ - * "inside" - * @sample {highstock} stock/xaxis/ticks/ - * Formatted ticks on X axis - * - * @validvalue ["inside", "outside"] - */ - tickPosition: 'outside', - /** - * A callback function returning array defining where the ticks are - * laid out on the axis. This overrides the default behaviour of - * [tickPixelInterval](#xAxis.tickPixelInterval) and [tickInterval]( - * #xAxis.tickInterval). The automatic tick positions are accessible - * through `this.tickPositions` and can be modified by the callback. - * - * @see [tickPositions](#xAxis.tickPositions) - * - * @sample {highcharts} highcharts/xaxis/tickpositions-tickpositioner/ - * Demo of tickPositions and tickPositioner - * @sample {highstock} highcharts/xaxis/tickpositions-tickpositioner/ - * Demo of tickPositions and tickPositioner - * - * @type {Highcharts.AxisTickPositionerCallbackFunction} - * @apioption xAxis.tickPositioner - */ - /** - * An array defining where the ticks are laid out on the axis. This - * overrides the default behaviour of [tickPixelInterval]( - * #xAxis.tickPixelInterval) and [tickInterval](#xAxis.tickInterval). - * - * @see [tickPositioner](#xAxis.tickPositioner) - * - * @sample {highcharts} highcharts/xaxis/tickpositions-tickpositioner/ - * Demo of tickPositions and tickPositioner - * @sample {highstock} highcharts/xaxis/tickpositions-tickpositioner/ - * Demo of tickPositions and tickPositioner - * - * @type {Array<number>} - * @apioption xAxis.tickPositions - */ - /** - * The pixel width of the major tick marks. Defaults to 0 on category - * axes, otherwise 1. - * - * In styled mode, the stroke width is given in the `.highcharts-tick` - * class, but in order for the element to be generated on category axes, - * the option must be explicitly set to 1. - * - * @sample {highcharts} highcharts/xaxis/tickwidth/ - * 10 px width - * @sample {highcharts} highcharts/css/axis-grid/ - * Styled mode - * @sample {highstock} stock/xaxis/ticks/ - * Formatted ticks on X axis - * @sample {highstock} highcharts/css/axis-grid/ - * Styled mode - * - * @type {undefined|number} - * @default {highstock} 1 - * @default {highmaps} 0 - * @apioption xAxis.tickWidth - */ - /** - * The axis title, showing next to the axis line. - * - * @productdesc {highmaps} - * In Highmaps, the axis is hidden by default, but adding an axis title - * is still possible. X axis and Y axis titles will appear at the bottom - * and left by default. - */ - title: { - /** - * Deprecated. Set the `text` to `null` to disable the title. - * - * @deprecated - * @type {boolean} - * @product highcharts - * @apioption xAxis.title.enabled - */ - /** - * The pixel distance between the axis labels or line and the title. - * Defaults to 0 for horizontal axes, 10 for vertical - * - * @sample {highcharts} highcharts/xaxis/title-margin/ - * Y axis title margin of 60 - * - * @type {number} - * @apioption xAxis.title.margin - */ - /** - * The distance of the axis title from the axis line. By default, - * this distance is computed from the offset width of the labels, - * the labels' distance from the axis and the title's margin. - * However when the offset option is set, it overrides all this. - * - * @sample {highcharts} highcharts/yaxis/title-offset/ - * Place the axis title on top of the axis - * @sample {highstock} highcharts/yaxis/title-offset/ - * Place the axis title on top of the Y axis - * - * @type {number} - * @since 2.2.0 - * @apioption xAxis.title.offset - */ - /** - * Whether to reserve space for the title when laying out the axis. - * - * @type {boolean} - * @default true - * @since 5.0.11 - * @product highcharts highstock gantt - * @apioption xAxis.title.reserveSpace - */ - /** - * The rotation of the text in degrees. 0 is horizontal, 270 is - * vertical reading from bottom to top. - * - * @sample {highcharts} highcharts/yaxis/title-offset/ - * Horizontal - * - * @type {number} - * @default 0 - * @apioption xAxis.title.rotation - */ - /** - * The actual text of the axis title. It can contain basic HTML tags - * like `b`, `i` and `span` with style. - * - * @sample {highcharts} highcharts/xaxis/title-text/ - * Custom HTML - * @sample {highstock} stock/xaxis/title-text/ - * Titles for both axes - * - * @type {string|null} - * @apioption xAxis.title.text - */ - /** - * Alignment of the text, can be `"left"`, `"right"` or `"center"`. - * Default alignment depends on the - * [title.align](xAxis.title.align): - * - * Horizontal axes: - * - for `align` = `"low"`, `textAlign` is set to `left` - * - for `align` = `"middle"`, `textAlign` is set to `center` - * - for `align` = `"high"`, `textAlign` is set to `right` - * - * Vertical axes: - * - for `align` = `"low"` and `opposite` = `true`, `textAlign` is - * set to `right` - * - for `align` = `"low"` and `opposite` = `false`, `textAlign` is - * set to `left` - * - for `align` = `"middle"`, `textAlign` is set to `center` - * - for `align` = `"high"` and `opposite` = `true` `textAlign` is - * set to `left` - * - for `align` = `"high"` and `opposite` = `false` `textAlign` is - * set to `right` - * - * @type {Highcharts.AlignValue} - * @apioption xAxis.title.textAlign - */ - /** - * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the axis title. - * - * @type {boolean} - * @default false - * @product highcharts highstock gantt - * @apioption xAxis.title.useHTML - */ - /** - * Horizontal pixel offset of the title position. - * - * @type {number} - * @default 0 - * @since 4.1.6 - * @product highcharts highstock gantt - * @apioption xAxis.title.x - */ - /** - * Vertical pixel offset of the title position. - * - * @type {number} - * @product highcharts highstock gantt - * @apioption xAxis.title.y - */ - /** - * Alignment of the title relative to the axis values. Possible - * values are "low", "middle" or "high". - * - * @sample {highcharts} highcharts/xaxis/title-align-low/ - * "low" - * @sample {highcharts} highcharts/xaxis/title-align-center/ - * "middle" by default - * @sample {highcharts} highcharts/xaxis/title-align-high/ - * "high" - * @sample {highcharts} highcharts/yaxis/title-offset/ - * Place the Y axis title on top of the axis - * @sample {highstock} stock/xaxis/title-align/ - * Aligned to "high" value - * - * @type {Highcharts.AxisTitleAlignValue} - */ - align: 'middle', - /** - * CSS styles for the title. If the title text is longer than the - * axis length, it will wrap to multiple lines by default. This can - * be customized by setting `textOverflow: 'ellipsis'`, by - * setting a specific `width` or by setting `whiteSpace: 'nowrap'`. - * - * In styled mode, the stroke width is given in the - * `.highcharts-axis-title` class. - * - * @sample {highcharts} highcharts/xaxis/title-style/ - * Red - * @sample {highcharts} highcharts/css/axis/ - * Styled mode - * - * @type {Highcharts.CSSObject} - */ - style: { - /** @internal */ - color: '#666666' - } - }, - /** - * The type of axis. Can be one of `linear`, `logarithmic`, `datetime` - * or `category`. In a datetime axis, the numbers are given in - * milliseconds, and tick marks are placed on appropriate values like - * full hours or days. In a category axis, the - * [point names](#series.line.data.name) of the chart's series are used - * for categories, if not a [categories](#xAxis.categories) array is - * defined. - * - * @sample {highcharts} highcharts/xaxis/type-linear/ - * Linear - * @sample {highcharts} highcharts/yaxis/type-log/ - * Logarithmic - * @sample {highcharts} highcharts/yaxis/type-log-minorgrid/ - * Logarithmic with minor grid lines - * @sample {highcharts} highcharts/xaxis/type-log-both/ - * Logarithmic on two axes - * @sample {highcharts} highcharts/yaxis/type-log-negative/ - * Logarithmic with extension to emulate negative values - * - * @type {Highcharts.AxisTypeValue} - * @product highcharts gantt - */ - type: 'linear', - /** - * If there are multiple axes on the same side of the chart, the pixel - * margin between the axes. Defaults to 0 on vertical axes, 15 on - * horizontal axes. - * - * @type {number} - * @since 7.0.3 - * @apioption xAxis.margin - */ - /** - * Applies only when the axis `type` is `category`. When `uniqueNames` - * is true, points are placed on the X axis according to their names. - * If the same point name is repeated in the same or another series, - * the point is placed on the same X position as other points of the - * same name. When `uniqueNames` is false, the points are laid out in - * increasing X positions regardless of their names, and the X axis - * category will take the name of the last point in each position. - * - * @sample {highcharts} highcharts/xaxis/uniquenames-true/ - * True by default - * @sample {highcharts} highcharts/xaxis/uniquenames-false/ - * False - * - * @type {boolean} - * @default true - * @since 4.2.7 - * @product highcharts gantt - * @apioption xAxis.uniqueNames - */ - /** - * Datetime axis only. An array determining what time intervals the - * ticks are allowed to fall on. Each array item is an array where the - * first value is the time unit and the second value another array of - * allowed multiples. - * - * Defaults to: - * ```js - * units: [[ - * 'millisecond', // unit name - * [1, 2, 5, 10, 20, 25, 50, 100, 200, 500] // allowed multiples - * ], [ - * 'second', - * [1, 2, 5, 10, 15, 30] - * ], [ - * 'minute', - * [1, 2, 5, 10, 15, 30] - * ], [ - * 'hour', - * [1, 2, 3, 4, 6, 8, 12] - * ], [ - * 'day', - * [1] - * ], [ - * 'week', - * [1] - * ], [ - * 'month', - * [1, 3, 6] - * ], [ - * 'year', - * null - * ]] - * ``` - * - * @type {Array<Array<string,(Array<number>|null)>>} - * @product highcharts highstock gantt - * @apioption xAxis.units - */ - /** - * Whether axis, including axis title, line, ticks and labels, should - * be visible. - * - * @type {boolean} - * @default true - * @since 4.1.9 - * @product highcharts highstock gantt - * @apioption xAxis.visible - */ - /** - * Color of the minor, secondary grid lines. - * - * In styled mode, the stroke width is given in the - * `.highcharts-minor-grid-line` class. - * - * @sample {highcharts} highcharts/yaxis/minorgridlinecolor/ - * Bright grey lines from Y axis - * @sample {highcharts|highstock} highcharts/css/axis-grid/ - * Styled mode - * @sample {highstock} stock/xaxis/minorgridlinecolor/ - * Bright grey lines from Y axis - * - * @type {Highcharts.ColorType} - * @default #f2f2f2 - */ - minorGridLineColor: '#f2f2f2', - /** - * Width of the minor, secondary grid lines. - * - * In styled mode, the stroke width is given in the - * `.highcharts-grid-line` class. - * - * @sample {highcharts} highcharts/yaxis/minorgridlinewidth/ - * 2px lines from Y axis - * @sample {highcharts|highstock} highcharts/css/axis-grid/ - * Styled mode - * @sample {highstock} stock/xaxis/minorgridlinewidth/ - * 2px lines from Y axis - */ - minorGridLineWidth: 1, - /** - * Color for the minor tick marks. - * - * @sample {highcharts} highcharts/yaxis/minortickcolor/ - * Black tick marks on Y axis - * @sample {highstock} stock/xaxis/minorticks/ - * Black tick marks on Y axis - * - * @type {Highcharts.ColorType} - * @default #999999 - */ - minorTickColor: '#999999', - /** - * The color of the line marking the axis itself. - * - * In styled mode, the line stroke is given in the - * `.highcharts-axis-line` or `.highcharts-xaxis-line` class. - * - * @productdesc {highmaps} - * In Highmaps, the axis line is hidden by default, because the axis is - * not visible by default. - * - * @sample {highcharts} highcharts/yaxis/linecolor/ - * A red line on Y axis - * @sample {highcharts|highstock} highcharts/css/axis/ - * Axes in styled mode - * @sample {highstock} stock/xaxis/linecolor/ - * A red line on X axis - * - * @type {Highcharts.ColorType} - * @default #ccd6eb - */ - lineColor: '#ccd6eb', - /** - * The width of the line marking the axis itself. - * - * In styled mode, the stroke width is given in the - * `.highcharts-axis-line` or `.highcharts-xaxis-line` class. - * - * @sample {highcharts} highcharts/yaxis/linecolor/ - * A 1px line on Y axis - * @sample {highcharts|highstock} highcharts/css/axis/ - * Axes in styled mode - * @sample {highstock} stock/xaxis/linewidth/ - * A 2px line on X axis - * - * @default {highcharts|highstock} 1 - * @default {highmaps} 0 - */ - lineWidth: 1, - /** - * Color of the grid lines extending the ticks across the plot area. - * - * In styled mode, the stroke is given in the `.highcharts-grid-line` - * class. - * - * @productdesc {highmaps} - * In Highmaps, the grid lines are hidden by default. - * - * @sample {highcharts} highcharts/yaxis/gridlinecolor/ - * Green lines - * @sample {highcharts|highstock} highcharts/css/axis-grid/ - * Styled mode - * @sample {highstock} stock/xaxis/gridlinecolor/ - * Green lines - * - * @type {Highcharts.ColorType} - * @default #e6e6e6 - */ - gridLineColor: '#e6e6e6', - // gridLineDashStyle: 'solid', - /** - * The width of the grid lines extending the ticks across the plot area. - * - * In styled mode, the stroke width is given in the - * `.highcharts-grid-line` class. - * - * @sample {highcharts} highcharts/yaxis/gridlinewidth/ - * 2px lines - * @sample {highcharts|highstock} highcharts/css/axis-grid/ - * Styled mode - * @sample {highstock} stock/xaxis/gridlinewidth/ - * 2px lines - * - * @type {number} - * @default 0 - * @apioption xAxis.gridLineWidth - */ - // gridLineWidth: 0, - /** - * The height as the vertical axis. If it's a number, it is - * interpreted as pixels. - * - * Since Highcharts 2: If it's a percentage string, it is interpreted - * as percentages of the total plot height. - * - * @type {number|string} - * @product highcharts highstock - * @apioption xAxis.height - */ - /** - * The width as the horizontal axis. If it's a number, it is interpreted - * as pixels. - * - * Since Highcharts v5.0.13: If it's a percentage string, it is - * interpreted as percentages of the total plot width. - * - * @type {number|string} - * @product highcharts highstock - * @apioption xAxis.width - */ - /** - * Color for the main tick marks. - * - * In styled mode, the stroke is given in the `.highcharts-tick` - * class. - * - * @sample {highcharts} highcharts/xaxis/tickcolor/ - * Red ticks on X axis - * @sample {highcharts|highstock} highcharts/css/axis-grid/ - * Styled mode - * @sample {highstock} stock/xaxis/ticks/ - * Formatted ticks on X axis - * - * @type {Highcharts.ColorType} - * @default #ccd6eb - */ - tickColor: '#ccd6eb' - // tickWidth: 1 - }; - /** - * The Y axis or value axis. Normally this is the vertical axis, - * though if the chart is inverted this is the horizontal axis. - * In case of multiple axes, the yAxis node is an array of - * configuration objects. - * - * See [the Axis object](/class-reference/Highcharts.Axis) for programmatic - * access to the axis. - * - * @type {*|Array<*>} - * @extends xAxis - * @excluding currentDateIndicator,ordinal,overscroll - * @optionparent yAxis - * - * @private - */ - Axis.defaultYAxisOptions = { - /** - * The type of axis. Can be one of `linear`, `logarithmic`, `datetime`, - * `category` or `treegrid`. Defaults to `treegrid` for Gantt charts, - * `linear` for other chart types. - * - * In a datetime axis, the numbers are given in milliseconds, and tick - * marks are placed on appropriate values, like full hours or days. In a - * category or treegrid axis, the [point names](#series.line.data.name) - * of the chart's series are used for categories, if a - * [categories](#xAxis.categories) array is not defined. - * - * @sample {highcharts} highcharts/yaxis/type-log-minorgrid/ - * Logarithmic with minor grid lines - * @sample {highcharts} highcharts/yaxis/type-log-negative/ - * Logarithmic with extension to emulate negative values - * @sample {gantt} gantt/treegrid-axis/demo - * Treegrid axis - * - * @type {Highcharts.AxisTypeValue} - * @default {highcharts} linear - * @default {gantt} treegrid - * @product highcharts gantt - * @apioption yAxis.type - */ - /** - * The height of the Y axis. If it's a number, it is interpreted as - * pixels. - * - * Since Highcharts 2: If it's a percentage string, it is interpreted as - * percentages of the total plot height. - * - * @see [yAxis.top](#yAxis.top) - * - * @sample {highstock} stock/demo/candlestick-and-volume/ - * Percentage height panes - * - * @type {number|string} - * @product highcharts highstock - * @apioption yAxis.height - */ - /** - * Solid gauge only. Unless [stops](#yAxis.stops) are set, the color - * to represent the maximum value of the Y axis. - * - * @sample {highcharts} highcharts/yaxis/mincolor-maxcolor/ - * Min and max colors - * - * @type {Highcharts.ColorType} - * @default #003399 - * @since 4.0 - * @product highcharts - * @apioption yAxis.maxColor - */ - /** - * Solid gauge only. Unless [stops](#yAxis.stops) are set, the color - * to represent the minimum value of the Y axis. - * - * @sample {highcharts} highcharts/yaxis/mincolor-maxcolor/ - * Min and max color - * - * @type {Highcharts.ColorType} - * @default #e6ebf5 - * @since 4.0 - * @product highcharts - * @apioption yAxis.minColor - */ - /** - * Whether to reverse the axis so that the highest number is closest - * to the origin. - * - * @sample {highcharts} highcharts/yaxis/reversed/ - * Reversed Y axis - * @sample {highstock} stock/xaxis/reversed/ - * Reversed Y axis - * - * @type {boolean} - * @default {highcharts} false - * @default {highstock} false - * @default {highmaps} true - * @default {gantt} true - * @apioption yAxis.reversed - */ - /** - * If `true`, the first series in a stack will be drawn on top in a - * positive, non-reversed Y axis. If `false`, the first series is in - * the base of the stack. - * - * @sample {highcharts} highcharts/yaxis/reversedstacks-false/ - * Non-reversed stacks - * @sample {highstock} highcharts/yaxis/reversedstacks-false/ - * Non-reversed stacks - * - * @type {boolean} - * @default true - * @since 3.0.10 - * @product highcharts highstock - * @apioption yAxis.reversedStacks - */ - /** - * Solid gauge series only. Color stops for the solid gauge. Use this - * in cases where a linear gradient between a `minColor` and `maxColor` - * is not sufficient. The stops is an array of tuples, where the first - * item is a float between 0 and 1 assigning the relative position in - * the gradient, and the second item is the color. - * - * For solid gauges, the Y axis also inherits the concept of - * [data classes](https://api.highcharts.com/highmaps#colorAxis.dataClasses) - * from the Highmaps color axis. - * - * @see [minColor](#yAxis.minColor) - * @see [maxColor](#yAxis.maxColor) - * - * @sample {highcharts} highcharts/demo/gauge-solid/ - * True by default - * - * @type {Array<Array<number,Highcharts.ColorType>>} - * @since 4.0 - * @product highcharts - * @apioption yAxis.stops - */ - /** - * The pixel width of the major tick marks. - * - * @sample {highcharts} highcharts/xaxis/tickwidth/ 10 px width - * @sample {highstock} stock/xaxis/ticks/ Formatted ticks on X axis - * - * @type {number} - * @default 0 - * @product highcharts highstock gantt - * @apioption yAxis.tickWidth - */ - /** - * Whether to force the axis to end on a tick. Use this option with - * the `maxPadding` option to control the axis end. - * - * This option is always disabled, when panning type is - * either `y` or `xy`. - * - * @see [type](#chart.panning.type) - * - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * True by default - * @sample {highcharts} highcharts/yaxis/endontick/ - * False - * @sample {highstock} stock/demo/basic-line/ - * True by default - * @sample {highstock} stock/xaxis/endontick/ - * False for Y axis - * - * @since 1.2.0 - */ - endOnTick: true, - /** - * Padding of the max value relative to the length of the axis. A - * padding of 0.05 will make a 100px axis 5px longer. This is useful - * when you don't want the highest data value to appear on the edge - * of the plot area. When the axis' `max` option is set or a max extreme - * is set using `axis.setExtremes()`, the maxPadding will be ignored. - * - * Also the `softThreshold` option takes precedence over `maxPadding`, - * so if the data is tangent to the threshold, `maxPadding` may not - * apply unless `softThreshold` is set to false. - * - * @sample {highcharts} highcharts/yaxis/maxpadding-02/ - * Max padding of 0.2 - * @sample {highstock} stock/xaxis/minpadding-maxpadding/ - * Greater min- and maxPadding - * - * @since 1.2.0 - * @product highcharts highstock gantt - */ - maxPadding: 0.05, - /** - * Padding of the min value relative to the length of the axis. A - * padding of 0.05 will make a 100px axis 5px longer. This is useful - * when you don't want the lowest data value to appear on the edge - * of the plot area. When the axis' `min` option is set or a max extreme - * is set using `axis.setExtremes()`, the maxPadding will be ignored. - * - * Also the `softThreshold` option takes precedence over `minPadding`, - * so if the data is tangent to the threshold, `minPadding` may not - * apply unless `softThreshold` is set to false. - * - * @sample {highcharts} highcharts/yaxis/minpadding/ - * Min padding of 0.2 - * @sample {highstock} stock/xaxis/minpadding-maxpadding/ - * Greater min- and maxPadding - * - * @since 1.2.0 - * @product highcharts highstock gantt - */ - minPadding: 0.05, - /** - * @productdesc {highstock} - * In Highstock 1.x, the Y axis was placed on the left side by default. - * - * @sample {highcharts} highcharts/yaxis/opposite/ - * Secondary Y axis opposite - * @sample {highstock} stock/xaxis/opposite/ - * Y axis on left side - * - * @type {boolean} - * @default {highstock} true - * @default {highcharts} false - * @product highstock highcharts gantt - * @apioption yAxis.opposite - */ - /** - * @see [tickInterval](#xAxis.tickInterval) - * @see [tickPositioner](#xAxis.tickPositioner) - * @see [tickPositions](#xAxis.tickPositions) - */ - tickPixelInterval: 72, - showLastLabel: true, - /** - * @extends xAxis.labels - */ - labels: { - /** - * Angular gauges and solid gauges only. - * The label's pixel distance from the perimeter of the plot area. - * - * Since v7.1.2: If it's a percentage string, it is interpreted the - * same as [series.radius](#plotOptions.gauge.radius), so label can be - * aligned under the gauge's shape. - * - * @sample {highcharts} highcharts/yaxis/labels-distance/ - * Labels centered under the arc - * - * @type {number|string} - * @default -25 - * @product highcharts - * @apioption yAxis.labels.distance - */ - /** - * The y position offset of all labels relative to the tick - * positions on the axis. For polar and radial axis consider the use - * of the [distance](#yAxis.labels.distance) option. - * - * @sample {highcharts} highcharts/xaxis/labels-x/ - * Y axis labels placed on grid lines - * - * @type {number} - * @default {highcharts} 3 - * @default {highstock} -2 - * @default {highmaps} 3 - * @apioption yAxis.labels.y - */ - /** - * What part of the string the given position is anchored to. Can - * be one of `"left"`, `"center"` or `"right"`. The exact position - * also depends on the `labels.x` setting. - * - * Angular gauges and solid gauges defaults to `"center"`. - * Solid gauges with two labels have additional option `"auto"` - * for automatic horizontal and vertical alignment. - * - * @see [yAxis.labels.distance](#yAxis.labels.distance) - * - * @sample {highcharts} highcharts/yaxis/labels-align-left/ - * Left - * @sample {highcharts} highcharts/series-solidgauge/labels-auto-aligned/ - * Solid gauge labels auto aligned - * - * @type {Highcharts.AlignValue} - * @default {highcharts|highmaps} right - * @default {highstock} left - * @apioption yAxis.labels.align - */ - /** - * The x position offset of all labels relative to the tick - * positions on the axis. Defaults to -15 for left axis, 15 for - * right axis. - * - * @sample {highcharts} highcharts/xaxis/labels-x/ - * Y axis labels placed on grid lines - */ - x: -8 - }, - /** - * @productdesc {highmaps} - * In Highmaps, the axis line is hidden by default, because the axis is - * not visible by default. - * - * @type {Highcharts.ColorType} - * @apioption yAxis.lineColor - */ - /** - * @sample {highcharts} highcharts/yaxis/max-200/ - * Y axis max of 200 - * @sample {highcharts} highcharts/yaxis/max-logarithmic/ - * Y axis max on logarithmic axis - * @sample {highstock} stock/yaxis/min-max/ - * Fixed min and max on Y axis - * @sample {highmaps} maps/axis/min-max/ - * Pre-zoomed to a specific area - * - * @apioption yAxis.max - */ - /** - * @sample {highcharts} highcharts/yaxis/min-startontick-false/ - * -50 with startOnTick to false - * @sample {highcharts} highcharts/yaxis/min-startontick-true/ - * -50 with startOnTick true by default - * @sample {highstock} stock/yaxis/min-max/ - * Fixed min and max on Y axis - * @sample {highmaps} maps/axis/min-max/ - * Pre-zoomed to a specific area - * - * @apioption yAxis.min - */ - /** - * An optional scrollbar to display on the Y axis in response to - * limiting the minimum an maximum of the axis values. - * - * In styled mode, all the presentational options for the scrollbar - * are replaced by the classes `.highcharts-scrollbar-thumb`, - * `.highcharts-scrollbar-arrow`, `.highcharts-scrollbar-button`, - * `.highcharts-scrollbar-rifles` and `.highcharts-scrollbar-track`. - * - * @sample {highstock} stock/yaxis/scrollbar/ - * Scrollbar on the Y axis - * - * @extends scrollbar - * @since 4.2.6 - * @product highstock - * @excluding height - * @apioption yAxis.scrollbar - */ - /** - * Enable the scrollbar on the Y axis. - * - * @sample {highstock} stock/yaxis/scrollbar/ - * Enabled on Y axis - * - * @type {boolean} - * @default false - * @since 4.2.6 - * @product highstock - * @apioption yAxis.scrollbar.enabled - */ - /** - * Pixel margin between the scrollbar and the axis elements. - * - * @type {number} - * @default 10 - * @since 4.2.6 - * @product highstock - * @apioption yAxis.scrollbar.margin - */ - /** - * Whether to show the scrollbar when it is fully zoomed out at max - * range. Setting it to `false` on the Y axis makes the scrollbar stay - * hidden until the user zooms in, like common in browsers. - * - * @type {boolean} - * @default true - * @since 4.2.6 - * @product highstock - * @apioption yAxis.scrollbar.showFull - */ - /** - * The width of a vertical scrollbar or height of a horizontal - * scrollbar. Defaults to 20 on touch devices. - * - * @type {number} - * @default 14 - * @since 4.2.6 - * @product highstock - * @apioption yAxis.scrollbar.size - */ - /** - * Z index of the scrollbar elements. - * - * @type {number} - * @default 3 - * @since 4.2.6 - * @product highstock - * @apioption yAxis.scrollbar.zIndex - */ - /** - * A soft maximum for the axis. If the series data maximum is less - * than this, the axis will stay at this maximum, but if the series - * data maximum is higher, the axis will flex to show all data. - * - * **Note**: The [series.softThreshold]( - * #plotOptions.series.softThreshold) option takes precedence over this - * option. - * - * @sample highcharts/yaxis/softmin-softmax/ - * Soft min and max - * - * @type {number} - * @since 5.0.1 - * @product highcharts highstock gantt - * @apioption yAxis.softMax - */ - /** - * A soft minimum for the axis. If the series data minimum is greater - * than this, the axis will stay at this minimum, but if the series - * data minimum is lower, the axis will flex to show all data. - * - * **Note**: The [series.softThreshold]( - * #plotOptions.series.softThreshold) option takes precedence over this - * option. - * - * @sample highcharts/yaxis/softmin-softmax/ - * Soft min and max - * - * @type {number} - * @since 5.0.1 - * @product highcharts highstock gantt - * @apioption yAxis.softMin - */ - /** - * Defines the horizontal alignment of the stack total label. Can be one - * of `"left"`, `"center"` or `"right"`. The default value is calculated - * at runtime and depends on orientation and whether the stack is - * positive or negative. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-align-left/ - * Aligned to the left - * @sample {highcharts} highcharts/yaxis/stacklabels-align-center/ - * Aligned in center - * @sample {highcharts} highcharts/yaxis/stacklabels-align-right/ - * Aligned to the right - * - * @type {Highcharts.AlignValue} - * @since 2.1.5 - * @product highcharts - * @apioption yAxis.stackLabels.align - */ - /** - * A format string for the data label. Available variables are the same - * as for `formatter`. - * - * @type {string} - * @default {total} - * @since 3.0.2 - * @product highcharts highstock - * @apioption yAxis.stackLabels.format - */ - /** - * Rotation of the labels in degrees. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-rotation/ - * Labels rotated 45° - * - * @type {number} - * @default 0 - * @since 2.1.5 - * @product highcharts - * @apioption yAxis.stackLabels.rotation - */ - /** - * The text alignment for the label. While `align` determines where the - * texts anchor point is placed with regards to the stack, `textAlign` - * determines how the text is aligned against its anchor point. Possible - * values are `"left"`, `"center"` and `"right"`. The default value is - * calculated at runtime and depends on orientation and whether the - * stack is positive or negative. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-textalign-left/ - * Label in center position but text-aligned left - * - * @type {Highcharts.AlignValue} - * @since 2.1.5 - * @product highcharts - * @apioption yAxis.stackLabels.textAlign - */ - /** - * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the labels. - * - * @type {boolean} - * @default false - * @since 3.0 - * @product highcharts highstock - * @apioption yAxis.stackLabels.useHTML - */ - /** - * Defines the vertical alignment of the stack total label. Can be one - * of `"top"`, `"middle"` or `"bottom"`. The default value is calculated - * at runtime and depends on orientation and whether the stack is - * positive or negative. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-top/ - * Vertically aligned top - * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-middle/ - * Vertically aligned middle - * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-bottom/ - * Vertically aligned bottom - * - * @type {Highcharts.VerticalAlignValue} - * @since 2.1.5 - * @product highcharts - * @apioption yAxis.stackLabels.verticalAlign - */ - /** - * The x position offset of the label relative to the left of the - * stacked bar. The default value is calculated at runtime and depends - * on orientation and whether the stack is positive or negative. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-x/ - * Stack total labels with x offset - * - * @type {number} - * @since 2.1.5 - * @product highcharts - * @apioption yAxis.stackLabels.x - */ - /** - * The y position offset of the label relative to the tick position - * on the axis. The default value is calculated at runtime and depends - * on orientation and whether the stack is positive or negative. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-y/ - * Stack total labels with y offset - * - * @type {number} - * @since 2.1.5 - * @product highcharts - * @apioption yAxis.stackLabels.y - */ - /** - * Whether to force the axis to start on a tick. Use this option with - * the `maxPadding` option to control the axis start. - * - * This option is always disabled, when panning type is - * either `y` or `xy`. - * - * @see [type](#chart.panning.type) - * - * @sample {highcharts} highcharts/xaxis/startontick-false/ - * False by default - * @sample {highcharts} highcharts/xaxis/startontick-true/ - * True - * @sample {highstock} stock/xaxis/endontick/ - * False for Y axis - * - * @since 1.2.0 - * @product highcharts highstock gantt - */ - startOnTick: true, - title: { - /** - * The pixel distance between the axis labels and the title. - * Positive values are outside the axis line, negative are inside. - * - * @sample {highcharts} highcharts/xaxis/title-margin/ - * Y axis title margin of 60 - * - * @type {number} - * @default 40 - * @apioption yAxis.title.margin - */ - /** - * The rotation of the text in degrees. 0 is horizontal, 270 is - * vertical reading from bottom to top. - * - * @sample {highcharts} highcharts/yaxis/title-offset/ - * Horizontal - */ - rotation: 270, - /** - * The actual text of the axis title. Horizontal texts can contain - * HTML, but rotated texts are painted using vector techniques and - * must be clean text. The Y axis title is disabled by setting the - * `text` option to `undefined`. - * - * @sample {highcharts} highcharts/xaxis/title-text/ - * Custom HTML - * - * @type {string|null} - * @default {highcharts} Values - * @default {highstock} undefined - * @product highcharts highstock gantt - */ - text: 'Values' - }, - /** - * The top position of the Y axis. If it's a number, it is interpreted - * as pixel position relative to the chart. - * - * Since Highcharts 2: If it's a percentage string, it is interpreted as - * percentages of the plot height, offset from plot area top. - * - * @see [yAxis.height](#yAxis.height) - * - * @sample {highstock} stock/demo/candlestick-and-volume/ - * Percentage height panes - * - * @type {number|string} - * @product highcharts highstock - * @apioption yAxis.top - */ - /** - * The stack labels show the total value for each bar in a stacked - * column or bar chart. The label will be placed on top of positive - * columns and below negative columns. In case of an inverted column - * chart or a bar chart the label is placed to the right of positive - * bars and to the left of negative bars. - * - * @product highcharts - */ - stackLabels: { - /** - * Enable or disable the initial animation when a series is - * displayed for the `stackLabels`. The animation can also be set as - * a configuration object. Please note that this option only - * applies to the initial animation. - * For other animations, see [chart.animation](#chart.animation) - * and the animation parameter under the API methods. - * The following properties are supported: - * - * - `defer`: The animation delay time in milliseconds. - * - * @sample {highcharts} highcharts/plotoptions/animation-defer/ - * Animation defer settings - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - * @since 8.2.0 - * @apioption yAxis.stackLabels.animation - */ - animation: {}, - /** - * The animation delay time in milliseconds. - * Set to `0` renders stackLabel immediately. - * As `undefined` inherits defer time from the [series.animation.defer](#plotOptions.series.animation.defer). - * - * @type {number} - * @since 8.2.0 - * @apioption yAxis.stackLabels.animation.defer - */ - /** - * Allow the stack labels to overlap. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-allowoverlap-false/ - * Default false - * - * @since 5.0.13 - * @product highcharts - */ - allowOverlap: false, - /** - * The background color or gradient for the stack label. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-box/ - * Stack labels box options - * @type {Highcharts.ColorType} - * @since 8.1.0 - * @apioption yAxis.stackLabels.backgroundColor - */ - /** - * The border color for the stack label. Defaults to `undefined`. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-box/ - * Stack labels box options - * @type {Highcharts.ColorType} - * @since 8.1.0 - * @apioption yAxis.stackLabels.borderColor - */ - /** - * The border radius in pixels for the stack label. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-box/ - * Stack labels box options - * @type {number} - * @default 0 - * @since 8.1.0 - * @apioption yAxis.stackLabels.borderRadius - */ - /** - * The border width in pixels for the stack label. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-box/ - * Stack labels box options - * @type {number} - * @default 0 - * @since 8.1.0 - * @apioption yAxis.stackLabels.borderWidth - */ - /** - * Enable or disable the stack total labels. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-enabled/ - * Enabled stack total labels - * @sample {highcharts} highcharts/yaxis/stacklabels-enabled-waterfall/ - * Enabled stack labels in waterfall chart - * - * @since 2.1.5 - * @product highcharts - */ - enabled: false, - /** - * Whether to hide stack labels that are outside the plot area. - * By default, the stack label is moved - * inside the plot area according to the - * [overflow](/highcharts/#yAxis/stackLabels/overflow) - * option. - * - * @type {boolean} - * @since 7.1.3 - */ - crop: true, - /** - * How to handle stack total labels that flow outside the plot area. - * The default is set to `"justify"`, - * which aligns them inside the plot area. - * For columns and bars, this means it will be moved inside the bar. - * To display stack labels outside the plot area, - * set `crop` to `false` and `overflow` to `"allow"`. - * - * @sample highcharts/yaxis/stacklabels-overflow/ - * Stack labels flows outside the plot area. - * - * @type {Highcharts.DataLabelsOverflowValue} - * @since 7.1.3 - */ - overflow: 'justify', - /* eslint-disable valid-jsdoc */ - /** - * Callback JavaScript function to format the label. The value is - * given by `this.total`. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-formatter/ - * Added units to stack total value - * - * @type {Highcharts.FormatterCallbackFunction<Highcharts.StackItemObject>} - * @since 2.1.5 - * @product highcharts - */ - formatter: function () { - var numberFormatter = this.axis.chart.numberFormatter; - /* eslint-enable valid-jsdoc */ - return numberFormatter(this.total, -1); - }, - /** - * CSS styles for the label. - * - * In styled mode, the styles are set in the - * `.highcharts-stack-label` class. - * - * @sample {highcharts} highcharts/yaxis/stacklabels-style/ - * Red stack total labels - * - * @type {Highcharts.CSSObject} - * @since 2.1.5 - * @product highcharts - */ - style: { - /** @internal */ - color: '#000000', - /** @internal */ - fontSize: '11px', - /** @internal */ - fontWeight: 'bold', - /** @internal */ - textOutline: '1px contrast' - } - }, - gridLineWidth: 1, - lineWidth: 0 - // tickWidth: 0 - }; - /** - * The Z axis or depth axis for 3D plots. - * - * See the [Axis class](/class-reference/Highcharts.Axis) for programmatic - * access to the axis. - * - * @sample {highcharts} highcharts/3d/scatter-zaxis-categories/ - * Z-Axis with Categories - * @sample {highcharts} highcharts/3d/scatter-zaxis-grid/ - * Z-Axis with styling - * - * @type {*|Array<*>} - * @extends xAxis - * @since 5.0.0 - * @product highcharts - * @excluding breaks, crosshair, height, left, lineColor, lineWidth, - * nameToX, showEmpty, top, width - * @apioption zAxis - * - * @private - */ - // This variable extends the defaultOptions for left axes. - Axis.defaultLeftAxisOptions = { - labels: { - x: -15 - }, - title: { - rotation: 270 - } - }; - // This variable extends the defaultOptions for right axes. - Axis.defaultRightAxisOptions = { - labels: { - x: 15 - }, - title: { - rotation: 90 - } - }; - // This variable extends the defaultOptions for bottom axes. - Axis.defaultBottomAxisOptions = { - labels: { - autoRotation: [-45], - x: 0 - // overflow: undefined, - // staggerLines: null - }, - margin: 15, - title: { - rotation: 0 - } - }; - // This variable extends the defaultOptions for top axes. - Axis.defaultTopAxisOptions = { - labels: { - autoRotation: [-45], - x: 0 - // overflow: undefined - // staggerLines: null - }, - margin: 15, - title: { - rotation: 0 - } - }; - // Properties to survive after destroy, needed for Axis.update (#4317, - // #5773, #5881). - Axis.keepProps = ['extKey', 'hcEvents', 'names', 'series', 'userMax', 'userMin']; - return Axis; - }()); - H.Axis = Axis; - - return H.Axis; - }); - _registerModule(_modules, 'Core/Axis/DateTimeAxis.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Utilities.js']], function (Axis, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var addEvent = U.addEvent, - getMagnitude = U.getMagnitude, - normalizeTickInterval = U.normalizeTickInterval, - timeUnits = U.timeUnits; - /* eslint-disable valid-jsdoc */ - var DateTimeAxisAdditions = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function DateTimeAxisAdditions(axis) { - this.axis = axis; - } - /* * - * - * Functions - * - * */ - /** - * Get a normalized tick interval for dates. Returns a configuration object - * with unit range (interval), count and name. Used to prepare data for - * `getTimeTicks`. Previously this logic was part of getTimeTicks, but as - * `getTimeTicks` now runs of segments in stock charts, the normalizing - * logic was extracted in order to prevent it for running over again for - * each segment having the same interval. #662, #697. - * @private - */ - /** - * Get a normalized tick interval for dates. Returns a configuration object - * with unit range (interval), count and name. Used to prepare data for - * `getTimeTicks`. Previously this logic was part of getTimeTicks, but as - * `getTimeTicks` now runs of segments in stock charts, the normalizing - * logic was extracted in order to prevent it for running over again for - * each segment having the same interval. #662, #697. - * @private - */ - DateTimeAxisAdditions.prototype.normalizeTimeTickInterval = function (tickInterval, unitsOption) { - var units = unitsOption || [[ - 'millisecond', - [1, 2, 5, 10, 20, 25, 50, 100, 200, 500] // allowed multiples - ], - [ - 'second', - [1, 2, 5, 10, 15, 30] - ], - [ - 'minute', - [1, 2, 5, 10, 15, 30] - ], - [ - 'hour', - [1, 2, 3, 4, 6, 8, 12] - ], - [ - 'day', - [1, 2] - ], - [ - 'week', - [1, 2] - ], - [ - 'month', - [1, 2, 3, 4, 6] - ], - [ - 'year', - null - ]], - unit = units[units.length - 1], // default unit is years - interval = timeUnits[unit[0]], - multiples = unit[1], - count, - i; - // loop through the units to find the one that best fits the - // tickInterval - for (i = 0; i < units.length; i++) { - unit = units[i]; - interval = timeUnits[unit[0]]; - multiples = unit[1]; - if (units[i + 1]) { - // lessThan is in the middle between the highest multiple and - // the next unit. - var lessThan = (interval * - multiples[multiples.length - 1] + - timeUnits[units[i + 1][0]]) / 2; - // break and keep the current unit - if (tickInterval <= lessThan) { - break; - } - } - } - // prevent 2.5 years intervals, though 25, 250 etc. are allowed - if (interval === timeUnits.year && tickInterval < 5 * interval) { - multiples = [1, 2, 5]; - } - // get the count - count = normalizeTickInterval(tickInterval / interval, multiples, unit[0] === 'year' ? // #1913, #2360 - Math.max(getMagnitude(tickInterval / interval), 1) : - 1); - return { - unitRange: interval, - count: count, - unitName: unit[0] - }; - }; - return DateTimeAxisAdditions; - }()); - /** - * Date and time support for axes. - * - * @private - * @class - */ - var DateTimeAxis = /** @class */ (function () { - function DateTimeAxis() { - } - /* * - * - * Static Functions - * - * */ - /** - * Extends axis class with date and time support. - * @private - */ - DateTimeAxis.compose = function (AxisClass) { - AxisClass.keepProps.push('dateTime'); - var axisProto = AxisClass.prototype; - /** - * Set the tick positions to a time unit that makes sense, for example - * on the first of each month or on every Monday. Return an array with - * the time positions. Used in datetime axes as well as for grouping - * data on a datetime axis. - * - * @private - * @function Highcharts.Axis#getTimeTicks - * - * @param {Highcharts.TimeNormalizeObject} normalizedInterval - * The interval in axis values (ms) and thecount. - * - * @param {number} min - * The minimum in axis values. - * - * @param {number} max - * The maximum in axis values. - * - * @param {number} startOfWeek - * - * @return {Highcharts.AxisTickPositionsArray} - */ - axisProto.getTimeTicks = function () { - return this.chart.time.getTimeTicks.apply(this.chart.time, arguments); - }; - /* eslint-disable no-invalid-this */ - addEvent(AxisClass, 'init', function (e) { - var axis = this; - var options = e.userOptions; - if (options.type !== 'datetime') { - axis.dateTime = void 0; - return; - } - if (!axis.dateTime) { - axis.dateTime = new DateTimeAxisAdditions(axis); - } - }); - /* eslint-enable no-invalid-this */ - }; - /* * - * - * Static Properties - * - * */ - DateTimeAxis.AdditionsClass = DateTimeAxisAdditions; - return DateTimeAxis; - }()); - DateTimeAxis.compose(Axis); - - return DateTimeAxis; - }); - _registerModule(_modules, 'Core/Axis/LogarithmicAxis.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Utilities.js']], function (Axis, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var addEvent = U.addEvent, - getMagnitude = U.getMagnitude, - normalizeTickInterval = U.normalizeTickInterval, - pick = U.pick; - /* eslint-disable valid-jsdoc */ - /** - * Provides logarithmic support for axes. - * - * @private - * @class - */ - var LogarithmicAxisAdditions = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function LogarithmicAxisAdditions(axis) { - this.axis = axis; - } - /* * - * - * Functions - * - * */ - /** - * Set the tick positions of a logarithmic axis. - */ - LogarithmicAxisAdditions.prototype.getLogTickPositions = function (interval, min, max, minor) { - var log = this; - var axis = log.axis; - var axisLength = axis.len; - var options = axis.options; - // Since we use this method for both major and minor ticks, - // use a local variable and return the result - var positions = []; - // Reset - if (!minor) { - log.minorAutoInterval = void 0; - } - // First case: All ticks fall on whole logarithms: 1, 10, 100 etc. - if (interval >= 0.5) { - interval = Math.round(interval); - positions = axis.getLinearTickPositions(interval, min, max); - // Second case: We need intermediary ticks. For example - // 1, 2, 4, 6, 8, 10, 20, 40 etc. - } - else if (interval >= 0.08) { - var roundedMin = Math.floor(min), - intermediate, - i, - j, - len, - pos, - lastPos, - break2; - if (interval > 0.3) { - intermediate = [1, 2, 4]; - // 0.2 equals five minor ticks per 1, 10, 100 etc - } - else if (interval > 0.15) { - intermediate = [1, 2, 4, 6, 8]; - } - else { // 0.1 equals ten minor ticks per 1, 10, 100 etc - intermediate = [1, 2, 3, 4, 5, 6, 7, 8, 9]; - } - for (i = roundedMin; i < max + 1 && !break2; i++) { - len = intermediate.length; - for (j = 0; j < len && !break2; j++) { - pos = log.log2lin(log.lin2log(i) * intermediate[j]); - // #1670, lastPos is #3113 - if (pos > min && - (!minor || lastPos <= max) && - typeof lastPos !== 'undefined') { - positions.push(lastPos); - } - if (lastPos > max) { - break2 = true; - } - lastPos = pos; - } - } - // Third case: We are so deep in between whole logarithmic values that - // we might as well handle the tick positions like a linear axis. For - // example 1.01, 1.02, 1.03, 1.04. - } - else { - var realMin = log.lin2log(min), - realMax = log.lin2log(max), - tickIntervalOption = minor ? - axis.getMinorTickInterval() : - options.tickInterval, - filteredTickIntervalOption = tickIntervalOption === 'auto' ? - null : - tickIntervalOption, - tickPixelIntervalOption = options.tickPixelInterval / (minor ? 5 : 1), - totalPixelLength = minor ? - axisLength / axis.tickPositions.length : - axisLength; - interval = pick(filteredTickIntervalOption, log.minorAutoInterval, (realMax - realMin) * - tickPixelIntervalOption / (totalPixelLength || 1)); - interval = normalizeTickInterval(interval, void 0, getMagnitude(interval)); - positions = axis.getLinearTickPositions(interval, realMin, realMax).map(log.log2lin); - if (!minor) { - log.minorAutoInterval = interval / 5; - } - } - // Set the axis-level tickInterval variable - if (!minor) { - axis.tickInterval = interval; - } - return positions; - }; - LogarithmicAxisAdditions.prototype.lin2log = function (num) { - return Math.pow(10, num); - }; - LogarithmicAxisAdditions.prototype.log2lin = function (num) { - return Math.log(num) / Math.LN10; - }; - return LogarithmicAxisAdditions; - }()); - var LogarithmicAxis = /** @class */ (function () { - function LogarithmicAxis() { - } - /** - * Provides logarithmic support for axes. - * - * @private - */ - LogarithmicAxis.compose = function (AxisClass) { - AxisClass.keepProps.push('logarithmic'); - // HC <= 8 backwards compatibility, allow wrapping - // Axis.prototype.lin2log and log2lin - // @todo Remove this in next major - var axisProto = AxisClass.prototype; - var logAxisProto = LogarithmicAxisAdditions.prototype; - axisProto.log2lin = logAxisProto.log2lin; - axisProto.lin2log = logAxisProto.lin2log; - /* eslint-disable no-invalid-this */ - addEvent(AxisClass, 'init', function (e) { - var axis = this; - var options = e.userOptions; - var logarithmic = axis.logarithmic; - if (options.type !== 'logarithmic') { - axis.logarithmic = void 0; - } - else { - if (!logarithmic) { - logarithmic = axis.logarithmic = new LogarithmicAxisAdditions(axis); - } - // HC <= 8 backwards compatibility, allow wrapping - // Axis.prototype.lin2log and log2lin - // @todo Remove this in next major - if (axis.log2lin !== logarithmic.log2lin) { - logarithmic.log2lin = axis.log2lin.bind(axis); - } - if (axis.lin2log !== logarithmic.lin2log) { - logarithmic.lin2log = axis.lin2log.bind(axis); - } - } - }); - addEvent(AxisClass, 'afterInit', function () { - var axis = this; - var log = axis.logarithmic; - // extend logarithmic axis - if (log) { - axis.lin2val = function (num) { - return log.lin2log(num); - }; - axis.val2lin = function (num) { - return log.log2lin(num); - }; - } - }); - }; - return LogarithmicAxis; - }()); - LogarithmicAxis.compose(Axis); // @todo move to factory functions - - return LogarithmicAxis; - }); - _registerModule(_modules, 'Core/Axis/PlotLineOrBand.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Axis, H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - /** - * Options for plot bands on axes. - * - * @typedef {Highcharts.XAxisPlotBandsOptions|Highcharts.YAxisPlotBandsOptions|Highcharts.ZAxisPlotBandsOptions} Highcharts.AxisPlotBandsOptions - */ - /** - * Options for plot band labels on axes. - * - * @typedef {Highcharts.XAxisPlotBandsLabelOptions|Highcharts.YAxisPlotBandsLabelOptions|Highcharts.ZAxisPlotBandsLabelOptions} Highcharts.AxisPlotBandsLabelOptions - */ - /** - * Options for plot lines on axes. - * - * @typedef {Highcharts.XAxisPlotLinesOptions|Highcharts.YAxisPlotLinesOptions|Highcharts.ZAxisPlotLinesOptions} Highcharts.AxisPlotLinesOptions - */ - /** - * Options for plot line labels on axes. - * - * @typedef {Highcharts.XAxisPlotLinesLabelOptions|Highcharts.YAxisPlotLinesLabelOptions|Highcharts.ZAxisPlotLinesLabelOptions} Highcharts.AxisPlotLinesLabelOptions - */ - var arrayMax = U.arrayMax, - arrayMin = U.arrayMin, - defined = U.defined, - destroyObjectProperties = U.destroyObjectProperties, - erase = U.erase, - extend = U.extend, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick; - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The object wrapper for plot lines and plot bands - * - * @class - * @name Highcharts.PlotLineOrBand - * - * @param {Highcharts.Axis} axis - * - * @param {Highcharts.AxisPlotLinesOptions|Highcharts.AxisPlotBandsOptions} [options] - */ - var PlotLineOrBand = /** @class */ (function () { - function PlotLineOrBand(axis, options) { - this.axis = axis; - if (options) { - this.options = options; - this.id = options.id; - } - } - /** - * Render the plot line or plot band. If it is already existing, - * move it. - * - * @private - * @function Highcharts.PlotLineOrBand#render - * @return {Highcharts.PlotLineOrBand|undefined} - */ - PlotLineOrBand.prototype.render = function () { - H.fireEvent(this, 'render'); - var plotLine = this, - axis = plotLine.axis, - horiz = axis.horiz, - log = axis.logarithmic, - options = plotLine.options, - optionsLabel = options.label, - label = plotLine.label, - to = options.to, - from = options.from, - value = options.value, - isBand = defined(from) && defined(to), - isLine = defined(value), - svgElem = plotLine.svgElem, - isNew = !svgElem, - path = [], - color = options.color, - zIndex = pick(options.zIndex, 0), - events = options.events, - attribs = { - 'class': 'highcharts-plot-' + (isBand ? 'band ' : 'line ') + - (options.className || '') - }, - groupAttribs = {}, - renderer = axis.chart.renderer, - groupName = isBand ? 'bands' : 'lines', - group; - // logarithmic conversion - if (log) { - from = log.log2lin(from); - to = log.log2lin(to); - value = log.log2lin(value); - } - // Set the presentational attributes - if (!axis.chart.styledMode) { - if (isLine) { - attribs.stroke = color || '#999999'; - attribs['stroke-width'] = pick(options.width, 1); - if (options.dashStyle) { - attribs.dashstyle = - options.dashStyle; - } - } - else if (isBand) { // plot band - attribs.fill = color || '#e6ebf5'; - if (options.borderWidth) { - attribs.stroke = options.borderColor; - attribs['stroke-width'] = options.borderWidth; - } - } - } - // Grouping and zIndex - groupAttribs.zIndex = zIndex; - groupName += '-' + zIndex; - group = axis.plotLinesAndBandsGroups[groupName]; - if (!group) { - axis.plotLinesAndBandsGroups[groupName] = group = - renderer.g('plot-' + groupName) - .attr(groupAttribs).add(); - } - // Create the path - if (isNew) { - /** - * SVG element of the plot line or band. - * - * @name Highcharts.PlotLineOrBand#svgElement - * @type {Highcharts.SVGElement} - */ - plotLine.svgElem = svgElem = renderer - .path() - .attr(attribs) - .add(group); - } - // Set the path or return - if (isLine) { - path = axis.getPlotLinePath({ - value: value, - lineWidth: svgElem.strokeWidth(), - acrossPanes: options.acrossPanes - }); - } - else if (isBand) { // plot band - path = axis.getPlotBandPath(from, to, options); - } - else { - return; - } - // common for lines and bands - // Add events only if they were not added before. - if (!plotLine.eventsAdded && events) { - objectEach(events, function (event, eventType) { - svgElem.on(eventType, function (e) { - events[eventType].apply(plotLine, [e]); - }); - }); - plotLine.eventsAdded = true; - } - if ((isNew || !svgElem.d) && path && path.length) { - svgElem.attr({ d: path }); - } - else if (svgElem) { - if (path) { - svgElem.show(true); - svgElem.animate({ d: path }); - } - else if (svgElem.d) { - svgElem.hide(); - if (label) { - plotLine.label = label = label.destroy(); - } - } - } - // the plot band/line label - if (optionsLabel && - (defined(optionsLabel.text) || defined(optionsLabel.formatter)) && - path && - path.length && - axis.width > 0 && - axis.height > 0 && - !path.isFlat) { - // apply defaults - optionsLabel = merge({ - align: horiz && isBand && 'center', - x: horiz ? !isBand && 4 : 10, - verticalAlign: !horiz && isBand && 'middle', - y: horiz ? isBand ? 16 : 10 : isBand ? 6 : -4, - rotation: horiz && !isBand && 90 - }, optionsLabel); - this.renderLabel(optionsLabel, path, isBand, zIndex); - } - else if (label) { // move out of sight - label.hide(); - } - // chainable - return plotLine; - }; - /** - * Render and align label for plot line or band. - * - * @private - * @function Highcharts.PlotLineOrBand#renderLabel - * @param {Highcharts.AxisPlotLinesLabelOptions|Highcharts.AxisPlotBandsLabelOptions} optionsLabel - * @param {Highcharts.SVGPathArray} path - * @param {boolean} [isBand] - * @param {number} [zIndex] - * @return {void} - */ - PlotLineOrBand.prototype.renderLabel = function (optionsLabel, path, isBand, zIndex) { - var plotLine = this, - label = plotLine.label, - renderer = plotLine.axis.chart.renderer, - attribs, - xBounds, - yBounds, - x, - y, - labelText; - // add the SVG element - if (!label) { - attribs = { - align: optionsLabel.textAlign || optionsLabel.align, - rotation: optionsLabel.rotation, - 'class': 'highcharts-plot-' + (isBand ? 'band' : 'line') + - '-label ' + (optionsLabel.className || '') - }; - attribs.zIndex = zIndex; - labelText = this.getLabelText(optionsLabel); - /** - * SVG element of the label. - * - * @name Highcharts.PlotLineOrBand#label - * @type {Highcharts.SVGElement} - */ - plotLine.label = label = renderer - .text(labelText, 0, 0, optionsLabel.useHTML) - .attr(attribs) - .add(); - if (!this.axis.chart.styledMode) { - label.css(optionsLabel.style); - } - } - // get the bounding box and align the label - // #3000 changed to better handle choice between plotband or plotline - xBounds = path.xBounds || - [path[0][1], path[1][1], (isBand ? path[2][1] : path[0][1])]; - yBounds = path.yBounds || - [path[0][2], path[1][2], (isBand ? path[2][2] : path[0][2])]; - x = arrayMin(xBounds); - y = arrayMin(yBounds); - label.align(optionsLabel, false, { - x: x, - y: y, - width: arrayMax(xBounds) - x, - height: arrayMax(yBounds) - y - }); - label.show(true); - }; - /** - * Get label's text content. - * - * @private - * @function Highcharts.PlotLineOrBand#getLabelText - * @param {Highcharts.AxisPlotLinesLabelOptions|Highcharts.AxisPlotBandsLabelOptions} optionsLabel - * @return {string} - */ - PlotLineOrBand.prototype.getLabelText = function (optionsLabel) { - return defined(optionsLabel.formatter) ? - optionsLabel.formatter - .call(this) : - optionsLabel.text; - }; - /** - * Remove the plot line or band. - * - * @function Highcharts.PlotLineOrBand#destroy - * @return {void} - */ - PlotLineOrBand.prototype.destroy = function () { - // remove it from the lookup - erase(this.axis.plotLinesAndBands, this); - delete this.axis; - destroyObjectProperties(this); - }; - return PlotLineOrBand; - }()); - /* eslint-enable no-invalid-this, valid-jsdoc */ - // Object with members for extending the Axis prototype - extend(Axis.prototype, /** @lends Highcharts.Axis.prototype */ { - /** - * An array of colored bands stretching across the plot area marking an - * interval on the axis. - * - * In styled mode, the plot bands are styled by the `.highcharts-plot-band` - * class in addition to the `className` option. - * - * @productdesc {highcharts} - * In a gauge, a plot band on the Y axis (value axis) will stretch along the - * perimeter of the gauge. - * - * @type {Array<*>} - * @product highcharts highstock gantt - * @apioption xAxis.plotBands - */ - /** - * Flag to decide if plotBand should be rendered across all panes. - * - * @since 7.1.2 - * @product highstock - * @type {boolean} - * @default true - * @apioption xAxis.plotBands.acrossPanes - */ - /** - * Border color for the plot band. Also requires `borderWidth` to be set. - * - * @type {Highcharts.ColorString} - * @apioption xAxis.plotBands.borderColor - */ - /** - * Border width for the plot band. Also requires `borderColor` to be set. - * - * @type {number} - * @default 0 - * @apioption xAxis.plotBands.borderWidth - */ - /** - * A custom class name, in addition to the default `highcharts-plot-band`, - * to apply to each individual band. - * - * @type {string} - * @since 5.0.0 - * @apioption xAxis.plotBands.className - */ - /** - * The color of the plot band. - * - * @sample {highcharts} highcharts/xaxis/plotbands-color/ - * Color band - * @sample {highstock} stock/xaxis/plotbands/ - * Plot band on Y axis - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @default #e6ebf5 - * @apioption xAxis.plotBands.color - */ - /** - * An object defining mouse events for the plot band. Supported properties - * are `click`, `mouseover`, `mouseout`, `mousemove`. - * - * @sample {highcharts} highcharts/xaxis/plotbands-events/ - * Mouse events demonstrated - * - * @since 1.2 - * @apioption xAxis.plotBands.events - */ - /** - * Click event on a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotBands.events.click - */ - /** - * Mouse move event on a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotBands.events.mousemove - */ - /** - * Mouse out event on the corner of a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotBands.events.mouseout - */ - /** - * Mouse over event on a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotBands.events.mouseover - */ - /** - * The start position of the plot band in axis units. - * - * @sample {highcharts} highcharts/xaxis/plotbands-color/ - * Datetime axis - * @sample {highcharts} highcharts/xaxis/plotbands-from/ - * Categorized axis - * @sample {highstock} stock/xaxis/plotbands/ - * Plot band on Y axis - * - * @type {number} - * @apioption xAxis.plotBands.from - */ - /** - * An id used for identifying the plot band in Axis.removePlotBand. - * - * @sample {highcharts} highcharts/xaxis/plotbands-id/ - * Remove plot band by id - * @sample {highstock} highcharts/xaxis/plotbands-id/ - * Remove plot band by id - * - * @type {string} - * @apioption xAxis.plotBands.id - */ - /** - * The end position of the plot band in axis units. - * - * @sample {highcharts} highcharts/xaxis/plotbands-color/ - * Datetime axis - * @sample {highcharts} highcharts/xaxis/plotbands-from/ - * Categorized axis - * @sample {highstock} stock/xaxis/plotbands/ - * Plot band on Y axis - * - * @type {number} - * @apioption xAxis.plotBands.to - */ - /** - * The z index of the plot band within the chart, relative to other - * elements. Using the same z index as another element may give - * unpredictable results, as the last rendered element will be on top. - * Values from 0 to 20 make sense. - * - * @sample {highcharts} highcharts/xaxis/plotbands-color/ - * Behind plot lines by default - * @sample {highcharts} highcharts/xaxis/plotbands-zindex/ - * Above plot lines - * @sample {highcharts} highcharts/xaxis/plotbands-zindex-above-series/ - * Above plot lines and series - * - * @type {number} - * @since 1.2 - * @apioption xAxis.plotBands.zIndex - */ - /** - * Text labels for the plot bands - * - * @product highcharts highstock gantt - * @apioption xAxis.plotBands.label - */ - /** - * Horizontal alignment of the label. Can be one of "left", "center" or - * "right". - * - * @sample {highcharts} highcharts/xaxis/plotbands-label-align/ - * Aligned to the right - * @sample {highstock} stock/xaxis/plotbands-label/ - * Plot band with labels - * - * @type {Highcharts.AlignValue} - * @default center - * @since 2.1 - * @apioption xAxis.plotBands.label.align - */ - /** - * Rotation of the text label in degrees . - * - * @sample {highcharts} highcharts/xaxis/plotbands-label-rotation/ - * Vertical text - * - * @type {number} - * @default 0 - * @since 2.1 - * @apioption xAxis.plotBands.label.rotation - */ - /** - * CSS styles for the text label. - * - * In styled mode, the labels are styled by the - * `.highcharts-plot-band-label` class. - * - * @sample {highcharts} highcharts/xaxis/plotbands-label-style/ - * Blue and bold label - * - * @type {Highcharts.CSSObject} - * @since 2.1 - * @apioption xAxis.plotBands.label.style - */ - /** - * The string text itself. A subset of HTML is supported. - * - * @type {string} - * @since 2.1 - * @apioption xAxis.plotBands.label.text - */ - /** - * The text alignment for the label. While `align` determines where the - * texts anchor point is placed within the plot band, `textAlign` determines - * how the text is aligned against its anchor point. Possible values are - * "left", "center" and "right". Defaults to the same as the `align` option. - * - * @sample {highcharts} highcharts/xaxis/plotbands-label-rotation/ - * Vertical text in center position but text-aligned left - * - * @type {Highcharts.AlignValue} - * @since 2.1 - * @apioption xAxis.plotBands.label.textAlign - */ - /** - * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the labels. - * - * @type {boolean} - * @default false - * @since 3.0.3 - * @apioption xAxis.plotBands.label.useHTML - */ - /** - * Vertical alignment of the label relative to the plot band. Can be one of - * "top", "middle" or "bottom". - * - * @sample {highcharts} highcharts/xaxis/plotbands-label-verticalalign/ - * Vertically centered label - * @sample {highstock} stock/xaxis/plotbands-label/ - * Plot band with labels - * - * @type {Highcharts.VerticalAlignValue} - * @default top - * @since 2.1 - * @apioption xAxis.plotBands.label.verticalAlign - */ - /** - * Horizontal position relative the alignment. Default varies by - * orientation. - * - * @sample {highcharts} highcharts/xaxis/plotbands-label-align/ - * Aligned 10px from the right edge - * @sample {highstock} stock/xaxis/plotbands-label/ - * Plot band with labels - * - * @type {number} - * @since 2.1 - * @apioption xAxis.plotBands.label.x - */ - /** - * Vertical position of the text baseline relative to the alignment. Default - * varies by orientation. - * - * @sample {highcharts} highcharts/xaxis/plotbands-label-y/ - * Label on x axis - * @sample {highstock} stock/xaxis/plotbands-label/ - * Plot band with labels - * - * @type {number} - * @since 2.1 - * @apioption xAxis.plotBands.label.y - */ - /** - * An array of lines stretching across the plot area, marking a specific - * value on one of the axes. - * - * In styled mode, the plot lines are styled by the - * `.highcharts-plot-line` class in addition to the `className` option. - * - * @type {Array<*>} - * @product highcharts highstock gantt - * @sample {highcharts} highcharts/xaxis/plotlines-color/ - * Basic plot line - * @sample {highcharts} highcharts/series-solidgauge/labels-auto-aligned/ - * Solid gauge plot line - * @apioption xAxis.plotLines - */ - /** - * Flag to decide if plotLine should be rendered across all panes. - * - * @sample {highstock} stock/xaxis/plotlines-acrosspanes/ - * Plot lines on different panes - * - * @since 7.1.2 - * @product highstock - * @type {boolean} - * @default true - * @apioption xAxis.plotLines.acrossPanes - */ - /** - * A custom class name, in addition to the default `highcharts-plot-line`, - * to apply to each individual line. - * - * @type {string} - * @since 5.0.0 - * @apioption xAxis.plotLines.className - */ - /** - * The color of the line. - * - * @sample {highcharts} highcharts/xaxis/plotlines-color/ - * A red line from X axis - * @sample {highstock} stock/xaxis/plotlines/ - * Plot line on Y axis - * - * @type {Highcharts.ColorString} - * @default #999999 - * @apioption xAxis.plotLines.color - */ - /** - * The dashing or dot style for the plot line. For possible values see - * [this overview](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/). - * - * @sample {highcharts} highcharts/xaxis/plotlines-dashstyle/ - * Dash and dot pattern - * @sample {highstock} stock/xaxis/plotlines/ - * Plot line on Y axis - * - * @type {Highcharts.DashStyleValue} - * @default Solid - * @since 1.2 - * @apioption xAxis.plotLines.dashStyle - */ - /** - * An object defining mouse events for the plot line. Supported - * properties are `click`, `mouseover`, `mouseout`, `mousemove`. - * - * @sample {highcharts} highcharts/xaxis/plotlines-events/ - * Mouse events demonstrated - * - * @since 1.2 - * @apioption xAxis.plotLines.events - */ - /** - * Click event on a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotLines.events.click - */ - /** - * Mouse move event on a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotLines.events.mousemove - */ - /** - * Mouse out event on the corner of a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotLines.events.mouseout - */ - /** - * Mouse over event on a plot band. - * - * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotLines.events.mouseover - */ - /** - * An id used for identifying the plot line in Axis.removePlotLine. - * - * @sample {highcharts} highcharts/xaxis/plotlines-id/ - * Remove plot line by id - * - * @type {string} - * @apioption xAxis.plotLines.id - */ - /** - * The position of the line in axis units. - * - * @sample {highcharts} highcharts/xaxis/plotlines-color/ - * Between two categories on X axis - * @sample {highstock} stock/xaxis/plotlines/ - * Plot line on Y axis - * - * @type {number} - * @apioption xAxis.plotLines.value - */ - /** - * The width or thickness of the plot line. - * - * @sample {highcharts} highcharts/xaxis/plotlines-color/ - * 2px wide line from X axis - * @sample {highstock} stock/xaxis/plotlines/ - * Plot line on Y axis - * - * @type {number} - * @default 2 - * @apioption xAxis.plotLines.width - */ - /** - * The z index of the plot line within the chart. - * - * @sample {highcharts} highcharts/xaxis/plotlines-zindex-behind/ - * Behind plot lines by default - * @sample {highcharts} highcharts/xaxis/plotlines-zindex-above/ - * Above plot lines - * @sample {highcharts} highcharts/xaxis/plotlines-zindex-above-all/ - * Above plot lines and series - * - * @type {number} - * @since 1.2 - * @apioption xAxis.plotLines.zIndex - */ - /** - * Text labels for the plot bands - * - * @apioption xAxis.plotLines.label - */ - /** - * Horizontal alignment of the label. Can be one of "left", "center" or - * "right". - * - * @sample {highcharts} highcharts/xaxis/plotlines-label-align-right/ - * Aligned to the right - * @sample {highstock} stock/xaxis/plotlines/ - * Plot line on Y axis - * - * @type {Highcharts.AlignValue} - * @default left - * @since 2.1 - * @apioption xAxis.plotLines.label.align - */ - /** - * Callback JavaScript function to format the label. Useful properties like - * the value of plot line or the range of plot band (`from` & `to` - * properties) can be found in `this.options` object. - * - * @sample {highcharts} highcharts/xaxis/plotlines-plotbands-label-formatter - * Label formatters for plot line and plot band. - * @type {Highcharts.FormatterCallbackFunction<Highcharts.PlotLineOrBand>} - * @apioption xAxis.plotLines.label.formatter - */ - /** - * Rotation of the text label in degrees. Defaults to 0 for horizontal plot - * lines and 90 for vertical lines. - * - * @sample {highcharts} highcharts/xaxis/plotlines-label-verticalalign-middle/ - * Slanted text - * - * @type {number} - * @since 2.1 - * @apioption xAxis.plotLines.label.rotation - */ - /** - * CSS styles for the text label. - * - * In styled mode, the labels are styled by the - * `.highcharts-plot-line-label` class. - * - * @sample {highcharts} highcharts/xaxis/plotlines-label-style/ - * Blue and bold label - * - * @type {Highcharts.CSSObject} - * @since 2.1 - * @apioption xAxis.plotLines.label.style - */ - /** - * The text itself. A subset of HTML is supported. - * - * @type {string} - * @since 2.1 - * @apioption xAxis.plotLines.label.text - */ - /** - * The text alignment for the label. While `align` determines where the - * texts anchor point is placed within the plot band, `textAlign` determines - * how the text is aligned against its anchor point. Possible values are - * "left", "center" and "right". Defaults to the same as the `align` option. - * - * @sample {highcharts} highcharts/xaxis/plotlines-label-textalign/ - * Text label in bottom position - * - * @type {Highcharts.AlignValue} - * @since 2.1 - * @apioption xAxis.plotLines.label.textAlign - */ - /** - * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the labels. - * - * @type {boolean} - * @default false - * @since 3.0.3 - * @apioption xAxis.plotLines.label.useHTML - */ - /** - * Vertical alignment of the label relative to the plot line. Can be - * one of "top", "middle" or "bottom". - * - * @sample {highcharts} highcharts/xaxis/plotlines-label-verticalalign-middle/ - * Vertically centered label - * - * @type {Highcharts.VerticalAlignValue} - * @default {highcharts} top - * @default {highstock} top - * @since 2.1 - * @apioption xAxis.plotLines.label.verticalAlign - */ - /** - * Horizontal position relative the alignment. Default varies by - * orientation. - * - * @sample {highcharts} highcharts/xaxis/plotlines-label-align-right/ - * Aligned 10px from the right edge - * @sample {highstock} stock/xaxis/plotlines/ - * Plot line on Y axis - * - * @type {number} - * @since 2.1 - * @apioption xAxis.plotLines.label.x - */ - /** - * Vertical position of the text baseline relative to the alignment. Default - * varies by orientation. - * - * @sample {highcharts} highcharts/xaxis/plotlines-label-y/ - * Label below the plot line - * @sample {highstock} stock/xaxis/plotlines/ - * Plot line on Y axis - * - * @type {number} - * @since 2.1 - * @apioption xAxis.plotLines.label.y - */ - /** - * - * @type {Array<*>} - * @extends xAxis.plotBands - * @apioption yAxis.plotBands - */ - /** - * In a gauge chart, this option determines the inner radius of the - * plot band that stretches along the perimeter. It can be given as - * a percentage string, like `"100%"`, or as a pixel number, like `100`. - * By default, the inner radius is controlled by the [thickness]( - * #yAxis.plotBands.thickness) option. - * - * @sample {highcharts} highcharts/xaxis/plotbands-gauge - * Gauge plot band - * - * @type {number|string} - * @since 2.3 - * @product highcharts - * @apioption yAxis.plotBands.innerRadius - */ - /** - * In a gauge chart, this option determines the outer radius of the - * plot band that stretches along the perimeter. It can be given as - * a percentage string, like `"100%"`, or as a pixel number, like `100`. - * - * @sample {highcharts} highcharts/xaxis/plotbands-gauge - * Gauge plot band - * - * @type {number|string} - * @default 100% - * @since 2.3 - * @product highcharts - * @apioption yAxis.plotBands.outerRadius - */ - /** - * In a gauge chart, this option sets the width of the plot band - * stretching along the perimeter. It can be given as a percentage - * string, like `"10%"`, or as a pixel number, like `10`. The default - * value 10 is the same as the default [tickLength](#yAxis.tickLength), - * thus making the plot band act as a background for the tick markers. - * - * @sample {highcharts} highcharts/xaxis/plotbands-gauge - * Gauge plot band - * - * @type {number|string} - * @default 10 - * @since 2.3 - * @product highcharts - * @apioption yAxis.plotBands.thickness - */ - /** - * @type {Array<*>} - * @extends xAxis.plotLines - * @apioption yAxis.plotLines - */ - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * Internal function to create the SVG path definition for a plot band. - * - * @function Highcharts.Axis#getPlotBandPath - * - * @param {number} from - * The axis value to start from. - * - * @param {number} to - * The axis value to end on. - * - * @param {Highcharts.AxisPlotBandsOptions|Highcharts.AxisPlotLinesOptions} options - * The plotBand or plotLine configuration object. - * - * @return {Highcharts.SVGPathArray} - * The SVG path definition in array form. - */ - getPlotBandPath: function (from, to, options) { - if (options === void 0) { options = this.options; } - var toPath = this.getPlotLinePath({ - value: to, - force: true, - acrossPanes: options.acrossPanes - }), - path = this.getPlotLinePath({ - value: from, - force: true, - acrossPanes: options.acrossPanes - }), - result = [], - i, - // #4964 check if chart is inverted or plotband is on yAxis - horiz = this.horiz, - plus = 1, - isFlat, - outside = (from < this.min && to < this.min) || - (from > this.max && to > this.max); - if (path && toPath) { - // Flat paths don't need labels (#3836) - if (outside) { - isFlat = path.toString() === toPath.toString(); - plus = 0; - } - // Go over each subpath - for panes in Highstock - for (i = 0; i < path.length; i += 2) { - var pathStart = path[i], - pathEnd = path[i + 1], - toPathStart = toPath[i], - toPathEnd = toPath[i + 1]; - // Type checking all affected path segments. Consider something - // smarter. - if ((pathStart[0] === 'M' || pathStart[0] === 'L') && - (pathEnd[0] === 'M' || pathEnd[0] === 'L') && - (toPathStart[0] === 'M' || toPathStart[0] === 'L') && - (toPathEnd[0] === 'M' || toPathEnd[0] === 'L')) { - // Add 1 pixel when coordinates are the same - if (horiz && toPathStart[1] === pathStart[1]) { - toPathStart[1] += plus; - toPathEnd[1] += plus; - } - else if (!horiz && toPathStart[2] === pathStart[2]) { - toPathStart[2] += plus; - toPathEnd[2] += plus; - } - result.push(['M', pathStart[1], pathStart[2]], ['L', pathEnd[1], pathEnd[2]], ['L', toPathEnd[1], toPathEnd[2]], ['L', toPathStart[1], toPathStart[2]], ['Z']); - } - result.isFlat = isFlat; - } - } - else { // outside the axis area - path = null; - } - return result; - }, - /** - * Add a plot band after render time. - * - * @sample highcharts/members/axis-addplotband/ - * Toggle the plot band from a button - * - * @function Highcharts.Axis#addPlotBand - * - * @param {Highcharts.AxisPlotBandsOptions} options - * A configuration object for the plot band, as defined in - * [xAxis.plotBands](https://api.highcharts.com/highcharts/xAxis.plotBands). - * - * @return {Highcharts.PlotLineOrBand|undefined} - * The added plot band. - */ - addPlotBand: function (options) { - return this.addPlotBandOrLine(options, 'plotBands'); - }, - /** - * Add a plot line after render time. - * - * @sample highcharts/members/axis-addplotline/ - * Toggle the plot line from a button - * - * @function Highcharts.Axis#addPlotLine - * - * @param {Highcharts.AxisPlotLinesOptions} options - * A configuration object for the plot line, as defined in - * [xAxis.plotLines](https://api.highcharts.com/highcharts/xAxis.plotLines). - * - * @return {Highcharts.PlotLineOrBand|undefined} - * The added plot line. - */ - addPlotLine: function (options) { - return this.addPlotBandOrLine(options, 'plotLines'); - }, - /** - * Add a plot band or plot line after render time. Called from addPlotBand - * and addPlotLine internally. - * - * @private - * @function Highcharts.Axis#addPlotBandOrLine - * - * @param {Highcharts.AxisPlotBandsOptions|Highcharts.AxisPlotLinesOptions} options - * The plotBand or plotLine configuration object. - * - * @param {"plotBands"|"plotLines"} [coll] - * - * @return {Highcharts.PlotLineOrBand|undefined} - */ - addPlotBandOrLine: function (options, coll) { - var obj = new H.PlotLineOrBand(this, - options), - userOptions = this.userOptions; - if (this.visible) { - obj = obj.render(); - } - if (obj) { // #2189 - // Add it to the user options for exporting and Axis.update - if (coll) { - // Workaround Microsoft/TypeScript issue #32693 - var updatedOptions = (userOptions[coll] || []); - updatedOptions.push(options); - userOptions[coll] = updatedOptions; - } - this.plotLinesAndBands.push(obj); - this._addedPlotLB = true; - } - return obj; - }, - /** - * Remove a plot band or plot line from the chart by id. Called internally - * from `removePlotBand` and `removePlotLine`. - * - * @private - * @function Highcharts.Axis#removePlotBandOrLine - * @param {string} id - * @return {void} - */ - removePlotBandOrLine: function (id) { - var plotLinesAndBands = this.plotLinesAndBands, - options = this.options, - userOptions = this.userOptions, - i = plotLinesAndBands.length; - while (i--) { - if (plotLinesAndBands[i].id === id) { - plotLinesAndBands[i].destroy(); - } - } - ([ - options.plotLines || [], - userOptions.plotLines || [], - options.plotBands || [], - userOptions.plotBands || [] - ]).forEach(function (arr) { - i = arr.length; - while (i--) { - if ((arr[i] || {}).id === id) { - erase(arr, arr[i]); - } - } - }); - }, - /** - * Remove a plot band by its id. - * - * @sample highcharts/members/axis-removeplotband/ - * Remove plot band by id - * @sample highcharts/members/axis-addplotband/ - * Toggle the plot band from a button - * - * @function Highcharts.Axis#removePlotBand - * - * @param {string} id - * The plot band's `id` as given in the original configuration - * object or in the `addPlotBand` option. - * - * @return {void} - */ - removePlotBand: function (id) { - this.removePlotBandOrLine(id); - }, - /** - * Remove a plot line by its id. - * - * @sample highcharts/xaxis/plotlines-id/ - * Remove plot line by id - * @sample highcharts/members/axis-addplotline/ - * Toggle the plot line from a button - * - * @function Highcharts.Axis#removePlotLine - * - * @param {string} id - * The plot line's `id` as given in the original configuration - * object or in the `addPlotLine` option. - */ - removePlotLine: function (id) { - this.removePlotBandOrLine(id); - } - }); - H.PlotLineOrBand = PlotLineOrBand; - - return H.PlotLineOrBand; - }); - _registerModule(_modules, 'Core/Tooltip.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var doc = H.doc; - var clamp = U.clamp, - css = U.css, - defined = U.defined, - discardElement = U.discardElement, - extend = U.extend, - fireEvent = U.fireEvent, - format = U.format, - isNumber = U.isNumber, - isString = U.isString, - merge = U.merge, - pick = U.pick, - splat = U.splat, - syncTimeout = U.syncTimeout, - timeUnits = U.timeUnits; - /** - * Callback function to format the text of the tooltip from scratch. - * - * In case of single or shared tooltips, a string should be be returned. In case - * of splitted tooltips, it should return an array where the first item is the - * header, and subsequent items are mapped to the points. Return `false` to - * disable tooltip for a specific point on series. - * - * @callback Highcharts.TooltipFormatterCallbackFunction - * - * @param {Highcharts.TooltipFormatterContextObject} this - * Context to format - * - * @param {Highcharts.Tooltip} tooltip - * The tooltip instance - * - * @return {false|string|Array<(string|null|undefined)>|null|undefined} - * Formatted text or false - */ - /** - * @interface Highcharts.TooltipFormatterContextObject - */ /** - * @name Highcharts.TooltipFormatterContextObject#color - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ /** - * @name Highcharts.TooltipFormatterContextObject#colorIndex - * @type {number|undefined} - */ /** - * @name Highcharts.TooltipFormatterContextObject#key - * @type {number} - */ /** - * @name Highcharts.TooltipFormatterContextObject#percentage - * @type {number|undefined} - */ /** - * @name Highcharts.TooltipFormatterContextObject#point - * @type {Highcharts.Point} - */ /** - * @name Highcharts.TooltipFormatterContextObject#points - * @type {Array<Highcharts.TooltipFormatterContextObject>|undefined} - */ /** - * @name Highcharts.TooltipFormatterContextObject#series - * @type {Highcharts.Series} - */ /** - * @name Highcharts.TooltipFormatterContextObject#total - * @type {number|undefined} - */ /** - * @name Highcharts.TooltipFormatterContextObject#x - * @type {number} - */ /** - * @name Highcharts.TooltipFormatterContextObject#y - * @type {number} - */ - /** - * A callback function to place the tooltip in a specific position. - * - * @callback Highcharts.TooltipPositionerCallbackFunction - * - * @param {Highcharts.Tooltip} this - * Tooltip context of the callback. - * - * @param {number} labelWidth - * Width of the tooltip. - * - * @param {number} labelHeight - * Height of the tooltip. - * - * @param {Highcharts.TooltipPositionerPointObject} point - * Point information for positioning a tooltip. - * - * @return {Highcharts.PositionObject} - * New position for the tooltip. - */ - /** - * Point information for positioning a tooltip. - * - * @interface Highcharts.TooltipPositionerPointObject - * @extends Highcharts.Point - */ /** - * If `tooltip.split` option is enabled and positioner is called for each of the - * boxes separately, this property indicates the call on the xAxis header, which - * is not a point itself. - * @name Highcharts.TooltipPositionerPointObject#isHeader - * @type {boolean} - */ /** - * The reference point relative to the plot area. Add chart.plotLeft to get the - * full coordinates. - * @name Highcharts.TooltipPositionerPointObject#plotX - * @type {number} - */ /** - * The reference point relative to the plot area. Add chart.plotTop to get the - * full coordinates. - * @name Highcharts.TooltipPositionerPointObject#plotY - * @type {number} - */ - /** - * @typedef {"callout"|"circle"|"square"} Highcharts.TooltipShapeValue - */ - ''; // separates doclets above from variables below - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * Tooltip of a chart. - * - * @class - * @name Highcharts.Tooltip - * - * @param {Highcharts.Chart} chart - * The chart instance. - * - * @param {Highcharts.TooltipOptions} options - * Tooltip options. - */ - var Tooltip = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function Tooltip(chart, options) { - this.container = void 0; - this.crosshairs = []; - this.distance = 0; - this.isHidden = true; - this.isSticky = false; - this.now = {}; - this.options = {}; - this.outside = false; - this.chart = chart; - this.init(chart, options); - } - /* * - * - * Functions - * - * */ - /** - * In styled mode, apply the default filter for the tooltip drop-shadow. It - * needs to have an id specific to the chart, otherwise there will be issues - * when one tooltip adopts the filter of a different chart, specifically one - * where the container is hidden. - * - * @private - * @function Highcharts.Tooltip#applyFilter - */ - Tooltip.prototype.applyFilter = function () { - var chart = this.chart; - chart.renderer.definition({ - tagName: 'filter', - id: 'drop-shadow-' + chart.index, - opacity: 0.5, - children: [{ - tagName: 'feGaussianBlur', - 'in': 'SourceAlpha', - stdDeviation: 1 - }, { - tagName: 'feOffset', - dx: 1, - dy: 1 - }, { - tagName: 'feComponentTransfer', - children: [{ - tagName: 'feFuncA', - type: 'linear', - slope: 0.3 - }] - }, { - tagName: 'feMerge', - children: [{ - tagName: 'feMergeNode' - }, { - tagName: 'feMergeNode', - 'in': 'SourceGraphic' - }] - }] - }); - chart.renderer.definition({ - tagName: 'style', - textContent: '.highcharts-tooltip-' + chart.index + '{' + - 'filter:url(#drop-shadow-' + chart.index + ')' + - '}' - }); - }; - /** - * Build the body (lines) of the tooltip by iterating over the items and - * returning one entry for each item, abstracting this functionality allows - * to easily overwrite and extend it. - * - * @private - * @function Highcharts.Tooltip#bodyFormatter - * @param {Array<(Highcharts.Point|Highcharts.Series)>} items - * @return {Array<string>} - */ - Tooltip.prototype.bodyFormatter = function (items) { - return items.map(function (item) { - var tooltipOptions = item.series.tooltipOptions; - return (tooltipOptions[(item.point.formatPrefix || 'point') + 'Formatter'] || - item.point.tooltipFormatter).call(item.point, tooltipOptions[(item.point.formatPrefix || 'point') + 'Format'] || ''); - }); - }; - /** - * Destroy the single tooltips in a split tooltip. - * If the tooltip is active then it is not destroyed, unless forced to. - * - * @private - * @function Highcharts.Tooltip#cleanSplit - * - * @param {boolean} [force] - * Force destroy all tooltips. - */ - Tooltip.prototype.cleanSplit = function (force) { - this.chart.series.forEach(function (series) { - var tt = series && series.tt; - if (tt) { - if (!tt.isActive || force) { - series.tt = tt.destroy(); - } - else { - tt.isActive = false; - } - } - }); - }; - /** - * In case no user defined formatter is given, this will be used. Note that - * the context here is an object holding point, series, x, y etc. - * - * @function Highcharts.Tooltip#defaultFormatter - * - * @param {Highcharts.Tooltip} tooltip - * - * @return {Array<string>} - */ - Tooltip.prototype.defaultFormatter = function (tooltip) { - var items = this.points || splat(this), - s; - // Build the header - s = [tooltip.tooltipFooterHeaderFormatter(items[0])]; - // build the values - s = s.concat(tooltip.bodyFormatter(items)); - // footer - s.push(tooltip.tooltipFooterHeaderFormatter(items[0], true)); - return s; - }; - /** - * Removes and destroys the tooltip and its elements. - * - * @function Highcharts.Tooltip#destroy - */ - Tooltip.prototype.destroy = function () { - // Destroy and clear local variables - if (this.label) { - this.label = this.label.destroy(); - } - if (this.split && this.tt) { - this.cleanSplit(this.chart, true); - this.tt = this.tt.destroy(); - } - if (this.renderer) { - this.renderer = this.renderer.destroy(); - discardElement(this.container); - } - U.clearTimeout(this.hideTimer); - U.clearTimeout(this.tooltipTimeout); - }; - /** - * Extendable method to get the anchor position of the tooltip - * from a point or set of points - * - * @private - * @function Highcharts.Tooltip#getAnchor - * - * @param {Highcharts.Point|Array<Highcharts.Point>} points - * - * @param {Highcharts.PointerEventObject} [mouseEvent] - * - * @return {Array<number>} - */ - Tooltip.prototype.getAnchor = function (points, mouseEvent) { - var ret, - chart = this.chart, - pointer = chart.pointer, - inverted = chart.inverted, - plotTop = chart.plotTop, - plotLeft = chart.plotLeft, - plotX = 0, - plotY = 0, - yAxis, - xAxis; - points = splat(points); - // When tooltip follows mouse, relate the position to the mouse - if (this.followPointer && mouseEvent) { - if (typeof mouseEvent.chartX === 'undefined') { - mouseEvent = pointer.normalize(mouseEvent); - } - ret = [ - mouseEvent.chartX - plotLeft, - mouseEvent.chartY - plotTop - ]; - // Some series types use a specificly calculated tooltip position for - // each point - } - else if (points[0].tooltipPos) { - ret = points[0].tooltipPos; - // When shared, use the average position - } - else { - points.forEach(function (point) { - yAxis = point.series.yAxis; - xAxis = point.series.xAxis; - plotX += point.plotX + - (!inverted && xAxis ? xAxis.left - plotLeft : 0); - plotY += (point.plotLow ? - (point.plotLow + point.plotHigh) / 2 : - point.plotY) + (!inverted && yAxis ? yAxis.top - plotTop : 0); // #1151 - }); - plotX /= points.length; - plotY /= points.length; - ret = [ - inverted ? chart.plotWidth - plotY : plotX, - this.shared && !inverted && points.length > 1 && mouseEvent ? - // place shared tooltip next to the mouse (#424) - mouseEvent.chartY - plotTop : - inverted ? chart.plotHeight - plotX : plotY - ]; - } - return ret.map(Math.round); - }; - /** - * Get the optimal date format for a point, based on a range. - * - * @private - * @function Highcharts.Tooltip#getDateFormat - * - * @param {number} range - * The time range - * - * @param {number} date - * The date of the point in question - * - * @param {number} startOfWeek - * An integer representing the first day of the week, where 0 is - * Sunday. - * - * @param {Highcharts.Dictionary<string>} dateTimeLabelFormats - * A map of time units to formats. - * - * @return {string} - * The optimal date format for a point. - */ - Tooltip.prototype.getDateFormat = function (range, date, startOfWeek, dateTimeLabelFormats) { - var time = this.chart.time, dateStr = time.dateFormat('%m-%d %H:%M:%S.%L', date), format, n, blank = '01-01 00:00:00.000', strpos = { - millisecond: 15, - second: 12, - minute: 9, - hour: 6, - day: 3 - }, lastN = 'millisecond'; // for sub-millisecond data, #4223 - for (n in timeUnits) { // eslint-disable-line guard-for-in - // If the range is exactly one week and we're looking at a - // Sunday/Monday, go for the week format - if (range === timeUnits.week && - +time.dateFormat('%w', date) === startOfWeek && - dateStr.substr(6) === blank.substr(6)) { - n = 'week'; - break; - } - // The first format that is too great for the range - if (timeUnits[n] > range) { - n = lastN; - break; - } - // If the point is placed every day at 23:59, we need to show - // the minutes as well. #2637. - if (strpos[n] && - dateStr.substr(strpos[n]) !== blank.substr(strpos[n])) { - break; - } - // Weeks are outside the hierarchy, only apply them on - // Mondays/Sundays like in the first condition - if (n !== 'week') { - lastN = n; - } - } - if (n) { - format = time.resolveDTLFormat(dateTimeLabelFormats[n]).main; - } - return format; - }; - /** - * Creates the Tooltip label element if it does not exist, then returns it. - * - * @function Highcharts.Tooltip#getLabel - * @return {Highcharts.SVGElement} - */ - Tooltip.prototype.getLabel = function () { - var _a, - _b; - var tooltip = this, - renderer = this.chart.renderer, - styledMode = this.chart.styledMode, - options = this.options, - className = ('tooltip' + (defined(options.className) ? - ' ' + options.className : - '')), - pointerEvents = (((_a = options.style) === null || _a === void 0 ? void 0 : _a.pointerEvents) || - (!this.followPointer && options.stickOnContact ? 'auto' : 'none')), - container, - set, - onMouseEnter = function () { - tooltip.inContact = true; - }, onMouseLeave = function () { - var series = tooltip.chart.hoverSeries; - tooltip.inContact = false; - if (series && - series.onMouseOut) { - series.onMouseOut(); - } - }; - if (!this.label) { - if (this.outside) { - /** - * Reference to the tooltip's container, when - * [Highcharts.Tooltip#outside] is set to true, otherwise - * it's undefined. - * - * @name Highcharts.Tooltip#container - * @type {Highcharts.HTMLDOMElement|undefined} - */ - this.container = container = H.doc.createElement('div'); - container.className = 'highcharts-tooltip-container'; - css(container, { - position: 'absolute', - top: '1px', - pointerEvents: pointerEvents, - zIndex: 3 - }); - H.doc.body.appendChild(container); - /** - * Reference to the tooltip's renderer, when - * [Highcharts.Tooltip#outside] is set to true, otherwise - * it's undefined. - * - * @name Highcharts.Tooltip#renderer - * @type {Highcharts.SVGRenderer|undefined} - */ - this.renderer = renderer = new H.Renderer(container, 0, 0, (_b = this.chart.options.chart) === null || _b === void 0 ? void 0 : _b.style, void 0, void 0, renderer.styledMode); - } - // Create the label - if (this.split) { - this.label = renderer.g(className); - } - else { - this.label = renderer - .label('', 0, 0, options.shape || 'callout', null, null, options.useHTML, null, className) - .attr({ - padding: options.padding, - r: options.borderRadius - }); - if (!styledMode) { - this.label - .attr({ - fill: options.backgroundColor, - 'stroke-width': options.borderWidth - }) - // #2301, #2657 - .css(options.style) - .css({ pointerEvents: pointerEvents }) - .shadow(options.shadow); - } - } - if (styledMode) { - // Apply the drop-shadow filter - this.applyFilter(); - this.label.addClass('highcharts-tooltip-' + this.chart.index); - } - // Split tooltip use updateTooltipContainer to position the tooltip - // container. - if (tooltip.outside && !tooltip.split) { - var label_1 = this.label; - var xSetter_1 = label_1.xSetter, - ySetter_1 = label_1.ySetter; - label_1.xSetter = function (value) { - xSetter_1.call(label_1, tooltip.distance); - container.style.left = value + 'px'; - }; - label_1.ySetter = function (value) { - ySetter_1.call(label_1, tooltip.distance); - container.style.top = value + 'px'; - }; - } - this.label - .on('mouseenter', onMouseEnter) - .on('mouseleave', onMouseLeave) - .attr({ zIndex: 8 }) - .add(); - } - return this.label; - }; - /** - * Place the tooltip in a chart without spilling over - * and not covering the point it self. - * - * @private - * @function Highcharts.Tooltip#getPosition - * - * @param {number} boxWidth - * - * @param {number} boxHeight - * - * @param {Highcharts.Point} point - * - * @return {Highcharts.PositionObject} - */ - Tooltip.prototype.getPosition = function (boxWidth, boxHeight, point) { - var chart = this.chart, - distance = this.distance, - ret = {}, - // Don't use h if chart isn't inverted (#7242) ??? - h = (chart.inverted && point.h) || 0, // #4117 ??? - swapped, - outside = this.outside, - outerWidth = outside ? - // substract distance to prevent scrollbars - doc.documentElement.clientWidth - 2 * distance : - chart.chartWidth, - outerHeight = outside ? - Math.max(doc.body.scrollHeight, - doc.documentElement.scrollHeight, - doc.body.offsetHeight, - doc.documentElement.offsetHeight, - doc.documentElement.clientHeight) : - chart.chartHeight, - chartPosition = chart.pointer.getChartPosition(), - containerScaling = chart.containerScaling, - scaleX = function (val) { return ( // eslint-disable-line no-confusing-arrow - containerScaling ? val * containerScaling.scaleX : val); }, - scaleY = function (val) { return ( // eslint-disable-line no-confusing-arrow - containerScaling ? val * containerScaling.scaleY : val); }, - // Build parameter arrays for firstDimension()/secondDimension() - buildDimensionArray = function (dim) { - var isX = dim === 'x'; - return [ - dim, - isX ? outerWidth : outerHeight, - isX ? boxWidth : boxHeight - ].concat(outside ? [ - // If we are using tooltip.outside, we need to scale the - // position to match scaling of the container in case there - // is a transform/zoom on the container. #11329 - isX ? scaleX(boxWidth) : scaleY(boxHeight), - isX ? chartPosition.left - distance + - scaleX(point.plotX + chart.plotLeft) : - chartPosition.top - distance + - scaleY(point.plotY + chart.plotTop), - 0, - isX ? outerWidth : outerHeight - ] : [ - // Not outside, no scaling is needed - isX ? boxWidth : boxHeight, - isX ? point.plotX + chart.plotLeft : - point.plotY + chart.plotTop, - isX ? chart.plotLeft : chart.plotTop, - isX ? chart.plotLeft + chart.plotWidth : - chart.plotTop + chart.plotHeight - ]); - }, first = buildDimensionArray('y'), second = buildDimensionArray('x'), - // The far side is right or bottom - preferFarSide = !this.followPointer && pick(point.ttBelow, !chart.inverted === !!point.negative), // #4984 - /* - * Handle the preferred dimension. When the preferred dimension is - * tooltip on top or bottom of the point, it will look for space - * there. - * - * @private - */ - firstDimension = function (dim, outerSize, innerSize, scaledInnerSize, // #11329 - point, min, max) { - var scaledDist = dim === 'y' ? - scaleY(distance) : scaleX(distance), - scaleDiff = (innerSize - scaledInnerSize) / 2, - roomLeft = scaledInnerSize < point - distance, - roomRight = point + distance + scaledInnerSize < outerSize, - alignedLeft = point - scaledDist - innerSize + scaleDiff, - alignedRight = point + scaledDist - scaleDiff; - if (preferFarSide && roomRight) { - ret[dim] = alignedRight; - } - else if (!preferFarSide && roomLeft) { - ret[dim] = alignedLeft; - } - else if (roomLeft) { - ret[dim] = Math.min(max - scaledInnerSize, alignedLeft - h < 0 ? alignedLeft : alignedLeft - h); - } - else if (roomRight) { - ret[dim] = Math.max(min, alignedRight + h + innerSize > outerSize ? - alignedRight : - alignedRight + h); - } - else { - return false; - } - }, - /* - * Handle the secondary dimension. If the preferred dimension is - * tooltip on top or bottom of the point, the second dimension is to - * align the tooltip above the point, trying to align center but - * allowing left or right align within the chart box. - * - * @private - */ - secondDimension = function (dim, outerSize, innerSize, scaledInnerSize, // #11329 - point) { - var retVal; - // Too close to the edge, return false and swap dimensions - if (point < distance || point > outerSize - distance) { - retVal = false; - // Align left/top - } - else if (point < innerSize / 2) { - ret[dim] = 1; - // Align right/bottom - } - else if (point > outerSize - scaledInnerSize / 2) { - ret[dim] = outerSize - scaledInnerSize - 2; - // Align center - } - else { - ret[dim] = point - innerSize / 2; - } - return retVal; - }, - /* - * Swap the dimensions - */ - swap = function (count) { - var temp = first; - first = second; - second = temp; - swapped = count; - }, run = function () { - if (firstDimension.apply(0, first) !== false) { - if (secondDimension.apply(0, second) === false && - !swapped) { - swap(true); - run(); - } - } - else if (!swapped) { - swap(true); - run(); - } - else { - ret.x = ret.y = 0; - } - }; - // Under these conditions, prefer the tooltip on the side of the point - if (chart.inverted || this.len > 1) { - swap(); - } - run(); - return ret; - }; - /** - * Get the best X date format based on the closest point range on the axis. - * - * @private - * @function Highcharts.Tooltip#getXDateFormat - * - * @param {Highcharts.Point} point - * - * @param {Highcharts.TooltipOptions} options - * - * @param {Highcharts.Axis} xAxis - * - * @return {string} - */ - Tooltip.prototype.getXDateFormat = function (point, options, xAxis) { - var xDateFormat, - dateTimeLabelFormats = options.dateTimeLabelFormats, - closestPointRange = xAxis && xAxis.closestPointRange; - if (closestPointRange) { - xDateFormat = this.getDateFormat(closestPointRange, point.x, xAxis.options.startOfWeek, dateTimeLabelFormats); - } - else { - xDateFormat = dateTimeLabelFormats.day; - } - return xDateFormat || dateTimeLabelFormats.year; // #2546, 2581 - }; - /** - * Hides the tooltip with a fade out animation. - * - * @function Highcharts.Tooltip#hide - * - * @param {number} [delay] - * The fade out in milliseconds. If no value is provided the value - * of the tooltip.hideDelay option is used. A value of 0 disables - * the fade out animation. - */ - Tooltip.prototype.hide = function (delay) { - var tooltip = this; - // disallow duplicate timers (#1728, #1766) - U.clearTimeout(this.hideTimer); - delay = pick(delay, this.options.hideDelay, 500); - if (!this.isHidden) { - this.hideTimer = syncTimeout(function () { - // If there is a delay, do fadeOut with the default duration. If - // the hideDelay is 0, we assume no animation is wanted, so we - // pass 0 duration. #12994. - tooltip.getLabel().fadeOut(delay ? void 0 : delay); - tooltip.isHidden = true; - }, delay); - } - }; - /** - * @private - * @function Highcharts.Tooltip#init - * - * @param {Highcharts.Chart} chart - * The chart instance. - * - * @param {Highcharts.TooltipOptions} options - * Tooltip options. - */ - Tooltip.prototype.init = function (chart, options) { - /** - * Chart of the tooltip. - * - * @readonly - * @name Highcharts.Tooltip#chart - * @type {Highcharts.Chart} - */ - this.chart = chart; - /** - * Used tooltip options. - * - * @readonly - * @name Highcharts.Tooltip#options - * @type {Highcharts.TooltipOptions} - */ - this.options = options; - /** - * List of crosshairs. - * - * @private - * @readonly - * @name Highcharts.Tooltip#crosshairs - * @type {Array<null>} - */ - this.crosshairs = []; - /** - * Current values of x and y when animating. - * - * @private - * @readonly - * @name Highcharts.Tooltip#now - * @type {Highcharts.PositionObject} - */ - this.now = { x: 0, y: 0 }; - /** - * Tooltips are initially hidden. - * - * @private - * @readonly - * @name Highcharts.Tooltip#isHidden - * @type {boolean} - */ - this.isHidden = true; - /** - * True, if the tooltip is split into one label per series, with the - * header close to the axis. - * - * @readonly - * @name Highcharts.Tooltip#split - * @type {boolean|undefined} - */ - this.split = options.split && !chart.inverted && !chart.polar; - /** - * When the tooltip is shared, the entire plot area will capture mouse - * movement or touch events. - * - * @readonly - * @name Highcharts.Tooltip#shared - * @type {boolean|undefined} - */ - this.shared = options.shared || this.split; - /** - * Whether to allow the tooltip to render outside the chart's SVG - * element box. By default (false), the tooltip is rendered within the - * chart's SVG element, which results in the tooltip being aligned - * inside the chart area. - * - * @readonly - * @name Highcharts.Tooltip#outside - * @type {boolean} - * - * @todo - * Split tooltip does not support outside in the first iteration. Should - * not be too complicated to implement. - */ - this.outside = pick(options.outside, Boolean(chart.scrollablePixelsX || chart.scrollablePixelsY)); - }; - /** - * Returns true, if the pointer is in contact with the tooltip tracker. - */ - Tooltip.prototype.isStickyOnContact = function () { - return !!(!this.followPointer && - this.options.stickOnContact && - this.inContact); - }; - /** - * Moves the tooltip with a soft animation to a new position. - * - * @private - * @function Highcharts.Tooltip#move - * - * @param {number} x - * - * @param {number} y - * - * @param {number} anchorX - * - * @param {number} anchorY - */ - Tooltip.prototype.move = function (x, y, anchorX, anchorY) { - var tooltip = this, - now = tooltip.now, - animate = tooltip.options.animation !== false && - !tooltip.isHidden && - // When we get close to the target position, abort animation and - // land on the right place (#3056) - (Math.abs(x - now.x) > 1 || Math.abs(y - now.y) > 1), - skipAnchor = tooltip.followPointer || tooltip.len > 1; - // Get intermediate values for animation - extend(now, { - x: animate ? (2 * now.x + x) / 3 : x, - y: animate ? (now.y + y) / 2 : y, - anchorX: skipAnchor ? - void 0 : - animate ? (2 * now.anchorX + anchorX) / 3 : anchorX, - anchorY: skipAnchor ? - void 0 : - animate ? (now.anchorY + anchorY) / 2 : anchorY - }); - // Move to the intermediate value - tooltip.getLabel().attr(now); - tooltip.drawTracker(); - // Run on next tick of the mouse tracker - if (animate) { - // Never allow two timeouts - U.clearTimeout(this.tooltipTimeout); - // Set the fixed interval ticking for the smooth tooltip - this.tooltipTimeout = setTimeout(function () { - // The interval function may still be running during destroy, - // so check that the chart is really there before calling. - if (tooltip) { - tooltip.move(x, y, anchorX, anchorY); - } - }, 32); - } - }; - /** - * Refresh the tooltip's text and position. - * - * @function Highcharts.Tooltip#refresh - * - * @param {Highcharts.Point|Array<Highcharts.Point>} pointOrPoints - * Either a point or an array of points. - * - * @param {Highcharts.PointerEventObject} [mouseEvent] - * Mouse event, that is responsible for the refresh and should be - * used for the tooltip update. - */ - Tooltip.prototype.refresh = function (pointOrPoints, mouseEvent) { - var tooltip = this, - chart = this.chart, - options = tooltip.options, - x, - y, - point = pointOrPoints, - anchor, - textConfig = {}, - text, - pointConfig = [], - formatter = options.formatter || tooltip.defaultFormatter, - shared = tooltip.shared, - currentSeries, - styledMode = chart.styledMode; - if (!options.enabled) { - return; - } - U.clearTimeout(this.hideTimer); - // get the reference point coordinates (pie charts use tooltipPos) - tooltip.followPointer = splat(point)[0].series.tooltipOptions - .followPointer; - anchor = tooltip.getAnchor(point, mouseEvent); - x = anchor[0]; - y = anchor[1]; - // shared tooltip, array is sent over - if (shared && - !(point.series && - point.series.noSharedTooltip)) { - chart.pointer.applyInactiveState(point); - // Now set hover state for the choosen ones: - point.forEach(function (item) { - item.setState('hover'); - pointConfig.push(item.getLabelConfig()); - }); - textConfig = { - x: point[0].category, - y: point[0].y - }; - textConfig.points = pointConfig; - point = point[0]; - // single point tooltip - } - else { - textConfig = point.getLabelConfig(); - } - this.len = pointConfig.length; // #6128 - text = formatter.call(textConfig, tooltip); - // register the current series - currentSeries = point.series; - this.distance = pick(currentSeries.tooltipOptions.distance, 16); - // update the inner HTML - if (text === false) { - this.hide(); - } - else { - // update text - if (tooltip.split) { - this.renderSplit(text, splat(pointOrPoints)); - } - else { - var label = tooltip.getLabel(); - // Prevent the tooltip from flowing over the chart box (#6659) - if (!options.style.width || styledMode) { - label.css({ - width: this.chart.spacingBox.width + 'px' - }); - } - label.attr({ - text: text && text.join ? - text.join('') : - text - }); - // Set the stroke color of the box to reflect the point - label.removeClass(/highcharts-color-[\d]+/g) - .addClass('highcharts-color-' + - pick(point.colorIndex, currentSeries.colorIndex)); - if (!styledMode) { - label.attr({ - stroke: (options.borderColor || - point.color || - currentSeries.color || - '#666666') - }); - } - tooltip.updatePosition({ - plotX: x, - plotY: y, - negative: point.negative, - ttBelow: point.ttBelow, - h: anchor[2] || 0 - }); - } - // show it - if (tooltip.isHidden && tooltip.label) { - tooltip.label.attr({ - opacity: 1 - }).show(); - } - tooltip.isHidden = false; - } - fireEvent(this, 'refresh'); - }; - /** - * Render the split tooltip. Loops over each point's text and adds - * a label next to the point, then uses the distribute function to - * find best non-overlapping positions. - * - * @private - * @function Highcharts.Tooltip#renderSplit - * - * @param {string|Array<(boolean|string)>} labels - * - * @param {Array<Highcharts.Point>} points - */ - Tooltip.prototype.renderSplit = function (labels, points) { - var tooltip = this; - var chart = tooltip.chart, - _a = tooltip.chart, - chartWidth = _a.chartWidth, - chartHeight = _a.chartHeight, - plotHeight = _a.plotHeight, - plotLeft = _a.plotLeft, - plotTop = _a.plotTop, - pointer = _a.pointer, - ren = _a.renderer, - _b = _a.scrollablePixelsY, - scrollablePixelsY = _b === void 0 ? 0 : _b, - _c = _a.scrollingContainer, - _d = _c === void 0 ? { scrollLeft: 0, - scrollTop: 0 } : _c, - scrollLeft = _d.scrollLeft, - scrollTop = _d.scrollTop, - styledMode = _a.styledMode, - distance = tooltip.distance, - options = tooltip.options, - positioner = tooltip.options.positioner; - // The area which the tooltip should be limited to. Limit to scrollable - // plot area if enabled, otherwise limit to the chart container. - var bounds = { - left: scrollLeft, - right: scrollLeft + chartWidth, - top: scrollTop, - bottom: scrollTop + chartHeight - }; - var tooltipLabel = tooltip.getLabel(); - var headerTop = Boolean(chart.xAxis[0] && chart.xAxis[0].opposite); - var distributionBoxTop = plotTop + scrollTop; - var headerHeight = 0; - var adjustedPlotHeight = plotHeight - scrollablePixelsY; - /** - * Calculates the anchor position for the partial tooltip - * - * @private - * @param {Highcharts.Point} point The point related to the tooltip - * @return {object} Returns an object with anchorX and anchorY - */ - function getAnchor(point) { - var isHeader = point.isHeader, - _a = point.plotX, - plotX = _a === void 0 ? 0 : _a, - _b = point.plotY, - plotY = _b === void 0 ? 0 : _b, - series = point.series; - var anchorX; - var anchorY; - if (isHeader) { - // Set anchorX to plotX - anchorX = plotLeft + plotX; - // Set anchorY to center of visible plot area. - anchorY = plotTop + plotHeight / 2; - } - else { - var xAxis = series.xAxis, - yAxis = series.yAxis; - // Set anchorX to plotX. Limit to within xAxis. - anchorX = xAxis.pos + clamp(plotX, -distance, xAxis.len + distance); - // Set anchorY, limit to the scrollable plot area - if (yAxis.pos + plotY >= scrollTop + plotTop && - yAxis.pos + plotY <= scrollTop + plotTop + plotHeight - scrollablePixelsY) { - anchorY = yAxis.pos + plotY; - } - } - // Limit values to plot area - anchorX = clamp(anchorX, bounds.left - distance, bounds.right + distance); - return { anchorX: anchorX, anchorY: anchorY }; - } - /** - * Calculates the position of the partial tooltip - * - * @private - * @param {number} anchorX The partial tooltip anchor x position - * @param {number} anchorY The partial tooltip anchor y position - * @param {boolean} isHeader Whether the partial tooltip is a header - * @param {number} boxWidth Width of the partial tooltip - * @return {Highcharts.PositionObject} Returns the partial tooltip x and - * y position - */ - function defaultPositioner(anchorX, anchorY, isHeader, boxWidth, alignedLeft) { - if (alignedLeft === void 0) { alignedLeft = true; } - var y; - var x; - if (isHeader) { - y = headerTop ? 0 : adjustedPlotHeight; - x = clamp(anchorX - (boxWidth / 2), bounds.left, bounds.right - boxWidth); - } - else { - y = anchorY - distributionBoxTop; - x = alignedLeft ? - anchorX - boxWidth - distance : - anchorX + distance; - x = clamp(x, alignedLeft ? x : bounds.left, bounds.right); - } - // NOTE: y is relative to distributionBoxTop - return { x: x, y: y }; - } - /** - * Updates the attributes and styling of the partial tooltip. Creates a - * new partial tooltip if it does not exists. - * - * @private - * @param {Highcharts.SVGElement|undefined} partialTooltip - * The partial tooltip to update - * @param {Highcharts.Point} point - * The point related to the partial tooltip - * @param {boolean|string} str The text for the partial tooltip - * @return {Highcharts.SVGElement} Returns the updated partial tooltip - */ - function updatePartialTooltip(partialTooltip, point, str) { - var tt = partialTooltip; - var isHeader = point.isHeader, - series = point.series; - var colorClass = 'highcharts-color-' + pick(point.colorIndex, series.colorIndex, 'none'); - if (!tt) { - var attribs = { - padding: options.padding, - r: options.borderRadius - }; - if (!styledMode) { - attribs.fill = options.backgroundColor; - attribs['stroke-width'] = options.borderWidth; - } - tt = ren - .label('', 0, 0, (options[isHeader ? 'headerShape' : 'shape']) || - 'callout', void 0, void 0, options.useHTML) - .addClass((isHeader ? 'highcharts-tooltip-header ' : '') + - 'highcharts-tooltip-box ' + - colorClass) - .attr(attribs) - .add(tooltipLabel); - } - tt.isActive = true; - tt.attr({ - text: str - }); - if (!styledMode) { - tt.css(options.style) - .shadow(options.shadow) - .attr({ - stroke: (options.borderColor || - point.color || - series.color || - '#333333') - }); - } - return tt; - } - // Graceful degradation for legacy formatters - if (isString(labels)) { - labels = [false, labels]; - } - // Create the individual labels for header and points, ignore footer - var boxes = labels.slice(0, - points.length + 1).reduce(function (boxes, - str, - i) { - if (str !== false && str !== '') { - var point = (points[i - 1] || - { - // Item 0 is the header. Instead of this, we could also - // use the crosshair label - isHeader: true, - plotX: points[0].plotX, - plotY: plotHeight, - series: {} - }); - var isHeader = point.isHeader; - // Store the tooltip label referance on the series - var owner = isHeader ? tooltip : point.series; - var tt = owner.tt = updatePartialTooltip(owner.tt, - point, - str); - // Get X position now, so we can move all to the other side in - // case of overflow - var bBox = tt.getBBox(); - var boxWidth = bBox.width + tt.strokeWidth(); - if (isHeader) { - headerHeight = bBox.height; - adjustedPlotHeight += headerHeight; - if (headerTop) { - distributionBoxTop -= headerHeight; - } - } - var _a = getAnchor(point), - anchorX = _a.anchorX, - anchorY = _a.anchorY; - if (typeof anchorY === 'number') { - var size = bBox.height + 1; - var boxPosition = (positioner ? - positioner.call(tooltip, - boxWidth, - size, - point) : - defaultPositioner(anchorX, - anchorY, - isHeader, - boxWidth)); - boxes.push({ - // 0-align to the top, 1-align to the bottom - align: positioner ? 0 : void 0, - anchorX: anchorX, - anchorY: anchorY, - boxWidth: boxWidth, - point: point, - rank: pick(boxPosition.rank, isHeader ? 1 : 0), - size: size, - target: boxPosition.y, - tt: tt, - x: boxPosition.x - }); - } - else { - // Hide tooltips which anchorY is outside the visible plot - // area - tt.isActive = false; - } - } - return boxes; - }, []); - // If overflow left then align all labels to the right - if (!positioner && boxes.some(function (box) { return box.x < bounds.left; })) { - boxes = boxes.map(function (box) { - var _a = defaultPositioner(box.anchorX, - box.anchorY, - box.point.isHeader, - box.boxWidth, - false), - x = _a.x, - y = _a.y; - return extend(box, { - target: y, - x: x - }); - }); - } - // Clean previous run (for missing points) - tooltip.cleanSplit(); - // Distribute and put in place - H.distribute(boxes, adjustedPlotHeight); - boxes.forEach(function (box) { - var anchorX = box.anchorX, - anchorY = box.anchorY, - pos = box.pos, - x = box.x; - // Put the label in place - box.tt.attr({ - visibility: typeof pos === 'undefined' ? 'hidden' : 'inherit', - x: x, - /* NOTE: y should equal pos to be consistent with !split - * tooltip, but is currently relative to plotTop. Is left as is - * to avoid breaking change. Remove distributionBoxTop to make - * it consistent. - */ - y: pos + distributionBoxTop, - anchorX: anchorX, - anchorY: anchorY - }); - }); - /* If we have a seperate tooltip container, then update the necessary - * container properties. - * Test that tooltip has its own container and renderer before executing - * the operation. - */ - var container = tooltip.container, - outside = tooltip.outside, - renderer = tooltip.renderer; - if (outside && container && renderer) { - // Set container size to fit the tooltip - var _e = tooltipLabel.getBBox(), - width = _e.width, - height = _e.height, - x = _e.x, - y = _e.y; - renderer.setSize(width + x, height + y, false); - // Position the tooltip container to the chart container - var chartPosition = pointer.getChartPosition(); - container.style.left = chartPosition.left + 'px'; - container.style.top = chartPosition.top + 'px'; - } - }; - /** - * If the `stickOnContact` option is active, this will add a tracker shape. - * - * @private - * @function Highcharts.Tooltip#drawTracker - */ - Tooltip.prototype.drawTracker = function () { - var tooltip = this; - if (tooltip.followPointer || - !tooltip.options.stickOnContact) { - if (tooltip.tracker) { - tooltip.tracker.destroy(); - } - return; - } - var chart = tooltip.chart; - var label = tooltip.label; - var point = chart.hoverPoint; - if (!label || !point) { - return; - } - var box = { - x: 0, - y: 0, - width: 0, - height: 0 - }; - // Combine anchor and tooltip - var anchorPos = this.getAnchor(point); - var labelBBox = label.getBBox(); - anchorPos[0] += chart.plotLeft - label.translateX; - anchorPos[1] += chart.plotTop - label.translateY; - // When the mouse pointer is between the anchor point and the label, - // the label should stick. - box.x = Math.min(0, anchorPos[0]); - box.y = Math.min(0, anchorPos[1]); - box.width = (anchorPos[0] < 0 ? - Math.max(Math.abs(anchorPos[0]), (labelBBox.width - anchorPos[0])) : - Math.max(Math.abs(anchorPos[0]), labelBBox.width)); - box.height = (anchorPos[1] < 0 ? - Math.max(Math.abs(anchorPos[1]), (labelBBox.height - Math.abs(anchorPos[1]))) : - Math.max(Math.abs(anchorPos[1]), labelBBox.height)); - if (tooltip.tracker) { - tooltip.tracker.attr(box); - } - else { - tooltip.tracker = label.renderer - .rect(box) - .addClass('highcharts-tracker') - .add(label); - if (!chart.styledMode) { - tooltip.tracker.attr({ - fill: 'rgba(0,0,0,0)' - }); - } - } - }; - /** - * @private - */ - Tooltip.prototype.styledModeFormat = function (formatString) { - return formatString - .replace('style="font-size: 10px"', 'class="highcharts-header"') - .replace(/style="color:{(point|series)\.color}"/g, 'class="highcharts-color-{$1.colorIndex}"'); - }; - /** - * Format the footer/header of the tooltip - * #3397: abstraction to enable formatting of footer and header - * - * @private - * @function Highcharts.Tooltip#tooltipFooterHeaderFormatter - * @param {Highcharts.PointLabelObject} labelConfig - * @param {boolean} [isFooter] - * @return {string} - */ - Tooltip.prototype.tooltipFooterHeaderFormatter = function (labelConfig, isFooter) { - var footOrHead = isFooter ? 'footer' : 'header', - series = labelConfig.series, - tooltipOptions = series.tooltipOptions, - xDateFormat = tooltipOptions.xDateFormat, - xAxis = series.xAxis, - isDateTime = (xAxis && - xAxis.options.type === 'datetime' && - isNumber(labelConfig.key)), - formatString = tooltipOptions[footOrHead + 'Format'], - e = { - isFooter: isFooter, - labelConfig: labelConfig - }; - fireEvent(this, 'headerFormatter', e, function (e) { - // Guess the best date format based on the closest point distance - // (#568, #3418) - if (isDateTime && !xDateFormat) { - xDateFormat = this.getXDateFormat(labelConfig, tooltipOptions, xAxis); - } - // Insert the footer date format if any - if (isDateTime && xDateFormat) { - ((labelConfig.point && labelConfig.point.tooltipDateKeys) || - ['key']).forEach(function (key) { - formatString = formatString.replace('{point.' + key + '}', '{point.' + key + ':' + xDateFormat + '}'); - }); - } - // Replace default header style with class name - if (series.chart.styledMode) { - formatString = this.styledModeFormat(formatString); - } - e.text = format(formatString, { - point: labelConfig, - series: series - }, this.chart); - }); - return e.text; - }; - /** - * Updates the tooltip with the provided tooltip options. - * - * @function Highcharts.Tooltip#update - * - * @param {Highcharts.TooltipOptions} options - * The tooltip options to update. - */ - Tooltip.prototype.update = function (options) { - this.destroy(); - // Update user options (#6218) - merge(true, this.chart.options.tooltip.userOptions, options); - this.init(this.chart, merge(true, this.options, options)); - }; - /** - * Find the new position and perform the move - * - * @private - * @function Highcharts.Tooltip#updatePosition - * - * @param {Highcharts.Point} point - */ - Tooltip.prototype.updatePosition = function (point) { - var chart = this.chart, - pointer = chart.pointer, - label = this.getLabel(), - pos, - anchorX = point.plotX + chart.plotLeft, - anchorY = point.plotY + chart.plotTop, - pad; - // Needed for outside: true (#11688) - var chartPosition = pointer.getChartPosition(); - pos = (this.options.positioner || this.getPosition).call(this, label.width, label.height, point); - // Set the renderer size dynamically to prevent document size to change - if (this.outside) { - pad = (this.options.borderWidth || 0) + 2 * this.distance; - this.renderer.setSize(label.width + pad, label.height + pad, false); - // Anchor and tooltip container need scaling if chart container has - // scale transform/css zoom. #11329. - var containerScaling = chart.containerScaling; - if (containerScaling) { - css(this.container, { - transform: "scale(" + containerScaling.scaleX + ", " + containerScaling.scaleY + ")" - }); - anchorX *= containerScaling.scaleX; - anchorY *= containerScaling.scaleY; - } - anchorX += chartPosition.left - pos.x; - anchorY += chartPosition.top - pos.y; - } - // do the move - this.move(Math.round(pos.x), Math.round(pos.y || 0), // can be undefined (#3977) - anchorX, anchorY); - }; - return Tooltip; - }()); - H.Tooltip = Tooltip; - - return H.Tooltip; - }); - _registerModule(_modules, 'Core/Pointer.js', [_modules['Core/Color/Color.js'], _modules['Core/Globals.js'], _modules['Core/Tooltip.js'], _modules['Core/Utilities.js']], function (Color, H, Tooltip, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var color = Color.parse; - var charts = H.charts, - noop = H.noop; - var addEvent = U.addEvent, - attr = U.attr, - css = U.css, - defined = U.defined, - extend = U.extend, - find = U.find, - fireEvent = U.fireEvent, - isNumber = U.isNumber, - isObject = U.isObject, - objectEach = U.objectEach, - offset = U.offset, - pick = U.pick, - splat = U.splat; - /** - * One position in relation to an axis. - * - * @interface Highcharts.PointerAxisCoordinateObject - */ /** - * Related axis. - * - * @name Highcharts.PointerAxisCoordinateObject#axis - * @type {Highcharts.Axis} - */ /** - * Axis value. - * - * @name Highcharts.PointerAxisCoordinateObject#value - * @type {number} - */ - /** - * Positions in terms of axis values. - * - * @interface Highcharts.PointerAxisCoordinatesObject - */ /** - * Positions on the x-axis. - * @name Highcharts.PointerAxisCoordinatesObject#xAxis - * @type {Array<Highcharts.PointerAxisCoordinateObject>} - */ /** - * Positions on the y-axis. - * @name Highcharts.PointerAxisCoordinatesObject#yAxis - * @type {Array<Highcharts.PointerAxisCoordinateObject>} - */ - /** - * Pointer coordinates. - * - * @interface Highcharts.PointerCoordinatesObject - */ /** - * @name Highcharts.PointerCoordinatesObject#chartX - * @type {number} - */ /** - * @name Highcharts.PointerCoordinatesObject#chartY - * @type {number} - */ - /** - * A native browser mouse or touch event, extended with position information - * relative to the {@link Chart.container}. - * - * @interface Highcharts.PointerEventObject - * @extends global.PointerEvent - */ /** - * The X coordinate of the pointer interaction relative to the chart. - * - * @name Highcharts.PointerEventObject#chartX - * @type {number} - */ /** - * The Y coordinate of the pointer interaction relative to the chart. - * - * @name Highcharts.PointerEventObject#chartY - * @type {number} - */ - /** - * Axis-specific data of a selection. - * - * @interface Highcharts.SelectDataObject - */ /** - * @name Highcharts.SelectDataObject#axis - * @type {Highcharts.Axis} - */ /** - * @name Highcharts.SelectDataObject#max - * @type {number} - */ /** - * @name Highcharts.SelectDataObject#min - * @type {number} - */ - /** - * Object for select events. - * - * @interface Highcharts.SelectEventObject - */ /** - * @name Highcharts.SelectEventObject#originalEvent - * @type {global.Event} - */ /** - * @name Highcharts.SelectEventObject#xAxis - * @type {Array<Highcharts.SelectDataObject>} - */ /** - * @name Highcharts.SelectEventObject#yAxis - * @type {Array<Highcharts.SelectDataObject>} - */ - ''; // detach doclets above - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The mouse and touch tracker object. Each {@link Chart} item has one - * assosiated Pointer item that can be accessed from the {@link Chart.pointer} - * property. - * - * @class - * @name Highcharts.Pointer - * - * @param {Highcharts.Chart} chart - * The chart instance. - * - * @param {Highcharts.Options} options - * The root options object. The pointer uses options from the chart and - * tooltip structures. - */ - var Pointer = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function Pointer(chart, options) { - this.lastValidTouch = {}; - this.pinchDown = []; - this.runChartClick = false; - this.chart = chart; - this.hasDragged = false; - this.options = options; - this.unbindContainerMouseLeave = function () { }; - this.unbindContainerMouseEnter = function () { }; - this.init(chart, options); - } - /* * - * - * Functions - * - * */ - /** - * Set inactive state to all series that are not currently hovered, - * or, if `inactiveOtherPoints` is set to true, set inactive state to - * all points within that series. - * - * @private - * @function Highcharts.Pointer#applyInactiveState - * @param {Array<Highcharts.Point>} points - * Currently hovered points - */ - Pointer.prototype.applyInactiveState = function (points) { - var activeSeries = [], - series; - // Get all active series from the hovered points - (points || []).forEach(function (item) { - series = item.series; - // Include itself - activeSeries.push(series); - // Include parent series - if (series.linkedParent) { - activeSeries.push(series.linkedParent); - } - // Include all child series - if (series.linkedSeries) { - activeSeries = activeSeries.concat(series.linkedSeries); - } - // Include navigator series - if (series.navigatorSeries) { - activeSeries.push(series.navigatorSeries); - } - }); - // Now loop over all series, filtering out active series - this.chart.series.forEach(function (inactiveSeries) { - if (activeSeries.indexOf(inactiveSeries) === -1) { - // Inactive series - inactiveSeries.setState('inactive', true); - } - else if (inactiveSeries.options.inactiveOtherPoints) { - // Active series, but other points should be inactivated - inactiveSeries.setAllPointsToState('inactive'); - } - }); - }; - /** - * Destroys the Pointer object and disconnects DOM events. - * - * @function Highcharts.Pointer#destroy - */ - Pointer.prototype.destroy = function () { - var pointer = this; - if (typeof pointer.unDocMouseMove !== 'undefined') { - pointer.unDocMouseMove(); - } - this.unbindContainerMouseLeave(); - if (!H.chartCount) { - if (H.unbindDocumentMouseUp) { - H.unbindDocumentMouseUp = H.unbindDocumentMouseUp(); - } - if (H.unbindDocumentTouchEnd) { - H.unbindDocumentTouchEnd = H.unbindDocumentTouchEnd(); - } - } - // memory and CPU leak - clearInterval(pointer.tooltipTimeout); - objectEach(pointer, function (_val, prop) { - pointer[prop] = void 0; - }); - }; - /** - * Perform a drag operation in response to a mousemove event while the mouse - * is down. - * - * @private - * @function Highcharts.Pointer#drag - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - Pointer.prototype.drag = function (e) { - var chart = this.chart, - chartOptions = chart.options.chart, - chartX = e.chartX, - chartY = e.chartY, - zoomHor = this.zoomHor, - zoomVert = this.zoomVert, - plotLeft = chart.plotLeft, - plotTop = chart.plotTop, - plotWidth = chart.plotWidth, - plotHeight = chart.plotHeight, - clickedInside, - size, - selectionMarker = this.selectionMarker, - mouseDownX = (this.mouseDownX || 0), - mouseDownY = (this.mouseDownY || 0), - panningEnabled = isObject(chartOptions.panning) ? - chartOptions.panning && chartOptions.panning.enabled : - chartOptions.panning, - panKey = (chartOptions.panKey && e[chartOptions.panKey + 'Key']); - // If the device supports both touch and mouse (like IE11), and we are - // touch-dragging inside the plot area, don't handle the mouse event. - // #4339. - if (selectionMarker && selectionMarker.touch) { - return; - } - // If the mouse is outside the plot area, adjust to cooordinates - // inside to prevent the selection marker from going outside - if (chartX < plotLeft) { - chartX = plotLeft; - } - else if (chartX > plotLeft + plotWidth) { - chartX = plotLeft + plotWidth; - } - if (chartY < plotTop) { - chartY = plotTop; - } - else if (chartY > plotTop + plotHeight) { - chartY = plotTop + plotHeight; - } - // determine if the mouse has moved more than 10px - this.hasDragged = Math.sqrt(Math.pow(mouseDownX - chartX, 2) + - Math.pow(mouseDownY - chartY, 2)); - if (this.hasDragged > 10) { - clickedInside = chart.isInsidePlot(mouseDownX - plotLeft, mouseDownY - plotTop); - // make a selection - if (chart.hasCartesianSeries && - (this.zoomX || this.zoomY) && - clickedInside && - !panKey) { - if (!selectionMarker) { - this.selectionMarker = selectionMarker = - chart.renderer.rect(plotLeft, plotTop, zoomHor ? 1 : plotWidth, zoomVert ? 1 : plotHeight, 0) - .attr({ - 'class': 'highcharts-selection-marker', - zIndex: 7 - }) - .add(); - if (!chart.styledMode) { - selectionMarker.attr({ - fill: (chartOptions.selectionMarkerFill || - color('#335cad') - .setOpacity(0.25).get()) - }); - } - } - } - // adjust the width of the selection marker - if (selectionMarker && zoomHor) { - size = chartX - mouseDownX; - selectionMarker.attr({ - width: Math.abs(size), - x: (size > 0 ? 0 : size) + mouseDownX - }); - } - // adjust the height of the selection marker - if (selectionMarker && zoomVert) { - size = chartY - mouseDownY; - selectionMarker.attr({ - height: Math.abs(size), - y: (size > 0 ? 0 : size) + mouseDownY - }); - } - // panning - if (clickedInside && - !selectionMarker && - panningEnabled) { - chart.pan(e, chartOptions.panning); - } - } - }; - /** - * Start a drag operation. - * - * @private - * @function Highcharts.Pointer#dragStart - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - Pointer.prototype.dragStart = function (e) { - var chart = this.chart; - // Record the start position - chart.mouseIsDown = e.type; - chart.cancelClick = false; - chart.mouseDownX = this.mouseDownX = e.chartX; - chart.mouseDownY = this.mouseDownY = e.chartY; - }; - /** - * On mouse up or touch end across the entire document, drop the selection. - * - * @private - * @function Highcharts.Pointer#drop - * - * @param {global.Event} e - */ - Pointer.prototype.drop = function (e) { - var pointer = this, - chart = this.chart, - hasPinched = this.hasPinched; - if (this.selectionMarker) { - var selectionData = { - originalEvent: e, - xAxis: [], - yAxis: [] - }, - selectionBox = this.selectionMarker, - selectionLeft = selectionBox.attr ? - selectionBox.attr('x') : - selectionBox.x, - selectionTop = selectionBox.attr ? - selectionBox.attr('y') : - selectionBox.y, - selectionWidth = selectionBox.attr ? - selectionBox.attr('width') : - selectionBox.width, - selectionHeight = selectionBox.attr ? - selectionBox.attr('height') : - selectionBox.height, - runZoom; - // a selection has been made - if (this.hasDragged || hasPinched) { - // record each axis' min and max - chart.axes.forEach(function (axis) { - if (axis.zoomEnabled && - defined(axis.min) && - (hasPinched || - pointer[{ - xAxis: 'zoomX', - yAxis: 'zoomY' - }[axis.coll]]) && - isNumber(selectionLeft) && - isNumber(selectionTop)) { // #859, #3569 - var horiz = axis.horiz, - minPixelPadding = e.type === 'touchend' ? - axis.minPixelPadding : - 0, // #1207, #3075 - selectionMin = axis.toValue((horiz ? selectionLeft : selectionTop) + - minPixelPadding), - selectionMax = axis.toValue((horiz ? - selectionLeft + selectionWidth : - selectionTop + selectionHeight) - minPixelPadding); - selectionData[axis.coll].push({ - axis: axis, - // Min/max for reversed axes - min: Math.min(selectionMin, selectionMax), - max: Math.max(selectionMin, selectionMax) - }); - runZoom = true; - } - }); - if (runZoom) { - fireEvent(chart, 'selection', selectionData, function (args) { - chart.zoom(extend(args, hasPinched ? - { animation: false } : - null)); - }); - } - } - if (isNumber(chart.index)) { - this.selectionMarker = this.selectionMarker.destroy(); - } - // Reset scaling preview - if (hasPinched) { - this.scaleGroups(); - } - } - // Reset all. Check isNumber because it may be destroyed on mouse up - // (#877) - if (chart && isNumber(chart.index)) { - css(chart.container, { cursor: chart._cursor }); - chart.cancelClick = this.hasDragged > 10; // #370 - chart.mouseIsDown = this.hasDragged = this.hasPinched = false; - this.pinchDown = []; - } - }; - /** - * Finds the closest point to a set of coordinates, using the k-d-tree - * algorithm. - * - * @function Highcharts.Pointer#findNearestKDPoint - * - * @param {Array<Highcharts.Series>} series - * All the series to search in. - * - * @param {boolean|undefined} shared - * Whether it is a shared tooltip or not. - * - * @param {Highcharts.PointerEventObject} e - * The pointer event object, containing chart coordinates of the - * pointer. - * - * @return {Highcharts.Point|undefined} - * The point closest to given coordinates. - */ - Pointer.prototype.findNearestKDPoint = function (series, shared, e) { - var chart = this.chart; - var hoverPoint = chart.hoverPoint; - var tooltip = chart.tooltip; - if (hoverPoint && - tooltip && - tooltip.isStickyOnContact()) { - return hoverPoint; - } - var closest; - /** @private */ - function sort(p1, p2) { - var isCloserX = p1.distX - p2.distX, - isCloser = p1.dist - p2.dist, - isAbove = (p2.series.group && p2.series.group.zIndex) - - (p1.series.group && p1.series.group.zIndex), - result; - // We have two points which are not in the same place on xAxis - // and shared tooltip: - if (isCloserX !== 0 && shared) { // #5721 - result = isCloserX; - // Points are not exactly in the same place on x/yAxis: - } - else if (isCloser !== 0) { - result = isCloser; - // The same xAxis and yAxis position, sort by z-index: - } - else if (isAbove !== 0) { - result = isAbove; - // The same zIndex, sort by array index: - } - else { - result = - p1.series.index > p2.series.index ? - -1 : - 1; - } - return result; - } - series.forEach(function (s) { - var noSharedTooltip = s.noSharedTooltip && shared, - compareX = (!noSharedTooltip && - s.options.findNearestPointBy.indexOf('y') < 0), - point = s.searchPoint(e, - compareX); - if ( // Check that we actually found a point on the series. - isObject(point, true) && - // Use the new point if it is closer. - (!isObject(closest, true) || - (sort(closest, point) > 0))) { - closest = point; - } - }); - return closest; - }; - /** - * @private - * @function Highcharts.Pointer#getChartCoordinatesFromPoint - * @param {Highcharts.Point} point - * @param {boolean} [inverted] - * @return {Highcharts.PointerCoordinatesObject|undefined} - */ - Pointer.prototype.getChartCoordinatesFromPoint = function (point, inverted) { - var series = point.series, - xAxis = series.xAxis, - yAxis = series.yAxis, - plotX = pick(point.clientX, - point.plotX), - shapeArgs = point.shapeArgs; - if (xAxis && yAxis) { - return inverted ? { - chartX: xAxis.len + xAxis.pos - plotX, - chartY: yAxis.len + yAxis.pos - point.plotY - } : { - chartX: plotX + xAxis.pos, - chartY: point.plotY + yAxis.pos - }; - } - if (shapeArgs && shapeArgs.x && shapeArgs.y) { - // E.g. pies do not have axes - return { - chartX: shapeArgs.x, - chartY: shapeArgs.y - }; - } - }; - /** - * Return the cached chartPosition if it is available on the Pointer, - * otherwise find it. Running offset is quite expensive, so it should be - * avoided when we know the chart hasn't moved. - * - * @function Highcharts.Pointer#getChartPosition - * - * @return {Highcharts.OffsetObject} - * The offset of the chart container within the page - */ - Pointer.prototype.getChartPosition = function () { - return (this.chartPosition || - (this.chartPosition = offset(this.chart.container))); - }; - /** - * Get the click position in terms of axis values. - * - * @function Highcharts.Pointer#getCoordinates - * - * @param {Highcharts.PointerEventObject} e - * Pointer event, extended with `chartX` and `chartY` properties. - * - * @return {Highcharts.PointerAxisCoordinatesObject} - */ - Pointer.prototype.getCoordinates = function (e) { - var coordinates = { - xAxis: [], - yAxis: [] - }; - this.chart.axes.forEach(function (axis) { - coordinates[axis.isXAxis ? 'xAxis' : 'yAxis'].push({ - axis: axis, - value: axis.toValue(e[axis.horiz ? 'chartX' : 'chartY']) - }); - }); - return coordinates; - }; - /** - * Calculates what is the current hovered point/points and series. - * - * @private - * @function Highcharts.Pointer#getHoverData - * - * @param {Highcharts.Point|undefined} existingHoverPoint - * The point currrently beeing hovered. - * - * @param {Highcharts.Series|undefined} existingHoverSeries - * The series currently beeing hovered. - * - * @param {Array<Highcharts.Series>} series - * All the series in the chart. - * - * @param {boolean} isDirectTouch - * Is the pointer directly hovering the point. - * - * @param {boolean|undefined} shared - * Whether it is a shared tooltip or not. - * - * @param {Highcharts.PointerEventObject} [e] - * The triggering event, containing chart coordinates of the pointer. - * - * @return {object} - * Object containing resulting hover data: hoverPoint, hoverSeries, - * and hoverPoints. - */ - Pointer.prototype.getHoverData = function (existingHoverPoint, existingHoverSeries, series, isDirectTouch, shared, e) { - var hoverPoint, - hoverPoints = [], - hoverSeries = existingHoverSeries, - useExisting = !!(isDirectTouch && existingHoverPoint), - notSticky = hoverSeries && !hoverSeries.stickyTracking, - // Which series to look in for the hover point - searchSeries, - // Parameters needed for beforeGetHoverData event. - eventArgs = { - chartX: e ? e.chartX : void 0, - chartY: e ? e.chartY : void 0, - shared: shared - }, - filter = function (s) { - return (s.visible && - !(!shared && s.directTouch) && // #3821 - pick(s.options.enableMouseTracking, - true)); - }; - // Find chart.hoverPane and update filter method in polar. - fireEvent(this, 'beforeGetHoverData', eventArgs); - searchSeries = notSticky ? - // Only search on hovered series if it has stickyTracking false - [hoverSeries] : - // Filter what series to look in. - series.filter(function (s) { - return eventArgs.filter ? eventArgs.filter(s) : filter(s) && - s.stickyTracking; - }); - // Use existing hovered point or find the one closest to coordinates. - hoverPoint = useExisting || !e ? - existingHoverPoint : - this.findNearestKDPoint(searchSeries, shared, e); - // Assign hover series - hoverSeries = hoverPoint && hoverPoint.series; - // If we have a hoverPoint, assign hoverPoints. - if (hoverPoint) { - // When tooltip is shared, it displays more than one point - if (shared && !hoverSeries.noSharedTooltip) { - searchSeries = series.filter(function (s) { - return eventArgs.filter ? - eventArgs.filter(s) : filter(s) && !s.noSharedTooltip; - }); - // Get all points with the same x value as the hoverPoint - searchSeries.forEach(function (s) { - var point = find(s.points, - function (p) { - return p.x === hoverPoint.x && !p.isNull; - }); - if (isObject(point)) { - /* - * Boost returns a minimal point. Convert it to a usable - * point for tooltip and states. - */ - if (s.chart.isBoosting) { - point = s.getPoint(point); - } - hoverPoints.push(point); - } - }); - } - else { - hoverPoints.push(hoverPoint); - } - } - // Check whether the hoverPoint is inside pane we are hovering over. - eventArgs = { hoverPoint: hoverPoint }; - fireEvent(this, 'afterGetHoverData', eventArgs); - return { - hoverPoint: eventArgs.hoverPoint, - hoverSeries: hoverSeries, - hoverPoints: hoverPoints - }; - }; - /** - * @private - * @function Highcharts.Pointer#getPointFromEvent - * - * @param {global.Event} e - * - * @return {Highcharts.Point|undefined} - */ - Pointer.prototype.getPointFromEvent = function (e) { - var target = e.target, - point; - while (target && !point) { - point = target.point; - target = target.parentNode; - } - return point; - }; - /** - * @private - * @function Highcharts.Pointer#onTrackerMouseOut - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - Pointer.prototype.onTrackerMouseOut = function (e) { - var chart = this.chart; - var relatedTarget = e.relatedTarget || e.toElement; - var series = chart.hoverSeries; - this.isDirectTouch = false; - if (series && - relatedTarget && - !series.stickyTracking && - !this.inClass(relatedTarget, 'highcharts-tooltip') && - (!this.inClass(relatedTarget, 'highcharts-series-' + series.index) || // #2499, #4465, #5553 - !this.inClass(relatedTarget, 'highcharts-tracker'))) { - series.onMouseOut(); - } - }; - /** - * Utility to detect whether an element has, or has a parent with, a - * specificclass name. Used on detection of tracker objects and on deciding - * whether hovering the tooltip should cause the active series to mouse out. - * - * @function Highcharts.Pointer#inClass - * - * @param {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} element - * The element to investigate. - * - * @param {string} className - * The class name to look for. - * - * @return {boolean|undefined} - * True if either the element or one of its parents has the given - * class name. - */ - Pointer.prototype.inClass = function (element, className) { - var elemClassName; - while (element) { - elemClassName = attr(element, 'class'); - if (elemClassName) { - if (elemClassName.indexOf(className) !== -1) { - return true; - } - if (elemClassName.indexOf('highcharts-container') !== -1) { - return false; - } - } - element = element.parentNode; - } - }; - /** - * Initialize the Pointer. - * - * @private - * @function Highcharts.Pointer#init - * - * @param {Highcharts.Chart} chart - * The Chart instance. - * - * @param {Highcharts.Options} options - * The root options object. The pointer uses options from the chart - * and tooltip structures. - * - * @return {void} - */ - Pointer.prototype.init = function (chart, options) { - // Store references - this.options = options; - this.chart = chart; - // Do we need to handle click on a touch device? - this.runChartClick = - options.chart.events && - !!options.chart.events.click; - this.pinchDown = []; - this.lastValidTouch = {}; - if (Tooltip) { - /** - * Tooltip object for points of series. - * - * @name Highcharts.Chart#tooltip - * @type {Highcharts.Tooltip} - */ - chart.tooltip = new Tooltip(chart, options.tooltip); - this.followTouchMove = pick(options.tooltip.followTouchMove, true); - } - this.setDOMEvents(); - }; - /** - * Takes a browser event object and extends it with custom Highcharts - * properties `chartX` and `chartY` in order to work on the internal - * coordinate system. - * - * @function Highcharts.Pointer#normalize - * - * @param {global.MouseEvent|global.PointerEvent|global.TouchEvent} e - * Event object in standard browsers. - * - * @param {Highcharts.OffsetObject} [chartPosition] - * Additional chart offset. - * - * @return {Highcharts.PointerEventObject} - * A browser event with extended properties `chartX` and `chartY`. - */ - Pointer.prototype.normalize = function (e, chartPosition) { - var touches = e.touches; - // iOS (#2757) - var ePos = (touches ? - touches.length ? - touches.item(0) : - (pick(// #13534 - touches.changedTouches, - e.changedTouches))[0] : - e); - // Get mouse position - if (!chartPosition) { - chartPosition = this.getChartPosition(); - } - var chartX = ePos.pageX - chartPosition.left, - chartY = ePos.pageY - chartPosition.top; - // #11329 - when there is scaling on a parent element, we need to take - // this into account - var containerScaling = this.chart.containerScaling; - if (containerScaling) { - chartX /= containerScaling.scaleX; - chartY /= containerScaling.scaleY; - } - return extend(e, { - chartX: Math.round(chartX), - chartY: Math.round(chartY) - }); - }; - /** - * @private - * @function Highcharts.Pointer#onContainerClick - */ - Pointer.prototype.onContainerClick = function (e) { - var chart = this.chart; - var hoverPoint = chart.hoverPoint; - var pEvt = this.normalize(e); - var plotLeft = chart.plotLeft; - var plotTop = chart.plotTop; - if (!chart.cancelClick) { - // On tracker click, fire the series and point events. #783, #1583 - if (hoverPoint && - this.inClass(pEvt.target, 'highcharts-tracker')) { - // the series click event - fireEvent(hoverPoint.series, 'click', extend(pEvt, { - point: hoverPoint - })); - // the point click event - if (chart.hoverPoint) { // it may be destroyed (#1844) - hoverPoint.firePointEvent('click', pEvt); - } - // When clicking outside a tracker, fire a chart event - } - else { - extend(pEvt, this.getCoordinates(pEvt)); - // fire a click event in the chart - if (chart.isInsidePlot((pEvt.chartX - plotLeft), (pEvt.chartY - plotTop))) { - fireEvent(chart, 'click', pEvt); - } - } - } - }; - /** - * @private - * @function Highcharts.Pointer#onContainerMouseDown - * - * @param {global.MouseEvent} e - */ - Pointer.prototype.onContainerMouseDown = function (e) { - var isPrimaryButton = ((e.buttons || e.button) & 1) === 1; - // Normalize before the 'if' for the legacy IE (#7850) - e = this.normalize(e); - // #11635, Firefox does not reliable fire move event after click scroll - if (H.isFirefox && - e.button !== 0) { - this.onContainerMouseMove(e); - } - // #11635, limiting to primary button (incl. IE 8 support) - if (typeof e.button === 'undefined' || - isPrimaryButton) { - this.zoomOption(e); - // #295, #13737 solve conflict between container drag and chart zoom - if (isPrimaryButton && - e.preventDefault) { - e.preventDefault(); - } - this.dragStart(e); - } - }; - /** - * When mouse leaves the container, hide the tooltip. - * - * @private - * @function Highcharts.Pointer#onContainerMouseLeave - * - * @param {global.MouseEvent} e - * - * @return {void} - */ - Pointer.prototype.onContainerMouseLeave = function (e) { - var chart = charts[pick(H.hoverChartIndex, -1)]; - var tooltip = this.chart.tooltip; - e = this.normalize(e); - // #4886, MS Touch end fires mouseleave but with no related target - if (chart && - (e.relatedTarget || e.toElement)) { - chart.pointer.reset(); - // Also reset the chart position, used in #149 fix - chart.pointer.chartPosition = void 0; - } - if ( // #11635, Firefox wheel scroll does not fire out events consistently - tooltip && - !tooltip.isHidden) { - this.reset(); - } - }; - /** - * When mouse enters the container, delete pointer's chartPosition. - * - * @private - * @function Highcharts.Pointer#onContainerMouseEnter - * - * @param {global.MouseEvent} e - * - * @return {void} - */ - Pointer.prototype.onContainerMouseEnter = function (e) { - delete this.chartPosition; - }; - /** - * The mousemove, touchmove and touchstart event handler - * - * @private - * @function Highcharts.Pointer#onContainerMouseMove - * - * @param {global.MouseEvent} e - * - * @return {void} - */ - Pointer.prototype.onContainerMouseMove = function (e) { - var chart = this.chart; - var pEvt = this.normalize(e); - this.setHoverChartIndex(); - // In IE8 we apparently need this returnValue set to false in order to - // avoid text being selected. But in Chrome, e.returnValue is prevented, - // plus we don't need to run e.preventDefault to prevent selected text - // in modern browsers. So we set it conditionally. Remove it when IE8 is - // no longer needed. #2251, #3224. - if (!pEvt.preventDefault) { - pEvt.returnValue = false; - } - if (chart.mouseIsDown === 'mousedown') { - this.drag(pEvt); - } - // Show the tooltip and run mouse over events (#977) - if (!chart.openMenu && - (this.inClass(pEvt.target, 'highcharts-tracker') || - chart.isInsidePlot((pEvt.chartX - chart.plotLeft), (pEvt.chartY - chart.plotTop)))) { - this.runPointActions(pEvt); - } - }; - /** - * @private - * @function Highcharts.Pointer#onDocumentTouchEnd - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - Pointer.prototype.onDocumentTouchEnd = function (e) { - if (charts[H.hoverChartIndex]) { - charts[H.hoverChartIndex].pointer.drop(e); - } - }; - /** - * @private - * @function Highcharts.Pointer#onContainerTouchMove - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - Pointer.prototype.onContainerTouchMove = function (e) { - this.touch(e); - }; - /** - * @private - * @function Highcharts.Pointer#onContainerTouchStart - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - Pointer.prototype.onContainerTouchStart = function (e) { - this.zoomOption(e); - this.touch(e, true); - }; - /** - * Special handler for mouse move that will hide the tooltip when the mouse - * leaves the plotarea. Issue #149 workaround. The mouseleave event does not - * always fire. - * - * @private - * @function Highcharts.Pointer#onDocumentMouseMove - * - * @param {global.MouseEvent} e - * - * @return {void} - */ - Pointer.prototype.onDocumentMouseMove = function (e) { - var chart = this.chart; - var chartPosition = this.chartPosition; - var pEvt = this.normalize(e, - chartPosition); - var tooltip = chart.tooltip; - // If we're outside, hide the tooltip - if (chartPosition && - (!tooltip || - !tooltip.isStickyOnContact()) && - !chart.isInsidePlot(pEvt.chartX - chart.plotLeft, pEvt.chartY - chart.plotTop) && - !this.inClass(pEvt.target, 'highcharts-tracker')) { - this.reset(); - } - }; - /** - * @private - * @function Highcharts.Pointer#onDocumentMouseUp - * - * @param {global.MouseEvent} e - * - * @return {void} - */ - Pointer.prototype.onDocumentMouseUp = function (e) { - var chart = charts[pick(H.hoverChartIndex, -1)]; - if (chart) { - chart.pointer.drop(e); - } - }; - /** - * Handle touch events with two touches - * - * @private - * @function Highcharts.Pointer#pinch - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - Pointer.prototype.pinch = function (e) { - var self = this, - chart = self.chart, - pinchDown = self.pinchDown, - touches = (e.touches || []), - touchesLength = touches.length, - lastValidTouch = self.lastValidTouch, - hasZoom = self.hasZoom, - selectionMarker = self.selectionMarker, - transform = {}, - fireClickEvent = touchesLength === 1 && ((self.inClass(e.target, 'highcharts-tracker') && - chart.runTrackerClick) || - self.runChartClick), - clip = {}; - // Don't initiate panning until the user has pinched. This prevents us - // from blocking page scrolling as users scroll down a long page - // (#4210). - if (touchesLength > 1) { - self.initiated = true; - } - // On touch devices, only proceed to trigger click if a handler is - // defined - if (hasZoom && self.initiated && !fireClickEvent && e.cancelable !== false) { - e.preventDefault(); - } - // Normalize each touch - [].map.call(touches, function (e) { - return self.normalize(e); - }); - // Register the touch start position - if (e.type === 'touchstart') { - [].forEach.call(touches, function (e, i) { - pinchDown[i] = { chartX: e.chartX, chartY: e.chartY }; - }); - lastValidTouch.x = [pinchDown[0].chartX, pinchDown[1] && - pinchDown[1].chartX]; - lastValidTouch.y = [pinchDown[0].chartY, pinchDown[1] && - pinchDown[1].chartY]; - // Identify the data bounds in pixels - chart.axes.forEach(function (axis) { - if (axis.zoomEnabled) { - var bounds = chart.bounds[axis.horiz ? 'h' : 'v'], - minPixelPadding = axis.minPixelPadding, - min = axis.toPixels(Math.min(pick(axis.options.min, - axis.dataMin), - axis.dataMin)), - max = axis.toPixels(Math.max(pick(axis.options.max, - axis.dataMax), - axis.dataMax)), - absMin = Math.min(min, - max), - absMax = Math.max(min, - max); - // Store the bounds for use in the touchmove handler - bounds.min = Math.min(axis.pos, absMin - minPixelPadding); - bounds.max = Math.max(axis.pos + axis.len, absMax + minPixelPadding); - } - }); - self.res = true; // reset on next move - // Optionally move the tooltip on touchmove - } - else if (self.followTouchMove && touchesLength === 1) { - this.runPointActions(self.normalize(e)); - // Event type is touchmove, handle panning and pinching - } - else if (pinchDown.length) { // can be 0 when releasing, if touchend - // fires first - // Set the marker - if (!selectionMarker) { - self.selectionMarker = selectionMarker = extend({ - destroy: noop, - touch: true - }, chart.plotBox); - } - self.pinchTranslate(pinchDown, touches, transform, selectionMarker, clip, lastValidTouch); - self.hasPinched = hasZoom; - // Scale and translate the groups to provide visual feedback during - // pinching - self.scaleGroups(transform, clip); - if (self.res) { - self.res = false; - this.reset(false, 0); - } - } - }; - /** - * Run translation operations - * - * @private - * @function Highcharts.Pointer#pinchTranslate - * - * @param {Array<*>} pinchDown - * - * @param {Array<Highcharts.PointerEventObject>} touches - * - * @param {*} transform - * - * @param {*} selectionMarker - * - * @param {*} clip - * - * @param {*} lastValidTouch - * - * @return {void} - */ - Pointer.prototype.pinchTranslate = function (pinchDown, touches, transform, selectionMarker, clip, lastValidTouch) { - if (this.zoomHor) { - this.pinchTranslateDirection(true, pinchDown, touches, transform, selectionMarker, clip, lastValidTouch); - } - if (this.zoomVert) { - this.pinchTranslateDirection(false, pinchDown, touches, transform, selectionMarker, clip, lastValidTouch); - } - }; - /** - * Run translation operations for each direction (horizontal and vertical) - * independently. - * - * @private - * @function Highcharts.Pointer#pinchTranslateDirection - * - * @param {boolean} horiz - * - * @param {Array<*>} pinchDown - * - * @param {Array<Highcharts.PointerEventObject>} touches - * - * @param {*} transform - * - * @param {*} selectionMarker - * - * @param {*} clip - * - * @param {*} lastValidTouch - * - * @param {number|undefined} [forcedScale=1] - * - * @return {void} - */ - Pointer.prototype.pinchTranslateDirection = function (horiz, pinchDown, touches, transform, selectionMarker, clip, lastValidTouch, forcedScale) { - var chart = this.chart, xy = horiz ? 'x' : 'y', XY = horiz ? 'X' : 'Y', sChartXY = ('chart' + XY), wh = horiz ? 'width' : 'height', plotLeftTop = chart['plot' + (horiz ? 'Left' : 'Top')], selectionWH, selectionXY, clipXY, scale = forcedScale || 1, inverted = chart.inverted, bounds = chart.bounds[horiz ? 'h' : 'v'], singleTouch = pinchDown.length === 1, touch0Start = pinchDown[0][sChartXY], touch0Now = touches[0][sChartXY], touch1Start = !singleTouch && pinchDown[1][sChartXY], touch1Now = !singleTouch && touches[1][sChartXY], outOfBounds, transformScale, scaleKey, setScale = function () { - // Don't zoom if fingers are too close on this axis - if (typeof touch1Now === 'number' && - Math.abs(touch0Start - touch1Start) > 20) { - scale = forcedScale || - Math.abs(touch0Now - touch1Now) / - Math.abs(touch0Start - touch1Start); - } - clipXY = ((plotLeftTop - touch0Now) / scale) + touch0Start; - selectionWH = chart['plot' + (horiz ? 'Width' : 'Height')] / scale; - }; - // Set the scale, first pass - setScale(); - // The clip position (x or y) is altered if out of bounds, the selection - // position is not - selectionXY = clipXY; - // Out of bounds - if (selectionXY < bounds.min) { - selectionXY = bounds.min; - outOfBounds = true; - } - else if (selectionXY + selectionWH > bounds.max) { - selectionXY = bounds.max - selectionWH; - outOfBounds = true; - } - // Is the chart dragged off its bounds, determined by dataMin and - // dataMax? - if (outOfBounds) { - // Modify the touchNow position in order to create an elastic drag - // movement. This indicates to the user that the chart is responsive - // but can't be dragged further. - touch0Now -= 0.8 * (touch0Now - lastValidTouch[xy][0]); - if (typeof touch1Now === 'number') { - touch1Now -= 0.8 * (touch1Now - lastValidTouch[xy][1]); - } - // Set the scale, second pass to adapt to the modified touchNow - // positions - setScale(); - } - else { - lastValidTouch[xy] = [touch0Now, touch1Now]; - } - // Set geometry for clipping, selection and transformation - if (!inverted) { - clip[xy] = clipXY - plotLeftTop; - clip[wh] = selectionWH; - } - scaleKey = inverted ? (horiz ? 'scaleY' : 'scaleX') : 'scale' + XY; - transformScale = inverted ? 1 / scale : scale; - selectionMarker[wh] = selectionWH; - selectionMarker[xy] = selectionXY; - transform[scaleKey] = scale; - transform['translate' + XY] = (transformScale * plotLeftTop) + - (touch0Now - (transformScale * touch0Start)); - }; - /** - * Reset the tracking by hiding the tooltip, the hover series state and the - * hover point - * - * @function Highcharts.Pointer#reset - * - * @param {boolean} [allowMove] - * Instead of destroying the tooltip altogether, allow moving it if - * possible. - * - * @param {number} [delay] - * - * @return {void} - */ - Pointer.prototype.reset = function (allowMove, delay) { - var pointer = this, - chart = pointer.chart, - hoverSeries = chart.hoverSeries, - hoverPoint = chart.hoverPoint, - hoverPoints = chart.hoverPoints, - tooltip = chart.tooltip, - tooltipPoints = tooltip && tooltip.shared ? - hoverPoints : - hoverPoint; - // Check if the points have moved outside the plot area (#1003, #4736, - // #5101) - if (allowMove && tooltipPoints) { - splat(tooltipPoints).forEach(function (point) { - if (point.series.isCartesian && - typeof point.plotX === 'undefined') { - allowMove = false; - } - }); - } - // Just move the tooltip, #349 - if (allowMove) { - if (tooltip && tooltipPoints && splat(tooltipPoints).length) { - tooltip.refresh(tooltipPoints); - if (tooltip.shared && hoverPoints) { // #8284 - hoverPoints.forEach(function (point) { - point.setState(point.state, true); - if (point.series.isCartesian) { - if (point.series.xAxis.crosshair) { - point.series.xAxis - .drawCrosshair(null, point); - } - if (point.series.yAxis.crosshair) { - point.series.yAxis - .drawCrosshair(null, point); - } - } - }); - } - else if (hoverPoint) { // #2500 - hoverPoint.setState(hoverPoint.state, true); - chart.axes.forEach(function (axis) { - if (axis.crosshair && - hoverPoint.series[axis.coll] === axis) { - axis.drawCrosshair(null, hoverPoint); - } - }); - } - } - // Full reset - } - else { - if (hoverPoint) { - hoverPoint.onMouseOut(); - } - if (hoverPoints) { - hoverPoints.forEach(function (point) { - point.setState(); - }); - } - if (hoverSeries) { - hoverSeries.onMouseOut(); - } - if (tooltip) { - tooltip.hide(delay); - } - if (pointer.unDocMouseMove) { - pointer.unDocMouseMove = pointer.unDocMouseMove(); - } - // Remove crosshairs - chart.axes.forEach(function (axis) { - axis.hideCrosshair(); - }); - pointer.hoverX = chart.hoverPoints = chart.hoverPoint = null; - } - }; - /** - * With line type charts with a single tracker, get the point closest to the - * mouse. Run Point.onMouseOver and display tooltip for the point or points. - * - * @private - * @function Highcharts.Pointer#runPointActions - * - * @param {global.Event} e - * - * @param {Highcharts.PointerEventObject} [p] - * - * @return {void} - * - * @fires Highcharts.Point#event:mouseOut - * @fires Highcharts.Point#event:mouseOver - */ - Pointer.prototype.runPointActions = function (e, p) { - var pointer = this, - chart = pointer.chart, - series = chart.series, - tooltip = (chart.tooltip && chart.tooltip.options.enabled ? - chart.tooltip : - void 0), - shared = (tooltip ? - tooltip.shared : - false), - hoverPoint = p || chart.hoverPoint, - hoverSeries = hoverPoint && hoverPoint.series || chart.hoverSeries, - // onMouseOver or already hovering a series with directTouch - isDirectTouch = (!e || e.type !== 'touchmove') && (!!p || ((hoverSeries && hoverSeries.directTouch) && - pointer.isDirectTouch)), - hoverData = this.getHoverData(hoverPoint, - hoverSeries, - series, - isDirectTouch, - shared, - e), - useSharedTooltip, - followPointer, - anchor, - points; - // Update variables from hoverData. - hoverPoint = hoverData.hoverPoint; - points = hoverData.hoverPoints; - hoverSeries = hoverData.hoverSeries; - followPointer = hoverSeries && hoverSeries.tooltipOptions.followPointer; - useSharedTooltip = (shared && - hoverSeries && - !hoverSeries.noSharedTooltip); - // Refresh tooltip for kdpoint if new hover point or tooltip was hidden - // #3926, #4200 - if (hoverPoint && - // !(hoverSeries && hoverSeries.directTouch) && - (hoverPoint !== chart.hoverPoint || (tooltip && tooltip.isHidden))) { - (chart.hoverPoints || []).forEach(function (p) { - if (points.indexOf(p) === -1) { - p.setState(); - } - }); - // Set normal state to previous series - if (chart.hoverSeries !== hoverSeries) { - hoverSeries.onMouseOver(); - } - pointer.applyInactiveState(points); - // Do mouseover on all points (#3919, #3985, #4410, #5622) - (points || []).forEach(function (p) { - p.setState('hover'); - }); - // If tracking is on series in stead of on each point, - // fire mouseOver on hover point. // #4448 - if (chart.hoverPoint) { - chart.hoverPoint.firePointEvent('mouseOut'); - } - // Hover point may have been destroyed in the event handlers (#7127) - if (!hoverPoint.series) { - return; - } - /** - * Contains all hovered points. - * - * @name Highcharts.Chart#hoverPoints - * @type {Array<Highcharts.Point>|null} - */ - chart.hoverPoints = points; - /** - * Contains the original hovered point. - * - * @name Highcharts.Chart#hoverPoint - * @type {Highcharts.Point|null} - */ - chart.hoverPoint = hoverPoint; - /** - * Hover state should not be lost when axis is updated (#12569) - * Axis.update runs pointer.reset which uses chart.hoverPoint.state - * to apply state which does not exist in hoverPoint yet. - * The mouseOver event should be triggered when hoverPoint - * is correct. - */ - hoverPoint.firePointEvent('mouseOver'); - // Draw tooltip if necessary - if (tooltip) { - tooltip.refresh(useSharedTooltip ? points : hoverPoint, e); - } - // Update positions (regardless of kdpoint or hoverPoint) - } - else if (followPointer && tooltip && !tooltip.isHidden) { - anchor = tooltip.getAnchor([{}], e); - tooltip.updatePosition({ plotX: anchor[0], plotY: anchor[1] }); - } - // Start the event listener to pick up the tooltip and crosshairs - if (!pointer.unDocMouseMove) { - pointer.unDocMouseMove = addEvent(chart.container.ownerDocument, 'mousemove', function (e) { - var chart = charts[H.hoverChartIndex]; - if (chart) { - chart.pointer.onDocumentMouseMove(e); - } - }); - } - // Issues related to crosshair #4927, #5269 #5066, #5658 - chart.axes.forEach(function drawAxisCrosshair(axis) { - var snap = pick((axis.crosshair || {}).snap, - true); - var point; - if (snap) { - point = chart.hoverPoint; // #13002 - if (!point || point.series[axis.coll] !== axis) { - point = find(points, function (p) { - return p.series[axis.coll] === axis; - }); - } - } - // Axis has snapping crosshairs, and one of the hover points belongs - // to axis. Always call drawCrosshair when it is not snap. - if (point || !snap) { - axis.drawCrosshair(e, point); - // Axis has snapping crosshairs, but no hover point belongs to axis - } - else { - axis.hideCrosshair(); - } - }); - }; - /** - * Scale series groups to a certain scale and translation. - * - * @private - * @function Highcharts.Pointer#scaleGroups - * - * @param {Highcharts.SeriesPlotBoxObject} [attribs] - * - * @param {boolean} [clip] - * - * @return {void} - */ - Pointer.prototype.scaleGroups = function (attribs, clip) { - var chart = this.chart, - seriesAttribs; - // Scale each series - chart.series.forEach(function (series) { - seriesAttribs = attribs || series.getPlotBox(); // #1701 - if (series.xAxis && series.xAxis.zoomEnabled && series.group) { - series.group.attr(seriesAttribs); - if (series.markerGroup) { - series.markerGroup.attr(seriesAttribs); - series.markerGroup.clip(clip ? chart.clipRect : null); - } - if (series.dataLabelsGroup) { - series.dataLabelsGroup.attr(seriesAttribs); - } - } - }); - // Clip - chart.clipRect.attr(clip || chart.clipBox); - }; - /** - * Set the JS DOM events on the container and document. This method should - * contain a one-to-one assignment between methods and their handlers. Any - * advanced logic should be moved to the handler reflecting the event's - * name. - * - * @private - * @function Highcharts.Pointer#setDOMEvents - * - * @return {void} - */ - Pointer.prototype.setDOMEvents = function () { - var container = this.chart.container, - ownerDoc = container.ownerDocument; - container.onmousedown = this.onContainerMouseDown.bind(this); - container.onmousemove = this.onContainerMouseMove.bind(this); - container.onclick = this.onContainerClick.bind(this); - this.unbindContainerMouseEnter = addEvent(container, 'mouseenter', this.onContainerMouseEnter.bind(this)); - this.unbindContainerMouseLeave = addEvent(container, 'mouseleave', this.onContainerMouseLeave.bind(this)); - if (!H.unbindDocumentMouseUp) { - H.unbindDocumentMouseUp = addEvent(ownerDoc, 'mouseup', this.onDocumentMouseUp.bind(this)); - } - if (H.hasTouch) { - addEvent(container, 'touchstart', this.onContainerTouchStart.bind(this)); - addEvent(container, 'touchmove', this.onContainerTouchMove.bind(this)); - if (!H.unbindDocumentTouchEnd) { - H.unbindDocumentTouchEnd = addEvent(ownerDoc, 'touchend', this.onDocumentTouchEnd.bind(this)); - } - } - }; - /** - * Sets the index of the hovered chart and leaves the previous hovered - * chart, to reset states like tooltip. - * - * @private - * @function Highcharts.Pointer#setHoverChartIndex - */ - Pointer.prototype.setHoverChartIndex = function () { - var chart = this.chart; - var hoverChart = H.charts[pick(H.hoverChartIndex, -1)]; - if (hoverChart && - hoverChart !== chart) { - hoverChart.pointer.onContainerMouseLeave({ relatedTarget: true }); - } - if (!hoverChart || - !hoverChart.mouseIsDown) { - H.hoverChartIndex = chart.index; - } - }; - /** - * General touch handler shared by touchstart and touchmove. - * - * @private - * @function Highcharts.Pointer#touch - * - * @param {Highcharts.PointerEventObject} e - * - * @param {boolean} [start] - * - * @return {void} - */ - Pointer.prototype.touch = function (e, start) { - var chart = this.chart, - hasMoved, - pinchDown, - isInside; - this.setHoverChartIndex(); - if (e.touches.length === 1) { - e = this.normalize(e); - isInside = chart.isInsidePlot(e.chartX - chart.plotLeft, e.chartY - chart.plotTop); - if (isInside && !chart.openMenu) { - // Run mouse events and display tooltip etc - if (start) { - this.runPointActions(e); - } - // Android fires touchmove events after the touchstart even if - // the finger hasn't moved, or moved only a pixel or two. In iOS - // however, the touchmove doesn't fire unless the finger moves - // more than ~4px. So we emulate this behaviour in Android by - // checking how much it moved, and cancelling on small - // distances. #3450. - if (e.type === 'touchmove') { - pinchDown = this.pinchDown; - hasMoved = pinchDown[0] ? Math.sqrt(// #5266 - Math.pow(pinchDown[0].chartX - e.chartX, 2) + - Math.pow(pinchDown[0].chartY - e.chartY, 2)) >= 4 : false; - } - if (pick(hasMoved, true)) { - this.pinch(e); - } - } - else if (start) { - // Hide the tooltip on touching outside the plot area (#1203) - this.reset(); - } - } - else if (e.touches.length === 2) { - this.pinch(e); - } - }; - /** - * Resolve the zoomType option, this is reset on all touch start and mouse - * down events. - * - * @private - * @function Highcharts.Pointer#zoomOption - * - * @param {global.Event} e - * Event object. - * - * @param {void} - */ - Pointer.prototype.zoomOption = function (e) { - var chart = this.chart, - options = chart.options.chart, - zoomType = options.zoomType || '', - inverted = chart.inverted, - zoomX, - zoomY; - // Look for the pinchType option - if (/touch/.test(e.type)) { - zoomType = pick(options.pinchType, zoomType); - } - this.zoomX = zoomX = /x/.test(zoomType); - this.zoomY = zoomY = /y/.test(zoomType); - this.zoomHor = (zoomX && !inverted) || (zoomY && inverted); - this.zoomVert = (zoomY && !inverted) || (zoomX && inverted); - this.hasZoom = zoomX || zoomY; - }; - return Pointer; - }()); - H.Pointer = Pointer; - - return Pointer; - }); - _registerModule(_modules, 'Core/MSPointer.js', [_modules['Core/Globals.js'], _modules['Core/Pointer.js'], _modules['Core/Utilities.js']], function (H, Pointer, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var __extends = (this && this.__extends) || (function () { - var extendStatics = function (d, - b) { - extendStatics = Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && function (d, - b) { d.__proto__ = b; }) || - function (d, - b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; - return extendStatics(d, b); - }; - return function (d, b) { - extendStatics(d, b); - function __() { this.constructor = d; } - d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); - }; - })(); - var charts = H.charts, - doc = H.doc, - noop = H.noop, - win = H.win; - var addEvent = U.addEvent, - css = U.css, - objectEach = U.objectEach, - removeEvent = U.removeEvent; - /* globals MSPointerEvent, PointerEvent */ - // The touches object keeps track of the points being touched at all times - var touches = {}; - var hasPointerEvent = !!win.PointerEvent; - /* eslint-disable valid-jsdoc */ - /** @private */ - function getWebkitTouches() { - var fake = []; - fake.item = function (i) { - return this[i]; - }; - objectEach(touches, function (touch) { - fake.push({ - pageX: touch.pageX, - pageY: touch.pageY, - target: touch.target - }); - }); - return fake; - } - /** @private */ - function translateMSPointer(e, method, wktype, func) { - var p; - if ((e.pointerType === 'touch' || - e.pointerType === e.MSPOINTER_TYPE_TOUCH) && charts[H.hoverChartIndex]) { - func(e); - p = charts[H.hoverChartIndex].pointer; - p[method]({ - type: wktype, - target: e.currentTarget, - preventDefault: noop, - touches: getWebkitTouches() - }); - } - } - /** @private */ - var MSPointer = /** @class */ (function (_super) { - __extends(MSPointer, _super); - function MSPointer() { - return _super !== null && _super.apply(this, arguments) || this; - } - /* * - * - * Functions - * - * */ - /** - * Add or remove the MS Pointer specific events - * - * @private - * @function Highcharts.Pointer#batchMSEvents - * - * @param {Function} fn - * - * @return {void} - */ - MSPointer.prototype.batchMSEvents = function (fn) { - fn(this.chart.container, hasPointerEvent ? 'pointerdown' : 'MSPointerDown', this.onContainerPointerDown); - fn(this.chart.container, hasPointerEvent ? 'pointermove' : 'MSPointerMove', this.onContainerPointerMove); - fn(doc, hasPointerEvent ? 'pointerup' : 'MSPointerUp', this.onDocumentPointerUp); - }; - // Destroy MS events also - MSPointer.prototype.destroy = function () { - this.batchMSEvents(removeEvent); - _super.prototype.destroy.call(this); - }; - // Disable default IE actions for pinch and such on chart element - MSPointer.prototype.init = function (chart, options) { - _super.prototype.init.call(this, chart, options); - if (this.hasZoom) { // #4014 - css(chart.container, { - '-ms-touch-action': 'none', - 'touch-action': 'none' - }); - } - }; - /** - * @private - * @function Highcharts.Pointer#onContainerPointerDown - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - MSPointer.prototype.onContainerPointerDown = function (e) { - translateMSPointer(e, 'onContainerTouchStart', 'touchstart', function (e) { - touches[e.pointerId] = { - pageX: e.pageX, - pageY: e.pageY, - target: e.currentTarget - }; - }); - }; - /** - * @private - * @function Highcharts.Pointer#onContainerPointerMove - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - MSPointer.prototype.onContainerPointerMove = function (e) { - translateMSPointer(e, 'onContainerTouchMove', 'touchmove', function (e) { - touches[e.pointerId] = ({ pageX: e.pageX, pageY: e.pageY }); - if (!touches[e.pointerId].target) { - touches[e.pointerId].target = e.currentTarget; - } - }); - }; - /** - * @private - * @function Highcharts.Pointer#onDocumentPointerUp - * - * @param {Highcharts.PointerEventObject} e - * - * @return {void} - */ - MSPointer.prototype.onDocumentPointerUp = function (e) { - translateMSPointer(e, 'onDocumentTouchEnd', 'touchend', function (e) { - delete touches[e.pointerId]; - }); - }; - // Add IE specific touch events to chart - MSPointer.prototype.setDOMEvents = function () { - _super.prototype.setDOMEvents.call(this); - if (this.hasZoom || this.followTouchMove) { - this.batchMSEvents(addEvent); - } - }; - return MSPointer; - }(Pointer)); - - return MSPointer; - }); - _registerModule(_modules, 'Core/Legend.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (A, H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var animObject = A.animObject, - setAnimation = A.setAnimation; - /** - * Gets fired when the legend item belonging to a point is clicked. The default - * action is to toggle the visibility of the point. This can be prevented by - * returning `false` or calling `event.preventDefault()`. - * - * @callback Highcharts.PointLegendItemClickCallbackFunction - * - * @param {Highcharts.Point} this - * The point on which the event occured. - * - * @param {Highcharts.PointLegendItemClickEventObject} event - * The event that occured. - */ - /** - * Information about the legend click event. - * - * @interface Highcharts.PointLegendItemClickEventObject - */ /** - * Related browser event. - * @name Highcharts.PointLegendItemClickEventObject#browserEvent - * @type {Highcharts.PointerEvent} - */ /** - * Prevent the default action of toggle the visibility of the point. - * @name Highcharts.PointLegendItemClickEventObject#preventDefault - * @type {Function} - */ /** - * Related point. - * @name Highcharts.PointLegendItemClickEventObject#target - * @type {Highcharts.Point} - */ /** - * Event type. - * @name Highcharts.PointLegendItemClickEventObject#type - * @type {"legendItemClick"} - */ - /** - * Gets fired when the legend item belonging to a series is clicked. The default - * action is to toggle the visibility of the series. This can be prevented by - * returning `false` or calling `event.preventDefault()`. - * - * @callback Highcharts.SeriesLegendItemClickCallbackFunction - * - * @param {Highcharts.Series} this - * The series where the event occured. - * - * @param {Highcharts.SeriesLegendItemClickEventObject} event - * The event that occured. - */ - /** - * Information about the legend click event. - * - * @interface Highcharts.SeriesLegendItemClickEventObject - */ /** - * Related browser event. - * @name Highcharts.SeriesLegendItemClickEventObject#browserEvent - * @type {Highcharts.PointerEvent} - */ /** - * Prevent the default action of toggle the visibility of the series. - * @name Highcharts.SeriesLegendItemClickEventObject#preventDefault - * @type {Function} - */ /** - * Related series. - * @name Highcharts.SeriesLegendItemClickEventObject#target - * @type {Highcharts.Series} - */ /** - * Event type. - * @name Highcharts.SeriesLegendItemClickEventObject#type - * @type {"legendItemClick"} - */ - var addEvent = U.addEvent, - css = U.css, - defined = U.defined, - discardElement = U.discardElement, - find = U.find, - fireEvent = U.fireEvent, - format = U.format, - isNumber = U.isNumber, - merge = U.merge, - pick = U.pick, - relativeLength = U.relativeLength, - stableSort = U.stableSort, - syncTimeout = U.syncTimeout, - wrap = U.wrap; - var isFirefox = H.isFirefox, - marginNames = H.marginNames, - win = H.win; - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The overview of the chart's series. The legend object is instanciated - * internally in the chart constructor, and is available from the `chart.legend` - * property. Each chart has only one legend. - * - * @class - * @name Highcharts.Legend - * - * @param {Highcharts.Chart} chart - * The chart instance. - * - * @param {Highcharts.LegendOptions} options - * Legend options. - */ - var Legend = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function Legend(chart, options) { - /* * - * - * Properties - * - * */ - this.allItems = []; - this.box = void 0; - this.contentGroup = void 0; - this.display = false; - this.group = void 0; - this.initialItemY = 0; - this.itemHeight = 0; - this.itemMarginBottom = 0; - this.itemMarginTop = 0; - this.itemX = 0; - this.itemY = 0; - this.lastItemY = 0; - this.lastLineHeight = 0; - this.legendHeight = 0; - this.legendWidth = 0; - this.maxItemWidth = 0; - this.maxLegendWidth = 0; - this.offsetWidth = 0; - this.options = {}; - this.padding = 0; - this.pages = []; - this.proximate = false; - this.scrollGroup = void 0; - this.symbolHeight = 0; - this.symbolWidth = 0; - this.titleHeight = 0; - this.totalItemWidth = 0; - this.widthOption = 0; - this.chart = chart; - this.init(chart, options); - } - /* * - * - * Functions - * - * */ - /** - * Initialize the legend. - * - * @private - * @function Highcharts.Legend#init - * - * @param {Highcharts.Chart} chart - * The chart instance. - * - * @param {Highcharts.LegendOptions} options - * Legend options. - */ - Legend.prototype.init = function (chart, options) { - /** - * Chart of this legend. - * - * @readonly - * @name Highcharts.Legend#chart - * @type {Highcharts.Chart} - */ - this.chart = chart; - this.setOptions(options); - if (options.enabled) { - // Render it - this.render(); - // move checkboxes - addEvent(this.chart, 'endResize', function () { - this.legend.positionCheckboxes(); - }); - if (this.proximate) { - this.unchartrender = addEvent(this.chart, 'render', function () { - this.legend.proximatePositions(); - this.legend.positionItems(); - }); - } - else if (this.unchartrender) { - this.unchartrender(); - } - } - }; - /** - * @private - * @function Highcharts.Legend#setOptions - * @param {Highcharts.LegendOptions} options - */ - Legend.prototype.setOptions = function (options) { - var padding = pick(options.padding, 8); - /** - * Legend options. - * - * @readonly - * @name Highcharts.Legend#options - * @type {Highcharts.LegendOptions} - */ - this.options = options; - if (!this.chart.styledMode) { - this.itemStyle = options.itemStyle; - this.itemHiddenStyle = merge(this.itemStyle, options.itemHiddenStyle); - } - this.itemMarginTop = options.itemMarginTop || 0; - this.itemMarginBottom = options.itemMarginBottom || 0; - this.padding = padding; - this.initialItemY = padding - 5; // 5 is pixels above the text - this.symbolWidth = pick(options.symbolWidth, 16); - this.pages = []; - this.proximate = options.layout === 'proximate' && !this.chart.inverted; - this.baseline = void 0; // #12705: baseline has to be reset on every update - }; - /** - * Update the legend with new options. Equivalent to running `chart.update` - * with a legend configuration option. - * - * @sample highcharts/legend/legend-update/ - * Legend update - * - * @function Highcharts.Legend#update - * - * @param {Highcharts.LegendOptions} options - * Legend options. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the axis is altered. If doing more - * operations on the chart, it is a good idea to set redraw to false and - * call {@link Chart#redraw} after. Whether to redraw the chart. - * - * @fires Highcharts.Legends#event:afterUpdate - */ - Legend.prototype.update = function (options, redraw) { - var chart = this.chart; - this.setOptions(merge(true, this.options, options)); - this.destroy(); - chart.isDirtyLegend = chart.isDirtyBox = true; - if (pick(redraw, true)) { - chart.redraw(); - } - fireEvent(this, 'afterUpdate'); - }; - /** - * Set the colors for the legend item. - * - * @private - * @function Highcharts.Legend#colorizeItem - * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item - * A Series or Point instance - * @param {boolean} [visible=false] - * Dimmed or colored - * - * @todo - * Make events official: Fires the event `afterColorizeItem`. - */ - Legend.prototype.colorizeItem = function (item, visible) { - item.legendGroup[visible ? 'removeClass' : 'addClass']('highcharts-legend-item-hidden'); - if (!this.chart.styledMode) { - var legend = this, - options = legend.options, - legendItem = item.legendItem, - legendLine = item.legendLine, - legendSymbol = item.legendSymbol, - hiddenColor = legend.itemHiddenStyle.color, - textColor = visible ? - options.itemStyle.color : - hiddenColor, - symbolColor = visible ? - (item.color || hiddenColor) : - hiddenColor, - markerOptions = item.options && item.options.marker, - symbolAttr = { fill: symbolColor }; - if (legendItem) { - legendItem.css({ - fill: textColor, - color: textColor // #1553, oldIE - }); - } - if (legendLine) { - legendLine.attr({ stroke: symbolColor }); - } - if (legendSymbol) { - // Apply marker options - if (markerOptions && legendSymbol.isMarker) { // #585 - symbolAttr = item.pointAttribs(); - if (!visible) { - // #6769 - symbolAttr.stroke = symbolAttr.fill = hiddenColor; - } - } - legendSymbol.attr(symbolAttr); - } - } - fireEvent(this, 'afterColorizeItem', { item: item, visible: visible }); - }; - /** - * @private - * @function Highcharts.Legend#positionItems - */ - Legend.prototype.positionItems = function () { - // Now that the legend width and height are established, put the items - // in the final position - this.allItems.forEach(this.positionItem, this); - if (!this.chart.isResizing) { - this.positionCheckboxes(); - } - }; - /** - * Position the legend item. - * - * @private - * @function Highcharts.Legend#positionItem - * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item - * The item to position - */ - Legend.prototype.positionItem = function (item) { - var _this = this; - var legend = this, - options = legend.options, - symbolPadding = options.symbolPadding, - ltr = !options.rtl, - legendItemPos = item._legendItemPos, - itemX = legendItemPos[0], - itemY = legendItemPos[1], - checkbox = item.checkbox, - legendGroup = item.legendGroup; - if (legendGroup && legendGroup.element) { - var attribs = { - translateX: ltr ? - itemX : - legend.legendWidth - itemX - 2 * symbolPadding - 4, - translateY: itemY - }; - var complete = function () { - fireEvent(_this, 'afterPositionItem', { item: item }); - }; - if (defined(legendGroup.translateY)) { - legendGroup.animate(attribs, void 0, complete); - } - else { - legendGroup.attr(attribs); - complete(); - } - } - if (checkbox) { - checkbox.x = itemX; - checkbox.y = itemY; - } - }; - /** - * Destroy a single legend item, used internally on removing series items. - * - * @private - * @function Highcharts.Legend#destroyItem - * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item - * The item to remove - */ - Legend.prototype.destroyItem = function (item) { - var checkbox = item.checkbox; - // destroy SVG elements - ['legendItem', 'legendLine', 'legendSymbol', 'legendGroup'].forEach(function (key) { - if (item[key]) { - item[key] = item[key].destroy(); - } - }); - if (checkbox) { - discardElement(item.checkbox); - } - }; - /** - * Destroy the legend. Used internally. To reflow objects, `chart.redraw` - * must be called after destruction. - * - * @private - * @function Highcharts.Legend#destroy - */ - Legend.prototype.destroy = function () { - /** - * @private - * @param {string} key - * @return {void} - */ - function destroyItems(key) { - if (this[key]) { - this[key] = this[key].destroy(); - } - } - // Destroy items - this.getAllItems().forEach(function (item) { - ['legendItem', 'legendGroup'].forEach(destroyItems, item); - }); - // Destroy legend elements - [ - 'clipRect', - 'up', - 'down', - 'pager', - 'nav', - 'box', - 'title', - 'group' - ].forEach(destroyItems, this); - this.display = null; // Reset in .render on update. - }; - /** - * Position the checkboxes after the width is determined. - * - * @private - * @function Highcharts.Legend#positionCheckboxes - */ - Legend.prototype.positionCheckboxes = function () { - var alignAttr = this.group && this.group.alignAttr, - translateY, - clipHeight = this.clipHeight || this.legendHeight, - titleHeight = this.titleHeight; - if (alignAttr) { - translateY = alignAttr.translateY; - this.allItems.forEach(function (item) { - var checkbox = item.checkbox, - top; - if (checkbox) { - top = translateY + titleHeight + checkbox.y + - (this.scrollOffset || 0) + 3; - css(checkbox, { - left: (alignAttr.translateX + item.checkboxOffset + - checkbox.x - 20) + 'px', - top: top + 'px', - display: this.proximate || (top > translateY - 6 && - top < translateY + clipHeight - 6) ? - '' : - 'none' - }); - } - }, this); - } - }; - /** - * Render the legend title on top of the legend. - * - * @private - * @function Highcharts.Legend#renderTitle - */ - Legend.prototype.renderTitle = function () { - var options = this.options, - padding = this.padding, - titleOptions = options.title, - titleHeight = 0, - bBox; - if (titleOptions.text) { - if (!this.title) { - /** - * SVG element of the legend title. - * - * @readonly - * @name Highcharts.Legend#title - * @type {Highcharts.SVGElement} - */ - this.title = this.chart.renderer.label(titleOptions.text, padding - 3, padding - 4, null, null, null, options.useHTML, null, 'legend-title') - .attr({ zIndex: 1 }); - if (!this.chart.styledMode) { - this.title.css(titleOptions.style); - } - this.title.add(this.group); - } - // Set the max title width (#7253) - if (!titleOptions.width) { - this.title.css({ - width: this.maxLegendWidth + 'px' - }); - } - bBox = this.title.getBBox(); - titleHeight = bBox.height; - this.offsetWidth = bBox.width; // #1717 - this.contentGroup.attr({ translateY: titleHeight }); - } - this.titleHeight = titleHeight; - }; - /** - * Set the legend item text. - * - * @function Highcharts.Legend#setText - * @param {Highcharts.Point|Highcharts.Series} item - * The item for which to update the text in the legend. - */ - Legend.prototype.setText = function (item) { - var options = this.options; - item.legendItem.attr({ - text: options.labelFormat ? - format(options.labelFormat, item, this.chart) : - options.labelFormatter.call(item) - }); - }; - /** - * Render a single specific legend item. Called internally from the `render` - * function. - * - * @private - * @function Highcharts.Legend#renderItem - * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item - * The item to render. - */ - Legend.prototype.renderItem = function (item) { - var legend = this, - chart = legend.chart, - renderer = chart.renderer, - options = legend.options, - horizontal = options.layout === 'horizontal', - symbolWidth = legend.symbolWidth, - symbolPadding = options.symbolPadding, - itemStyle = legend.itemStyle, - itemHiddenStyle = legend.itemHiddenStyle, - itemDistance = horizontal ? pick(options.itemDistance, 20) : 0, - ltr = !options.rtl, - bBox, - li = item.legendItem, - isSeries = !item.series, - series = !isSeries && item.series.drawLegendSymbol ? - item.series : - item, - seriesOptions = series.options, - showCheckbox = legend.createCheckboxForItem && - seriesOptions && - seriesOptions.showCheckbox, - // full width minus text width - itemExtraWidth = symbolWidth + symbolPadding + - itemDistance + (showCheckbox ? 20 : 0), - useHTML = options.useHTML, - itemClassName = item.options.className; - if (!li) { // generate it once, later move it - // Generate the group box, a group to hold the symbol and text. Text - // is to be appended in Legend class. - item.legendGroup = renderer - .g('legend-item') - .addClass('highcharts-' + series.type + '-series ' + - 'highcharts-color-' + item.colorIndex + - (itemClassName ? ' ' + itemClassName : '') + - (isSeries ? - ' highcharts-series-' + item.index : - '')) - .attr({ zIndex: 1 }) - .add(legend.scrollGroup); - // Generate the list item text and add it to the group - item.legendItem = li = renderer.text('', ltr ? - symbolWidth + symbolPadding : - -symbolPadding, legend.baseline || 0, useHTML); - if (!chart.styledMode) { - // merge to prevent modifying original (#1021) - li.css(merge(item.visible ? - itemStyle : - itemHiddenStyle)); - } - li - .attr({ - align: ltr ? 'left' : 'right', - zIndex: 2 - }) - .add(item.legendGroup); - // Get the baseline for the first item - the font size is equal for - // all - if (!legend.baseline) { - legend.fontMetrics = renderer.fontMetrics(chart.styledMode ? 12 : itemStyle.fontSize, li); - legend.baseline = - legend.fontMetrics.f + 3 + legend.itemMarginTop; - li.attr('y', legend.baseline); - } - // Draw the legend symbol inside the group box - legend.symbolHeight = - options.symbolHeight || legend.fontMetrics.f; - series.drawLegendSymbol(legend, item); - if (legend.setItemEvents) { - legend.setItemEvents(item, li, useHTML); - } - } - // Add the HTML checkbox on top - if (showCheckbox && !item.checkbox && legend.createCheckboxForItem) { - legend.createCheckboxForItem(item); - } - // Colorize the items - legend.colorizeItem(item, item.visible); - // Take care of max width and text overflow (#6659) - if (chart.styledMode || !itemStyle.width) { - li.css({ - width: ((options.itemWidth || - legend.widthOption || - chart.spacingBox.width) - itemExtraWidth) + 'px' - }); - } - // Always update the text - legend.setText(item); - // calculate the positions for the next line - bBox = li.getBBox(); - item.itemWidth = item.checkboxOffset = - options.itemWidth || - item.legendItemWidth || - bBox.width + itemExtraWidth; - legend.maxItemWidth = Math.max(legend.maxItemWidth, item.itemWidth); - legend.totalItemWidth += item.itemWidth; - legend.itemHeight = item.itemHeight = Math.round(item.legendItemHeight || bBox.height || legend.symbolHeight); - }; - /** - * Get the position of the item in the layout. We now know the - * maxItemWidth from the previous loop. - * - * @private - * @function Highcharts.Legend#layoutItem - * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item - */ - Legend.prototype.layoutItem = function (item) { - var options = this.options, - padding = this.padding, - horizontal = options.layout === 'horizontal', - itemHeight = item.itemHeight, - itemMarginBottom = this.itemMarginBottom, - itemMarginTop = this.itemMarginTop, - itemDistance = horizontal ? pick(options.itemDistance, 20) : 0, - maxLegendWidth = this.maxLegendWidth, - itemWidth = (options.alignColumns && - this.totalItemWidth > maxLegendWidth) ? - this.maxItemWidth : - item.itemWidth; - // If the item exceeds the width, start a new line - if (horizontal && - this.itemX - padding + itemWidth > maxLegendWidth) { - this.itemX = padding; - if (this.lastLineHeight) { // Not for the first line (#10167) - this.itemY += (itemMarginTop + - this.lastLineHeight + - itemMarginBottom); - } - this.lastLineHeight = 0; // reset for next line (#915, #3976) - } - // Set the edge positions - this.lastItemY = itemMarginTop + this.itemY + itemMarginBottom; - this.lastLineHeight = Math.max(// #915 - itemHeight, this.lastLineHeight); - // cache the position of the newly generated or reordered items - item._legendItemPos = [this.itemX, this.itemY]; - // advance - if (horizontal) { - this.itemX += itemWidth; - } - else { - this.itemY += - itemMarginTop + itemHeight + itemMarginBottom; - this.lastLineHeight = itemHeight; - } - // the width of the widest item - this.offsetWidth = this.widthOption || Math.max((horizontal ? this.itemX - padding - (item.checkbox ? - // decrease by itemDistance only when no checkbox #4853 - 0 : - itemDistance) : itemWidth) + padding, this.offsetWidth); - }; - /** - * Get all items, which is one item per series for most series and one - * item per point for pie series and its derivatives. Fires the event - * `afterGetAllItems`. - * - * @private - * @function Highcharts.Legend#getAllItems - * @return {Array<(Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series)>} - * The current items in the legend. - * @fires Highcharts.Legend#event:afterGetAllItems - */ - Legend.prototype.getAllItems = function () { - var allItems = []; - this.chart.series.forEach(function (series) { - var seriesOptions = series && series.options; - // Handle showInLegend. If the series is linked to another series, - // defaults to false. - if (series && pick(seriesOptions.showInLegend, !defined(seriesOptions.linkedTo) ? void 0 : false, true)) { - // Use points or series for the legend item depending on - // legendType - allItems = allItems.concat(series.legendItems || - (seriesOptions.legendType === 'point' ? - series.data : - series)); - } - }); - fireEvent(this, 'afterGetAllItems', { allItems: allItems }); - return allItems; - }; - /** - * Get a short, three letter string reflecting the alignment and layout. - * - * @private - * @function Highcharts.Legend#getAlignment - * @return {string} - * The alignment, empty string if floating - */ - Legend.prototype.getAlignment = function () { - var options = this.options; - // Use the first letter of each alignment option in order to detect - // the side. (#4189 - use charAt(x) notation instead of [x] for IE7) - if (this.proximate) { - return options.align.charAt(0) + 'tv'; - } - return options.floating ? '' : (options.align.charAt(0) + - options.verticalAlign.charAt(0) + - options.layout.charAt(0)); - }; - /** - * Adjust the chart margins by reserving space for the legend on only one - * side of the chart. If the position is set to a corner, top or bottom is - * reserved for horizontal legends and left or right for vertical ones. - * - * @private - * @function Highcharts.Legend#adjustMargins - * @param {Array<number>} margin - * @param {Array<number>} spacing - */ - Legend.prototype.adjustMargins = function (margin, spacing) { - var chart = this.chart, - options = this.options, - alignment = this.getAlignment(); - if (alignment) { - ([ - /(lth|ct|rth)/, - /(rtv|rm|rbv)/, - /(rbh|cb|lbh)/, - /(lbv|lm|ltv)/ - ]).forEach(function (alignments, side) { - if (alignments.test(alignment) && !defined(margin[side])) { - // Now we have detected on which side of the chart we should - // reserve space for the legend - chart[marginNames[side]] = Math.max(chart[marginNames[side]], (chart.legend[(side + 1) % 2 ? 'legendHeight' : 'legendWidth'] + - [1, -1, -1, 1][side] * options[(side % 2) ? 'x' : 'y'] + - pick(options.margin, 12) + - spacing[side] + - (chart.titleOffset[side] || 0))); - } - }); - } - }; - /** - * @private - * @function Highcharts.Legend#proximatePositions - */ - Legend.prototype.proximatePositions = function () { - var chart = this.chart, - boxes = [], - alignLeft = this.options.align === 'left'; - this.allItems.forEach(function (item) { - var lastPoint, - height, - useFirstPoint = alignLeft, - target, - top; - if (item.yAxis) { - if (item.xAxis.options.reversed) { - useFirstPoint = !useFirstPoint; - } - if (item.points) { - lastPoint = find(useFirstPoint ? - item.points : - item.points.slice(0).reverse(), function (item) { - return isNumber(item.plotY); - }); - } - height = this.itemMarginTop + - item.legendItem.getBBox().height + - this.itemMarginBottom; - top = item.yAxis.top - chart.plotTop; - if (item.visible) { - target = lastPoint ? - lastPoint.plotY : - item.yAxis.height; - target += top - 0.3 * height; - } - else { - target = top + item.yAxis.height; - } - boxes.push({ - target: target, - size: height, - item: item - }); - } - }, this); - H.distribute(boxes, chart.plotHeight); - boxes.forEach(function (box) { - box.item._legendItemPos[1] = - chart.plotTop - chart.spacing[0] + box.pos; - }); - }; - /** - * Render the legend. This method can be called both before and after - * `chart.render`. If called after, it will only rearrange items instead - * of creating new ones. Called internally on initial render and after - * redraws. - * - * @private - * @function Highcharts.Legend#render - */ - Legend.prototype.render = function () { - var legend = this, - chart = legend.chart, - renderer = chart.renderer, - legendGroup = legend.group, - allItems, - display, - legendWidth, - legendHeight, - box = legend.box, - options = legend.options, - padding = legend.padding, - allowedWidth; - legend.itemX = padding; - legend.itemY = legend.initialItemY; - legend.offsetWidth = 0; - legend.lastItemY = 0; - legend.widthOption = relativeLength(options.width, chart.spacingBox.width - padding); - // Compute how wide the legend is allowed to be - allowedWidth = - chart.spacingBox.width - 2 * padding - options.x; - if (['rm', 'lm'].indexOf(legend.getAlignment().substring(0, 2)) > -1) { - allowedWidth /= 2; - } - legend.maxLegendWidth = legend.widthOption || allowedWidth; - if (!legendGroup) { - /** - * SVG group of the legend. - * - * @readonly - * @name Highcharts.Legend#group - * @type {Highcharts.SVGElement} - */ - legend.group = legendGroup = renderer.g('legend') - .attr({ zIndex: 7 }) - .add(); - legend.contentGroup = renderer.g() - .attr({ zIndex: 1 }) // above background - .add(legendGroup); - legend.scrollGroup = renderer.g() - .add(legend.contentGroup); - } - legend.renderTitle(); - // add each series or point - allItems = legend.getAllItems(); - // sort by legendIndex - stableSort(allItems, function (a, b) { - return ((a.options && a.options.legendIndex) || 0) - - ((b.options && b.options.legendIndex) || 0); - }); - // reversed legend - if (options.reversed) { - allItems.reverse(); - } - /** - * All items for the legend, which is an array of series for most series - * and an array of points for pie series and its derivatives. - * - * @readonly - * @name Highcharts.Legend#allItems - * @type {Array<(Highcharts.Point|Highcharts.Series)>} - */ - legend.allItems = allItems; - legend.display = display = !!allItems.length; - // Render the items. First we run a loop to set the text and properties - // and read all the bounding boxes. The next loop computes the item - // positions based on the bounding boxes. - legend.lastLineHeight = 0; - legend.maxItemWidth = 0; - legend.totalItemWidth = 0; - legend.itemHeight = 0; - allItems.forEach(legend.renderItem, legend); - allItems.forEach(legend.layoutItem, legend); - // Get the box - legendWidth = (legend.widthOption || legend.offsetWidth) + padding; - legendHeight = legend.lastItemY + legend.lastLineHeight + - legend.titleHeight; - legendHeight = legend.handleOverflow(legendHeight); - legendHeight += padding; - // Draw the border and/or background - if (!box) { - /** - * SVG element of the legend box. - * - * @readonly - * @name Highcharts.Legend#box - * @type {Highcharts.SVGElement} - */ - legend.box = box = renderer.rect() - .addClass('highcharts-legend-box') - .attr({ - r: options.borderRadius - }) - .add(legendGroup); - box.isNew = true; - } - // Presentational - if (!chart.styledMode) { - box - .attr({ - stroke: options.borderColor, - 'stroke-width': options.borderWidth || 0, - fill: options.backgroundColor || 'none' - }) - .shadow(options.shadow); - } - if (legendWidth > 0 && legendHeight > 0) { - box[box.isNew ? 'attr' : 'animate'](box.crisp.call({}, { - x: 0, - y: 0, - width: legendWidth, - height: legendHeight - }, box.strokeWidth())); - box.isNew = false; - } - // hide the border if no items - box[display ? 'show' : 'hide'](); - // Open for responsiveness - if (chart.styledMode && legendGroup.getStyle('display') === 'none') { - legendWidth = legendHeight = 0; - } - legend.legendWidth = legendWidth; - legend.legendHeight = legendHeight; - if (display) { - legend.align(); - } - if (!this.proximate) { - this.positionItems(); - } - fireEvent(this, 'afterRender'); - }; - /** - * Align the legend to chart's box. - * - * @private - * @function Highcharts.align - * @param {Highcharts.BBoxObject} alignTo - * @return {void} - */ - Legend.prototype.align = function (alignTo) { - if (alignTo === void 0) { alignTo = this.chart.spacingBox; } - var chart = this.chart, - options = this.options; - // If aligning to the top and the layout is horizontal, adjust for - // the title (#7428) - var y = alignTo.y; - if (/(lth|ct|rth)/.test(this.getAlignment()) && - chart.titleOffset[0] > 0) { - y += chart.titleOffset[0]; - } - else if (/(lbh|cb|rbh)/.test(this.getAlignment()) && - chart.titleOffset[2] > 0) { - y -= chart.titleOffset[2]; - } - if (y !== alignTo.y) { - alignTo = merge(alignTo, { y: y }); - } - this.group.align(merge(options, { - width: this.legendWidth, - height: this.legendHeight, - verticalAlign: this.proximate ? 'top' : options.verticalAlign - }), true, alignTo); - }; - /** - * Set up the overflow handling by adding navigation with up and down arrows - * below the legend. - * - * @private - * @function Highcharts.Legend#handleOverflow - * @param {number} legendHeight - * @return {number} - */ - Legend.prototype.handleOverflow = function (legendHeight) { - var legend = this, - chart = this.chart, - renderer = chart.renderer, - options = this.options, - optionsY = options.y, - alignTop = options.verticalAlign === 'top', - padding = this.padding, - spaceHeight = (chart.spacingBox.height + - (alignTop ? -optionsY : optionsY) - padding), - maxHeight = options.maxHeight, - clipHeight, - clipRect = this.clipRect, - navOptions = options.navigation, - animation = pick(navOptions.animation, - true), - arrowSize = navOptions.arrowSize || 12, - nav = this.nav, - pages = this.pages, - lastY, - allItems = this.allItems, - clipToHeight = function (height) { - if (typeof height === 'number') { - clipRect.attr({ - height: height - }); - } - else if (clipRect) { // Reset (#5912) - legend.clipRect = clipRect.destroy(); - legend.contentGroup.clip(); - } - // useHTML - if (legend.contentGroup.div) { - legend.contentGroup.div.style.clip = height ? - 'rect(' + padding + 'px,9999px,' + - (padding + height) + 'px,0)' : - 'auto'; - } - }, addTracker = function (key) { - legend[key] = renderer - .circle(0, 0, arrowSize * 1.3) - .translate(arrowSize / 2, arrowSize / 2) - .add(nav); - if (!chart.styledMode) { - legend[key].attr('fill', 'rgba(0,0,0,0.0001)'); - } - return legend[key]; - }; - // Adjust the height - if (options.layout === 'horizontal' && - options.verticalAlign !== 'middle' && - !options.floating) { - spaceHeight /= 2; - } - if (maxHeight) { - spaceHeight = Math.min(spaceHeight, maxHeight); - } - // Reset the legend height and adjust the clipping rectangle - pages.length = 0; - if (legendHeight > spaceHeight && - navOptions.enabled !== false) { - this.clipHeight = clipHeight = - Math.max(spaceHeight - 20 - this.titleHeight - padding, 0); - this.currentPage = pick(this.currentPage, 1); - this.fullHeight = legendHeight; - // Fill pages with Y positions so that the top of each a legend item - // defines the scroll top for each page (#2098) - allItems.forEach(function (item, i) { - var y = item._legendItemPos[1], - h = Math.round(item.legendItem.getBBox().height), - len = pages.length; - if (!len || (y - pages[len - 1] > clipHeight && - (lastY || y) !== pages[len - 1])) { - pages.push(lastY || y); - len++; - } - // Keep track of which page each item is on - item.pageIx = len - 1; - if (lastY) { - allItems[i - 1].pageIx = len - 1; - } - if (i === allItems.length - 1 && - y + h - pages[len - 1] > clipHeight && - y !== lastY // #2617 - ) { - pages.push(y); - item.pageIx = len; - } - if (y !== lastY) { - lastY = y; - } - }); - // Only apply clipping if needed. Clipping causes blurred legend in - // PDF export (#1787) - if (!clipRect) { - clipRect = legend.clipRect = - renderer.clipRect(0, padding, 9999, 0); - legend.contentGroup.clip(clipRect); - } - clipToHeight(clipHeight); - // Add navigation elements - if (!nav) { - this.nav = nav = renderer.g() - .attr({ zIndex: 1 }) - .add(this.group); - this.up = renderer - .symbol('triangle', 0, 0, arrowSize, arrowSize) - .add(nav); - addTracker('upTracker') - .on('click', function () { - legend.scroll(-1, animation); - }); - this.pager = renderer.text('', 15, 10) - .addClass('highcharts-legend-navigation'); - if (!chart.styledMode) { - this.pager.css(navOptions.style); - } - this.pager.add(nav); - this.down = renderer - .symbol('triangle-down', 0, 0, arrowSize, arrowSize) - .add(nav); - addTracker('downTracker') - .on('click', function () { - legend.scroll(1, animation); - }); - } - // Set initial position - legend.scroll(0); - legendHeight = spaceHeight; - // Reset - } - else if (nav) { - clipToHeight(); - this.nav = nav.destroy(); // #6322 - this.scrollGroup.attr({ - translateY: 1 - }); - this.clipHeight = 0; // #1379 - } - return legendHeight; - }; - /** - * Scroll the legend by a number of pages. - * - * @private - * @function Highcharts.Legend#scroll - * - * @param {number} scrollBy - * The number of pages to scroll. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * Whether and how to apply animation. - * - * @return {void} - */ - Legend.prototype.scroll = function (scrollBy, animation) { - var _this = this; - var chart = this.chart, - pages = this.pages, - pageCount = pages.length, - currentPage = this.currentPage + scrollBy, - clipHeight = this.clipHeight, - navOptions = this.options.navigation, - pager = this.pager, - padding = this.padding; - // When resizing while looking at the last page - if (currentPage > pageCount) { - currentPage = pageCount; - } - if (currentPage > 0) { - if (typeof animation !== 'undefined') { - setAnimation(animation, chart); - } - this.nav.attr({ - translateX: padding, - translateY: clipHeight + this.padding + 7 + this.titleHeight, - visibility: 'visible' - }); - [this.up, this.upTracker].forEach(function (elem) { - elem.attr({ - 'class': currentPage === 1 ? - 'highcharts-legend-nav-inactive' : - 'highcharts-legend-nav-active' - }); - }); - pager.attr({ - text: currentPage + '/' + pageCount - }); - [this.down, this.downTracker].forEach(function (elem) { - elem.attr({ - // adjust to text width - x: 18 + this.pager.getBBox().width, - 'class': currentPage === pageCount ? - 'highcharts-legend-nav-inactive' : - 'highcharts-legend-nav-active' - }); - }, this); - if (!chart.styledMode) { - this.up - .attr({ - fill: currentPage === 1 ? - navOptions.inactiveColor : - navOptions.activeColor - }); - this.upTracker - .css({ - cursor: currentPage === 1 ? 'default' : 'pointer' - }); - this.down - .attr({ - fill: currentPage === pageCount ? - navOptions.inactiveColor : - navOptions.activeColor - }); - this.downTracker - .css({ - cursor: currentPage === pageCount ? - 'default' : - 'pointer' - }); - } - this.scrollOffset = -pages[currentPage - 1] + this.initialItemY; - this.scrollGroup.animate({ - translateY: this.scrollOffset - }); - this.currentPage = currentPage; - this.positionCheckboxes(); - // Fire event after scroll animation is complete - var animOptions = animObject(pick(animation, - chart.renderer.globalAnimation, - true)); - syncTimeout(function () { - fireEvent(_this, 'afterScroll', { currentPage: currentPage }); - }, animOptions.duration); - } - }; - return Legend; - }()); - // Workaround for #2030, horizontal legend items not displaying in IE11 Preview, - // and for #2580, a similar drawing flaw in Firefox 26. - // Explore if there's a general cause for this. The problem may be related - // to nested group elements, as the legend item texts are within 4 group - // elements. - if (/Trident\/7\.0/.test(win.navigator && win.navigator.userAgent) || - isFirefox) { - wrap(Legend.prototype, 'positionItem', function (proceed, item) { - var legend = this, - // If chart destroyed in sync, this is undefined (#2030) - runPositionItem = function () { - if (item._legendItemPos) { - proceed.call(legend, - item); - } - }; - // Do it now, for export and to get checkbox placement - runPositionItem(); - // Do it after to work around the core issue - if (!legend.bubbleLegend) { - setTimeout(runPositionItem); - } - }); - } - H.Legend = Legend; - - return H.Legend; - }); - _registerModule(_modules, 'Core/Series/Point.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (A, H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var animObject = A.animObject; - var defined = U.defined, - erase = U.erase, - extend = U.extend, - fireEvent = U.fireEvent, - format = U.format, - getNestedProperty = U.getNestedProperty, - isArray = U.isArray, - isNumber = U.isNumber, - isObject = U.isObject, - syncTimeout = U.syncTimeout, - pick = U.pick, - removeEvent = U.removeEvent, - uniqueKey = U.uniqueKey; - /** - * Function callback when a series point is clicked. Return false to cancel the - * action. - * - * @callback Highcharts.PointClickCallbackFunction - * - * @param {Highcharts.Point} this - * The point where the event occured. - * - * @param {Highcharts.PointClickEventObject} event - * Event arguments. - */ - /** - * Common information for a click event on a series point. - * - * @interface Highcharts.PointClickEventObject - * @extends Highcharts.PointerEventObject - */ /** - * Clicked point. - * @name Highcharts.PointClickEventObject#point - * @type {Highcharts.Point} - */ - /** - * Configuration hash for the data label and tooltip formatters. - * - * @interface Highcharts.PointLabelObject - */ /** - * The point's current color. - * @name Highcharts.PointLabelObject#color - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} - */ /** - * The point's current color index, used in styled mode instead of `color`. The - * color index is inserted in class names used for styling. - * @name Highcharts.PointLabelObject#colorIndex - * @type {number} - */ /** - * The name of the related point. - * @name Highcharts.PointLabelObject#key - * @type {string|undefined} - */ /** - * The percentage for related points in a stacked series or pies. - * @name Highcharts.PointLabelObject#percentage - * @type {number} - */ /** - * The related point. The point name, if defined, is available through - * `this.point.name`. - * @name Highcharts.PointLabelObject#point - * @type {Highcharts.Point} - */ /** - * The related series. The series name is available through `this.series.name`. - * @name Highcharts.PointLabelObject#series - * @type {Highcharts.Series} - */ /** - * The total of values in either a stack for stacked series, or a pie in a pie - * series. - * @name Highcharts.PointLabelObject#total - * @type {number|undefined} - */ /** - * For categorized axes this property holds the category name for the point. For - * other axes it holds the X value. - * @name Highcharts.PointLabelObject#x - * @type {number|string|undefined} - */ /** - * The y value of the point. - * @name Highcharts.PointLabelObject#y - * @type {number|undefined} - */ - /** - * Gets fired when the mouse leaves the area close to the point. - * - * @callback Highcharts.PointMouseOutCallbackFunction - * - * @param {Highcharts.Point} this - * Point where the event occured. - * - * @param {global.PointerEvent} event - * Event that occured. - */ - /** - * Gets fired when the mouse enters the area close to the point. - * - * @callback Highcharts.PointMouseOverCallbackFunction - * - * @param {Highcharts.Point} this - * Point where the event occured. - * - * @param {global.Event} event - * Event that occured. - */ - /** - * The generic point options for all series. - * - * In TypeScript you have to extend `PointOptionsObject` with an additional - * declaration to allow custom data options: - * - * ``` - * declare interface PointOptionsObject { - * customProperty: string; - * } - * ``` - * - * @interface Highcharts.PointOptionsObject - */ - /** - * Possible option types for a data point. Use `null` to indicate a gap. - * - * @typedef {number|string|Highcharts.PointOptionsObject|Array<(number|string|null)>|null} Highcharts.PointOptionsType - */ - /** - * Gets fired when the point is removed using the `.remove()` method. - * - * @callback Highcharts.PointRemoveCallbackFunction - * - * @param {Highcharts.Point} this - * Point where the event occured. - * - * @param {global.Event} event - * Event that occured. - */ - /** - * Possible key values for the point state options. - * - * @typedef {"hover"|"inactive"|"normal"|"select"} Highcharts.PointStateValue - */ - /** - * Gets fired when the point is updated programmatically through the `.update()` - * method. - * - * @callback Highcharts.PointUpdateCallbackFunction - * - * @param {Highcharts.Point} this - * Point where the event occured. - * - * @param {Highcharts.PointUpdateEventObject} event - * Event that occured. - */ - /** - * Information about the update event. - * - * @interface Highcharts.PointUpdateEventObject - * @extends global.Event - */ /** - * Options data of the update event. - * @name Highcharts.PointUpdateEventObject#options - * @type {Highcharts.PointOptionsType} - */ - ''; // detach doclet above - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The Point object. The point objects are generated from the `series.data` - * configuration objects or raw numbers. They can be accessed from the - * `Series.points` array. Other ways to instantiate points are through {@link - * Highcharts.Series#addPoint} or {@link Highcharts.Series#setData}. - * - * @class - * @name Highcharts.Point - */ - var Point = /** @class */ (function () { - function Point() { - /* * - * - * Properties - * - * */ - /** - * For categorized axes this property holds the category name for the - * point. For other axes it holds the X value. - * - * @name Highcharts.Point#category - * @type {string} - */ - this.category = void 0; - /** - * The point's current color index, used in styled mode instead of - * `color`. The color index is inserted in class names used for styling. - * - * @name Highcharts.Point#colorIndex - * @type {number} - */ - this.colorIndex = void 0; - this.formatPrefix = 'point'; - this.id = void 0; - this.isNull = false; - /** - * The name of the point. The name can be given as the first position of the - * point configuration array, or as a `name` property in the configuration: - * - * @example - * // Array config - * data: [ - * ['John', 1], - * ['Jane', 2] - * ] - * - * // Object config - * data: [{ - * name: 'John', - * y: 1 - * }, { - * name: 'Jane', - * y: 2 - * }] - * - * @name Highcharts.Point#name - * @type {string} - */ - this.name = void 0; - /** - * The point's options as applied in the initial configuration, or - * extended through `Point.update`. - * - * In TypeScript you have to extend `PointOptionsObject` via an - * additional interface to allow custom data options: - * - * ``` - * declare interface PointOptionsObject { - * customProperty: string; - * } - * ``` - * - * @name Highcharts.Point#options - * @type {Highcharts.PointOptionsObject} - */ - this.options = void 0; - /** - * The percentage for points in a stacked series or pies. - * - * @name Highcharts.Point#percentage - * @type {number|undefined} - */ - this.percentage = void 0; - this.selected = false; - /** - * The series object associated with the point. - * - * @name Highcharts.Point#series - * @type {Highcharts.Series} - */ - this.series = void 0; - /** - * The total of values in either a stack for stacked series, or a pie in a - * pie series. - * - * @name Highcharts.Point#total - * @type {number|undefined} - */ - this.total = void 0; - /** - * For certain series types, like pie charts, where individual points can - * be shown or hidden. - * - * @name Highcharts.Point#visible - * @type {boolean} - * @default true - */ - this.visible = true; - this.x = void 0; - } - /* * - * - * Functions - * - * */ - /** - * Animate SVG elements associated with the point. - * - * @private - * @function Highcharts.Point#animateBeforeDestroy - */ - Point.prototype.animateBeforeDestroy = function () { - var point = this, - animateParams = { x: point.startXPos, - opacity: 0 }, - isDataLabel, - graphicalProps = point.getGraphicalProps(); - graphicalProps.singular.forEach(function (prop) { - isDataLabel = prop === 'dataLabel'; - point[prop] = point[prop].animate(isDataLabel ? { - x: point[prop].startXPos, - y: point[prop].startYPos, - opacity: 0 - } : animateParams); - }); - graphicalProps.plural.forEach(function (plural) { - point[plural].forEach(function (item) { - if (item.element) { - item.animate(extend({ x: point.startXPos }, (item.startYPos ? { - x: item.startXPos, - y: item.startYPos - } : {}))); - } - }); - }); - }; - /** - * Apply the options containing the x and y data and possible some extra - * properties. Called on point init or from point.update. - * - * @private - * @function Highcharts.Point#applyOptions - * - * @param {Highcharts.PointOptionsType} options - * The point options as defined in series.data. - * - * @param {number} [x] - * Optionally, the x value. - * - * @return {Highcharts.Point} - * The Point instance. - */ - Point.prototype.applyOptions = function (options, x) { - var point = this, - series = point.series, - pointValKey = series.options.pointValKey || series.pointValKey; - options = Point.prototype.optionsToObject.call(this, options); - // copy options directly to point - extend(point, options); - point.options = point.options ? extend(point.options, options) : options; - // Since options are copied into the Point instance, some accidental - // options must be shielded (#5681) - if (options.group) { - delete point.group; - } - if (options.dataLabels) { - delete point.dataLabels; - } - /** - * The y value of the point. - * @name Highcharts.Point#y - * @type {number|undefined} - */ - // For higher dimension series types. For instance, for ranges, point.y - // is mapped to point.low. - if (pointValKey) { - point.y = Point.prototype.getNestedProperty.call(point, pointValKey); - } - point.isNull = pick(point.isValid && !point.isValid(), point.x === null || !isNumber(point.y)); // #3571, check for NaN - point.formatPrefix = point.isNull ? 'null' : 'point'; // #9233, #10874 - // The point is initially selected by options (#5777) - if (point.selected) { - point.state = 'select'; - } - /** - * The x value of the point. - * @name Highcharts.Point#x - * @type {number} - */ - // If no x is set by now, get auto incremented value. All points must - // have an x value, however the y value can be null to create a gap in - // the series - if ('name' in point && - typeof x === 'undefined' && - series.xAxis && - series.xAxis.hasNames) { - point.x = series.xAxis.nameToX(point); - } - if (typeof point.x === 'undefined' && series) { - if (typeof x === 'undefined') { - point.x = series.autoIncrement(point); - } - else { - point.x = x; - } - } - return point; - }; - /** - * Destroy a point to clear memory. Its reference still stays in - * `series.data`. - * - * @private - * @function Highcharts.Point#destroy - */ - Point.prototype.destroy = function () { - var point = this, - series = point.series, - chart = series.chart, - dataSorting = series.options.dataSorting, - hoverPoints = chart.hoverPoints, - globalAnimation = point.series.chart.renderer.globalAnimation, - animation = animObject(globalAnimation), - prop; - /** - * Allow to call after animation. - * @private - */ - function destroyPoint() { - // Remove all events and elements - if (point.graphic || point.dataLabel || point.dataLabels) { - removeEvent(point); - point.destroyElements(); - } - for (prop in point) { // eslint-disable-line guard-for-in - point[prop] = null; - } - } - if (point.legendItem) { // pies have legend items - chart.legend.destroyItem(point); - } - if (hoverPoints) { - point.setState(); - erase(hoverPoints, point); - if (!hoverPoints.length) { - chart.hoverPoints = null; - } - } - if (point === chart.hoverPoint) { - point.onMouseOut(); - } - // Remove properties after animation - if (!dataSorting || !dataSorting.enabled) { - destroyPoint(); - } - else { - this.animateBeforeDestroy(); - syncTimeout(destroyPoint, animation.duration); - } - chart.pointCount--; - }; - /** - * Destroy SVG elements associated with the point. - * - * @private - * @function Highcharts.Point#destroyElements - * @param {Highcharts.Dictionary<number>} [kinds] - */ - Point.prototype.destroyElements = function (kinds) { - var point = this, - props = point.getGraphicalProps(kinds); - props.singular.forEach(function (prop) { - point[prop] = point[prop].destroy(); - }); - props.plural.forEach(function (plural) { - point[plural].forEach(function (item) { - if (item.element) { - item.destroy(); - } - }); - delete point[plural]; - }); - }; - /** - * Fire an event on the Point object. - * - * @private - * @function Highcharts.Point#firePointEvent - * - * @param {string} eventType - * Type of the event. - * - * @param {Highcharts.Dictionary<any>|Event} [eventArgs] - * Additional event arguments. - * - * @param {Highcharts.EventCallbackFunction<Highcharts.Point>|Function} [defaultFunction] - * Default event handler. - * - * @fires Highcharts.Point#event:* - */ - Point.prototype.firePointEvent = function (eventType, eventArgs, defaultFunction) { - var point = this, - series = this.series, - seriesOptions = series.options; - // load event handlers on demand to save time on mouseover/out - if (seriesOptions.point.events[eventType] || - (point.options && - point.options.events && - point.options.events[eventType])) { - point.importEvents(); - } - // add default handler if in selection mode - if (eventType === 'click' && seriesOptions.allowPointSelect) { - defaultFunction = function (event) { - // Control key is for Windows, meta (= Cmd key) for Mac, Shift - // for Opera. - if (point.select) { // #2911 - point.select(null, event.ctrlKey || event.metaKey || event.shiftKey); - } - }; - } - fireEvent(point, eventType, eventArgs, defaultFunction); - }; - /** - * Get the CSS class names for individual points. Used internally where the - * returned value is set on every point. - * - * @function Highcharts.Point#getClassName - * - * @return {string} - * The class names. - */ - Point.prototype.getClassName = function () { - var point = this; - return 'highcharts-point' + - (point.selected ? ' highcharts-point-select' : '') + - (point.negative ? ' highcharts-negative' : '') + - (point.isNull ? ' highcharts-null-point' : '') + - (typeof point.colorIndex !== 'undefined' ? - ' highcharts-color-' + point.colorIndex : '') + - (point.options.className ? ' ' + point.options.className : '') + - (point.zone && point.zone.className ? ' ' + - point.zone.className.replace('highcharts-negative', '') : ''); - }; - /** - * Get props of all existing graphical point elements. - * - * @private - * @function Highcharts.Point#getGraphicalProps - * @param {Highcharts.Dictionary<number>} [kinds] - * @return {Highcharts.PointGraphicalProps} - */ - Point.prototype.getGraphicalProps = function (kinds) { - var point = this, - props = [], - prop, - i, - graphicalProps = { singular: [], - plural: [] }; - kinds = kinds || { graphic: 1, dataLabel: 1 }; - if (kinds.graphic) { - props.push('graphic', 'shadowGroup'); - } - if (kinds.dataLabel) { - props.push('dataLabel', 'dataLabelUpper', 'connector'); - } - i = props.length; - while (i--) { - prop = props[i]; - if (point[prop]) { - graphicalProps.singular.push(prop); - } - } - ['dataLabel', 'connector'].forEach(function (prop) { - var plural = prop + 's'; - if (kinds[prop] && point[plural]) { - graphicalProps.plural.push(plural); - } - }); - return graphicalProps; - }; - /** - * Return the configuration hash needed for the data label and tooltip - * formatters. - * - * @function Highcharts.Point#getLabelConfig - * - * @return {Highcharts.PointLabelObject} - * Abstract object used in formatters and formats. - */ - Point.prototype.getLabelConfig = function () { - return { - x: this.category, - y: this.y, - color: this.color, - colorIndex: this.colorIndex, - key: this.name || this.category, - series: this.series, - point: this, - percentage: this.percentage, - total: this.total || this.stackTotal - }; - }; - /** - * Returns the value of the point property for a given value. - * @private - */ - Point.prototype.getNestedProperty = function (key) { - if (!key) { - return; - } - if (key.indexOf('custom.') === 0) { - return getNestedProperty(key, this.options); - } - return this[key]; - }; - /** - * In a series with `zones`, return the zone that the point belongs to. - * - * @function Highcharts.Point#getZone - * - * @return {Highcharts.SeriesZonesOptionsObject} - * The zone item. - */ - Point.prototype.getZone = function () { - var series = this.series, - zones = series.zones, - zoneAxis = series.zoneAxis || 'y', - i = 0, - zone; - zone = zones[i]; - while (this[zoneAxis] >= zone.value) { - zone = zones[++i]; - } - // For resetting or reusing the point (#8100) - if (!this.nonZonedColor) { - this.nonZonedColor = this.color; - } - if (zone && zone.color && !this.options.color) { - this.color = zone.color; - } - else { - this.color = this.nonZonedColor; - } - return zone; - }; - /** - * Utility to check if point has new shape type. Used in column series and - * all others that are based on column series. - * - * @return boolean|undefined - */ - Point.prototype.hasNewShapeType = function () { - var point = this; - var oldShapeType = point.graphic && - (point.graphic.symbolName || point.graphic.element.nodeName); - return oldShapeType !== this.shapeType; - }; - /** - * Initialize the point. Called internally based on the `series.data` - * option. - * - * @function Highcharts.Point#init - * - * @param {Highcharts.Series} series - * The series object containing this point. - * - * @param {Highcharts.PointOptionsType} options - * The data in either number, array or object format. - * - * @param {number} [x] - * Optionally, the X value of the point. - * - * @return {Highcharts.Point} - * The Point instance. - * - * @fires Highcharts.Point#event:afterInit - */ - Point.prototype.init = function (series, options, x) { - this.series = series; - this.applyOptions(options, x); - // Add a unique ID to the point if none is assigned - this.id = defined(this.id) ? this.id : uniqueKey(); - this.resolveColor(); - series.chart.pointCount++; - fireEvent(this, 'afterInit'); - return this; - }; - /** - * Transform number or array configs into objects. Also called for object - * configs. Used internally to unify the different configuration formats for - * points. For example, a simple number `10` in a line series will be - * transformed to `{ y: 10 }`, and an array config like `[1, 10]` in a - * scatter series will be transformed to `{ x: 1, y: 10 }`. - * - * @function Highcharts.Point#optionsToObject - * - * @param {Highcharts.PointOptionsType} options - * The input option. - * - * @return {Highcharts.Dictionary<*>} - * Transformed options. - */ - Point.prototype.optionsToObject = function (options) { - var ret = {}, - series = this.series, - keys = series.options.keys, - pointArrayMap = keys || series.pointArrayMap || ['y'], - valueCount = pointArrayMap.length, - firstItemType, - i = 0, - j = 0; - if (isNumber(options) || options === null) { - ret[pointArrayMap[0]] = options; - } - else if (isArray(options)) { - // with leading x value - if (!keys && options.length > valueCount) { - firstItemType = typeof options[0]; - if (firstItemType === 'string') { - ret.name = options[0]; - } - else if (firstItemType === 'number') { - ret.x = options[0]; - } - i++; - } - while (j < valueCount) { - // Skip undefined positions for keys - if (!keys || typeof options[i] !== 'undefined') { - if (pointArrayMap[j].indexOf('.') > 0) { - // Handle nested keys, e.g. ['color.pattern.image'] - // Avoid function call unless necessary. - Point.prototype.setNestedProperty(ret, options[i], pointArrayMap[j]); - } - else { - ret[pointArrayMap[j]] = options[i]; - } - } - i++; - j++; - } - } - else if (typeof options === 'object') { - ret = options; - // This is the fastest way to detect if there are individual point - // dataLabels that need to be considered in drawDataLabels. These - // can only occur in object configs. - if (options.dataLabels) { - series._hasPointLabels = true; - } - // Same approach as above for markers - if (options.marker) { - series._hasPointMarkers = true; - } - } - return ret; - }; - /** - * @private - * @function Highcharts.Point#resolveColor - * @return {void} - */ - Point.prototype.resolveColor = function () { - var series = this.series, - colors, - optionsChart = series.chart.options.chart, - colorCount = optionsChart.colorCount, - styledMode = series.chart.styledMode, - colorIndex; - // remove points nonZonedColor for later recalculation - delete this.nonZonedColor; - /** - * The point's current color. - * - * @name Highcharts.Point#color - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} - */ - if (!styledMode && !this.options.color) { - this.color = series.color; // #3445 - } - if (series.options.colorByPoint) { - if (!styledMode) { - colors = series.options.colors || series.chart.options.colors; - this.color = this.color || colors[series.colorCounter]; - colorCount = colors.length; - } - colorIndex = series.colorCounter; - series.colorCounter++; - // loop back to zero - if (series.colorCounter === colorCount) { - series.colorCounter = 0; - } - } - else { - colorIndex = series.colorIndex; - } - this.colorIndex = pick(this.colorIndex, colorIndex); - }; - /** - * Set a value in an object, on the property defined by key. The key - * supports nested properties using dot notation. The function modifies the - * input object and does not make a copy. - * - * @function Highcharts.Point#setNestedProperty<T> - * - * @param {T} object - * The object to set the value on. - * - * @param {*} value - * The value to set. - * - * @param {string} key - * Key to the property to set. - * - * @return {T} - * The modified object. - */ - Point.prototype.setNestedProperty = function (object, value, key) { - var nestedKeys = key.split('.'); - nestedKeys.reduce(function (result, key, i, arr) { - var isLastKey = arr.length - 1 === i; - result[key] = (isLastKey ? - value : - isObject(result[key], true) ? - result[key] : - {}); - return result[key]; - }, object); - return object; - }; - /** - * Extendable method for formatting each point's tooltip line. - * - * @function Highcharts.Point#tooltipFormatter - * - * @param {string} pointFormat - * The point format. - * - * @return {string} - * A string to be concatenated in to the common tooltip text. - */ - Point.prototype.tooltipFormatter = function (pointFormat) { - // Insert options for valueDecimals, valuePrefix, and valueSuffix - var series = this.series, seriesTooltipOptions = series.tooltipOptions, valueDecimals = pick(seriesTooltipOptions.valueDecimals, ''), valuePrefix = seriesTooltipOptions.valuePrefix || '', valueSuffix = seriesTooltipOptions.valueSuffix || ''; - // Replace default point style with class name - if (series.chart.styledMode) { - pointFormat = - series.chart.tooltip.styledModeFormat(pointFormat); - } - // Loop over the point array map and replace unformatted values with - // sprintf formatting markup - (series.pointArrayMap || ['y']).forEach(function (key) { - key = '{point.' + key; // without the closing bracket - if (valuePrefix || valueSuffix) { - pointFormat = pointFormat.replace(RegExp(key + '}', 'g'), valuePrefix + key + '}' + valueSuffix); - } - pointFormat = pointFormat.replace(RegExp(key + '}', 'g'), key + ':,.' + valueDecimals + 'f}'); - }); - return format(pointFormat, { - point: this, - series: this.series - }, series.chart); - }; - return Point; - }()); - H.Point = Point; - - return Point; - }); - _registerModule(_modules, 'Core/Series/Series.js', [_modules['Core/Globals.js'], _modules['Core/Series/Point.js'], _modules['Core/Utilities.js']], function (H, Point, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var error = U.error, - extendClass = U.extendClass, - fireEvent = U.fireEvent, - getOptions = U.getOptions, - isObject = U.isObject, - merge = U.merge, - objectEach = U.objectEach; - /** - * @class - * @name Highcharts.Series - */ - var BaseSeries = /** @class */ (function () { - /* * - * - * Constructor - * - * */ - function BaseSeries(chart, options) { - var mergedOptions = merge(BaseSeries.defaultOptions, options); - this.chart = chart; - this._i = chart.series.length; - chart.series.push(this); - this.options = mergedOptions; - this.userOptions = merge(options); - } - /* * - * - * Static Functions - * - * */ - BaseSeries.addSeries = function (seriesName, seriesType) { - BaseSeries.seriesTypes[seriesName] = seriesType; - }; - BaseSeries.cleanRecursively = function (toClean, reference) { - var clean = {}; - objectEach(toClean, function (_val, key) { - var ob; - // Dive into objects (except DOM nodes) - if (isObject(toClean[key], true) && - !toClean.nodeType && // #10044 - reference[key]) { - ob = BaseSeries.cleanRecursively(toClean[key], reference[key]); - if (Object.keys(ob).length) { - clean[key] = ob; - } - // Arrays, primitives and DOM nodes are copied directly - } - else if (isObject(toClean[key]) || - toClean[key] !== reference[key]) { - clean[key] = toClean[key]; - } - }); - return clean; - }; - // eslint-disable-next-line valid-jsdoc - /** - * Internal function to initialize an individual series. - * @private - */ - BaseSeries.getSeries = function (chart, options) { - if (options === void 0) { options = {}; } - var optionsChart = chart.options.chart, - type = (options.type || - optionsChart.type || - optionsChart.defaultSeriesType || - ''), - Series = BaseSeries.seriesTypes[type]; - // No such series type - if (!Series) { - error(17, true, chart, { missingModuleFor: type }); - } - return new Series(chart, options); - }; - /** - * Factory to create new series prototypes. - * - * @function Highcharts.seriesType - * - * @param {string} type - * The series type name. - * - * @param {string} parent - * The parent series type name. Use `line` to inherit from the basic - * {@link Series} object. - * - * @param {Highcharts.SeriesOptionsType|Highcharts.Dictionary<*>} options - * The additional default options that are merged with the parent's options. - * - * @param {Highcharts.Dictionary<*>} [props] - * The properties (functions and primitives) to set on the new prototype. - * - * @param {Highcharts.Dictionary<*>} [pointProps] - * Members for a series-specific extension of the {@link Point} prototype if - * needed. - * - * @return {Highcharts.Series} - * The newly created prototype as extended from {@link Series} or its - * derivatives. - */ - // docs: add to API + extending Highcharts - BaseSeries.seriesType = function (type, parent, options, seriesProto, pointProto) { - var defaultOptions = getOptions().plotOptions || {}, - seriesTypes = BaseSeries.seriesTypes; - parent = parent || ''; - // Merge the options - defaultOptions[type] = merge(defaultOptions[parent], options); - // Create the class - BaseSeries.addSeries(type, extendClass(seriesTypes[parent] || function () { }, seriesProto)); - seriesTypes[type].prototype.type = type; - // Create the point class if needed - if (pointProto) { - seriesTypes[type].prototype.pointClass = - extendClass(Point, pointProto); - } - return seriesTypes[type]; - }; - BaseSeries.prototype.update = function (newOptions, redraw) { - if (redraw === void 0) { redraw = true; } - var series = this; - newOptions = BaseSeries.cleanRecursively(newOptions, this.userOptions); - var newType = newOptions.type; - if (typeof newType !== 'undefined' && - newType !== series.type) { - series = BaseSeries.getSeries(series.chart, newOptions); - } - fireEvent(series, 'update', { newOptions: newOptions }); - series.userOptions = merge(newOptions); - fireEvent(series, 'afterUpdate', { newOptions: newOptions }); - if (redraw) { - series.chart.redraw(); - } - return series; - }; - /* * - * - * Static Properties - * - * */ - BaseSeries.defaultOptions = { - type: 'base' - }; - BaseSeries.seriesTypes = {}; - return BaseSeries; - }()); - BaseSeries.prototype.pointClass = Point; - // backwards compatibility - H.seriesType = BaseSeries.seriesType; - H.seriesTypes = BaseSeries.seriesTypes; - /* * - * - * Export - * - * */ - - return BaseSeries; - }); - _registerModule(_modules, 'Core/Chart/Chart.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Axis/Axis.js'], _modules['Core/Series/Series.js'], _modules['Core/Globals.js'], _modules['Core/Legend.js'], _modules['Core/MSPointer.js'], _modules['Core/Options.js'], _modules['Core/Pointer.js'], _modules['Core/Time.js'], _modules['Core/Utilities.js']], function (A, Axis, BaseSeries, H, Legend, MSPointer, O, Pointer, Time, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var animate = A.animate, - animObject = A.animObject, - setAnimation = A.setAnimation; - var charts = H.charts, - doc = H.doc, - win = H.win; - var defaultOptions = O.defaultOptions; - var addEvent = U.addEvent, - attr = U.attr, - createElement = U.createElement, - css = U.css, - defined = U.defined, - discardElement = U.discardElement, - erase = U.erase, - error = U.error, - extend = U.extend, - find = U.find, - fireEvent = U.fireEvent, - getStyle = U.getStyle, - isArray = U.isArray, - isFunction = U.isFunction, - isNumber = U.isNumber, - isObject = U.isObject, - isString = U.isString, - merge = U.merge, - numberFormat = U.numberFormat, - objectEach = U.objectEach, - pick = U.pick, - pInt = U.pInt, - relativeLength = U.relativeLength, - removeEvent = U.removeEvent, - splat = U.splat, - syncTimeout = U.syncTimeout, - uniqueKey = U.uniqueKey; - /** - * Callback for chart constructors. - * - * @callback Highcharts.ChartCallbackFunction - * - * @param {Highcharts.Chart} chart - * Created chart. - */ - /** - * Format a number and return a string based on input settings. - * - * @callback Highcharts.NumberFormatterCallbackFunction - * - * @param {number} number - * The input number to format. - * - * @param {number} decimals - * The amount of decimals. A value of -1 preserves the amount in the - * input number. - * - * @param {string} [decimalPoint] - * The decimal point, defaults to the one given in the lang options, or - * a dot. - * - * @param {string} [thousandsSep] - * The thousands separator, defaults to the one given in the lang - * options, or a space character. - * - * @return {string} The formatted number. - */ - /** - * The chart title. The title has an `update` method that allows modifying the - * options directly or indirectly via `chart.update`. - * - * @interface Highcharts.TitleObject - * @extends Highcharts.SVGElement - */ /** - * Modify options for the title. - * - * @function Highcharts.TitleObject#update - * - * @param {Highcharts.TitleOptions} titleOptions - * Options to modify. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the title is altered. If doing more - * operations on the chart, it is a good idea to set redraw to false and - * call {@link Chart#redraw} after. - */ - /** - * The chart subtitle. The subtitle has an `update` method that - * allows modifying the options directly or indirectly via - * `chart.update`. - * - * @interface Highcharts.SubtitleObject - * @extends Highcharts.SVGElement - */ /** - * Modify options for the subtitle. - * - * @function Highcharts.SubtitleObject#update - * - * @param {Highcharts.SubtitleOptions} subtitleOptions - * Options to modify. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the subtitle is altered. If doing - * more operations on the chart, it is a good idea to set redraw to false - * and call {@link Chart#redraw} after. - */ - /** - * The chart caption. The caption has an `update` method that - * allows modifying the options directly or indirectly via - * `chart.update`. - * - * @interface Highcharts.CaptionObject - * @extends Highcharts.SVGElement - */ /** - * Modify options for the caption. - * - * @function Highcharts.CaptionObject#update - * - * @param {Highcharts.CaptionOptions} captionOptions - * Options to modify. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the caption is altered. If doing - * more operations on the chart, it is a good idea to set redraw to false - * and call {@link Chart#redraw} after. - */ - var marginNames = H.marginNames; - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** - * The Chart class. The recommended constructor is {@link Highcharts#chart}. - * - * @example - * var chart = Highcharts.chart('container', { - * title: { - * text: 'My chart' - * }, - * series: [{ - * data: [1, 3, 2, 4] - * }] - * }) - * - * @class - * @name Highcharts.Chart - * - * @param {string|Highcharts.HTMLDOMElement} [renderTo] - * The DOM element to render to, or its id. - * - * @param {Highcharts.Options} options - * The chart options structure. - * - * @param {Highcharts.ChartCallbackFunction} [callback] - * Function to run when the chart has loaded and and all external images - * are loaded. Defining a - * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load) - * handler is equivalent. - */ - var Chart = /** @class */ (function () { - function Chart(a, b, c) { - this.axes = void 0; - this.axisOffset = void 0; - this.bounds = void 0; - this.chartHeight = void 0; - this.chartWidth = void 0; - this.clipBox = void 0; - this.colorCounter = void 0; - this.container = void 0; - this.index = void 0; - this.isResizing = void 0; - this.labelCollectors = void 0; - this.legend = void 0; - this.margin = void 0; - this.numberFormatter = void 0; - this.options = void 0; - this.plotBox = void 0; - this.plotHeight = void 0; - this.plotLeft = void 0; - this.plotTop = void 0; - this.plotWidth = void 0; - this.pointCount = void 0; - this.pointer = void 0; - this.renderer = void 0; - this.renderTo = void 0; - this.series = void 0; - this.spacing = void 0; - this.spacingBox = void 0; - this.symbolCounter = void 0; - this.time = void 0; - this.titleOffset = void 0; - this.userOptions = void 0; - this.xAxis = void 0; - this.yAxis = void 0; - this.getArgs(a, b, c); - } - /* * - * - * Functions - * - * */ - /** - * Handle the arguments passed to the constructor. - * - * @private - * @function Highcharts.Chart#getArgs - * - * @param {...Array<*>} arguments - * All arguments for the constructor. - * - * @fires Highcharts.Chart#event:init - * @fires Highcharts.Chart#event:afterInit - */ - Chart.prototype.getArgs = function (a, b, c) { - // Remove the optional first argument, renderTo, and - // set it on this. - if (isString(a) || a.nodeName) { - this.renderTo = a; - this.init(b, c); - } - else { - this.init(a, b); - } - }; - /** - * Overridable function that initializes the chart. The constructor's - * arguments are passed on directly. - * - * @function Highcharts.Chart#init - * - * @param {Highcharts.Options} userOptions - * Custom options. - * - * @param {Function} [callback] - * Function to run when the chart has loaded and and all external - * images are loaded. - * - * @return {void} - * - * @fires Highcharts.Chart#event:init - * @fires Highcharts.Chart#event:afterInit - */ - Chart.prototype.init = function (userOptions, callback) { - // Handle regular options - var options, - // skip merging data points to increase performance - seriesOptions = userOptions.series, - userPlotOptions = userOptions.plotOptions || {}; - // Fire the event with a default function - fireEvent(this, 'init', { args: arguments }, function () { - userOptions.series = null; - options = merge(defaultOptions, userOptions); // do the merge - var optionsChart = options.chart || {}; - // Override (by copy of user options) or clear tooltip options - // in chart.options.plotOptions (#6218) - objectEach(options.plotOptions, function (typeOptions, type) { - if (isObject(typeOptions)) { // #8766 - typeOptions.tooltip = (userPlotOptions[type] && // override by copy: - merge(userPlotOptions[type].tooltip)) || void 0; // or clear - } - }); - // User options have higher priority than default options - // (#6218). In case of exporting: path is changed - options.tooltip.userOptions = (userOptions.chart && - userOptions.chart.forExport && - userOptions.tooltip.userOptions) || userOptions.tooltip; - // set back the series data - options.series = userOptions.series = seriesOptions; - /** - * The original options given to the constructor or a chart factory - * like {@link Highcharts.chart} and {@link Highcharts.stockChart}. - * - * @name Highcharts.Chart#userOptions - * @type {Highcharts.Options} - */ - this.userOptions = userOptions; - var chartEvents = optionsChart.events; - this.margin = []; - this.spacing = []; - // Pixel data bounds for touch zoom - this.bounds = { h: {}, v: {} }; - // An array of functions that returns labels that should be - // considered for anti-collision - this.labelCollectors = []; - this.callback = callback; - this.isResizing = 0; - /** - * The options structure for the chart after merging - * {@link #defaultOptions} and {@link #userOptions}. It contains - * members for the sub elements like series, legend, tooltip etc. - * - * @name Highcharts.Chart#options - * @type {Highcharts.Options} - */ - this.options = options; - /** - * All the axes in the chart. - * - * @see Highcharts.Chart.xAxis - * @see Highcharts.Chart.yAxis - * - * @name Highcharts.Chart#axes - * @type {Array<Highcharts.Axis>} - */ - this.axes = []; - /** - * All the current series in the chart. - * - * @name Highcharts.Chart#series - * @type {Array<Highcharts.Series>} - */ - this.series = []; - /** - * The `Time` object associated with the chart. Since v6.0.5, - * time settings can be applied individually for each chart. If - * no individual settings apply, the `Time` object is shared by - * all instances. - * - * @name Highcharts.Chart#time - * @type {Highcharts.Time} - */ - this.time = - userOptions.time && Object.keys(userOptions.time).length ? - new Time(userOptions.time) : - H.time; - /** - * Callback function to override the default function that formats - * all the numbers in the chart. Returns a string with the formatted - * number. - * - * @name Highcharts.Chart#numberFormatter - * @type {Highcharts.NumberFormatterCallbackFunction} - */ - this.numberFormatter = optionsChart.numberFormatter || numberFormat; - /** - * Whether the chart is in styled mode, meaning all presentatinoal - * attributes are avoided. - * - * @name Highcharts.Chart#styledMode - * @type {boolean} - */ - this.styledMode = optionsChart.styledMode; - this.hasCartesianSeries = optionsChart.showAxes; - var chart = this; - /** - * Index position of the chart in the {@link Highcharts#charts} - * property. - * - * @name Highcharts.Chart#index - * @type {number} - * @readonly - */ - chart.index = charts.length; // Add the chart to the global lookup - charts.push(chart); - H.chartCount++; - // Chart event handlers - if (chartEvents) { - objectEach(chartEvents, function (event, eventType) { - if (isFunction(event)) { - addEvent(chart, eventType, event); - } - }); - } - /** - * A collection of the X axes in the chart. - * - * @name Highcharts.Chart#xAxis - * @type {Array<Highcharts.Axis>} - */ - chart.xAxis = []; - /** - * A collection of the Y axes in the chart. - * - * @name Highcharts.Chart#yAxis - * @type {Array<Highcharts.Axis>} - * - * @todo - * Make events official: Fire the event `afterInit`. - */ - chart.yAxis = []; - chart.pointCount = chart.colorCounter = chart.symbolCounter = 0; - // Fire after init but before first render, before axes and series - // have been initialized. - fireEvent(chart, 'afterInit'); - chart.firstRender(); - }); - }; - /** - * Internal function to unitialize an individual series. - * - * @private - * @function Highcharts.Chart#initSeries - */ - Chart.prototype.initSeries = function (options) { - var chart = this, - optionsChart = chart.options.chart, - type = (options.type || - optionsChart.type || - optionsChart.defaultSeriesType), - series, - Constr = BaseSeries.seriesTypes[type]; - // No such series type - if (!Constr) { - error(17, true, chart, { missingModuleFor: type }); - } - series = new Constr(chart, options); - if (typeof series.init === 'function') { - series.init(this, options); - } - return series; - }; - /** - * Internal function to set data for all series with enabled sorting. - * - * @private - * @function Highcharts.Chart#setSeriesData - */ - Chart.prototype.setSeriesData = function () { - this.getSeriesOrderByLinks().forEach(function (series) { - // We need to set data for series with sorting after series init - if (!series.points && !series.data && series.enabledDataSorting) { - series.setData(series.options.data, false); - } - }); - }; - /** - * Sort and return chart series in order depending on the number of linked - * series. - * - * @private - * @function Highcharts.Series#getSeriesOrderByLinks - * @return {Array<Highcharts.Series>} - */ - Chart.prototype.getSeriesOrderByLinks = function () { - return this.series.concat().sort(function (a, b) { - if (a.linkedSeries.length || b.linkedSeries.length) { - return b.linkedSeries.length - a.linkedSeries.length; - } - return 0; - }); - }; - /** - * Order all series above a given index. When series are added and ordered - * by configuration, only the last series is handled (#248, #1123, #2456, - * #6112). This function is called on series initialization and destroy. - * - * @private - * @function Highcharts.Series#orderSeries - * @param {number} [fromIndex] - * If this is given, only the series above this index are handled. - */ - Chart.prototype.orderSeries = function (fromIndex) { - var series = this.series, - i = fromIndex || 0; - for (; i < series.length; i++) { - if (series[i]) { - /** - * Contains the series' index in the `Chart.series` array. - * - * @name Highcharts.Series#index - * @type {number} - * @readonly - */ - series[i].index = i; - series[i].name = series[i].getName(); - } - } - }; - /** - * Check whether a given point is within the plot area. - * - * @function Highcharts.Chart#isInsidePlot - * - * @param {number} plotX - * Pixel x relative to the plot area. - * - * @param {number} plotY - * Pixel y relative to the plot area. - * - * @param {boolean} [inverted] - * Whether the chart is inverted. - * - * @return {boolean} - * Returns true if the given point is inside the plot area. - */ - Chart.prototype.isInsidePlot = function (plotX, plotY, inverted) { - var x = inverted ? plotY : plotX, - y = inverted ? plotX : plotY, - e = { - x: x, - y: y, - isInsidePlot: x >= 0 && - x <= this.plotWidth && - y >= 0 && - y <= this.plotHeight - }; - fireEvent(this, 'afterIsInsidePlot', e); - return e.isInsidePlot; - }; - /** - * Redraw the chart after changes have been done to the data, axis extremes - * chart size or chart elements. All methods for updating axes, series or - * points have a parameter for redrawing the chart. This is `true` by - * default. But in many cases you want to do more than one operation on the - * chart before redrawing, for example add a number of points. In those - * cases it is a waste of resources to redraw the chart for each new point - * added. So you add the points and call `chart.redraw()` after. - * - * @function Highcharts.Chart#redraw - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * If or how to apply animation to the redraw. - * - * @fires Highcharts.Chart#event:afterSetExtremes - * @fires Highcharts.Chart#event:beforeRedraw - * @fires Highcharts.Chart#event:predraw - * @fires Highcharts.Chart#event:redraw - * @fires Highcharts.Chart#event:render - * @fires Highcharts.Chart#event:updatedData - */ - Chart.prototype.redraw = function (animation) { - fireEvent(this, 'beforeRedraw'); - var chart = this, - axes = chart.axes, - series = chart.series, - pointer = chart.pointer, - legend = chart.legend, - legendUserOptions = chart.userOptions.legend, - redrawLegend = chart.isDirtyLegend, - hasStackedSeries, - hasDirtyStacks, - hasCartesianSeries = chart.hasCartesianSeries, - isDirtyBox = chart.isDirtyBox, - i, - serie, - renderer = chart.renderer, - isHiddenChart = renderer.isHidden(), - afterRedraw = []; - // Handle responsive rules, not only on resize (#6130) - if (chart.setResponsive) { - chart.setResponsive(false); - } - // Set the global animation. When chart.hasRendered is not true, the - // redraw call comes from a responsive rule and animation should not - // occur. - setAnimation(chart.hasRendered ? animation : false, chart); - if (isHiddenChart) { - chart.temporaryDisplay(); - } - // Adjust title layout (reflow multiline text) - chart.layOutTitles(); - // link stacked series - i = series.length; - while (i--) { - serie = series[i]; - if (serie.options.stacking) { - hasStackedSeries = true; - if (serie.isDirty) { - hasDirtyStacks = true; - break; - } - } - } - if (hasDirtyStacks) { // mark others as dirty - i = series.length; - while (i--) { - serie = series[i]; - if (serie.options.stacking) { - serie.isDirty = true; - } - } - } - // Handle updated data in the series - series.forEach(function (serie) { - if (serie.isDirty) { - if (serie.options.legendType === 'point') { - if (typeof serie.updateTotals === 'function') { - serie.updateTotals(); - } - redrawLegend = true; - } - else if (legendUserOptions && - (legendUserOptions.labelFormatter || - legendUserOptions.labelFormat)) { - redrawLegend = true; // #2165 - } - } - if (serie.isDirtyData) { - fireEvent(serie, 'updatedData'); - } - }); - // handle added or removed series - if (redrawLegend && legend && legend.options.enabled) { - // draw legend graphics - legend.render(); - chart.isDirtyLegend = false; - } - // reset stacks - if (hasStackedSeries) { - chart.getStacks(); - } - if (hasCartesianSeries) { - // set axes scales - axes.forEach(function (axis) { - // Don't do setScale again if we're only resizing. Regression - // #13507. But we need it after chart.update (responsive), as - // axis is initialized again (#12137). - if (!chart.isResizing || !isNumber(axis.min)) { - axis.updateNames(); - axis.setScale(); - } - }); - } - chart.getMargins(); // #3098 - if (hasCartesianSeries) { - // If one axis is dirty, all axes must be redrawn (#792, #2169) - axes.forEach(function (axis) { - if (axis.isDirty) { - isDirtyBox = true; - } - }); - // redraw axes - axes.forEach(function (axis) { - // Fire 'afterSetExtremes' only if extremes are set - var key = axis.min + ',' + axis.max; - if (axis.extKey !== key) { // #821, #4452 - axis.extKey = key; - // prevent a recursive call to chart.redraw() (#1119) - afterRedraw.push(function () { - fireEvent(axis, 'afterSetExtremes', extend(axis.eventArgs, axis.getExtremes())); // #747, #751 - delete axis.eventArgs; - }); - } - if (isDirtyBox || hasStackedSeries) { - axis.redraw(); - } - }); - } - // the plot areas size has changed - if (isDirtyBox) { - chart.drawChartBox(); - } - // Fire an event before redrawing series, used by the boost module to - // clear previous series renderings. - fireEvent(chart, 'predraw'); - // redraw affected series - series.forEach(function (serie) { - if ((isDirtyBox || serie.isDirty) && serie.visible) { - serie.redraw(); - } - // Set it here, otherwise we will have unlimited 'updatedData' calls - // for a hidden series after setData(). Fixes #6012 - serie.isDirtyData = false; - }); - // move tooltip or reset - if (pointer) { - pointer.reset(true); - } - // redraw if canvas - renderer.draw(); - // Fire the events - fireEvent(chart, 'redraw'); - fireEvent(chart, 'render'); - if (isHiddenChart) { - chart.temporaryDisplay(true); - } - // Fire callbacks that are put on hold until after the redraw - afterRedraw.forEach(function (callback) { - callback.call(); - }); - }; - /** - * Get an axis, series or point object by `id` as given in the configuration - * options. Returns `undefined` if no item is found. - * - * @sample highcharts/plotoptions/series-id/ - * Get series by id - * - * @function Highcharts.Chart#get - * - * @param {string} id - * The id as given in the configuration options. - * - * @return {Highcharts.Axis|Highcharts.Series|Highcharts.Point|undefined} - * The retrieved item. - */ - Chart.prototype.get = function (id) { - var ret, - series = this.series, - i; - /** - * @private - * @param {Highcharts.Axis|Highcharts.Series} item - * @return {boolean} - */ - function itemById(item) { - return (item.id === id || - (item.options && item.options.id === id)); - } - ret = - // Search axes - find(this.axes, itemById) || - // Search series - find(this.series, itemById); - // Search points - for (i = 0; !ret && i < series.length; i++) { - ret = find(series[i].points || [], itemById); - } - return ret; - }; - /** - * Create the Axis instances based on the config options. - * - * @private - * @function Highcharts.Chart#getAxes - * @fires Highcharts.Chart#event:afterGetAxes - * @fires Highcharts.Chart#event:getAxes - */ - Chart.prototype.getAxes = function () { - var chart = this, - options = this.options, - xAxisOptions = options.xAxis = splat(options.xAxis || {}), - yAxisOptions = options.yAxis = splat(options.yAxis || {}), - optionsArray; - fireEvent(this, 'getAxes'); - // make sure the options are arrays and add some members - xAxisOptions.forEach(function (axis, i) { - axis.index = i; - axis.isX = true; - }); - yAxisOptions.forEach(function (axis, i) { - axis.index = i; - }); - // concatenate all axis options into one array - optionsArray = xAxisOptions.concat(yAxisOptions); - optionsArray.forEach(function (axisOptions) { - new Axis(chart, axisOptions); // eslint-disable-line no-new - }); - fireEvent(this, 'afterGetAxes'); - }; - /** - * Returns an array of all currently selected points in the chart. Points - * can be selected by clicking or programmatically by the - * {@link Highcharts.Point#select} - * function. - * - * @sample highcharts/plotoptions/series-allowpointselect-line/ - * Get selected points - * - * @function Highcharts.Chart#getSelectedPoints - * - * @return {Array<Highcharts.Point>} - * The currently selected points. - */ - Chart.prototype.getSelectedPoints = function () { - var points = []; - this.series.forEach(function (serie) { - // For one-to-one points inspect series.data in order to retrieve - // points outside the visible range (#6445). For grouped data, - // inspect the generated series.points. - points = points.concat(serie.getPointsCollection().filter(function (point) { - return pick(point.selectedStaging, point.selected); - })); - }); - return points; - }; - /** - * Returns an array of all currently selected series in the chart. Series - * can be selected either programmatically by the - * {@link Highcharts.Series#select} - * function or by checking the checkbox next to the legend item if - * [series.showCheckBox](https://api.highcharts.com/highcharts/plotOptions.series.showCheckbox) - * is true. - * - * @sample highcharts/members/chart-getselectedseries/ - * Get selected series - * - * @function Highcharts.Chart#getSelectedSeries - * - * @return {Array<Highcharts.Series>} - * The currently selected series. - */ - Chart.prototype.getSelectedSeries = function () { - return this.series.filter(function (serie) { - return serie.selected; - }); - }; - /** - * Set a new title or subtitle for the chart. - * - * @sample highcharts/members/chart-settitle/ - * Set title text and styles - * - * @function Highcharts.Chart#setTitle - * - * @param {Highcharts.TitleOptions} [titleOptions] - * New title options. The title text itself is set by the - * `titleOptions.text` property. - * - * @param {Highcharts.SubtitleOptions} [subtitleOptions] - * New subtitle options. The subtitle text itself is set by the - * `subtitleOptions.text` property. - * - * @param {boolean} [redraw] - * Whether to redraw the chart or wait for a later call to - * `chart.redraw()`. - */ - Chart.prototype.setTitle = function (titleOptions, subtitleOptions, redraw) { - this.applyDescription('title', titleOptions); - this.applyDescription('subtitle', subtitleOptions); - // The initial call also adds the caption. On update, chart.update will - // relay to Chart.setCaption. - this.applyDescription('caption', void 0); - this.layOutTitles(redraw); - }; - /** - * Apply a title, subtitle or caption for the chart - * - * @private - * @function Highcharts.Chart#applyDescription - * @param name {string} - * Either title, subtitle or caption - * @param {Highcharts.TitleOptions|Highcharts.SubtitleOptions|Highcharts.CaptionOptions|undefined} explicitOptions - * The options to set, will be merged with default options. - */ - Chart.prototype.applyDescription = function (name, explicitOptions) { - var chart = this; - // Default style - var style = name === 'title' ? { - color: '#333333', - fontSize: this.options.isStock ? '16px' : '18px' // #2944 - } : { - color: '#666666' - }; - // Merge default options with explicit options - var options = this.options[name] = merge( - // Default styles - (!this.styledMode && { style: style }), - this.options[name], - explicitOptions); - var elem = this[name]; - if (elem && explicitOptions) { - this[name] = elem = elem.destroy(); // remove old - } - if (options && !elem) { - elem = this.renderer.text(options.text, 0, 0, options.useHTML) - .attr({ - align: options.align, - 'class': 'highcharts-' + name, - zIndex: options.zIndex || 4 - }) - .add(); - // Update methods, shortcut to Chart.setTitle, Chart.setSubtitle and - // Chart.setCaption - elem.update = function (updateOptions) { - var fn = { - title: 'setTitle', - subtitle: 'setSubtitle', - caption: 'setCaption' - }[name]; - chart[fn](updateOptions); - }; - // Presentational - if (!this.styledMode) { - elem.css(options.style); - } - /** - * The chart title. The title has an `update` method that allows - * modifying the options directly or indirectly via - * `chart.update`. - * - * @sample highcharts/members/title-update/ - * Updating titles - * - * @name Highcharts.Chart#title - * @type {Highcharts.TitleObject} - */ - /** - * The chart subtitle. The subtitle has an `update` method that - * allows modifying the options directly or indirectly via - * `chart.update`. - * - * @name Highcharts.Chart#subtitle - * @type {Highcharts.SubtitleObject} - */ - this[name] = elem; - } - }; - /** - * Internal function to lay out the chart title, subtitle and caption, and - * cache the full offset height for use in `getMargins`. The result is - * stored in `this.titleOffset`. - * - * @private - * @function Highcharts.Chart#layOutTitles - * - * @param {boolean} [redraw=true] - * @fires Highcharts.Chart#event:afterLayOutTitles - */ - Chart.prototype.layOutTitles = function (redraw) { - var titleOffset = [0, 0, 0], - requiresDirtyBox, - renderer = this.renderer, - spacingBox = this.spacingBox; - // Lay out the title and the subtitle respectively - ['title', 'subtitle', 'caption'].forEach(function (key) { - var title = this[key], titleOptions = this.options[key], verticalAlign = titleOptions.verticalAlign || 'top', offset = key === 'title' ? -3 : - // Floating subtitle (#6574) - verticalAlign === 'top' ? titleOffset[0] + 2 : 0, titleSize, height; - if (title) { - if (!this.styledMode) { - titleSize = titleOptions.style.fontSize; - } - titleSize = renderer.fontMetrics(titleSize, title).b; - title - .css({ - width: (titleOptions.width || - spacingBox.width + (titleOptions.widthAdjust || 0)) + 'px' - }); - // Skip the cache for HTML (#3481, #11666) - height = Math.round(title.getBBox(titleOptions.useHTML).height); - title.align(extend({ - y: verticalAlign === 'bottom' ? - titleSize : - offset + titleSize, - height: height - }, titleOptions), false, 'spacingBox'); - if (!titleOptions.floating) { - if (verticalAlign === 'top') { - titleOffset[0] = Math.ceil(titleOffset[0] + - height); - } - else if (verticalAlign === 'bottom') { - titleOffset[2] = Math.ceil(titleOffset[2] + - height); - } - } - } - }, this); - // Handle title.margin and caption.margin - if (titleOffset[0] && - (this.options.title.verticalAlign || 'top') === 'top') { - titleOffset[0] += this.options.title.margin; - } - if (titleOffset[2] && - this.options.caption.verticalAlign === 'bottom') { - titleOffset[2] += this.options.caption.margin; - } - requiresDirtyBox = (!this.titleOffset || - this.titleOffset.join(',') !== titleOffset.join(',')); - // Used in getMargins - this.titleOffset = titleOffset; - fireEvent(this, 'afterLayOutTitles'); - if (!this.isDirtyBox && requiresDirtyBox) { - this.isDirtyBox = this.isDirtyLegend = requiresDirtyBox; - // Redraw if necessary (#2719, #2744) - if (this.hasRendered && pick(redraw, true) && this.isDirtyBox) { - this.redraw(); - } - } - }; - /** - * Internal function to get the chart width and height according to options - * and container size. Sets {@link Chart.chartWidth} and - * {@link Chart.chartHeight}. - * - * @private - * @function Highcharts.Chart#getChartSize - */ - Chart.prototype.getChartSize = function () { - var chart = this, - optionsChart = chart.options.chart, - widthOption = optionsChart.width, - heightOption = optionsChart.height, - renderTo = chart.renderTo; - // Get inner width and height - if (!defined(widthOption)) { - chart.containerWidth = getStyle(renderTo, 'width'); - } - if (!defined(heightOption)) { - chart.containerHeight = getStyle(renderTo, 'height'); - } - /** - * The current pixel width of the chart. - * - * @name Highcharts.Chart#chartWidth - * @type {number} - */ - chart.chartWidth = Math.max(// #1393 - 0, widthOption || chart.containerWidth || 600 // #1460 - ); - /** - * The current pixel height of the chart. - * - * @name Highcharts.Chart#chartHeight - * @type {number} - */ - chart.chartHeight = Math.max(0, relativeLength(heightOption, chart.chartWidth) || - (chart.containerHeight > 1 ? - chart.containerHeight : - 400)); - }; - /** - * If the renderTo element has no offsetWidth, most likely one or more of - * its parents are hidden. Loop up the DOM tree to temporarily display the - * parents, then save the original display properties, and when the true - * size is retrieved, reset them. Used on first render and on redraws. - * - * @private - * @function Highcharts.Chart#temporaryDisplay - * - * @param {boolean} [revert] - * Revert to the saved original styles. - */ - Chart.prototype.temporaryDisplay = function (revert) { - var node = this.renderTo, - tempStyle; - if (!revert) { - while (node && node.style) { - // When rendering to a detached node, it needs to be temporarily - // attached in order to read styling and bounding boxes (#5783, - // #7024). - if (!doc.body.contains(node) && !node.parentNode) { - node.hcOrigDetached = true; - doc.body.appendChild(node); - } - if (getStyle(node, 'display', false) === 'none' || - node.hcOricDetached) { - node.hcOrigStyle = { - display: node.style.display, - height: node.style.height, - overflow: node.style.overflow - }; - tempStyle = { - display: 'block', - overflow: 'hidden' - }; - if (node !== this.renderTo) { - tempStyle.height = 0; - } - css(node, tempStyle); - // If it still doesn't have an offset width after setting - // display to block, it probably has an !important priority - // #2631, 6803 - if (!node.offsetWidth) { - node.style.setProperty('display', 'block', 'important'); - } - } - node = node.parentNode; - if (node === doc.body) { - break; - } - } - } - else { - while (node && node.style) { - if (node.hcOrigStyle) { - css(node, node.hcOrigStyle); - delete node.hcOrigStyle; - } - if (node.hcOrigDetached) { - doc.body.removeChild(node); - node.hcOrigDetached = false; - } - node = node.parentNode; - } - } - }; - /** - * Set the {@link Chart.container|chart container's} class name, in - * addition to `highcharts-container`. - * - * @function Highcharts.Chart#setClassName - * - * @param {string} [className] - * The additional class name. - */ - Chart.prototype.setClassName = function (className) { - this.container.className = 'highcharts-container ' + (className || ''); - }; - /** - * Get the containing element, determine the size and create the inner - * container div to hold the chart. - * - * @private - * @function Highcharts.Chart#afterGetContainer - * @fires Highcharts.Chart#event:afterGetContainer - */ - Chart.prototype.getContainer = function () { - var chart = this, - container, - options = chart.options, - optionsChart = options.chart, - chartWidth, - chartHeight, - renderTo = chart.renderTo, - indexAttrName = 'data-highcharts-chart', - oldChartIndex, - Ren, - containerId = uniqueKey(), - containerStyle, - key; - if (!renderTo) { - chart.renderTo = renderTo = - optionsChart.renderTo; - } - if (isString(renderTo)) { - chart.renderTo = renderTo = - doc.getElementById(renderTo); - } - // Display an error if the renderTo is wrong - if (!renderTo) { - error(13, true, chart); - } - // If the container already holds a chart, destroy it. The check for - // hasRendered is there because web pages that are saved to disk from - // the browser, will preserve the data-highcharts-chart attribute and - // the SVG contents, but not an interactive chart. So in this case, - // charts[oldChartIndex] will point to the wrong chart if any (#2609). - oldChartIndex = pInt(attr(renderTo, indexAttrName)); - if (isNumber(oldChartIndex) && - charts[oldChartIndex] && - charts[oldChartIndex].hasRendered) { - charts[oldChartIndex].destroy(); - } - // Make a reference to the chart from the div - attr(renderTo, indexAttrName, chart.index); - // remove previous chart - renderTo.innerHTML = ''; - // If the container doesn't have an offsetWidth, it has or is a child of - // a node that has display:none. We need to temporarily move it out to a - // visible state to determine the size, else the legend and tooltips - // won't render properly. The skipClone option is used in sparklines as - // a micro optimization, saving about 1-2 ms each chart. - if (!optionsChart.skipClone && !renderTo.offsetWidth) { - chart.temporaryDisplay(); - } - // get the width and height - chart.getChartSize(); - chartWidth = chart.chartWidth; - chartHeight = chart.chartHeight; - // Allow table cells and flex-boxes to shrink without the chart blocking - // them out (#6427) - css(renderTo, { overflow: 'hidden' }); - // Create the inner container - if (!chart.styledMode) { - containerStyle = extend({ - position: 'relative', - // needed for context menu (avoidscrollbars) and content - // overflow in IE - overflow: 'hidden', - width: chartWidth + 'px', - height: chartHeight + 'px', - textAlign: 'left', - lineHeight: 'normal', - zIndex: 0, - '-webkit-tap-highlight-color': 'rgba(0,0,0,0)', - userSelect: 'none' // #13503 - }, optionsChart.style); - } - /** - * The containing HTML element of the chart. The container is - * dynamically inserted into the element given as the `renderTo` - * parameter in the {@link Highcharts#chart} constructor. - * - * @name Highcharts.Chart#container - * @type {Highcharts.HTMLDOMElement} - */ - container = createElement('div', { - id: containerId - }, containerStyle, renderTo); - chart.container = container; - // cache the cursor (#1650) - chart._cursor = container.style.cursor; - // Initialize the renderer - Ren = H[optionsChart.renderer] || H.Renderer; - /** - * The renderer instance of the chart. Each chart instance has only one - * associated renderer. - * - * @name Highcharts.Chart#renderer - * @type {Highcharts.SVGRenderer} - */ - chart.renderer = new Ren(container, chartWidth, chartHeight, null, optionsChart.forExport, options.exporting && options.exporting.allowHTML, chart.styledMode); - // Set the initial animation from the options - setAnimation(void 0, chart); - chart.setClassName(optionsChart.className); - if (!chart.styledMode) { - chart.renderer.setStyle(optionsChart.style); - } - else { - // Initialize definitions - for (key in options.defs) { // eslint-disable-line guard-for-in - this.renderer.definition(options.defs[key]); - } - } - // Add a reference to the charts index - chart.renderer.chartIndex = chart.index; - fireEvent(this, 'afterGetContainer'); - }; - /** - * Calculate margins by rendering axis labels in a preliminary position. - * Title, subtitle and legend have already been rendered at this stage, but - * will be moved into their final positions. - * - * @private - * @function Highcharts.Chart#getMargins - * @fires Highcharts.Chart#event:getMargins - */ - Chart.prototype.getMargins = function (skipAxes) { - var _a = this, - spacing = _a.spacing, - margin = _a.margin, - titleOffset = _a.titleOffset; - this.resetMargins(); - // Adjust for title and subtitle - if (titleOffset[0] && !defined(margin[0])) { - this.plotTop = Math.max(this.plotTop, titleOffset[0] + spacing[0]); - } - if (titleOffset[2] && !defined(margin[2])) { - this.marginBottom = Math.max(this.marginBottom, titleOffset[2] + spacing[2]); - } - // Adjust for legend - if (this.legend && this.legend.display) { - this.legend.adjustMargins(margin, spacing); - } - fireEvent(this, 'getMargins'); - if (!skipAxes) { - this.getAxisMargins(); - } - }; - /** - * @private - * @function Highcharts.Chart#getAxisMargins - */ - Chart.prototype.getAxisMargins = function () { - var chart = this, - // [top, right, bottom, left] - axisOffset = chart.axisOffset = [0, 0, 0, 0], - colorAxis = chart.colorAxis, - margin = chart.margin, - getOffset = function (axes) { - axes.forEach(function (axis) { - if (axis.visible) { - axis.getOffset(); - } - }); - }; - // pre-render axes to get labels offset width - if (chart.hasCartesianSeries) { - getOffset(chart.axes); - } - else if (colorAxis && colorAxis.length) { - getOffset(colorAxis); - } - // Add the axis offsets - marginNames.forEach(function (m, side) { - if (!defined(margin[side])) { - chart[m] += axisOffset[side]; - } - }); - chart.setChartSize(); - }; - /** - * Reflows the chart to its container. By default, the chart reflows - * automatically to its container following a `window.resize` event, as per - * the [chart.reflow](https://api.highcharts.com/highcharts/chart.reflow) - * option. However, there are no reliable events for div resize, so if the - * container is resized without a window resize event, this must be called - * explicitly. - * - * @sample highcharts/members/chart-reflow/ - * Resize div and reflow - * @sample highcharts/chart/events-container/ - * Pop up and reflow - * - * @function Highcharts.Chart#reflow - * - * @param {global.Event} [e] - * Event arguments. Used primarily when the function is called - * internally as a response to window resize. - */ - Chart.prototype.reflow = function (e) { - var chart = this, optionsChart = chart.options.chart, renderTo = chart.renderTo, hasUserSize = (defined(optionsChart.width) && - defined(optionsChart.height)), width = optionsChart.width || getStyle(renderTo, 'width'), height = optionsChart.height || getStyle(renderTo, 'height'), target = e ? e.target : win; - // Width and height checks for display:none. Target is doc in IE8 and - // Opera, win in Firefox, Chrome and IE9. - if (!hasUserSize && - !chart.isPrinting && - width && - height && - (target === win || target === doc)) { - if (width !== chart.containerWidth || - height !== chart.containerHeight) { - U.clearTimeout(chart.reflowTimeout); - // When called from window.resize, e is set, else it's called - // directly (#2224) - chart.reflowTimeout = syncTimeout(function () { - // Set size, it may have been destroyed in the meantime - // (#1257) - if (chart.container) { - chart.setSize(void 0, void 0, false); - } - }, e ? 100 : 0); - } - chart.containerWidth = width; - chart.containerHeight = height; - } - }; - /** - * Toggle the event handlers necessary for auto resizing, depending on the - * `chart.reflow` option. - * - * @private - * @function Highcharts.Chart#setReflow - */ - Chart.prototype.setReflow = function (reflow) { - var chart = this; - if (reflow !== false && !this.unbindReflow) { - this.unbindReflow = addEvent(win, 'resize', function (e) { - // a removed event listener still runs in Edge and IE if the - // listener was removed while the event runs, so check if the - // chart is not destroyed (#11609) - if (chart.options) { - chart.reflow(e); - } - }); - addEvent(this, 'destroy', this.unbindReflow); - } - else if (reflow === false && this.unbindReflow) { - // Unbind and unset - this.unbindReflow = this.unbindReflow(); - } - // The following will add listeners to re-fit the chart before and after - // printing (#2284). However it only works in WebKit. Should have worked - // in Firefox, but not supported in IE. - /* - if (win.matchMedia) { - win.matchMedia('print').addListener(function reflow() { - chart.reflow(); - }); - } - //*/ - }; - /** - * Resize the chart to a given width and height. In order to set the width - * only, the height argument may be skipped. To set the height only, pass - * `undefined` for the width. - * - * @sample highcharts/members/chart-setsize-button/ - * Test resizing from buttons - * @sample highcharts/members/chart-setsize-jquery-resizable/ - * Add a jQuery UI resizable - * @sample stock/members/chart-setsize/ - * Highstock with UI resizable - * - * @function Highcharts.Chart#setSize - * - * @param {number|null} [width] - * The new pixel width of the chart. Since v4.2.6, the argument can - * be `undefined` in order to preserve the current value (when - * setting height only), or `null` to adapt to the width of the - * containing element. - * - * @param {number|null} [height] - * The new pixel height of the chart. Since v4.2.6, the argument can - * be `undefined` in order to preserve the current value, or `null` - * in order to adapt to the height of the containing element. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] - * Whether and how to apply animation. - * - * @return {void} - * - * @fires Highcharts.Chart#event:endResize - * @fires Highcharts.Chart#event:resize - */ - Chart.prototype.setSize = function (width, height, animation) { - var chart = this, - renderer = chart.renderer, - globalAnimation; - // Handle the isResizing counter - chart.isResizing += 1; - // set the animation for the current process - setAnimation(animation, chart); - globalAnimation = renderer.globalAnimation; - chart.oldChartHeight = chart.chartHeight; - chart.oldChartWidth = chart.chartWidth; - if (typeof width !== 'undefined') { - chart.options.chart.width = width; - } - if (typeof height !== 'undefined') { - chart.options.chart.height = height; - } - chart.getChartSize(); - // Resize the container with the global animation applied if enabled - // (#2503) - if (!chart.styledMode) { - (globalAnimation ? animate : css)(chart.container, { - width: chart.chartWidth + 'px', - height: chart.chartHeight + 'px' - }, globalAnimation); - } - chart.setChartSize(true); - renderer.setSize(chart.chartWidth, chart.chartHeight, globalAnimation); - // handle axes - chart.axes.forEach(function (axis) { - axis.isDirty = true; - axis.setScale(); - }); - chart.isDirtyLegend = true; // force legend redraw - chart.isDirtyBox = true; // force redraw of plot and chart border - chart.layOutTitles(); // #2857 - chart.getMargins(); - chart.redraw(globalAnimation); - chart.oldChartHeight = null; - fireEvent(chart, 'resize'); - // Fire endResize and set isResizing back. If animation is disabled, - // fire without delay - syncTimeout(function () { - if (chart) { - fireEvent(chart, 'endResize', null, function () { - chart.isResizing -= 1; - }); - } - }, animObject(globalAnimation).duration); - }; - /** - * Set the public chart properties. This is done before and after the - * pre-render to determine margin sizes. - * - * @private - * @function Highcharts.Chart#setChartSize - * @fires Highcharts.Chart#event:afterSetChartSize - */ - Chart.prototype.setChartSize = function (skipAxes) { - var chart = this, - inverted = chart.inverted, - renderer = chart.renderer, - chartWidth = chart.chartWidth, - chartHeight = chart.chartHeight, - optionsChart = chart.options.chart, - spacing = chart.spacing, - clipOffset = chart.clipOffset, - clipX, - clipY, - plotLeft, - plotTop, - plotWidth, - plotHeight, - plotBorderWidth; - /** - * The current left position of the plot area in pixels. - * - * @name Highcharts.Chart#plotLeft - * @type {number} - */ - chart.plotLeft = plotLeft = Math.round(chart.plotLeft); - /** - * The current top position of the plot area in pixels. - * - * @name Highcharts.Chart#plotTop - * @type {number} - */ - chart.plotTop = plotTop = Math.round(chart.plotTop); - /** - * The current width of the plot area in pixels. - * - * @name Highcharts.Chart#plotWidth - * @type {number} - */ - chart.plotWidth = plotWidth = Math.max(0, Math.round(chartWidth - plotLeft - chart.marginRight)); - /** - * The current height of the plot area in pixels. - * - * @name Highcharts.Chart#plotHeight - * @type {number} - */ - chart.plotHeight = plotHeight = Math.max(0, Math.round(chartHeight - plotTop - chart.marginBottom)); - chart.plotSizeX = inverted ? plotHeight : plotWidth; - chart.plotSizeY = inverted ? plotWidth : plotHeight; - chart.plotBorderWidth = optionsChart.plotBorderWidth || 0; - // Set boxes used for alignment - chart.spacingBox = renderer.spacingBox = { - x: spacing[3], - y: spacing[0], - width: chartWidth - spacing[3] - spacing[1], - height: chartHeight - spacing[0] - spacing[2] - }; - chart.plotBox = renderer.plotBox = { - x: plotLeft, - y: plotTop, - width: plotWidth, - height: plotHeight - }; - plotBorderWidth = 2 * Math.floor(chart.plotBorderWidth / 2); - clipX = Math.ceil(Math.max(plotBorderWidth, clipOffset[3]) / 2); - clipY = Math.ceil(Math.max(plotBorderWidth, clipOffset[0]) / 2); - chart.clipBox = { - x: clipX, - y: clipY, - width: Math.floor(chart.plotSizeX - - Math.max(plotBorderWidth, clipOffset[1]) / 2 - - clipX), - height: Math.max(0, Math.floor(chart.plotSizeY - - Math.max(plotBorderWidth, clipOffset[2]) / 2 - - clipY)) - }; - if (!skipAxes) { - chart.axes.forEach(function (axis) { - axis.setAxisSize(); - axis.setAxisTranslation(); - }); - } - fireEvent(chart, 'afterSetChartSize', { skipAxes: skipAxes }); - }; - /** - * Initial margins before auto size margins are applied. - * - * @private - * @function Highcharts.Chart#resetMargins - */ - Chart.prototype.resetMargins = function () { - fireEvent(this, 'resetMargins'); - var chart = this, - chartOptions = chart.options.chart; - // Create margin and spacing array - ['margin', 'spacing'].forEach(function splashArrays(target) { - var value = chartOptions[target], - values = isObject(value) ? value : [value, - value, - value, - value]; - [ - 'Top', - 'Right', - 'Bottom', - 'Left' - ].forEach(function (sideName, side) { - chart[target][side] = pick(chartOptions[target + sideName], values[side]); - }); - }); - // Set margin names like chart.plotTop, chart.plotLeft, - // chart.marginRight, chart.marginBottom. - marginNames.forEach(function (m, side) { - chart[m] = pick(chart.margin[side], chart.spacing[side]); - }); - chart.axisOffset = [0, 0, 0, 0]; // top, right, bottom, left - chart.clipOffset = [0, 0, 0, 0]; - }; - /** - * Internal function to draw or redraw the borders and backgrounds for chart - * and plot area. - * - * @private - * @function Highcharts.Chart#drawChartBox - * @fires Highcharts.Chart#event:afterDrawChartBox - */ - Chart.prototype.drawChartBox = function () { - var chart = this, - optionsChart = chart.options.chart, - renderer = chart.renderer, - chartWidth = chart.chartWidth, - chartHeight = chart.chartHeight, - chartBackground = chart.chartBackground, - plotBackground = chart.plotBackground, - plotBorder = chart.plotBorder, - chartBorderWidth, - styledMode = chart.styledMode, - plotBGImage = chart.plotBGImage, - chartBackgroundColor = optionsChart.backgroundColor, - plotBackgroundColor = optionsChart.plotBackgroundColor, - plotBackgroundImage = optionsChart.plotBackgroundImage, - mgn, - bgAttr, - plotLeft = chart.plotLeft, - plotTop = chart.plotTop, - plotWidth = chart.plotWidth, - plotHeight = chart.plotHeight, - plotBox = chart.plotBox, - clipRect = chart.clipRect, - clipBox = chart.clipBox, - verb = 'animate'; - // Chart area - if (!chartBackground) { - chart.chartBackground = chartBackground = renderer.rect() - .addClass('highcharts-background') - .add(); - verb = 'attr'; - } - if (!styledMode) { - // Presentational - chartBorderWidth = optionsChart.borderWidth || 0; - mgn = chartBorderWidth + (optionsChart.shadow ? 8 : 0); - bgAttr = { - fill: chartBackgroundColor || 'none' - }; - if (chartBorderWidth || chartBackground['stroke-width']) { // #980 - bgAttr.stroke = optionsChart.borderColor; - bgAttr['stroke-width'] = chartBorderWidth; - } - chartBackground - .attr(bgAttr) - .shadow(optionsChart.shadow); - } - else { - chartBorderWidth = mgn = chartBackground.strokeWidth(); - } - chartBackground[verb]({ - x: mgn / 2, - y: mgn / 2, - width: chartWidth - mgn - chartBorderWidth % 2, - height: chartHeight - mgn - chartBorderWidth % 2, - r: optionsChart.borderRadius - }); - // Plot background - verb = 'animate'; - if (!plotBackground) { - verb = 'attr'; - chart.plotBackground = plotBackground = renderer.rect() - .addClass('highcharts-plot-background') - .add(); - } - plotBackground[verb](plotBox); - if (!styledMode) { - // Presentational attributes for the background - plotBackground - .attr({ - fill: plotBackgroundColor || 'none' - }) - .shadow(optionsChart.plotShadow); - // Create the background image - if (plotBackgroundImage) { - if (!plotBGImage) { - chart.plotBGImage = renderer.image(plotBackgroundImage, plotLeft, plotTop, plotWidth, plotHeight).add(); - } - else { - if (plotBackgroundImage !== plotBGImage.attr('href')) { - plotBGImage.attr('href', plotBackgroundImage); - } - plotBGImage.animate(plotBox); - } - } - } - // Plot clip - if (!clipRect) { - chart.clipRect = renderer.clipRect(clipBox); - } - else { - clipRect.animate({ - width: clipBox.width, - height: clipBox.height - }); - } - // Plot area border - verb = 'animate'; - if (!plotBorder) { - verb = 'attr'; - chart.plotBorder = plotBorder = renderer.rect() - .addClass('highcharts-plot-border') - .attr({ - zIndex: 1 // Above the grid - }) - .add(); - } - if (!styledMode) { - // Presentational - plotBorder.attr({ - stroke: optionsChart.plotBorderColor, - 'stroke-width': optionsChart.plotBorderWidth || 0, - fill: 'none' - }); - } - plotBorder[verb](plotBorder.crisp({ - x: plotLeft, - y: plotTop, - width: plotWidth, - height: plotHeight - }, -plotBorder.strokeWidth())); // #3282 plotBorder should be negative; - // reset - chart.isDirtyBox = false; - fireEvent(this, 'afterDrawChartBox'); - }; - /** - * Detect whether a certain chart property is needed based on inspecting its - * options and series. This mainly applies to the chart.inverted property, - * and in extensions to the chart.angular and chart.polar properties. - * - * @private - * @function Highcharts.Chart#propFromSeries - * @return {void} - */ - Chart.prototype.propFromSeries = function () { - var chart = this, - optionsChart = chart.options.chart, - klass, - seriesOptions = chart.options.series, - i, - value; - /** - * The flag is set to `true` if a series of the chart is inverted. - * - * @name Highcharts.Chart#inverted - * @type {boolean|undefined} - */ - ['inverted', 'angular', 'polar'].forEach(function (key) { - // The default series type's class - klass = BaseSeries.seriesTypes[(optionsChart.type || - optionsChart.defaultSeriesType)]; - // Get the value from available chart-wide properties - value = - // It is set in the options: - optionsChart[key] || - // The default series class: - (klass && klass.prototype[key]); - // requires it - // 4. Check if any the chart's series require it - i = seriesOptions && seriesOptions.length; - while (!value && i--) { - klass = BaseSeries.seriesTypes[seriesOptions[i].type]; - if (klass && klass.prototype[key]) { - value = true; - } - } - // Set the chart property - chart[key] = value; - }); - }; - /** - * Internal function to link two or more series together, based on the - * `linkedTo` option. This is done from `Chart.render`, and after - * `Chart.addSeries` and `Series.remove`. - * - * @private - * @function Highcharts.Chart#linkSeries - * @fires Highcharts.Chart#event:afterLinkSeries - */ - Chart.prototype.linkSeries = function () { - var chart = this, - chartSeries = chart.series; - // Reset links - chartSeries.forEach(function (series) { - series.linkedSeries.length = 0; - }); - // Apply new links - chartSeries.forEach(function (series) { - var linkedTo = series.options.linkedTo; - if (isString(linkedTo)) { - if (linkedTo === ':previous') { - linkedTo = chart.series[series.index - 1]; - } - else { - linkedTo = chart.get(linkedTo); - } - // #3341 avoid mutual linking - if (linkedTo && linkedTo.linkedParent !== series) { - linkedTo.linkedSeries.push(series); - series.linkedParent = linkedTo; - if (linkedTo.enabledDataSorting) { - series.setDataSortingOptions(); - } - series.visible = pick(series.options.visible, linkedTo.options.visible, series.visible); // #3879 - } - } - }); - fireEvent(this, 'afterLinkSeries'); - }; - /** - * Render series for the chart. - * - * @private - * @function Highcharts.Chart#renderSeries - */ - Chart.prototype.renderSeries = function () { - this.series.forEach(function (serie) { - serie.translate(); - serie.render(); - }); - }; - /** - * Render labels for the chart. - * - * @private - * @function Highcharts.Chart#renderLabels - */ - Chart.prototype.renderLabels = function () { - var chart = this, - labels = chart.options.labels; - if (labels.items) { - labels.items.forEach(function (label) { - var style = extend(labels.style, - label.style), - x = pInt(style.left) + chart.plotLeft, - y = pInt(style.top) + chart.plotTop + 12; - // delete to prevent rewriting in IE - delete style.left; - delete style.top; - chart.renderer.text(label.html, x, y) - .attr({ zIndex: 2 }) - .css(style) - .add(); - }); - } - }; - /** - * Render all graphics for the chart. Runs internally on initialization. - * - * @private - * @function Highcharts.Chart#render - */ - Chart.prototype.render = function () { - var chart = this, - axes = chart.axes, - colorAxis = chart.colorAxis, - renderer = chart.renderer, - options = chart.options, - correction = 0, // correction for X axis labels - tempWidth, - tempHeight, - redoHorizontal, - redoVertical, - renderAxes = function (axes) { - axes.forEach(function (axis) { - if (axis.visible) { - axis.render(); - } - }); - }; - // Title - chart.setTitle(); - /** - * The overview of the chart's series. - * - * @name Highcharts.Chart#legend - * @type {Highcharts.Legend} - */ - chart.legend = new Legend(chart, options.legend); - // Get stacks - if (chart.getStacks) { - chart.getStacks(); - } - // Get chart margins - chart.getMargins(true); - chart.setChartSize(); - // Record preliminary dimensions for later comparison - tempWidth = chart.plotWidth; - axes.some(function (axis) { - if (axis.horiz && - axis.visible && - axis.options.labels.enabled && - axis.series.length) { - // 21 is the most common correction for X axis labels - correction = 21; - return true; - } - }); - // use Math.max to prevent negative plotHeight - chart.plotHeight = Math.max(chart.plotHeight - correction, 0); - tempHeight = chart.plotHeight; - // Get margins by pre-rendering axes - axes.forEach(function (axis) { - axis.setScale(); - }); - chart.getAxisMargins(); - // If the plot area size has changed significantly, calculate tick - // positions again - redoHorizontal = tempWidth / chart.plotWidth > 1.1; - // Height is more sensitive, use lower threshold - redoVertical = tempHeight / chart.plotHeight > 1.05; - if (redoHorizontal || redoVertical) { - axes.forEach(function (axis) { - if ((axis.horiz && redoHorizontal) || - (!axis.horiz && redoVertical)) { - // update to reflect the new margins - axis.setTickInterval(true); - } - }); - chart.getMargins(); // second pass to check for new labels - } - // Draw the borders and backgrounds - chart.drawChartBox(); - // Axes - if (chart.hasCartesianSeries) { - renderAxes(axes); - } - else if (colorAxis && colorAxis.length) { - renderAxes(colorAxis); - } - // The series - if (!chart.seriesGroup) { - chart.seriesGroup = renderer.g('series-group') - .attr({ zIndex: 3 }) - .add(); - } - chart.renderSeries(); - // Labels - chart.renderLabels(); - // Credits - chart.addCredits(); - // Handle responsiveness - if (chart.setResponsive) { - chart.setResponsive(); - } - // Handle scaling - chart.updateContainerScaling(); - // Set flag - chart.hasRendered = true; - }; - /** - * Set a new credits label for the chart. - * - * @sample highcharts/credits/credits-update/ - * Add and update credits - * - * @function Highcharts.Chart#addCredits - * - * @param {Highcharts.CreditsOptions} [credits] - * A configuration object for the new credits. - */ - Chart.prototype.addCredits = function (credits) { - var chart = this, - creds = merge(true, - this.options.credits, - credits); - if (creds.enabled && !this.credits) { - /** - * The chart's credits label. The label has an `update` method that - * allows setting new options as per the - * [credits options set](https://api.highcharts.com/highcharts/credits). - * - * @name Highcharts.Chart#credits - * @type {Highcharts.SVGElement} - */ - this.credits = this.renderer.text(creds.text + (this.mapCredits || ''), 0, 0) - .addClass('highcharts-credits') - .on('click', function () { - if (creds.href) { - win.location.href = creds.href; - } - }) - .attr({ - align: creds.position.align, - zIndex: 8 - }); - if (!chart.styledMode) { - this.credits.css(creds.style); - } - this.credits - .add() - .align(creds.position); - // Dynamically update - this.credits.update = function (options) { - chart.credits = chart.credits.destroy(); - chart.addCredits(options); - }; - } - }; - /** - * Handle scaling, #11329 - when there is scaling/transform on the container - * or on a parent element, we need to take this into account. We calculate - * the scaling once here and it is picked up where we need to use it - * (Pointer, Tooltip). - * - * @private - * @function Highcharts.Chart#updateContainerScaling - */ - Chart.prototype.updateContainerScaling = function () { - var container = this.container; - // #13342 - tooltip was not visible in Chrome, when chart - // updates height. - if (container.offsetWidth > 2 && // #13342 - container.offsetHeight > 2 && // #13342 - container.getBoundingClientRect) { - var bb = container.getBoundingClientRect(), - scaleX = bb.width / container.offsetWidth, - scaleY = bb.height / container.offsetHeight; - if (scaleX !== 1 || scaleY !== 1) { - this.containerScaling = { scaleX: scaleX, scaleY: scaleY }; - } - else { - delete this.containerScaling; - } - } - }; - /** - * Remove the chart and purge memory. This method is called internally - * before adding a second chart into the same container, as well as on - * window unload to prevent leaks. - * - * @sample highcharts/members/chart-destroy/ - * Destroy the chart from a button - * @sample stock/members/chart-destroy/ - * Destroy with Highstock - * - * @function Highcharts.Chart#destroy - * - * @fires Highcharts.Chart#event:destroy - */ - Chart.prototype.destroy = function () { - var chart = this, - axes = chart.axes, - series = chart.series, - container = chart.container, - i, - parentNode = container && container.parentNode; - // fire the chart.destoy event - fireEvent(chart, 'destroy'); - // Delete the chart from charts lookup array - if (chart.renderer.forExport) { - erase(charts, chart); // #6569 - } - else { - charts[chart.index] = void 0; - } - H.chartCount--; - chart.renderTo.removeAttribute('data-highcharts-chart'); - // remove events - removeEvent(chart); - // ==== Destroy collections: - // Destroy axes - i = axes.length; - while (i--) { - axes[i] = axes[i].destroy(); - } - // Destroy scroller & scroller series before destroying base series - if (this.scroller && this.scroller.destroy) { - this.scroller.destroy(); - } - // Destroy each series - i = series.length; - while (i--) { - series[i] = series[i].destroy(); - } - // ==== Destroy chart properties: - [ - 'title', 'subtitle', 'chartBackground', 'plotBackground', - 'plotBGImage', 'plotBorder', 'seriesGroup', 'clipRect', 'credits', - 'pointer', 'rangeSelector', 'legend', 'resetZoomButton', 'tooltip', - 'renderer' - ].forEach(function (name) { - var prop = chart[name]; - if (prop && prop.destroy) { - chart[name] = prop.destroy(); - } - }); - // Remove container and all SVG, check container as it can break in IE - // when destroyed before finished loading - if (container) { - container.innerHTML = ''; - removeEvent(container); - if (parentNode) { - discardElement(container); - } - } - // clean it all up - objectEach(chart, function (val, key) { - delete chart[key]; - }); - }; - /** - * Prepare for first rendering after all data are loaded. - * - * @private - * @function Highcharts.Chart#firstRender - * @fires Highcharts.Chart#event:beforeRender - */ - Chart.prototype.firstRender = function () { - var chart = this, - options = chart.options; - // Hook for oldIE to check whether the chart is ready to render - if (chart.isReadyToRender && !chart.isReadyToRender()) { - return; - } - // Create the container - chart.getContainer(); - chart.resetMargins(); - chart.setChartSize(); - // Set the common chart properties (mainly invert) from the given series - chart.propFromSeries(); - // get axes - chart.getAxes(); - // Initialize the series - (isArray(options.series) ? options.series : []).forEach( - // #9680 - function (serieOptions) { - chart.initSeries(serieOptions); - }); - chart.linkSeries(); - chart.setSeriesData(); - // Run an event after axes and series are initialized, but before - // render. At this stage, the series data is indexed and cached in the - // xData and yData arrays, so we can access those before rendering. Used - // in Highstock. - fireEvent(chart, 'beforeRender'); - // depends on inverted and on margins being set - if (Pointer) { - if (!H.hasTouch && (win.PointerEvent || win.MSPointerEvent)) { - chart.pointer = new MSPointer(chart, options); - } - else { - /** - * The Pointer that keeps track of mouse and touch interaction. - * - * @memberof Highcharts.Chart - * @name pointer - * @type {Highcharts.Pointer} - * @instance - */ - chart.pointer = new Pointer(chart, options); - } - } - chart.render(); - // Fire the load event if there are no external images - if (!chart.renderer.imgCount && !chart.hasLoaded) { - chart.onload(); - } - // If the chart was rendered outside the top container, put it back in - // (#3679) - chart.temporaryDisplay(true); - }; - /** - * Internal function that runs on chart load, async if any images are loaded - * in the chart. Runs the callbacks and triggers the `load` and `render` - * events. - * - * @private - * @function Highcharts.Chart#onload - * @fires Highcharts.Chart#event:load - * @fires Highcharts.Chart#event:render - */ - Chart.prototype.onload = function () { - // Run callbacks, first the ones registered by modules, then user's one - this.callbacks.concat([this.callback]).forEach(function (fn) { - // Chart destroyed in its own callback (#3600) - if (fn && typeof this.index !== 'undefined') { - fn.apply(this, [this]); - } - }, this); - fireEvent(this, 'load'); - fireEvent(this, 'render'); - // Set up auto resize, check for not destroyed (#6068) - if (defined(this.index)) { - this.setReflow(this.options.chart.reflow); - } - // Don't run again - this.hasLoaded = true; - }; - return Chart; - }()); - // Hook for adding callbacks in modules - Chart.prototype.callbacks = []; - /** - * Factory function for basic charts. - * - * @example - * // Render a chart in to div#container - * var chart = Highcharts.chart('container', { - * title: { - * text: 'My chart' - * }, - * series: [{ - * data: [1, 3, 2, 4] - * }] - * }); - * - * @function Highcharts.chart - * - * @param {string|Highcharts.HTMLDOMElement} [renderTo] - * The DOM element to render to, or its id. - * - * @param {Highcharts.Options} options - * The chart options structure. - * - * @param {Highcharts.ChartCallbackFunction} [callback] - * Function to run when the chart has loaded and and all external images - * are loaded. Defining a - * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load) - * handler is equivalent. - * - * @return {Highcharts.Chart} - * Returns the Chart object. - */ - function chart(a, b, c) { - return new Chart(a, b, c); - } - H.chart = chart; - H.Chart = Chart; - - return Chart; - }); - _registerModule(_modules, 'Extensions/ScrollablePlotArea.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (A, Chart, H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * Highcharts feature to make the Y axis stay fixed when scrolling the chart - * horizontally on mobile devices. Supports left and right side axes. - */ - /* - WIP on vertical scrollable plot area (#9378). To do: - - Bottom axis positioning - - Test with Gantt - - Look for size optimizing the code - - API and demos - */ - var stop = A.stop; - var addEvent = U.addEvent, - createElement = U.createElement, - pick = U.pick; - /** - * Options for a scrollable plot area. This feature provides a minimum size for - * the plot area of the chart. If the size gets smaller than this, typically - * on mobile devices, a native browser scrollbar is presented. This scrollbar - * provides smooth scrolling for the contents of the plot area, whereas the - * title, legend and unaffected axes are fixed. - * - * Since v7.1.2, a scrollable plot area can be defined for either horizontal or - * vertical scrolling, depending on whether the `minWidth` or `minHeight` - * option is set. - * - * @sample highcharts/chart/scrollable-plotarea - * Scrollable plot area - * @sample highcharts/chart/scrollable-plotarea-vertical - * Vertically scrollable plot area - * @sample {gantt} highcharts/chart/scrollable-plotarea-vertical - * Gantt chart with vertically scrollable plot area - * - * @since 6.1.0 - * @product highcharts gantt - * @apioption chart.scrollablePlotArea - */ - /** - * The minimum height for the plot area. If it gets smaller than this, the plot - * area will become scrollable. - * - * @type {number} - * @apioption chart.scrollablePlotArea.minHeight - */ - /** - * The minimum width for the plot area. If it gets smaller than this, the plot - * area will become scrollable. - * - * @type {number} - * @apioption chart.scrollablePlotArea.minWidth - */ - /** - * The initial scrolling position of the scrollable plot area. Ranges from 0 to - * 1, where 0 aligns the plot area to the left and 1 aligns it to the right. - * Typically we would use 1 if the chart has right aligned Y axes. - * - * @type {number} - * @apioption chart.scrollablePlotArea.scrollPositionX - */ - /** - * The initial scrolling position of the scrollable plot area. Ranges from 0 to - * 1, where 0 aligns the plot area to the top and 1 aligns it to the bottom. - * - * @type {number} - * @apioption chart.scrollablePlotArea.scrollPositionY - */ - /** - * The opacity of mask applied on one of the sides of the plot - * area. - * - * @sample {highcharts} highcharts/chart/scrollable-plotarea-opacity - * Disabled opacity for the mask - * - * @type {number} - * @default 0.85 - * @since 7.1.1 - * @apioption chart.scrollablePlotArea.opacity - */ - ''; // detach API doclets - /* eslint-disable no-invalid-this, valid-jsdoc */ - addEvent(Chart, 'afterSetChartSize', function (e) { - var scrollablePlotArea = this.options.chart.scrollablePlotArea, - scrollableMinWidth = scrollablePlotArea && scrollablePlotArea.minWidth, - scrollableMinHeight = scrollablePlotArea && scrollablePlotArea.minHeight, - scrollablePixelsX, - scrollablePixelsY, - corrections; - if (!this.renderer.forExport) { - // The amount of pixels to scroll, the difference between chart - // width and scrollable width - if (scrollableMinWidth) { - this.scrollablePixelsX = scrollablePixelsX = Math.max(0, scrollableMinWidth - this.chartWidth); - if (scrollablePixelsX) { - this.plotWidth += scrollablePixelsX; - if (this.inverted) { - this.clipBox.height += scrollablePixelsX; - this.plotBox.height += scrollablePixelsX; - } - else { - this.clipBox.width += scrollablePixelsX; - this.plotBox.width += scrollablePixelsX; - } - corrections = { - // Corrections for right side - 1: { name: 'right', value: scrollablePixelsX } - }; - } - // Currently we can only do either X or Y - } - else if (scrollableMinHeight) { - this.scrollablePixelsY = scrollablePixelsY = Math.max(0, scrollableMinHeight - this.chartHeight); - if (scrollablePixelsY) { - this.plotHeight += scrollablePixelsY; - if (this.inverted) { - this.clipBox.width += scrollablePixelsY; - this.plotBox.width += scrollablePixelsY; - } - else { - this.clipBox.height += scrollablePixelsY; - this.plotBox.height += scrollablePixelsY; - } - corrections = { - 2: { name: 'bottom', value: scrollablePixelsY } - }; - } - } - if (corrections && !e.skipAxes) { - this.axes.forEach(function (axis) { - // For right and bottom axes, only fix the plot line length - if (corrections[axis.side]) { - // Get the plot lines right in getPlotLinePath, - // temporarily set it to the adjusted plot width. - axis.getPlotLinePath = function () { - var marginName = corrections[axis.side].name, - correctionValue = corrections[axis.side].value, - // axis.right or axis.bottom - margin = this[marginName], - path; - // Temporarily adjust - this[marginName] = margin - correctionValue; - path = H.Axis.prototype.getPlotLinePath.apply(this, arguments); - // Reset - this[marginName] = margin; - return path; - }; - } - else { - // Apply the corrected plotWidth - axis.setAxisSize(); - axis.setAxisTranslation(); - } - }); - } - } - }); - addEvent(Chart, 'render', function () { - if (this.scrollablePixelsX || this.scrollablePixelsY) { - if (this.setUpScrolling) { - this.setUpScrolling(); - } - this.applyFixed(); - } - else if (this.fixedDiv) { // Has been in scrollable mode - this.applyFixed(); - } - }); - /** - * @private - * @function Highcharts.Chart#setUpScrolling - * @return {void} - */ - Chart.prototype.setUpScrolling = function () { - var _this = this; - var attribs = { - WebkitOverflowScrolling: 'touch', - overflowX: 'hidden', - overflowY: 'hidden' - }; - if (this.scrollablePixelsX) { - attribs.overflowX = 'auto'; - } - if (this.scrollablePixelsY) { - attribs.overflowY = 'auto'; - } - // Insert a container with position relative - // that scrolling and fixed container renders to (#10555) - this.scrollingParent = createElement('div', { - className: 'highcharts-scrolling-parent' - }, { - position: 'relative' - }, this.renderTo); - // Add the necessary divs to provide scrolling - this.scrollingContainer = createElement('div', { - 'className': 'highcharts-scrolling' - }, attribs, this.scrollingParent); - // On scroll, reset the chart position because it applies to the scrolled - // container - addEvent(this.scrollingContainer, 'scroll', function () { - if (_this.pointer) { - delete _this.pointer.chartPosition; - } - }); - this.innerContainer = createElement('div', { - 'className': 'highcharts-inner-container' - }, null, this.scrollingContainer); - // Now move the container inside - this.innerContainer.appendChild(this.container); - // Don't run again - this.setUpScrolling = null; - }; - /** - * These elements are moved over to the fixed renderer and stay fixed when the - * user scrolls the chart - * @private - */ - Chart.prototype.moveFixedElements = function () { - var container = this.container, - fixedRenderer = this.fixedRenderer, - fixedSelectors = [ - '.highcharts-contextbutton', - '.highcharts-credits', - '.highcharts-legend', - '.highcharts-legend-checkbox', - '.highcharts-navigator-series', - '.highcharts-navigator-xaxis', - '.highcharts-navigator-yaxis', - '.highcharts-navigator', - '.highcharts-reset-zoom', - '.highcharts-scrollbar', - '.highcharts-subtitle', - '.highcharts-title' - ], - axisClass; - if (this.scrollablePixelsX && !this.inverted) { - axisClass = '.highcharts-yaxis'; - } - else if (this.scrollablePixelsX && this.inverted) { - axisClass = '.highcharts-xaxis'; - } - else if (this.scrollablePixelsY && !this.inverted) { - axisClass = '.highcharts-xaxis'; - } - else if (this.scrollablePixelsY && this.inverted) { - axisClass = '.highcharts-yaxis'; - } - fixedSelectors.push(axisClass, axisClass + '-labels'); - fixedSelectors.forEach(function (className) { - [].forEach.call(container.querySelectorAll(className), function (elem) { - (elem.namespaceURI === fixedRenderer.SVG_NS ? - fixedRenderer.box : - fixedRenderer.box.parentNode).appendChild(elem); - elem.style.pointerEvents = 'auto'; - }); - }); - }; - /** - * @private - * @function Highcharts.Chart#applyFixed - * @return {void} - */ - Chart.prototype.applyFixed = function () { - var _a, - _b; - var fixedRenderer, - scrollableWidth, - scrollableHeight, - firstTime = !this.fixedDiv, - scrollableOptions = this.options.chart.scrollablePlotArea; - // First render - if (firstTime) { - this.fixedDiv = createElement('div', { - className: 'highcharts-fixed' - }, { - position: 'absolute', - overflow: 'hidden', - pointerEvents: 'none', - zIndex: 2, - top: 0 - }, null, true); - (_a = this.scrollingContainer) === null || _a === void 0 ? void 0 : _a.parentNode.insertBefore(this.fixedDiv, this.scrollingContainer); - this.renderTo.style.overflow = 'visible'; - this.fixedRenderer = fixedRenderer = new H.Renderer(this.fixedDiv, this.chartWidth, this.chartHeight, (_b = this.options.chart) === null || _b === void 0 ? void 0 : _b.style); - // Mask - this.scrollableMask = fixedRenderer - .path() - .attr({ - fill: this.options.chart.backgroundColor || '#fff', - 'fill-opacity': pick(scrollableOptions.opacity, 0.85), - zIndex: -1 - }) - .addClass('highcharts-scrollable-mask') - .add(); - this.moveFixedElements(); - addEvent(this, 'afterShowResetZoom', this.moveFixedElements); - addEvent(this, 'afterLayOutTitles', this.moveFixedElements); - } - else { - // Set the size of the fixed renderer to the visible width - this.fixedRenderer.setSize(this.chartWidth, this.chartHeight); - } - // Increase the size of the scrollable renderer and background - scrollableWidth = this.chartWidth + (this.scrollablePixelsX || 0); - scrollableHeight = this.chartHeight + (this.scrollablePixelsY || 0); - stop(this.container); - this.container.style.width = scrollableWidth + 'px'; - this.container.style.height = scrollableHeight + 'px'; - this.renderer.boxWrapper.attr({ - width: scrollableWidth, - height: scrollableHeight, - viewBox: [0, 0, scrollableWidth, scrollableHeight].join(' ') - }); - this.chartBackground.attr({ - width: scrollableWidth, - height: scrollableHeight - }); - this.scrollingContainer.style.height = this.chartHeight + 'px'; - // Set scroll position - if (firstTime) { - if (scrollableOptions.scrollPositionX) { - this.scrollingContainer.scrollLeft = - this.scrollablePixelsX * - scrollableOptions.scrollPositionX; - } - if (scrollableOptions.scrollPositionY) { - this.scrollingContainer.scrollTop = - this.scrollablePixelsY * - scrollableOptions.scrollPositionY; - } - } - // Mask behind the left and right side - var axisOffset = this.axisOffset, - maskTop = this.plotTop - axisOffset[0] - 1, - maskLeft = this.plotLeft - axisOffset[3] - 1, - maskBottom = this.plotTop + this.plotHeight + axisOffset[2] + 1, - maskRight = this.plotLeft + this.plotWidth + axisOffset[1] + 1, - maskPlotRight = this.plotLeft + this.plotWidth - - (this.scrollablePixelsX || 0), - maskPlotBottom = this.plotTop + this.plotHeight - - (this.scrollablePixelsY || 0), - d; - if (this.scrollablePixelsX) { - d = [ - // Left side - ['M', 0, maskTop], - ['L', this.plotLeft - 1, maskTop], - ['L', this.plotLeft - 1, maskBottom], - ['L', 0, maskBottom], - ['Z'], - // Right side - ['M', maskPlotRight, maskTop], - ['L', this.chartWidth, maskTop], - ['L', this.chartWidth, maskBottom], - ['L', maskPlotRight, maskBottom], - ['Z'] - ]; - } - else if (this.scrollablePixelsY) { - d = [ - // Top side - ['M', maskLeft, 0], - ['L', maskLeft, this.plotTop - 1], - ['L', maskRight, this.plotTop - 1], - ['L', maskRight, 0], - ['Z'], - // Bottom side - ['M', maskLeft, maskPlotBottom], - ['L', maskLeft, this.chartHeight], - ['L', maskRight, this.chartHeight], - ['L', maskRight, maskPlotBottom], - ['Z'] - ]; - } - else { - d = [['M', 0, 0]]; - } - if (this.redrawTrigger !== 'adjustHeight') { - this.scrollableMask.attr({ d: d }); - } - }; - - }); - _registerModule(_modules, 'Core/Axis/StackingAxis.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Utilities.js']], function (A, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var getDeferredAnimation = A.getDeferredAnimation; - var addEvent = U.addEvent, - destroyObjectProperties = U.destroyObjectProperties, - fireEvent = U.fireEvent, - objectEach = U.objectEach, - pick = U.pick; - /* eslint-disable valid-jsdoc */ - /** - * Adds stacking support to axes. - * @private - * @class - */ - var StackingAxisAdditions = /** @class */ (function () { - /* * - * - * Constructors - * - * */ - function StackingAxisAdditions(axis) { - this.oldStacks = {}; - this.stacks = {}; - this.stacksTouched = 0; - this.axis = axis; - } - /* * - * - * Functions - * - * */ - /** - * Build the stacks from top down - * @private - */ - StackingAxisAdditions.prototype.buildStacks = function () { - var stacking = this; - var axis = stacking.axis; - var axisSeries = axis.series; - var reversedStacks = pick(axis.options.reversedStacks, - true); - var len = axisSeries.length; - var actualSeries, - i; - if (!axis.isXAxis) { - stacking.usePercentage = false; - i = len; - while (i--) { - actualSeries = axisSeries[reversedStacks ? i : len - i - 1]; - actualSeries.setStackedPoints(); - actualSeries.setGroupedPoints(); - } - // Loop up again to compute percent and stream stack - for (i = 0; i < len; i++) { - axisSeries[i].modifyStacks(); - } - fireEvent(axis, 'afterBuildStacks'); - } - }; - /** - * @private - */ - StackingAxisAdditions.prototype.cleanStacks = function () { - var stacking = this; - var axis = stacking.axis; - var stacks; - if (!axis.isXAxis) { - if (stacking.oldStacks) { - stacks = stacking.stacks = stacking.oldStacks; - } - // reset stacks - objectEach(stacks, function (type) { - objectEach(type, function (stack) { - stack.cumulative = stack.total; - }); - }); - } - }; - /** - * Set all the stacks to initial states and destroy unused ones. - * @private - */ - StackingAxisAdditions.prototype.resetStacks = function () { - var stacking = this; - var axis = stacking.axis; - var stacks = stacking.stacks; - if (!axis.isXAxis) { - objectEach(stacks, function (type) { - objectEach(type, function (stack, key) { - // Clean up memory after point deletion (#1044, #4320) - if (stack.touched < stacking.stacksTouched) { - stack.destroy(); - delete type[key]; - // Reset stacks - } - else { - stack.total = null; - stack.cumulative = null; - } - }); - }); - } - }; - /** - * @private - */ - StackingAxisAdditions.prototype.renderStackTotals = function () { - var stacking = this; - var axis = stacking.axis; - var chart = axis.chart; - var renderer = chart.renderer; - var stacks = stacking.stacks; - var stackLabelsAnim = axis.options.stackLabels.animation; - var animationConfig = getDeferredAnimation(chart, - stackLabelsAnim); - var stackTotalGroup = stacking.stackTotalGroup = (stacking.stackTotalGroup || - renderer - .g('stack-labels') - .attr({ - visibility: 'visible', - zIndex: 6, - opacity: 0 - }) - .add()); - // plotLeft/Top will change when y axis gets wider so we need to - // translate the stackTotalGroup at every render call. See bug #506 - // and #516 - stackTotalGroup.translate(chart.plotLeft, chart.plotTop); - // Render each stack total - objectEach(stacks, function (type) { - objectEach(type, function (stack) { - stack.render(stackTotalGroup); - }); - }); - stackTotalGroup.animate({ - opacity: 1 - }, animationConfig); - }; - return StackingAxisAdditions; - }()); - /** - * Axis with stacking support. - * @private - * @class - */ - var StackingAxis = /** @class */ (function () { - function StackingAxis() { - } - /* * - * - * Static Functions - * - * */ - /** - * Extends axis with stacking support. - * @private - */ - StackingAxis.compose = function (AxisClass) { - var axisProto = AxisClass.prototype; - addEvent(AxisClass, 'init', StackingAxis.onInit); - addEvent(AxisClass, 'destroy', StackingAxis.onDestroy); - }; - /** - * @private - */ - StackingAxis.onDestroy = function () { - var stacking = this.stacking; - if (!stacking) { - return; - } - var stacks = stacking.stacks; - // Destroy each stack total - objectEach(stacks, function (stack, stackKey) { - destroyObjectProperties(stack); - stacks[stackKey] = null; - }); - if (stacking && - stacking.stackTotalGroup) { - stacking.stackTotalGroup.destroy(); - } - }; - /** - * @private - */ - StackingAxis.onInit = function () { - var axis = this; - if (!axis.stacking) { - axis.stacking = new StackingAxisAdditions(axis); - } - }; - return StackingAxis; - }()); - - return StackingAxis; - }); - _registerModule(_modules, 'Mixins/LegendSymbol.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var merge = U.merge, - pick = U.pick; - /* eslint-disable valid-jsdoc */ - /** - * Legend symbol mixin. - * - * @private - * @mixin Highcharts.LegendSymbolMixin - */ - var LegendSymbolMixin = H.LegendSymbolMixin = { - /** - * Get the series' symbol in the legend - * - * @private - * @function Highcharts.LegendSymbolMixin.drawRectangle - * - * @param {Highcharts.Legend} legend - * The legend object - * - * @param {Highcharts.Point|Highcharts.Series} item - * The series (this) or point - */ - drawRectangle: function (legend, - item) { - var options = legend.options, - symbolHeight = legend.symbolHeight, - square = options.squareSymbol, - symbolWidth = square ? symbolHeight : legend.symbolWidth; - item.legendSymbol = this.chart.renderer.rect(square ? (legend.symbolWidth - symbolHeight) / 2 : 0, legend.baseline - symbolHeight + 1, // #3988 - symbolWidth, symbolHeight, pick(legend.options.symbolRadius, symbolHeight / 2)) - .addClass('highcharts-point') - .attr({ - zIndex: 3 - }).add(item.legendGroup); - }, - /** - * Get the series' symbol in the legend. This method should be overridable - * to create custom symbols through - * Highcharts.seriesTypes[type].prototype.drawLegendSymbols. - * - * @private - * @function Highcharts.LegendSymbolMixin.drawLineMarker - * - * @param {Highcharts.Legend} legend - * The legend object. - */ - drawLineMarker: function (legend) { - var options = this.options, - markerOptions = options.marker, - radius, - legendSymbol, - symbolWidth = legend.symbolWidth, - symbolHeight = legend.symbolHeight, - generalRadius = symbolHeight / 2, - renderer = this.chart.renderer, - legendItemGroup = this.legendGroup, - verticalCenter = legend.baseline - - Math.round(legend.fontMetrics.b * 0.3), - attr = {}; - // Draw the line - if (!this.chart.styledMode) { - attr = { - 'stroke-width': options.lineWidth || 0 - }; - if (options.dashStyle) { - attr.dashstyle = options.dashStyle; - } - } - this.legendLine = renderer - .path([ - ['M', 0, verticalCenter], - ['L', symbolWidth, verticalCenter] - ]) - .addClass('highcharts-graph') - .attr(attr) - .add(legendItemGroup); - // Draw the marker - if (markerOptions && markerOptions.enabled !== false && symbolWidth) { - // Do not allow the marker to be larger than the symbolHeight - radius = Math.min(pick(markerOptions.radius, generalRadius), generalRadius); - // Restrict symbol markers size - if (this.symbol.indexOf('url') === 0) { - markerOptions = merge(markerOptions, { - width: symbolHeight, - height: symbolHeight - }); - radius = 0; - } - this.legendSymbol = legendSymbol = renderer.symbol(this.symbol, (symbolWidth / 2) - radius, verticalCenter - radius, 2 * radius, 2 * radius, markerOptions) - .addClass('highcharts-point') - .add(legendItemGroup); - legendSymbol.isMarker = true; - } - } - }; - - return LegendSymbolMixin; - }); - _registerModule(_modules, 'Core/Series/CartesianSeries.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Series/Series.js'], _modules['Core/Globals.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Utilities.js']], function (A, BaseSeries, H, LegendSymbolMixin, O, Point, SVGElement, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var animObject = A.animObject; - var defaultOptions = O.defaultOptions; - var addEvent = U.addEvent, - arrayMax = U.arrayMax, - arrayMin = U.arrayMin, - clamp = U.clamp, - correctFloat = U.correctFloat, - defined = U.defined, - erase = U.erase, - error = U.error, - extend = U.extend, - find = U.find, - fireEvent = U.fireEvent, - getNestedProperty = U.getNestedProperty, - isArray = U.isArray, - isFunction = U.isFunction, - isNumber = U.isNumber, - isString = U.isString, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick, - removeEvent = U.removeEvent, - splat = U.splat, - syncTimeout = U.syncTimeout; - /** - * This is a placeholder type of the possible series options for - * [Highcharts](../highcharts/series), [Highstock](../highstock/series), - * [Highmaps](../highmaps/series), and [Gantt](../gantt/series). - * - * In TypeScript is this dynamically generated to reference all possible types - * of series options. - * - * @ignore-declaration - * @typedef {Highcharts.SeriesOptions|Highcharts.Dictionary<*>} Highcharts.SeriesOptionsType - */ - /** - * Options for `dataSorting`. - * - * @interface Highcharts.DataSortingOptionsObject - * @since 8.0.0 - */ /** - * Enable or disable data sorting for the series. - * @name Highcharts.DataSortingOptionsObject#enabled - * @type {boolean|undefined} - */ /** - * Whether to allow matching points by name in an update. - * @name Highcharts.DataSortingOptionsObject#matchByName - * @type {boolean|undefined} - */ /** - * Determines what data value should be used to sort by. - * @name Highcharts.DataSortingOptionsObject#sortKey - * @type {string|undefined} - */ - /** - * Function callback when a series has been animated. - * - * @callback Highcharts.SeriesAfterAnimateCallbackFunction - * - * @param {Highcharts.Series} this - * The series where the event occured. - * - * @param {Highcharts.SeriesAfterAnimateEventObject} event - * Event arguments. - */ - /** - * Event information regarding completed animation of a series. - * - * @interface Highcharts.SeriesAfterAnimateEventObject - */ /** - * Animated series. - * @name Highcharts.SeriesAfterAnimateEventObject#target - * @type {Highcharts.Series} - */ /** - * Event type. - * @name Highcharts.SeriesAfterAnimateEventObject#type - * @type {"afterAnimate"} - */ - /** - * Function callback when the checkbox next to the series' name in the legend is - * clicked. - * - * @callback Highcharts.SeriesCheckboxClickCallbackFunction - * - * @param {Highcharts.Series} this - * The series where the event occured. - * - * @param {Highcharts.SeriesCheckboxClickEventObject} event - * Event arguments. - */ - /** - * Event information regarding check of a series box. - * - * @interface Highcharts.SeriesCheckboxClickEventObject - */ /** - * Whether the box has been checked. - * @name Highcharts.SeriesCheckboxClickEventObject#checked - * @type {boolean} - */ /** - * Related series. - * @name Highcharts.SeriesCheckboxClickEventObject#item - * @type {Highcharts.Series} - */ /** - * Related series. - * @name Highcharts.SeriesCheckboxClickEventObject#target - * @type {Highcharts.Series} - */ /** - * Event type. - * @name Highcharts.SeriesCheckboxClickEventObject#type - * @type {"checkboxClick"} - */ - /** - * Function callback when a series is clicked. Return false to cancel toogle - * actions. - * - * @callback Highcharts.SeriesClickCallbackFunction - * - * @param {Highcharts.Series} this - * The series where the event occured. - * - * @param {Highcharts.SeriesClickEventObject} event - * Event arguments. - */ - /** - * Common information for a click event on a series. - * - * @interface Highcharts.SeriesClickEventObject - * @extends global.Event - */ /** - * Nearest point on the graph. - * @name Highcharts.SeriesClickEventObject#point - * @type {Highcharts.Point} - */ - /** - * Gets fired when the series is hidden after chart generation time, either by - * clicking the legend item or by calling `.hide()`. - * - * @callback Highcharts.SeriesHideCallbackFunction - * - * @param {Highcharts.Series} this - * The series where the event occured. - * - * @param {global.Event} event - * The event that occured. - */ - /** - * The SVG value used for the `stroke-linecap` and `stroke-linejoin` of a line - * graph. - * - * @typedef {"butt"|"round"|"square"|string} Highcharts.SeriesLinecapValue - */ - /** - * Gets fired when the legend item belonging to the series is clicked. The - * default action is to toggle the visibility of the series. This can be - * prevented by returning `false` or calling `event.preventDefault()`. - * - * @callback Highcharts.SeriesLegendItemClickCallbackFunction - * - * @param {Highcharts.Series} this - * The series where the event occured. - * - * @param {Highcharts.SeriesLegendItemClickEventObject} event - * The event that occured. - */ - /** - * Information about the event. - * - * @interface Highcharts.SeriesLegendItemClickEventObject - */ /** - * Related browser event. - * @name Highcharts.SeriesLegendItemClickEventObject#browserEvent - * @type {global.PointerEvent} - */ /** - * Prevent the default action of toggle the visibility of the series. - * @name Highcharts.SeriesLegendItemClickEventObject#preventDefault - * @type {Function} - */ /** - * Related series. - * @name Highcharts.SeriesCheckboxClickEventObject#target - * @type {Highcharts.Series} - */ /** - * Event type. - * @name Highcharts.SeriesCheckboxClickEventObject#type - * @type {"checkboxClick"} - */ - /** - * Gets fired when the mouse leaves the graph. - * - * @callback Highcharts.SeriesMouseOutCallbackFunction - * - * @param {Highcharts.Series} this - * Series where the event occured. - * - * @param {global.PointerEvent} event - * Event that occured. - */ - /** - * Gets fired when the mouse enters the graph. - * - * @callback Highcharts.SeriesMouseOverCallbackFunction - * - * @param {Highcharts.Series} this - * Series where the event occured. - * - * @param {global.PointerEvent} event - * Event that occured. - */ - /** - * Translation and scale for the plot area of a series. - * - * @interface Highcharts.SeriesPlotBoxObject - */ /** - * @name Highcharts.SeriesPlotBoxObject#scaleX - * @type {number} - */ /** - * @name Highcharts.SeriesPlotBoxObject#scaleY - * @type {number} - */ /** - * @name Highcharts.SeriesPlotBoxObject#translateX - * @type {number} - */ /** - * @name Highcharts.SeriesPlotBoxObject#translateY - * @type {number} - */ - /** - * Gets fired when the series is shown after chart generation time, either by - * clicking the legend item or by calling `.show()`. - * - * @callback Highcharts.SeriesShowCallbackFunction - * - * @param {Highcharts.Series} this - * Series where the event occured. - * - * @param {global.Event} event - * Event that occured. - */ - /** - * Possible key values for the series state options. - * - * @typedef {"hover"|"inactive"|"normal"|"select"} Highcharts.SeriesStateValue - */ - ''; // detach doclets above - var seriesTypes = BaseSeries.seriesTypes, - win = H.win; - /** - * This is the base series prototype that all other series types inherit from. - * A new series is initialized either through the - * [series](https://api.highcharts.com/highcharts/series) - * option structure, or after the chart is initialized, through - * {@link Highcharts.Chart#addSeries}. - * - * The object can be accessed in a number of ways. All series and point event - * handlers give a reference to the `series` object. The chart object has a - * {@link Highcharts.Chart#series|series} property that is a collection of all - * the chart's series. The point objects and axis objects also have the same - * reference. - * - * Another way to reference the series programmatically is by `id`. Add an id - * in the series configuration options, and get the series object by - * {@link Highcharts.Chart#get}. - * - * Configuration options for the series are given in three levels. Options for - * all series in a chart are given in the - * [plotOptions.series](https://api.highcharts.com/highcharts/plotOptions.series) - * object. Then options for all series of a specific type - * are given in the plotOptions of that type, for example `plotOptions.line`. - * Next, options for one single series are given in the series array, or as - * arguments to `chart.addSeries`. - * - * The data in the series is stored in various arrays. - * - * - First, `series.options.data` contains all the original config options for - * each point whether added by options or methods like `series.addPoint`. - * - * - Next, `series.data` contains those values converted to points, but in case - * the series data length exceeds the `cropThreshold`, or if the data is - * grouped, `series.data` doesn't contain all the points. It only contains the - * points that have been created on demand. - * - * - Then there's `series.points` that contains all currently visible point - * objects. In case of cropping, the cropped-away points are not part of this - * array. The `series.points` array starts at `series.cropStart` compared to - * `series.data` and `series.options.data`. If however the series data is - * grouped, these can't be correlated one to one. - * - * - `series.xData` and `series.processedXData` contain clean x values, - * equivalent to `series.data` and `series.points`. - * - * - `series.yData` and `series.processedYData` contain clean y values, - * equivalent to `series.data` and `series.points`. - * - * @class - * @name Highcharts.Series - * - * @param {Highcharts.Chart} chart - * The chart instance. - * - * @param {Highcharts.SeriesOptionsType|object} options - * The series options. - */ /** - * The line series is the base type and is therefor the series base prototype. - * - * @private - * @class - * @name Highcharts.seriesTypes.line - * - * @augments Highcharts.Series - */ - var CartesianSeries = BaseSeries.seriesType('line', - /** - * Series options for specific data and the data itself. In TypeScript you - * have to cast the series options to specific series types, - to get all - * possible options for a series. - * - * @example - * // TypeScript example - * Highcharts.chart('container', { - * series: [{ - * color: '#06C', - * data: [[0, 1], - [2, 3]] - * } as Highcharts.SeriesLineOptions ] - * }); - * - * @type {Array<*>} - * @apioption series - */ - /** - * An id for the series. This can be used after render time to get a pointer - * to the series object through `chart.get()`. - * - * @sample {highcharts} highcharts/plotoptions/series-id/ - * Get series by id - * - * @type {string} - * @since 1.2.0 - * @apioption series.id - */ - /** - * The index of the series in the chart, affecting the internal index in the - * `chart.series` array, the visible Z index as well as the order in the - * legend. - * - * @type {number} - * @since 2.3.0 - * @apioption series.index - */ - /** - * The sequential index of the series in the legend. - * - * @see [legend.reversed](#legend.reversed), - * [yAxis.reversedStacks](#yAxis.reversedStacks) - * - * @sample {highcharts|highstock} highcharts/series/legendindex/ - * Legend in opposite order - * - * @type {number} - * @apioption series.legendIndex - */ - /** - * The name of the series as shown in the legend, tooltip etc. - * - * @sample {highcharts} highcharts/series/name/ - * Series name - * @sample {highmaps} maps/demo/category-map/ - * Series name - * - * @type {string} - * @apioption series.name - */ - /** - * This option allows grouping series in a stacked chart. The stack option - * can be a string or anything else, as long as the grouped series' stack - * options match each other after conversion into a string. - * - * @sample {highcharts} highcharts/series/stack/ - * Stacked and grouped columns - * - * @type {number|string} - * @since 2.1 - * @product highcharts highstock - * @apioption series.stack - */ - /** - * The type of series, for example `line` or `column`. By default, the - * series type is inherited from [chart.type](#chart.type), so unless the - * chart is a combination of series types, there is no need to set it on the - * series level. - * - * @sample {highcharts} highcharts/series/type/ - * Line and column in the same chart - * @sample highcharts/series/type-dynamic/ - * Dynamic types with button selector - * @sample {highmaps} maps/demo/mapline-mappoint/ - * Multiple types in the same map - * - * @type {string} - * @apioption series.type - */ - /** - * When using dual or multiple x axes, this number defines which xAxis the - * particular series is connected to. It refers to either the - * {@link #xAxis.id|axis id} - * or the index of the axis in the xAxis array, with 0 being the first. - * - * @type {number|string} - * @default 0 - * @product highcharts highstock - * @apioption series.xAxis - */ - /** - * When using dual or multiple y axes, this number defines which yAxis the - * particular series is connected to. It refers to either the - * {@link #yAxis.id|axis id} - * or the index of the axis in the yAxis array, with 0 being the first. - * - * @sample {highcharts} highcharts/series/yaxis/ - * Apply the column series to the secondary Y axis - * - * @type {number|string} - * @default 0 - * @product highcharts highstock - * @apioption series.yAxis - */ - /** - * Define the visual z index of the series. - * - * @sample {highcharts} highcharts/plotoptions/series-zindex-default/ - * With no z index, the series defined last are on top - * @sample {highcharts} highcharts/plotoptions/series-zindex/ - * With a z index, the series with the highest z index is on top - * @sample {highstock} highcharts/plotoptions/series-zindex-default/ - * With no z index, the series defined last are on top - * @sample {highstock} highcharts/plotoptions/series-zindex/ - * With a z index, the series with the highest z index is on top - * - * @type {number} - * @product highcharts highstock - * @apioption series.zIndex - */ - void 0, - /** - * General options for all series types. - * - * @optionparent plotOptions.series - */ - { - /** - * The SVG value used for the `stroke-linecap` and `stroke-linejoin` - * of a line graph. Round means that lines are rounded in the ends and - * bends. - * - * @type {Highcharts.SeriesLinecapValue} - * @default round - * @since 3.0.7 - * @apioption plotOptions.line.linecap - */ - /** - * Pixel width of the graph line. - * - * @see In styled mode, the line stroke-width can be set with the - * `.highcharts-graph` class name. - * - * @sample {highcharts} highcharts/plotoptions/series-linewidth-general/ - * On all series - * @sample {highcharts} highcharts/plotoptions/series-linewidth-specific/ - * On one single series - * - * @product highcharts highstock - * - * @private - */ - lineWidth: 2, - /** - * For some series, there is a limit that shuts down initial animation - * by default when the total number of points in the chart is too high. - * For example, for a column chart and its derivatives, animation does - * not run if there is more than 250 points totally. To disable this - * cap, set `animationLimit` to `Infinity`. - * - * @type {number} - * @apioption plotOptions.series.animationLimit - */ - /** - * Allow this series' points to be selected by clicking on the graphic - * (columns, point markers, pie slices, map areas etc). - * - * The selected points can be handled by point select and unselect - * events, or collectively by the [getSelectedPoints - * ](/class-reference/Highcharts.Chart#getSelectedPoints) function. - * - * And alternative way of selecting points is through dragging. - * - * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-line/ - * Line - * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-column/ - * Column - * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-pie/ - * Pie - * @sample {highcharts} highcharts/chart/events-selection-points/ - * Select a range of points through a drag selection - * @sample {highmaps} maps/plotoptions/series-allowpointselect/ - * Map area - * @sample {highmaps} maps/plotoptions/mapbubble-allowpointselect/ - * Map bubble - * - * @since 1.2.0 - * - * @private - */ - allowPointSelect: false, - /** - * When true, each point or column edge is rounded to its nearest pixel - * in order to render sharp on screen. In some cases, when there are a - * lot of densely packed columns, this leads to visible difference - * in column widths or distance between columns. In these cases, - * setting `crisp` to `false` may look better, even though each column - * is rendered blurry. - * - * @sample {highcharts} highcharts/plotoptions/column-crisp-false/ - * Crisp is false - * - * @since 5.0.10 - * @product highcharts highstock gantt - * - * @private - */ - crisp: true, - /** - * If true, a checkbox is displayed next to the legend item to allow - * selecting the series. The state of the checkbox is determined by - * the `selected` option. - * - * @productdesc {highmaps} - * Note that if a `colorAxis` is defined, the color axis is represented - * in the legend, not the series. - * - * @sample {highcharts} highcharts/plotoptions/series-showcheckbox-true/ - * Show select box - * - * @since 1.2.0 - * - * @private - */ - showCheckbox: false, - /** - * Enable or disable the initial animation when a series is displayed. - * The animation can also be set as a configuration object. Please - * note that this option only applies to the initial animation of the - * series itself. For other animations, see [chart.animation]( - * #chart.animation) and the animation parameter under the API methods. - * The following properties are supported: - * - * - `defer`: The animation delay time in milliseconds. - * - * - `duration`: The duration of the animation in milliseconds. - * - * - `easing`: Can be a string reference to an easing function set on - * the `Math` object or a function. See the _Custom easing function_ - * demo below. - * - * Due to poor performance, animation is disabled in old IE browsers - * for several chart types. - * - * @sample {highcharts} highcharts/plotoptions/series-animation-disabled/ - * Animation disabled - * @sample {highcharts} highcharts/plotoptions/series-animation-slower/ - * Slower animation - * @sample {highcharts} highcharts/plotoptions/series-animation-easing/ - * Custom easing function - * @sample {highstock} stock/plotoptions/animation-slower/ - * Slower animation - * @sample {highstock} stock/plotoptions/animation-easing/ - * Custom easing function - * @sample {highmaps} maps/plotoptions/series-animation-true/ - * Animation enabled on map series - * @sample {highmaps} maps/plotoptions/mapbubble-animation-false/ - * Disabled on mapbubble series - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - * @default {highcharts} true - * @default {highstock} true - * @default {highmaps} false - * - * @private - */ - animation: { - /** @internal */ - duration: 1000 - }, - /** - * @default 0 - * @type {number} - * @since 8.2.0 - * @apioption plotOptions.series.animation.defer - */ - /** - * An additional class name to apply to the series' graphical elements. - * This option does not replace default class names of the graphical - * element. - * - * @type {string} - * @since 5.0.0 - * @apioption plotOptions.series.className - */ - /** - * Disable this option to allow series rendering in the whole plotting - * area. - * - * **Note:** Clipping should be always enabled when - * [chart.zoomType](#chart.zoomType) is set - * - * @sample {highcharts} highcharts/plotoptions/series-clip/ - * Disabled clipping - * - * @default true - * @type {boolean} - * @since 3.0.0 - * @apioption plotOptions.series.clip - */ - /** - * The main color of the series. In line type series it applies to the - * line and the point markers unless otherwise specified. In bar type - * series it applies to the bars unless a color is specified per point. - * The default value is pulled from the `options.colors` array. - * - * In styled mode, the color can be defined by the - * [colorIndex](#plotOptions.series.colorIndex) option. Also, the series - * color can be set with the `.highcharts-series`, - * `.highcharts-color-{n}`, `.highcharts-{type}-series` or - * `.highcharts-series-{n}` class, or individual classes given by the - * `className` option. - * - * @productdesc {highmaps} - * In maps, the series color is rarely used, as most choropleth maps use - * the color to denote the value of each point. The series color can - * however be used in a map with multiple series holding categorized - * data. - * - * @sample {highcharts} highcharts/plotoptions/series-color-general/ - * General plot option - * @sample {highcharts} highcharts/plotoptions/series-color-specific/ - * One specific series - * @sample {highcharts} highcharts/plotoptions/series-color-area/ - * Area color - * @sample {highcharts} highcharts/series/infographic/ - * Pattern fill - * @sample {highmaps} maps/demo/category-map/ - * Category map by multiple series - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption plotOptions.series.color - */ - /** - * Styled mode only. A specific color index to use for the series, so - * its graphic representations are given the class name - * `highcharts-color-{n}`. - * - * @type {number} - * @since 5.0.0 - * @apioption plotOptions.series.colorIndex - */ - /** - * Whether to connect a graph line across null points, or render a gap - * between the two points on either side of the null. - * - * @sample {highcharts} highcharts/plotoptions/series-connectnulls-false/ - * False by default - * @sample {highcharts} highcharts/plotoptions/series-connectnulls-true/ - * True - * - * @type {boolean} - * @default false - * @product highcharts highstock - * @apioption plotOptions.series.connectNulls - */ - /** - * You can set the cursor to "pointer" if you have click events attached - * to the series, to signal to the user that the points and lines can - * be clicked. - * - * In styled mode, the series cursor can be set with the same classes - * as listed under [series.color](#plotOptions.series.color). - * - * @sample {highcharts} highcharts/plotoptions/series-cursor-line/ - * On line graph - * @sample {highcharts} highcharts/plotoptions/series-cursor-column/ - * On columns - * @sample {highcharts} highcharts/plotoptions/series-cursor-scatter/ - * On scatter markers - * @sample {highstock} stock/plotoptions/cursor/ - * Pointer on a line graph - * @sample {highmaps} maps/plotoptions/series-allowpointselect/ - * Map area - * @sample {highmaps} maps/plotoptions/mapbubble-allowpointselect/ - * Map bubble - * - * @type {string|Highcharts.CursorValue} - * @apioption plotOptions.series.cursor - */ - /** - * A reserved subspace to store options and values for customized - * functionality. Here you can add additional data for your own event - * callbacks and formatter callbacks. - * - * @sample {highcharts} highcharts/point/custom/ - * Point and series with custom data - * - * @type {Highcharts.Dictionary<*>} - * @apioption plotOptions.series.custom - */ - /** - * Name of the dash style to use for the graph, or for some series types - * the outline of each shape. - * - * In styled mode, the - * [stroke dash-array](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/css/series-dashstyle/) - * can be set with the same classes as listed under - * [series.color](#plotOptions.series.color). - * - * @sample {highcharts} highcharts/plotoptions/series-dashstyle-all/ - * Possible values demonstrated - * @sample {highcharts} highcharts/plotoptions/series-dashstyle/ - * Chart suitable for printing in black and white - * @sample {highstock} highcharts/plotoptions/series-dashstyle-all/ - * Possible values demonstrated - * @sample {highmaps} highcharts/plotoptions/series-dashstyle-all/ - * Possible values demonstrated - * @sample {highmaps} maps/plotoptions/series-dashstyle/ - * Dotted borders on a map - * - * @type {Highcharts.DashStyleValue} - * @default Solid - * @since 2.1 - * @apioption plotOptions.series.dashStyle - */ - /** - * A description of the series to add to the screen reader information - * about the series. - * - * @type {string} - * @since 5.0.0 - * @requires modules/accessibility - * @apioption plotOptions.series.description - */ - /** - * Options for the series data sorting. - * - * @type {Highcharts.DataSortingOptionsObject} - * @since 8.0.0 - * @product highcharts highstock - * @apioption plotOptions.series.dataSorting - */ - /** - * Enable or disable data sorting for the series. Use [xAxis.reversed]( - * #xAxis.reversed) to change the sorting order. - * - * @sample {highcharts} highcharts/datasorting/animation/ - * Data sorting in scatter-3d - * @sample {highcharts} highcharts/datasorting/labels-animation/ - * Axis labels animation - * @sample {highcharts} highcharts/datasorting/dependent-sorting/ - * Dependent series sorting - * @sample {highcharts} highcharts/datasorting/independent-sorting/ - * Independent series sorting - * - * @type {boolean} - * @since 8.0.0 - * @apioption plotOptions.series.dataSorting.enabled - */ - /** - * Whether to allow matching points by name in an update. If this option - * is disabled, points will be matched by order. - * - * @sample {highcharts} highcharts/datasorting/match-by-name/ - * Enabled match by name - * - * @type {boolean} - * @since 8.0.0 - * @apioption plotOptions.series.dataSorting.matchByName - */ - /** - * Determines what data value should be used to sort by. - * - * @sample {highcharts} highcharts/datasorting/sort-key/ - * Sort key as `z` value - * - * @type {string} - * @since 8.0.0 - * @default y - * @apioption plotOptions.series.dataSorting.sortKey - */ - /** - * Enable or disable the mouse tracking for a specific series. This - * includes point tooltips and click events on graphs and points. For - * large datasets it improves performance. - * - * @sample {highcharts} highcharts/plotoptions/series-enablemousetracking-false/ - * No mouse tracking - * @sample {highmaps} maps/plotoptions/series-enablemousetracking-false/ - * No mouse tracking - * - * @type {boolean} - * @default true - * @apioption plotOptions.series.enableMouseTracking - */ - /** - * Whether to use the Y extremes of the total chart width or only the - * zoomed area when zooming in on parts of the X axis. By default, the - * Y axis adjusts to the min and max of the visible data. Cartesian - * series only. - * - * @type {boolean} - * @default false - * @since 4.1.6 - * @product highcharts highstock gantt - * @apioption plotOptions.series.getExtremesFromAll - */ - /** - * An array specifying which option maps to which key in the data point - * array. This makes it convenient to work with unstructured data arrays - * from different sources. - * - * @see [series.data](#series.line.data) - * - * @sample {highcharts|highstock} highcharts/series/data-keys/ - * An extended data array with keys - * @sample {highcharts|highstock} highcharts/series/data-nested-keys/ - * Nested keys used to access object properties - * - * @type {Array<string>} - * @since 4.1.6 - * @apioption plotOptions.series.keys - */ - /** - * The line cap used for line ends and line joins on the graph. - * - * @type {Highcharts.SeriesLinecapValue} - * @default round - * @product highcharts highstock - * @apioption plotOptions.series.linecap - */ - /** - * The [id](#series.id) of another series to link to. Additionally, - * the value can be ":previous" to link to the previous series. When - * two series are linked, only the first one appears in the legend. - * Toggling the visibility of this also toggles the linked series. - * - * If master series uses data sorting and linked series does not have - * its own sorting definition, the linked series will be sorted in the - * same order as the master one. - * - * @sample {highcharts|highstock} highcharts/demo/arearange-line/ - * Linked series - * - * @type {string} - * @since 3.0 - * @product highcharts highstock gantt - * @apioption plotOptions.series.linkedTo - */ - /** - * Options for the corresponding navigator series if `showInNavigator` - * is `true` for this series. Available options are the same as any - * series, documented at [plotOptions](#plotOptions.series) and - * [series](#series). - * - * These options are merged with options in [navigator.series]( - * #navigator.series), and will take precedence if the same option is - * defined both places. - * - * @see [navigator.series](#navigator.series) - * - * @type {Highcharts.PlotSeriesOptions} - * @since 5.0.0 - * @product highstock - * @apioption plotOptions.series.navigatorOptions - */ - /** - * The color for the parts of the graph or points that are below the - * [threshold](#plotOptions.series.threshold). Note that `zones` takes - * precedence over the negative color. Using `negativeColor` is - * equivalent to applying a zone with value of 0. - * - * @see In styled mode, a negative color is applied by setting this option - * to `true` combined with the `.highcharts-negative` class name. - * - * @sample {highcharts} highcharts/plotoptions/series-negative-color/ - * Spline, area and column - * @sample {highcharts} highcharts/plotoptions/arearange-negativecolor/ - * Arearange - * @sample {highcharts} highcharts/css/series-negative-color/ - * Styled mode - * @sample {highstock} highcharts/plotoptions/series-negative-color/ - * Spline, area and column - * @sample {highstock} highcharts/plotoptions/arearange-negativecolor/ - * Arearange - * @sample {highmaps} highcharts/plotoptions/series-negative-color/ - * Spline, area and column - * @sample {highmaps} highcharts/plotoptions/arearange-negativecolor/ - * Arearange - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 3.0 - * @apioption plotOptions.series.negativeColor - */ - /** - * Same as - * [accessibility.pointDescriptionFormatter](#accessibility.pointDescriptionFormatter), - * but for an individual series. Overrides the chart wide configuration. - * - * @type {Function} - * @since 5.0.12 - * @apioption plotOptions.series.pointDescriptionFormatter - */ - /** - * If no x values are given for the points in a series, `pointInterval` - * defines the interval of the x values. For example, if a series - * contains one value every decade starting from year 0, set - * `pointInterval` to `10`. In true `datetime` axes, the `pointInterval` - * is set in milliseconds. - * - * It can be also be combined with `pointIntervalUnit` to draw irregular - * time intervals. - * - * Please note that this options applies to the _series data_, not the - * interval of the axis ticks, which is independent. - * - * @sample {highcharts} highcharts/plotoptions/series-pointstart-datetime/ - * Datetime X axis - * @sample {highstock} stock/plotoptions/pointinterval-pointstart/ - * Using pointStart and pointInterval - * - * @type {number} - * @default 1 - * @product highcharts highstock gantt - * @apioption plotOptions.series.pointInterval - */ - /** - * On datetime series, this allows for setting the - * [pointInterval](#plotOptions.series.pointInterval) to irregular time - * units, `day`, `month` and `year`. A day is usually the same as 24 - * hours, but `pointIntervalUnit` also takes the DST crossover into - * consideration when dealing with local time. Combine this option with - * `pointInterval` to draw weeks, quarters, 6 months, 10 years etc. - * - * Please note that this options applies to the _series data_, not the - * interval of the axis ticks, which is independent. - * - * @sample {highcharts} highcharts/plotoptions/series-pointintervalunit/ - * One point a month - * @sample {highstock} highcharts/plotoptions/series-pointintervalunit/ - * One point a month - * - * @type {string} - * @since 4.1.0 - * @product highcharts highstock gantt - * @validvalue ["day", "month", "year"] - * @apioption plotOptions.series.pointIntervalUnit - */ - /** - * Possible values: `"on"`, `"between"`, `number`. - * - * In a column chart, when pointPlacement is `"on"`, the point will not - * create any padding of the X axis. In a polar column chart this means - * that the first column points directly north. If the pointPlacement is - * `"between"`, the columns will be laid out between ticks. This is - * useful for example for visualising an amount between two points in - * time or in a certain sector of a polar chart. - * - * Since Highcharts 3.0.2, the point placement can also be numeric, - * where 0 is on the axis value, -0.5 is between this value and the - * previous, and 0.5 is between this value and the next. Unlike the - * textual options, numeric point placement options won't affect axis - * padding. - * - * Note that pointPlacement needs a [pointRange]( - * #plotOptions.series.pointRange) to work. For column series this is - * computed, but for line-type series it needs to be set. - * - * For the `xrange` series type and gantt charts, if the Y axis is a - * category axis, the `pointPlacement` applies to the Y axis rather than - * the (typically datetime) X axis. - * - * Defaults to `undefined` in cartesian charts, `"between"` in polar - * charts. - * - * @see [xAxis.tickmarkPlacement](#xAxis.tickmarkPlacement) - * - * @sample {highcharts|highstock} highcharts/plotoptions/series-pointplacement-between/ - * Between in a column chart - * @sample {highcharts|highstock} highcharts/plotoptions/series-pointplacement-numeric/ - * Numeric placement for custom layout - * @sample {highcharts|highstock} maps/plotoptions/heatmap-pointplacement/ - * Placement in heatmap - * - * @type {string|number} - * @since 2.3.0 - * @product highcharts highstock gantt - * @apioption plotOptions.series.pointPlacement - */ - /** - * If no x values are given for the points in a series, pointStart - * defines on what value to start. For example, if a series contains one - * yearly value starting from 1945, set pointStart to 1945. - * - * @sample {highcharts} highcharts/plotoptions/series-pointstart-linear/ - * Linear - * @sample {highcharts} highcharts/plotoptions/series-pointstart-datetime/ - * Datetime - * @sample {highstock} stock/plotoptions/pointinterval-pointstart/ - * Using pointStart and pointInterval - * - * @type {number} - * @default 0 - * @product highcharts highstock gantt - * @apioption plotOptions.series.pointStart - */ - /** - * Whether to select the series initially. If `showCheckbox` is true, - * the checkbox next to the series name in the legend will be checked - * for a selected series. - * - * @sample {highcharts} highcharts/plotoptions/series-selected/ - * One out of two series selected - * - * @type {boolean} - * @default false - * @since 1.2.0 - * @apioption plotOptions.series.selected - */ - /** - * Whether to apply a drop shadow to the graph line. Since 2.3 the - * shadow can be an object configuration containing `color`, `offsetX`, - * `offsetY`, `opacity` and `width`. - * - * @sample {highcharts} highcharts/plotoptions/series-shadow/ - * Shadow enabled - * - * @type {boolean|Highcharts.ShadowOptionsObject} - * @default false - * @apioption plotOptions.series.shadow - */ - /** - * Whether to display this particular series or series type in the - * legend. Standalone series are shown in legend by default, and linked - * series are not. Since v7.2.0 it is possible to show series that use - * colorAxis by setting this option to `true`. - * - * @sample {highcharts} highcharts/plotoptions/series-showinlegend/ - * One series in the legend, one hidden - * - * @type {boolean} - * @apioption plotOptions.series.showInLegend - */ - /** - * Whether or not to show the series in the navigator. Takes precedence - * over [navigator.baseSeries](#navigator.baseSeries) if defined. - * - * @type {boolean} - * @since 5.0.0 - * @product highstock - * @apioption plotOptions.series.showInNavigator - */ - /** - * If set to `true`, the accessibility module will skip past the points - * in this series for keyboard navigation. - * - * @type {boolean} - * @since 5.0.12 - * @apioption plotOptions.series.skipKeyboardNavigation - */ - /** - * Whether to stack the values of each series on top of each other. - * Possible values are `undefined` to disable, `"normal"` to stack by - * value or `"percent"`. - * - * When stacking is enabled, data must be sorted - * in ascending X order. - * - * Some stacking options are related to specific series types. In the - * streamgraph series type, the stacking option is set to `"stream"`. - * The second one is `"overlap"`, which only applies to waterfall - * series. - * - * @see [yAxis.reversedStacks](#yAxis.reversedStacks) - * - * @sample {highcharts} highcharts/plotoptions/series-stacking-line/ - * Line - * @sample {highcharts} highcharts/plotoptions/series-stacking-column/ - * Column - * @sample {highcharts} highcharts/plotoptions/series-stacking-bar/ - * Bar - * @sample {highcharts} highcharts/plotoptions/series-stacking-area/ - * Area - * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-line/ - * Line - * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-column/ - * Column - * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-bar/ - * Bar - * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-area/ - * Area - * @sample {highcharts} highcharts/plotoptions/series-waterfall-with-normal-stacking - * Waterfall with normal stacking - * @sample {highcharts} highcharts/plotoptions/series-waterfall-with-overlap-stacking - * Waterfall with overlap stacking - * @sample {highstock} stock/plotoptions/stacking/ - * Area - * - * @type {string} - * @product highcharts highstock - * @validvalue ["normal", "overlap", "percent", "stream"] - * @apioption plotOptions.series.stacking - */ - /** - * Whether to apply steps to the line. Possible values are `left`, - * `center` and `right`. - * - * @sample {highcharts} highcharts/plotoptions/line-step/ - * Different step line options - * @sample {highcharts} highcharts/plotoptions/area-step/ - * Stepped, stacked area - * @sample {highstock} stock/plotoptions/line-step/ - * Step line - * - * @type {string} - * @since 1.2.5 - * @product highcharts highstock - * @validvalue ["left", "center", "right"] - * @apioption plotOptions.series.step - */ - /** - * The threshold, also called zero level or base level. For line type - * series this is only used in conjunction with - * [negativeColor](#plotOptions.series.negativeColor). - * - * @see [softThreshold](#plotOptions.series.softThreshold). - * - * @type {number} - * @default 0 - * @since 3.0 - * @product highcharts highstock - * @apioption plotOptions.series.threshold - */ - /** - * Set the initial visibility of the series. - * - * @sample {highcharts} highcharts/plotoptions/series-visible/ - * Two series, one hidden and one visible - * @sample {highstock} stock/plotoptions/series-visibility/ - * Hidden series - * - * @type {boolean} - * @default true - * @apioption plotOptions.series.visible - */ - /** - * Defines the Axis on which the zones are applied. - * - * @see [zones](#plotOptions.series.zones) - * - * @sample {highcharts} highcharts/series/color-zones-zoneaxis-x/ - * Zones on the X-Axis - * @sample {highstock} highcharts/series/color-zones-zoneaxis-x/ - * Zones on the X-Axis - * - * @type {string} - * @default y - * @since 4.1.0 - * @product highcharts highstock - * @apioption plotOptions.series.zoneAxis - */ - /** - * General event handlers for the series items. These event hooks can - * also be attached to the series at run time using the - * `Highcharts.addEvent` function. - * - * @declare Highcharts.SeriesEventsOptionsObject - * - * @private - */ - events: {}, - /** - * Fires after the series has finished its initial animation, or in case - * animation is disabled, immediately as the series is displayed. - * - * @sample {highcharts} highcharts/plotoptions/series-events-afteranimate/ - * Show label after animate - * @sample {highstock} highcharts/plotoptions/series-events-afteranimate/ - * Show label after animate - * - * @type {Highcharts.SeriesAfterAnimateCallbackFunction} - * @since 4.0 - * @product highcharts highstock gantt - * @context Highcharts.Series - * @apioption plotOptions.series.events.afterAnimate - */ - /** - * Fires when the checkbox next to the series' name in the legend is - * clicked. One parameter, `event`, is passed to the function. The state - * of the checkbox is found by `event.checked`. The checked item is - * found by `event.item`. Return `false` to prevent the default action - * which is to toggle the select state of the series. - * - * @sample {highcharts} highcharts/plotoptions/series-events-checkboxclick/ - * Alert checkbox status - * - * @type {Highcharts.SeriesCheckboxClickCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Series - * @apioption plotOptions.series.events.checkboxClick - */ - /** - * Fires when the series is clicked. One parameter, `event`, is passed - * to the function, containing common event information. Additionally, - * `event.point` holds a pointer to the nearest point on the graph. - * - * @sample {highcharts} highcharts/plotoptions/series-events-click/ - * Alert click info - * @sample {highstock} stock/plotoptions/series-events-click/ - * Alert click info - * @sample {highmaps} maps/plotoptions/series-events-click/ - * Display click info in subtitle - * - * @type {Highcharts.SeriesClickCallbackFunction} - * @context Highcharts.Series - * @apioption plotOptions.series.events.click - */ - /** - * Fires when the series is hidden after chart generation time, either - * by clicking the legend item or by calling `.hide()`. - * - * @sample {highcharts} highcharts/plotoptions/series-events-hide/ - * Alert when the series is hidden by clicking the legend item - * - * @type {Highcharts.SeriesHideCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Series - * @apioption plotOptions.series.events.hide - */ - /** - * Fires when the legend item belonging to the series is clicked. One - * parameter, `event`, is passed to the function. The default action - * is to toggle the visibility of the series. This can be prevented - * by returning `false` or calling `event.preventDefault()`. - * - * @sample {highcharts} highcharts/plotoptions/series-events-legenditemclick/ - * Confirm hiding and showing - * - * @type {Highcharts.SeriesLegendItemClickCallbackFunction} - * @context Highcharts.Series - * @apioption plotOptions.series.events.legendItemClick - */ - /** - * Fires when the mouse leaves the graph. One parameter, `event`, is - * passed to the function, containing common event information. If the - * [stickyTracking](#plotOptions.series) option is true, `mouseOut` - * doesn't happen before the mouse enters another graph or leaves the - * plot area. - * - * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-sticky/ - * With sticky tracking by default - * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-no-sticky/ - * Without sticky tracking - * - * @type {Highcharts.SeriesMouseOutCallbackFunction} - * @context Highcharts.Series - * @apioption plotOptions.series.events.mouseOut - */ - /** - * Fires when the mouse enters the graph. One parameter, `event`, is - * passed to the function, containing common event information. - * - * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-sticky/ - * With sticky tracking by default - * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-no-sticky/ - * Without sticky tracking - * - * @type {Highcharts.SeriesMouseOverCallbackFunction} - * @context Highcharts.Series - * @apioption plotOptions.series.events.mouseOver - */ - /** - * Fires when the series is shown after chart generation time, either - * by clicking the legend item or by calling `.show()`. - * - * @sample {highcharts} highcharts/plotoptions/series-events-show/ - * Alert when the series is shown by clicking the legend item. - * - * @type {Highcharts.SeriesShowCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Series - * @apioption plotOptions.series.events.show - */ - /** - * Options for the point markers of line-like series. Properties like - * `fillColor`, `lineColor` and `lineWidth` define the visual appearance - * of the markers. Other series types, like column series, don't have - * markers, but have visual options on the series level instead. - * - * In styled mode, the markers can be styled with the - * `.highcharts-point`, `.highcharts-point-hover` and - * `.highcharts-point-select` class names. - * - * @declare Highcharts.PointMarkerOptionsObject - * - * @private - */ - marker: { - /** - * Enable or disable the point marker. If `undefined`, the markers - * are hidden when the data is dense, and shown for more widespread - * data points. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-enabled/ - * Disabled markers - * @sample {highcharts} highcharts/plotoptions/series-marker-enabled-false/ - * Disabled in normal state but enabled on hover - * @sample {highstock} stock/plotoptions/series-marker/ - * Enabled markers - * - * @type {boolean} - * @default {highcharts} undefined - * @default {highstock} false - * @apioption plotOptions.series.marker.enabled - */ - /** - * The threshold for how dense the point markers should be before - * they are hidden, given that `enabled` is not defined. The number - * indicates the horizontal distance between the two closest points - * in the series, as multiples of the `marker.radius`. In other - * words, the default value of 2 means points are hidden if - * overlapping horizontally. - * - * @sample highcharts/plotoptions/series-marker-enabledthreshold - * A higher threshold - * - * @since 6.0.5 - */ - enabledThreshold: 2, - /** - * The fill color of the point marker. When `undefined`, the series' - * or point's color is used. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/ - * White fill - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption plotOptions.series.marker.fillColor - */ - /** - * Image markers only. Set the image width explicitly. When using - * this option, a `width` must also be set. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-width-height/ - * Fixed width and height - * @sample {highstock} highcharts/plotoptions/series-marker-width-height/ - * Fixed width and height - * - * @type {number} - * @since 4.0.4 - * @apioption plotOptions.series.marker.height - */ - /** - * The color of the point marker's outline. When `undefined`, the - * series' or point's color is used. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/ - * Inherit from series color (undefined) - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - lineColor: '#ffffff', - /** - * The width of the point marker's outline. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/ - * 2px blue marker - */ - lineWidth: 0, - /** - * The radius of the point marker. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-radius/ - * Bigger markers - * - * @default {highstock} 2 - */ - radius: 4, - /** - * A predefined shape or symbol for the marker. When undefined, the - * symbol is pulled from options.symbols. Other possible values are - * `'circle'`, `'square'`,`'diamond'`, `'triangle'` and - * `'triangle-down'`. - * - * Additionally, the URL to a graphic can be given on this form: - * `'url(graphic.png)'`. Note that for the image to be applied to - * exported charts, its URL needs to be accessible by the export - * server. - * - * Custom callbacks for symbol path generation can also be added to - * `Highcharts.SVGRenderer.prototype.symbols`. The callback is then - * used by its method name, as shown in the demo. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-symbol/ - * Predefined, graphic and custom markers - * @sample {highstock} highcharts/plotoptions/series-marker-symbol/ - * Predefined, graphic and custom markers - * - * @type {string} - * @apioption plotOptions.series.marker.symbol - */ - /** - * Image markers only. Set the image width explicitly. When using - * this option, a `height` must also be set. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-width-height/ - * Fixed width and height - * @sample {highstock} highcharts/plotoptions/series-marker-width-height/ - * Fixed width and height - * - * @type {number} - * @since 4.0.4 - * @apioption plotOptions.series.marker.width - */ - /** - * States for a single point marker. - * - * @declare Highcharts.PointStatesOptionsObject - */ - states: { - /** - * The normal state of a single point marker. Currently only - * used for setting animation when returning to normal state - * from hover. - * - * @declare Highcharts.PointStatesNormalOptionsObject - */ - normal: { - /** - * Animation when returning to normal state after hovering. - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - */ - animation: true - }, - /** - * The hover state for a single point marker. - * - * @declare Highcharts.PointStatesHoverOptionsObject - */ - hover: { - /** - * Animation when hovering over the marker. - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - */ - animation: { - /** @internal */ - duration: 50 - }, - /** - * Enable or disable the point marker. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-enabled/ - * Disabled hover state - */ - enabled: true, - /** - * The fill color of the marker in hover state. When - * `undefined`, the series' or point's fillColor for normal - * state is used. - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption plotOptions.series.marker.states.hover.fillColor - */ - /** - * The color of the point marker's outline. When - * `undefined`, the series' or point's lineColor for normal - * state is used. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-linecolor/ - * White fill color, black line color - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption plotOptions.series.marker.states.hover.lineColor - */ - /** - * The width of the point marker's outline. When - * `undefined`, the series' or point's lineWidth for normal - * state is used. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-linewidth/ - * 3px line width - * - * @type {number} - * @apioption plotOptions.series.marker.states.hover.lineWidth - */ - /** - * The radius of the point marker. In hover state, it - * defaults to the normal state's radius + 2 as per the - * [radiusPlus](#plotOptions.series.marker.states.hover.radiusPlus) - * option. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-radius/ - * 10px radius - * - * @type {number} - * @apioption plotOptions.series.marker.states.hover.radius - */ - /** - * The number of pixels to increase the radius of the - * hovered point. - * - * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/ - * 5 pixels greater radius on hover - * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/ - * 5 pixels greater radius on hover - * - * @since 4.0.3 - */ - radiusPlus: 2, - /** - * The additional line width for a hovered point. - * - * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/ - * 2 pixels wider on hover - * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/ - * 2 pixels wider on hover - * - * @since 4.0.3 - */ - lineWidthPlus: 1 - }, - /** - * The appearance of the point marker when selected. In order to - * allow a point to be selected, set the - * `series.allowPointSelect` option to true. - * - * @declare Highcharts.PointStatesSelectOptionsObject - */ - select: { - /** - * Enable or disable visible feedback for selection. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-enabled/ - * Disabled select state - * - * @type {boolean} - * @default true - * @apioption plotOptions.series.marker.states.select.enabled - */ - /** - * The radius of the point marker. In hover state, it - * defaults to the normal state's radius + 2. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-radius/ - * 10px radius for selected points - * - * @type {number} - * @apioption plotOptions.series.marker.states.select.radius - */ - /** - * The fill color of the point marker. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-fillcolor/ - * Solid red discs for selected points - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - fillColor: '#cccccc', - /** - * The color of the point marker's outline. When - * `undefined`, the series' or point's color is used. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-linecolor/ - * Red line color for selected points - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - */ - lineColor: '#000000', - /** - * The width of the point marker's outline. - * - * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-linewidth/ - * 3px line width for selected points - */ - lineWidth: 2 - } - } - }, - /** - * Properties for each single point. - * - * @declare Highcharts.PlotSeriesPointOptions - * - * @private - */ - point: { - /** - * Fires when a point is clicked. One parameter, `event`, is passed - * to the function, containing common event information. - * - * If the `series.allowPointSelect` option is true, the default - * action for the point's click event is to toggle the point's - * select state. Returning `false` cancels this action. - * - * @sample {highcharts} highcharts/plotoptions/series-point-events-click/ - * Click marker to alert values - * @sample {highcharts} highcharts/plotoptions/series-point-events-click-column/ - * Click column - * @sample {highcharts} highcharts/plotoptions/series-point-events-click-url/ - * Go to URL - * @sample {highmaps} maps/plotoptions/series-point-events-click/ - * Click marker to display values - * @sample {highmaps} maps/plotoptions/series-point-events-click-url/ - * Go to URL - * - * @type {Highcharts.PointClickCallbackFunction} - * @context Highcharts.Point - * @apioption plotOptions.series.point.events.click - */ - /** - * Fires when the mouse leaves the area close to the point. One - * parameter, `event`, is passed to the function, containing common - * event information. - * - * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/ - * Show values in the chart's corner on mouse over - * - * @type {Highcharts.PointMouseOutCallbackFunction} - * @context Highcharts.Point - * @apioption plotOptions.series.point.events.mouseOut - */ - /** - * Fires when the mouse enters the area close to the point. One - * parameter, `event`, is passed to the function, containing common - * event information. - * - * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/ - * Show values in the chart's corner on mouse over - * - * @type {Highcharts.PointMouseOverCallbackFunction} - * @context Highcharts.Point - * @apioption plotOptions.series.point.events.mouseOver - */ - /** - * Fires when the point is removed using the `.remove()` method. One - * parameter, `event`, is passed to the function. Returning `false` - * cancels the operation. - * - * @sample {highcharts} highcharts/plotoptions/series-point-events-remove/ - * Remove point and confirm - * - * @type {Highcharts.PointRemoveCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Point - * @apioption plotOptions.series.point.events.remove - */ - /** - * Fires when the point is selected either programmatically or - * following a click on the point. One parameter, `event`, is passed - * to the function. Returning `false` cancels the operation. - * - * @sample {highcharts} highcharts/plotoptions/series-point-events-select/ - * Report the last selected point - * @sample {highmaps} maps/plotoptions/series-allowpointselect/ - * Report select and unselect - * - * @type {Highcharts.PointSelectCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Point - * @apioption plotOptions.series.point.events.select - */ - /** - * Fires when the point is unselected either programmatically or - * following a click on the point. One parameter, `event`, is passed - * to the function. - * Returning `false` cancels the operation. - * - * @sample {highcharts} highcharts/plotoptions/series-point-events-unselect/ - * Report the last unselected point - * @sample {highmaps} maps/plotoptions/series-allowpointselect/ - * Report select and unselect - * - * @type {Highcharts.PointUnselectCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Point - * @apioption plotOptions.series.point.events.unselect - */ - /** - * Fires when the point is updated programmatically through the - * `.update()` method. One parameter, `event`, is passed to the - * function. The new point options can be accessed through - * `event.options`. Returning `false` cancels the operation. - * - * @sample {highcharts} highcharts/plotoptions/series-point-events-update/ - * Confirm point updating - * - * @type {Highcharts.PointUpdateCallbackFunction} - * @since 1.2.0 - * @context Highcharts.Point - * @apioption plotOptions.series.point.events.update - */ - /** - * Events for each single point. - * - * @declare Highcharts.PointEventsOptionsObject - */ - events: {} - }, - /** - * Options for the series data labels, appearing next to each data - * point. - * - * Since v6.2.0, multiple data labels can be applied to each single - * point by defining them as an array of configs. - * - * In styled mode, the data labels can be styled with the - * `.highcharts-data-label-box` and `.highcharts-data-label` class names - * ([see example](https://www.highcharts.com/samples/highcharts/css/series-datalabels)). - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-enabled - * Data labels enabled - * @sample {highcharts} highcharts/plotoptions/series-datalabels-multiple - * Multiple data labels on a bar series - * @sample {highcharts} highcharts/css/series-datalabels - * Style mode example - * - * @type {*|Array<*>} - * @product highcharts highstock highmaps gantt - * - * @private - */ - dataLabels: { - /** - * Enable or disable the initial animation when a series is - * displayed for the `dataLabels`. The animation can also be set as - * a configuration object. Please note that this option only - * applies to the initial animation. - * For other animations, see [chart.animation](#chart.animation) - * and the animation parameter under the API methods. - * The following properties are supported: - * - * - `defer`: The animation delay time in milliseconds. - * - * @sample {highcharts} highcharts/plotoptions/animation-defer/ - * Animation defer settings - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - * @since 8.2.0 - * @apioption plotOptions.series.dataLabels.animation - */ - animation: {}, - /** - * The animation delay time in milliseconds. - * Set to `0` renders dataLabel immediately. - * As `undefined` inherits defer time from the [series.animation.defer](#plotOptions.series.animation.defer). - * - * @type {number} - * @since 8.2.0 - * @apioption plotOptions.series.dataLabels.animation.defer - */ - /** - * The alignment of the data label compared to the point. If - * `right`, the right side of the label should be touching the - * point. For points with an extent, like columns, the alignments - * also dictates how to align it inside the box, as given with the - * [inside](#plotOptions.column.dataLabels.inside) - * option. Can be one of `left`, `center` or `right`. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-align-left/ - * Left aligned - * @sample {highcharts} highcharts/plotoptions/bar-datalabels-align-inside-bar/ - * Data labels inside the bar - * - * @type {Highcharts.AlignValue|null} - */ - align: 'center', - /** - * Whether to allow data labels to overlap. To make the labels less - * sensitive for overlapping, the - * [dataLabels.padding](#plotOptions.series.dataLabels.padding) - * can be set to 0. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-allowoverlap-false/ - * Don't allow overlap - * - * @type {boolean} - * @default false - * @since 4.1.0 - * @apioption plotOptions.series.dataLabels.allowOverlap - */ - /** - * The background color or gradient for the data label. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ - * Data labels box options - * @sample {highmaps} maps/plotoptions/series-datalabels-box/ - * Data labels box options - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 2.2.1 - * @apioption plotOptions.series.dataLabels.backgroundColor - */ - /** - * The border color for the data label. Defaults to `undefined`. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ - * Data labels box options - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 2.2.1 - * @apioption plotOptions.series.dataLabels.borderColor - */ - /** - * The border radius in pixels for the data label. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ - * Data labels box options - * @sample {highmaps} maps/plotoptions/series-datalabels-box/ - * Data labels box options - * - * @type {number} - * @default 0 - * @since 2.2.1 - * @apioption plotOptions.series.dataLabels.borderRadius - */ - /** - * The border width in pixels for the data label. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ - * Data labels box options - * - * @type {number} - * @default 0 - * @since 2.2.1 - * @apioption plotOptions.series.dataLabels.borderWidth - */ - /** - * A class name for the data label. Particularly in styled mode, - * this can be used to give each series' or point's data label - * unique styling. In addition to this option, a default color class - * name is added so that we can give the labels a contrast text - * shadow. - * - * @sample {highcharts} highcharts/css/data-label-contrast/ - * Contrast text shadow - * @sample {highcharts} highcharts/css/series-datalabels/ - * Styling by CSS - * - * @type {string} - * @since 5.0.0 - * @apioption plotOptions.series.dataLabels.className - */ - /** - * The text color for the data labels. Defaults to `undefined`. For - * certain series types, like column or map, the data labels can be - * drawn inside the points. In this case the data label will be - * drawn with maximum contrast by default. Additionally, it will be - * given a `text-outline` style with the opposite color, to further - * increase the contrast. This can be overridden by setting the - * `text-outline` style to `none` in the `dataLabels.style` option. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-color/ - * Red data labels - * @sample {highmaps} maps/demo/color-axis/ - * White data labels - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @apioption plotOptions.series.dataLabels.color - */ - /** - * Whether to hide data labels that are outside the plot area. By - * default, the data label is moved inside the plot area according - * to the - * [overflow](#plotOptions.series.dataLabels.overflow) - * option. - * - * @type {boolean} - * @default true - * @since 2.3.3 - * @apioption plotOptions.series.dataLabels.crop - */ - /** - * Whether to defer displaying the data labels until the initial - * series animation has finished. Setting to `false` renders the - * data label immediately. If set to `true` inherits the defer - * time set in [plotOptions.series.animation](#plotOptions.series.animation). - * - * @sample highcharts/plotoptions/animation-defer - * Set defer time - * - * @since 4.0.0 - * @product highcharts highstock gantt - */ - defer: true, - /** - * Enable or disable the data labels. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-enabled/ - * Data labels enabled - * @sample {highmaps} maps/demo/color-axis/ - * Data labels enabled - * - * @type {boolean} - * @default false - * @apioption plotOptions.series.dataLabels.enabled - */ - /** - * A declarative filter to control of which data labels to display. - * The declarative filter is designed for use when callback - * functions are not available, like when the chart options require - * a pure JSON structure or for use with graphical editors. For - * programmatic control, use the `formatter` instead, and return - * `undefined` to disable a single data label. - * - * @example - * filter: { - * property: 'percentage', - * operator: '>', - * value: 4 - * } - * - * @sample {highcharts} highcharts/demo/pie-monochrome - * Data labels filtered by percentage - * - * @declare Highcharts.DataLabelsFilterOptionsObject - * @since 6.0.3 - * @apioption plotOptions.series.dataLabels.filter - */ - /** - * The operator to compare by. Can be one of `>`, `<`, `>=`, `<=`, - * `==`, and `===`. - * - * @type {string} - * @validvalue [">", "<", ">=", "<=", "==", "==="] - * @apioption plotOptions.series.dataLabels.filter.operator - */ - /** - * The point property to filter by. Point options are passed - * directly to properties, additionally there are `y` value, - * `percentage` and others listed under {@link Highcharts.Point} - * members. - * - * @type {string} - * @apioption plotOptions.series.dataLabels.filter.property - */ - /** - * The value to compare against. - * - * @type {number} - * @apioption plotOptions.series.dataLabels.filter.value - */ - /** - * A - * [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) - * for the data label. Available variables are the same as for - * `formatter`. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ - * Add a unit - * @sample {highmaps} maps/plotoptions/series-datalabels-format/ - * Formatted value in the data label - * - * @type {string} - * @default y - * @default point.value - * @since 3.0 - * @apioption plotOptions.series.dataLabels.format - */ - // eslint-disable-next-line valid-jsdoc - /** - * Callback JavaScript function to format the data label. Note that - * if a `format` is defined, the format takes precedence and the - * formatter is ignored. - * - * @sample {highmaps} maps/plotoptions/series-datalabels-format/ - * Formatted value - * - * @type {Highcharts.DataLabelsFormatterCallbackFunction} - */ - formatter: function () { - var numberFormatter = this.series.chart.numberFormatter; - return typeof this.y !== 'number' ? '' : numberFormatter(this.y, -1); - }, - /** - * For points with an extent, like columns or map areas, whether to - * align the data label inside the box or to the actual value point. - * Defaults to `false` in most cases, `true` in stacked columns. - * - * @type {boolean} - * @since 3.0 - * @apioption plotOptions.series.dataLabels.inside - */ - /** - * Format for points with the value of null. Works analogously to - * [format](#plotOptions.series.dataLabels.format). `nullFormat` can - * be applied only to series which support displaying null points. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ - * Format data label and tooltip for null point. - * - * @type {boolean|string} - * @since 7.1.0 - * @apioption plotOptions.series.dataLabels.nullFormat - */ - /** - * Callback JavaScript function that defines formatting for points - * with the value of null. Works analogously to - * [formatter](#plotOptions.series.dataLabels.formatter). - * `nullPointFormatter` can be applied only to series which support - * displaying null points. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ - * Format data label and tooltip for null point. - * - * @type {Highcharts.DataLabelsFormatterCallbackFunction} - * @since 7.1.0 - * @apioption plotOptions.series.dataLabels.nullFormatter - */ - /** - * How to handle data labels that flow outside the plot area. The - * default is `"justify"`, which aligns them inside the plot area. - * For columns and bars, this means it will be moved inside the bar. - * To display data labels outside the plot area, set `crop` to - * `false` and `overflow` to `"allow"`. - * - * @type {Highcharts.DataLabelsOverflowValue} - * @default justify - * @since 3.0.6 - * @apioption plotOptions.series.dataLabels.overflow - */ - /** - * When either the `borderWidth` or the `backgroundColor` is set, - * this is the padding within the box. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ - * Data labels box options - * @sample {highmaps} maps/plotoptions/series-datalabels-box/ - * Data labels box options - * - * @since 2.2.1 - */ - padding: 5, - /** - * Aligns data labels relative to points. If `center` alignment is - * not possible, it defaults to `right`. - * - * @type {Highcharts.AlignValue} - * @default center - * @apioption plotOptions.series.dataLabels.position - */ - /** - * Text rotation in degrees. Note that due to a more complex - * structure, backgrounds, borders and padding will be lost on a - * rotated data label. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/ - * Vertical labels - * - * @type {number} - * @default 0 - * @apioption plotOptions.series.dataLabels.rotation - */ - /** - * The shadow of the box. Works best with `borderWidth` or - * `backgroundColor`. Since 2.3 the shadow can be an object - * configuration containing `color`, `offsetX`, `offsetY`, `opacity` - * and `width`. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ - * Data labels box options - * - * @type {boolean|Highcharts.ShadowOptionsObject} - * @default false - * @since 2.2.1 - * @apioption plotOptions.series.dataLabels.shadow - */ - /** - * The name of a symbol to use for the border around the label. - * Symbols are predefined functions on the Renderer object. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-shape/ - * A callout for annotations - * - * @type {string} - * @default square - * @since 4.1.2 - * @apioption plotOptions.series.dataLabels.shape - */ - /** - * Styles for the label. The default `color` setting is - * `"contrast"`, which is a pseudo color that Highcharts picks up - * and applies the maximum contrast to the underlying point item, - * for example the bar in a bar chart. - * - * The `textOutline` is a pseudo property that applies an outline of - * the given width with the given color, which by default is the - * maximum contrast to the text. So a bright text color will result - * in a black text outline for maximum readability on a mixed - * background. In some cases, especially with grayscale text, the - * text outline doesn't work well, in which cases it can be disabled - * by setting it to `"none"`. When `useHTML` is true, the - * `textOutline` will not be picked up. In this, case, the same - * effect can be acheived through the `text-shadow` CSS property. - * - * For some series types, where each point has an extent, like for - * example tree maps, the data label may overflow the point. There - * are two strategies for handling overflow. By default, the text - * will wrap to multiple lines. The other strategy is to set - * `style.textOverflow` to `ellipsis`, which will keep the text on - * one line plus it will break inside long words. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-style/ - * Bold labels - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow/ - * Long labels truncated with an ellipsis in a pie - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow-wrap/ - * Long labels are wrapped in a pie - * @sample {highmaps} maps/demo/color-axis/ - * Bold labels - * - * @type {Highcharts.CSSObject} - * @since 4.1.0 - * @apioption plotOptions.series.dataLabels.style - */ - style: { - /** @internal */ - fontSize: '11px', - /** @internal */ - fontWeight: 'bold', - /** @internal */ - color: 'contrast', - /** @internal */ - textOutline: '1px contrast' - }, - /** - * Options for a label text which should follow marker's shape. - * Border and background are disabled for a label that follows a - * path. - * - * **Note:** Only SVG-based renderer supports this option. Setting - * `useHTML` to true will disable this option. - * - * @declare Highcharts.DataLabelsTextPathOptionsObject - * @since 7.1.0 - * @apioption plotOptions.series.dataLabels.textPath - */ - /** - * Presentation attributes for the text path. - * - * @type {Highcharts.SVGAttributes} - * @since 7.1.0 - * @apioption plotOptions.series.dataLabels.textPath.attributes - */ - /** - * Enable or disable `textPath` option for link's or marker's data - * labels. - * - * @type {boolean} - * @since 7.1.0 - * @apioption plotOptions.series.dataLabels.textPath.enabled - */ - /** - * Whether to - * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) - * to render the labels. - * - * @type {boolean} - * @default false - * @apioption plotOptions.series.dataLabels.useHTML - */ - /** - * The vertical alignment of a data label. Can be one of `top`, - * `middle` or `bottom`. The default value depends on the data, for - * instance in a column chart, the label is above positive values - * and below negative values. - * - * @type {Highcharts.VerticalAlignValue|null} - * @since 2.3.3 - */ - verticalAlign: 'bottom', - /** - * The x position offset of the label relative to the point in - * pixels. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/ - * Vertical and positioned - * @sample {highcharts} highcharts/plotoptions/bar-datalabels-align-inside-bar/ - * Data labels inside the bar - */ - x: 0, - /** - * The Z index of the data labels. The default Z index puts it above - * the series. Use a Z index of 2 to display it behind the series. - * - * @type {number} - * @default 6 - * @since 2.3.5 - * @apioption plotOptions.series.dataLabels.z - */ - /** - * The y position offset of the label relative to the point in - * pixels. - * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/ - * Vertical and positioned - */ - y: 0 + this.cross.e = e; + } + fireEvent(this, "afterDrawCrosshair", { e: e, point: point }); + }; + /** + * Hide the crosshair if visible. + * + * @function Highcharts.Axis#hideCrosshair + */ + Axis.prototype.hideCrosshair = function () { + if (this.cross) { + this.cross.hide(); + } + fireEvent(this, "afterHideCrosshair"); + }; + /** + * Check whether the chart has vertical panning ('y' or 'xy' type). + * + * @private + * @function Highcharts.Axis#hasVerticalPanning + * @return {boolean} + * + */ + Axis.prototype.hasVerticalPanning = function () { + var _a, _b; + return /y/.test( + ((_b = + (_a = this.chart.options.chart) === null || _a === void 0 + ? void 0 + : _a.panning) === null || _b === void 0 + ? void 0 + : _b.type) || "" + ); + }; + /** + * Check whether the given value is a positive valid axis value. + * + * @private + * @function Highcharts.Axis#validatePositiveValue + * + * @param {unknown} value + * The axis value + * @return {boolean} + * + */ + Axis.prototype.validatePositiveValue = function (value) { + return isNumber(value) && value > 0; + }; + /* * + * + * Static Properties + * + * */ + /** + * The X axis or category axis. Normally this is the horizontal axis, + * though if the chart is inverted this is the vertical axis. In case of + * multiple axes, the xAxis node is an array of configuration objects. + * + * See the [Axis class](/class-reference/Highcharts.Axis) for programmatic + * access to the axis. + * + * @productdesc {highmaps} + * In Highmaps, the axis is hidden, but it is used behind the scenes to + * control features like zooming and panning. Zooming is in effect the same + * as setting the extremes of one of the exes. + * + * @type {*|Array<*>} + * @optionparent xAxis + * + * @private + */ + Axis.defaultOptions = { + /** + * When using multiple axis, the ticks of two or more opposite axes + * will automatically be aligned by adding ticks to the axis or axes + * with the least ticks, as if `tickAmount` were specified. + * + * This can be prevented by setting `alignTicks` to false. If the grid + * lines look messy, it's a good idea to hide them for the secondary + * axis by setting `gridLineWidth` to 0. + * + * If `startOnTick` or `endOnTick` in an Axis options are set to false, + * then the `alignTicks ` will be disabled for the Axis. + * + * Disabled for logarithmic axes. + * + * @type {boolean} + * @default true + * @product highcharts highstock gantt + * @apioption xAxis.alignTicks + */ + /** + * Whether to allow decimals in this axis' ticks. When counting + * integers, like persons or hits on a web page, decimals should + * be avoided in the labels. + * + * @see [minTickInterval](#xAxis.minTickInterval) + * + * @sample {highcharts|highstock} highcharts/yaxis/allowdecimals-true/ + * True by default + * @sample {highcharts|highstock} highcharts/yaxis/allowdecimals-false/ + * False + * + * @type {boolean} + * @default true + * @since 2.0 + * @apioption xAxis.allowDecimals + */ + /** + * When using an alternate grid color, a band is painted across the + * plot area between every other grid line. + * + * @sample {highcharts} highcharts/yaxis/alternategridcolor/ + * Alternate grid color on the Y axis + * @sample {highstock} stock/xaxis/alternategridcolor/ + * Alternate grid color on the Y axis + * + * @type {Highcharts.ColorType} + * @apioption xAxis.alternateGridColor + */ + /** + * An array defining breaks in the axis, the sections defined will be + * left out and all the points shifted closer to each other. + * + * @productdesc {highcharts} + * Requires that the broken-axis.js module is loaded. + * + * @sample {highcharts} highcharts/axisbreak/break-simple/ + * Simple break + * @sample {highcharts|highstock} highcharts/axisbreak/break-visualized/ + * Advanced with callback + * @sample {highstock} stock/demo/intraday-breaks/ + * Break on nights and weekends + * + * @type {Array<*>} + * @since 4.1.0 + * @product highcharts highstock gantt + * @apioption xAxis.breaks + */ + /** + * A number indicating how much space should be left between the start + * and the end of the break. The break size is given in axis units, + * so for instance on a `datetime` axis, a break size of 3600000 would + * indicate the equivalent of an hour. + * + * @type {number} + * @default 0 + * @since 4.1.0 + * @product highcharts highstock gantt + * @apioption xAxis.breaks.breakSize + */ + /** + * The point where the break starts. + * + * @type {number} + * @since 4.1.0 + * @product highcharts highstock gantt + * @apioption xAxis.breaks.from + */ + /** + * Defines an interval after which the break appears again. By default + * the breaks do not repeat. + * + * @type {number} + * @default 0 + * @since 4.1.0 + * @product highcharts highstock gantt + * @apioption xAxis.breaks.repeat + */ + /** + * The point where the break ends. + * + * @type {number} + * @since 4.1.0 + * @product highcharts highstock gantt + * @apioption xAxis.breaks.to + */ + /** + * If categories are present for the xAxis, names are used instead of + * numbers for that axis. + * + * Since Highcharts 3.0, categories can also + * be extracted by giving each point a [name](#series.data) and setting + * axis [type](#xAxis.type) to `category`. However, if you have multiple + * series, best practice remains defining the `categories` array. + * + * Example: `categories: ['Apples', 'Bananas', 'Oranges']` + * + * @sample {highcharts} highcharts/demo/line-labels/ + * With + * @sample {highcharts} highcharts/xaxis/categories/ + * Without + * + * @type {Array<string>} + * @product highcharts gantt + * @apioption xAxis.categories + */ + /** + * The highest allowed value for automatically computed axis extremes. + * + * @see [floor](#xAxis.floor) + * + * @sample {highcharts|highstock} highcharts/yaxis/floor-ceiling/ + * Floor and ceiling + * + * @type {number} + * @since 4.0 + * @product highcharts highstock gantt + * @apioption xAxis.ceiling + */ + /** + * A class name that opens for styling the axis by CSS, especially in + * Highcharts styled mode. The class name is applied to group elements + * for the grid, axis elements and labels. + * + * @sample {highcharts|highstock|highmaps} highcharts/css/axis/ + * Multiple axes with separate styling + * + * @type {string} + * @since 5.0.0 + * @apioption xAxis.className + */ + /** + * Configure a crosshair that follows either the mouse pointer or the + * hovered point. + * + * In styled mode, the crosshairs are styled in the + * `.highcharts-crosshair`, `.highcharts-crosshair-thin` or + * `.highcharts-xaxis-category` classes. + * + * @productdesc {highstock} + * In Highstock, by default, the crosshair is enabled on the X axis and + * disabled on the Y axis. + * + * @sample {highcharts} highcharts/xaxis/crosshair-both/ + * Crosshair on both axes + * @sample {highstock} stock/xaxis/crosshairs-xy/ + * Crosshair on both axes + * @sample {highmaps} highcharts/xaxis/crosshair-both/ + * Crosshair on both axes + * + * @declare Highcharts.AxisCrosshairOptions + * @type {boolean|*} + * @default false + * @since 4.1 + * @apioption xAxis.crosshair + */ + /** + * A class name for the crosshair, especially as a hook for styling. + * + * @type {string} + * @since 5.0.0 + * @apioption xAxis.crosshair.className + */ + /** + * The color of the crosshair. Defaults to `#cccccc` for numeric and + * datetime axes, and `rgba(204,214,235,0.25)` for category axes, where + * the crosshair by default highlights the whole category. + * + * @sample {highcharts|highstock|highmaps} highcharts/xaxis/crosshair-customized/ + * Customized crosshairs + * + * @type {Highcharts.ColorType} + * @default #cccccc + * @since 4.1 + * @apioption xAxis.crosshair.color + */ + /** + * The dash style for the crosshair. See + * [plotOptions.series.dashStyle](#plotOptions.series.dashStyle) + * for possible values. + * + * @sample {highcharts|highmaps} highcharts/xaxis/crosshair-dotted/ + * Dotted crosshair + * @sample {highstock} stock/xaxis/crosshair-dashed/ + * Dashed X axis crosshair + * + * @type {Highcharts.DashStyleValue} + * @default Solid + * @since 4.1 + * @apioption xAxis.crosshair.dashStyle + */ + /** + * A label on the axis next to the crosshair. + * + * In styled mode, the label is styled with the + * `.highcharts-crosshair-label` class. + * + * @sample {highstock} stock/xaxis/crosshair-label/ + * Crosshair labels + * @sample {highstock} highcharts/css/crosshair-label/ + * Style mode + * + * @declare Highcharts.AxisCrosshairLabelOptions + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label + */ + /** + * Alignment of the label compared to the axis. Defaults to `"left"` for + * right-side axes, `"right"` for left-side axes and `"center"` for + * horizontal axes. + * + * @type {Highcharts.AlignValue} + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.align + */ + /** + * The background color for the label. Defaults to the related series + * color, or `#666666` if that is not available. + * + * @type {Highcharts.ColorType} + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.backgroundColor + */ + /** + * The border color for the crosshair label + * + * @type {Highcharts.ColorType} + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.borderColor + */ + /** + * The border corner radius of the crosshair label. + * + * @type {number} + * @default 3 + * @since 2.1.10 + * @product highstock + * @apioption xAxis.crosshair.label.borderRadius + */ + /** + * The border width for the crosshair label. + * + * @type {number} + * @default 0 + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.borderWidth + */ + /** + * Flag to enable crosshair's label. + * + * @sample {highstock} stock/xaxis/crosshairs-xy/ + * Enabled label for yAxis' crosshair + * + * @type {boolean} + * @default false + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.enabled + */ + /** + * A format string for the crosshair label. Defaults to `{value}` for + * numeric axes and `{value:%b %d, %Y}` for datetime axes. + * + * @type {string} + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.format + */ + /** + * Formatter function for the label text. + * + * @type {Highcharts.XAxisCrosshairLabelFormatterCallbackFunction} + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.formatter + */ + /** + * Padding inside the crosshair label. + * + * @type {number} + * @default 8 + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.padding + */ + /** + * The shape to use for the label box. + * + * @type {string} + * @default callout + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.shape + */ + /** + * Text styles for the crosshair label. + * + * @type {Highcharts.CSSObject} + * @default {"color": "white", "fontWeight": "normal", "fontSize": "11px", "textAlign": "center"} + * @since 2.1 + * @product highstock + * @apioption xAxis.crosshair.label.style + */ + /** + * Whether the crosshair should snap to the point or follow the pointer + * independent of points. + * + * @sample {highcharts|highstock} highcharts/xaxis/crosshair-snap-false/ + * True by default + * @sample {highmaps} maps/demo/latlon-advanced/ + * Snap is false + * + * @type {boolean} + * @default true + * @since 4.1 + * @apioption xAxis.crosshair.snap + */ + /** + * The pixel width of the crosshair. Defaults to 1 for numeric or + * datetime axes, and for one category width for category axes. + * + * @sample {highcharts} highcharts/xaxis/crosshair-customized/ + * Customized crosshairs + * @sample {highstock} highcharts/xaxis/crosshair-customized/ + * Customized crosshairs + * @sample {highmaps} highcharts/xaxis/crosshair-customized/ + * Customized crosshairs + * + * @type {number} + * @default 1 + * @since 4.1 + * @apioption xAxis.crosshair.width + */ + /** + * The Z index of the crosshair. Higher Z indices allow drawing the + * crosshair on top of the series or behind the grid lines. + * + * @type {number} + * @default 2 + * @since 4.1 + * @apioption xAxis.crosshair.zIndex + */ + /** + * Whether to zoom axis. If `chart.zoomType` is set, the option allows + * to disable zooming on an individual axis. + * + * @sample {highcharts} highcharts/xaxis/zoomenabled/ + * Zoom enabled is false + * + * + * @type {boolean} + * @default enabled + * @apioption xAxis.zoomEnabled + */ + /** + * For a datetime axis, the scale will automatically adjust to the + * appropriate unit. This member gives the default string + * representations used for each unit. For intermediate values, + * different units may be used, for example the `day` unit can be used + * on midnight and `hour` unit be used for intermediate values on the + * same axis. + * + * For an overview of the replacement codes, see + * [dateFormat](/class-reference/Highcharts#dateFormat). + * + * Defaults to: + * ```js + * { + * millisecond: '%H:%M:%S.%L', + * second: '%H:%M:%S', + * minute: '%H:%M', + * hour: '%H:%M', + * day: '%e. %b', + * week: '%e. %b', + * month: '%b \'%y', + * year: '%Y' + * } + * ``` + * + * @sample {highcharts} highcharts/xaxis/datetimelabelformats/ + * Different day format on X axis + * @sample {highstock} stock/xaxis/datetimelabelformats/ + * More information in x axis labels + * + * @declare Highcharts.AxisDateTimeLabelFormatsOptions + * @product highcharts highstock gantt + */ + dateTimeLabelFormats: { + /** + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} + */ + millisecond: { + main: "%H:%M:%S.%L", + range: false, }, /** - * When the series contains less points than the crop threshold, all - * points are drawn, even if the points fall outside the visible plot - * area at the current zoom. The advantage of drawing all points - * (including markers and columns), is that animation is performed on - * updates. On the other hand, when the series contains more points than - * the crop threshold, the series data is cropped to only contain points - * that fall within the plot area. The advantage of cropping away - * invisible points is to increase performance on large series. - * - * @since 2.2 - * @product highcharts highstock - * - * @private + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} */ - cropThreshold: 300, + second: { + main: "%H:%M:%S", + range: false, + }, /** - * Opacity of a series parts: line, fill (e.g. area) and dataLabels. - * - * @see [states.inactive.opacity](#plotOptions.series.states.inactive.opacity) - * - * @since 7.1.0 - * - * @private + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} */ - opacity: 1, + minute: { + main: "%H:%M", + range: false, + }, /** - * The width of each point on the x axis. For example in a column chart - * with one value each day, the pointRange would be 1 day (= 24 * 3600 - * * 1000 milliseconds). This is normally computed automatically, but - * this option can be used to override the automatic value. - * - * @product highstock - * - * @private + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} */ - pointRange: 0, + hour: { + main: "%H:%M", + range: false, + }, /** - * When this is true, the series will not cause the Y axis to cross - * the zero plane (or [threshold](#plotOptions.series.threshold) option) - * unless the data actually crosses the plane. - * - * For example, if `softThreshold` is `false`, a series of 0, 1, 2, - * 3 will make the Y axis show negative values according to the - * `minPadding` option. If `softThreshold` is `true`, the Y axis starts - * at 0. - * - * @since 4.1.9 - * @product highcharts highstock - * - * @private + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} */ - softThreshold: true, + day: { + main: "%e. %b", + }, /** - * @declare Highcharts.SeriesStatesOptionsObject - * - * @private + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} */ - states: { - /** - * The normal state of a series, or for point items in column, pie - * and similar series. Currently only used for setting animation - * when returning to normal state from hover. - * - * @declare Highcharts.SeriesStatesNormalOptionsObject - */ - normal: { - /** - * Animation when returning to normal state after hovering. - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - */ - animation: true - }, - /** - * Options for the hovered series. These settings override the - * normal state options when a series is moused over or touched. - * - * @declare Highcharts.SeriesStatesHoverOptionsObject - */ - hover: { - /** - * Enable separate styles for the hovered series to visualize - * that the user hovers either the series itself or the legend. - * - * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled/ - * Line - * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled-column/ - * Column - * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled-pie/ - * Pie - * - * @type {boolean} - * @default true - * @since 1.2 - * @apioption plotOptions.series.states.hover.enabled - */ - /** - * Animation setting for hovering the graph in line-type series. - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - * @since 5.0.8 - * @product highcharts highstock - */ - animation: { - /** - * The duration of the hover animation in milliseconds. By - * default the hover state animates quickly in, and slowly - * back to normal. - * - * @internal - */ - duration: 50 - }, - /** - * Pixel width of the graph line. By default this property is - * undefined, and the `lineWidthPlus` property dictates how much - * to increase the linewidth from normal state. - * - * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidth/ - * 5px line on hover - * - * @type {number} - * @product highcharts highstock - * @apioption plotOptions.series.states.hover.lineWidth - */ - /** - * The additional line width for the graph of a hovered series. - * - * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/ - * 5 pixels wider - * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/ - * 5 pixels wider - * - * @since 4.0.3 - * @product highcharts highstock - */ - lineWidthPlus: 1, - /** - * In Highcharts 1.0, the appearance of all markers belonging - * to the hovered series. For settings on the hover state of the - * individual point, see - * [marker.states.hover](#plotOptions.series.marker.states.hover). - * - * @deprecated - * - * @extends plotOptions.series.marker - * @excluding states - * @product highcharts highstock - */ - marker: { - // lineWidth: base + 1, - // radius: base + 1 - }, - /** - * Options for the halo appearing around the hovered point in - * line-type series as well as outside the hovered slice in pie - * charts. By default the halo is filled by the current point or - * series color with an opacity of 0.25\. The halo can be - * disabled by setting the `halo` option to `null`. - * - * In styled mode, the halo is styled with the - * `.highcharts-halo` class, with colors inherited from - * `.highcharts-color-{n}`. - * - * @sample {highcharts} highcharts/plotoptions/halo/ - * Halo options - * @sample {highstock} highcharts/plotoptions/halo/ - * Halo options - * - * @declare Highcharts.SeriesStatesHoverHaloOptionsObject - * @type {null|*} - * @since 4.0 - * @product highcharts highstock - */ - halo: { - /** - * A collection of SVG attributes to override the appearance - * of the halo, for example `fill`, `stroke` and - * `stroke-width`. - * - * @type {Highcharts.SVGAttributes} - * @since 4.0 - * @product highcharts highstock - * @apioption plotOptions.series.states.hover.halo.attributes - */ - /** - * The pixel size of the halo. For point markers this is the - * radius of the halo. For pie slices it is the width of the - * halo outside the slice. For bubbles it defaults to 5 and - * is the width of the halo outside the bubble. - * - * @since 4.0 - * @product highcharts highstock - */ - size: 10, - /** - * Opacity for the halo unless a specific fill is overridden - * using the `attributes` setting. Note that Highcharts is - * only able to apply opacity to colors of hex or rgb(a) - * formats. - * - * @since 4.0 - * @product highcharts highstock - */ - opacity: 0.25 - } - }, - /** - * Specific options for point in selected states, after being - * selected by - * [allowPointSelect](#plotOptions.series.allowPointSelect) - * or programmatically. - * - * @sample maps/plotoptions/series-allowpointselect/ - * Allow point select demo - * - * @declare Highcharts.SeriesStatesSelectOptionsObject - * @extends plotOptions.series.states.hover - * @excluding brightness - */ - select: { - animation: { - /** @internal */ - duration: 0 - } - }, - /** - * The opposite state of a hover for series. - * - * @sample highcharts/plotoptions/series-states-inactive-disabled - * Disabled inactive state - * - * @declare Highcharts.SeriesStatesInactiveOptionsObject - */ - inactive: { - /** - * Enable or disable the inactive state for a series - * - * @sample highcharts/plotoptions/series-states-inactive-disabled - * Disabled inactive state - * - * @type {boolean} - * @default true - * @apioption plotOptions.series.states.inactive.enabled - */ - /** - * The animation for entering the inactive state. - * - * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} - */ - animation: { - /** @internal */ - duration: 50 - }, - /** - * Opacity of series elements (dataLabels, line, area). - * - * @type {number} - */ - opacity: 0.2 - } + week: { + main: "%e. %b", }, /** - * Sticky tracking of mouse events. When true, the `mouseOut` event on a - * series isn't triggered until the mouse moves over another series, or - * out of the plot area. When false, the `mouseOut` event on a series is - * triggered when the mouse leaves the area around the series' graph or - * markers. This also implies the tooltip when not shared. When - * `stickyTracking` is false and `tooltip.shared` is false, the tooltip - * will be hidden when moving the mouse between series. Defaults to true - * for line and area type series, but to false for columns, pies etc. - * - * **Note:** The boost module will force this option because of - * technical limitations. - * - * @sample {highcharts} highcharts/plotoptions/series-stickytracking-true/ - * True by default - * @sample {highcharts} highcharts/plotoptions/series-stickytracking-false/ - * False - * - * @default {highcharts} true - * @default {highstock} true - * @default {highmaps} false - * @since 2.0 - * - * @private + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} */ - stickyTracking: true, + month: { + main: "%b '%y", + }, /** - * A configuration object for the tooltip rendering of each single - * series. Properties are inherited from [tooltip](#tooltip), but only - * the following properties can be defined on a series level. - * - * @declare Highcharts.SeriesTooltipOptionsObject - * @since 2.3 - * @extends tooltip - * @excluding animation, backgroundColor, borderColor, borderRadius, - * borderWidth, className, crosshairs, enabled, formatter, - * headerShape, hideDelay, outside, padding, positioner, - * shadow, shape, shared, snap, split, stickOnContact, - * style, useHTML - * @apioption plotOptions.series.tooltip + * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject + * @type {string|*} */ - /** - * When a series contains a data array that is longer than this, only - * one dimensional arrays of numbers, or two dimensional arrays with - * x and y values are allowed. Also, only the first point is tested, - * and the rest are assumed to be the same format. This saves expensive - * data checking and indexing in long series. Set it to `0` disable. + year: { + main: "%Y", + }, + }, + /** + * Whether to force the axis to end on a tick. Use this option with + * the `maxPadding` option to control the axis end. + * + * @productdesc {highstock} + * In Highstock, `endOnTick` is always `false` when the navigator + * is enabled, to prevent jumpy scrolling. + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * True by default + * @sample {highcharts} highcharts/yaxis/endontick/ + * False + * @sample {highstock} stock/demo/basic-line/ + * True by default + * @sample {highstock} stock/xaxis/endontick/ + * False + * + * @since 1.2.0 + */ + endOnTick: false, + /** + * Event handlers for the axis. + * + * @type {*} + * @apioption xAxis.events + */ + /** + * An event fired after the breaks have rendered. + * + * @see [breaks](#xAxis.breaks) + * + * @sample {highcharts} highcharts/axisbreak/break-event/ + * AfterBreak Event + * + * @type {Highcharts.AxisEventCallbackFunction} + * @since 4.1.0 + * @product highcharts gantt + * @apioption xAxis.events.afterBreaks + */ + /** + * As opposed to the `setExtremes` event, this event fires after the + * final min and max values are computed and corrected for `minRange`. + * + * Fires when the minimum and maximum is set for the axis, either by + * calling the `.setExtremes()` method or by selecting an area in the + * chart. One parameter, `event`, is passed to the function, containing + * common event information. + * + * The new user set minimum and maximum values can be found by + * `event.min` and `event.max`. These reflect the axis minimum and + * maximum in axis values. The actual data extremes are found in + * `event.dataMin` and `event.dataMax`. + * + * @type {Highcharts.AxisSetExtremesEventCallbackFunction} + * @since 2.3 + * @context Highcharts.Axis + * @apioption xAxis.events.afterSetExtremes + */ + /** + * An event fired when a break from this axis occurs on a point. + * + * @see [breaks](#xAxis.breaks) + * + * @sample {highcharts} highcharts/axisbreak/break-visualized/ + * Visualization of a Break + * + * @type {Highcharts.AxisPointBreakEventCallbackFunction} + * @since 4.1.0 + * @product highcharts gantt + * @context Highcharts.Axis + * @apioption xAxis.events.pointBreak + */ + /** + * An event fired when a point falls inside a break from this axis. + * + * @type {Highcharts.AxisPointBreakEventCallbackFunction} + * @product highcharts highstock gantt + * @context Highcharts.Axis + * @apioption xAxis.events.pointInBreak + */ + /** + * Fires when the minimum and maximum is set for the axis, either by + * calling the `.setExtremes()` method or by selecting an area in the + * chart. One parameter, `event`, is passed to the function, + * containing common event information. + * + * The new user set minimum and maximum values can be found by + * `event.min` and `event.max`. These reflect the axis minimum and + * maximum in data values. When an axis is zoomed all the way out from + * the "Reset zoom" button, `event.min` and `event.max` are null, and + * the new extremes are set based on `this.dataMin` and `this.dataMax`. + * + * @sample {highstock} stock/xaxis/events-setextremes/ + * Log new extremes on x axis + * + * @type {Highcharts.AxisSetExtremesEventCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Axis + * @apioption xAxis.events.setExtremes + */ + /** + * The lowest allowed value for automatically computed axis extremes. + * + * @see [ceiling](#yAxis.ceiling) + * + * @sample {highcharts} highcharts/yaxis/floor-ceiling/ + * Floor and ceiling + * @sample {highstock} stock/demo/lazy-loading/ + * Prevent negative stock price on Y axis + * + * @type {number} + * @since 4.0 + * @product highcharts highstock gantt + * @apioption xAxis.floor + */ + /** + * The dash or dot style of the grid lines. For possible values, see + * [this demonstration](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/). + * + * @sample {highcharts} highcharts/yaxis/gridlinedashstyle/ + * Long dashes + * @sample {highstock} stock/xaxis/gridlinedashstyle/ + * Long dashes + * + * @type {Highcharts.DashStyleValue} + * @default Solid + * @since 1.2 + * @apioption xAxis.gridLineDashStyle + */ + /** + * The Z index of the grid lines. + * + * @sample {highcharts|highstock} highcharts/xaxis/gridzindex/ + * A Z index of 4 renders the grid above the graph + * + * @type {number} + * @default 1 + * @product highcharts highstock gantt + * @apioption xAxis.gridZIndex + */ + /** + * An id for the axis. This can be used after render time to get + * a pointer to the axis object through `chart.get()`. + * + * @sample {highcharts} highcharts/xaxis/id/ + * Get the object + * @sample {highstock} stock/xaxis/id/ + * Get the object + * + * @type {string} + * @since 1.2.0 + * @apioption xAxis.id + */ + /** + * The axis labels show the number or category for each tick. + * + * Since v8.0.0: Labels are animated in categorized x-axis with + * updating data if `tickInterval` and `step` is set to 1. + * + * @productdesc {highmaps} + * X and Y axis labels are by default disabled in Highmaps, but the + * functionality is inherited from Highcharts and used on `colorAxis`, + * and can be enabled on X and Y axes too. + */ + labels: { + /** + * What part of the string the given position is anchored to. + * If `left`, the left side of the string is at the axis position. + * Can be one of `"left"`, `"center"` or `"right"`. Defaults to + * an intelligent guess based on which side of the chart the axis + * is on and the rotation of the label. + * + * @see [reserveSpace](#xAxis.labels.reserveSpace) + * + * @sample {highcharts} highcharts/xaxis/labels-align-left/ + * Left + * @sample {highcharts} highcharts/xaxis/labels-align-right/ + * Right + * @sample {highcharts} highcharts/xaxis/labels-reservespace-true/ + * Left-aligned labels on a vertical category axis * - * Note: - * In boost mode turbo threshold is forced. Only array of numbers or - * two dimensional arrays are allowed. + * @type {Highcharts.AlignValue} + * @apioption xAxis.labels.align + */ + /** + * For horizontal axes, the allowed degrees of label rotation + * to prevent overlapping labels. If there is enough space, + * labels are not rotated. As the chart gets narrower, it + * will start rotating the labels -45 degrees, then remove + * every second label and try again with rotations 0 and -45 etc. + * Set it to `false` to disable rotation, which will + * cause the labels to word-wrap if possible. * - * @since 2.2 - * @product highcharts highstock gantt + * @sample {highcharts|highstock} highcharts/xaxis/labels-autorotation-default/ + * Default auto rotation of 0 or -45 + * @sample {highcharts|highstock} highcharts/xaxis/labels-autorotation-0-90/ + * Custom graded auto rotation * - * @private + * @type {Array<number>|false} + * @default [-45] + * @since 4.1.0 + * @product highcharts highstock gantt + * @apioption xAxis.labels.autoRotation */ - turboThreshold: 1000, /** - * An array defining zones within a series. Zones can be applied to the - * X axis, Y axis or Z axis for bubbles, according to the `zoneAxis` - * option. The zone definitions have to be in ascending order regarding - * to the value. + * When each category width is more than this many pixels, we don't + * apply auto rotation. Instead, we lay out the axis label with word + * wrap. A lower limit makes sense when the label contains multiple + * short words that don't extend the available horizontal space for + * each label. * - * In styled mode, the color zones are styled with the - * `.highcharts-zone-{n}` class, or custom classed from the `className` - * option - * ([view live demo](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/css/color-zones/)). + * @sample {highcharts} highcharts/xaxis/labels-autorotationlimit/ + * Lower limit * - * @see [zoneAxis](#plotOptions.series.zoneAxis) - * - * @sample {highcharts} highcharts/series/color-zones-simple/ - * Color zones - * @sample {highstock} highcharts/series/color-zones-simple/ - * Color zones + * @type {number} + * @default 80 + * @since 4.1.5 + * @product highcharts gantt + * @apioption xAxis.labels.autoRotationLimit + */ + /** + * Polar charts only. The label's pixel distance from the perimeter + * of the plot area. * - * @declare Highcharts.SeriesZonesOptionsObject - * @type {Array<*>} - * @since 4.1.0 - * @product highcharts highstock - * @apioption plotOptions.series.zones + * @type {number} + * @default 15 + * @product highcharts gantt + * @apioption xAxis.labels.distance */ /** - * Styled mode only. A custom class name for the zone. + * Enable or disable the axis labels. * - * @sample highcharts/css/color-zones/ - * Zones styled by class name + * @sample {highcharts} highcharts/xaxis/labels-enabled/ + * X axis labels disabled + * @sample {highstock} stock/xaxis/labels-enabled/ + * X axis labels disabled * - * @type {string} - * @since 5.0.0 - * @apioption plotOptions.series.zones.className + * @default {highcharts|highstock|gantt} true + * @default {highmaps} false */ + enabled: true, /** - * Defines the color of the series. + * A [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) + * for the axis label. * - * @see [series color](#plotOptions.series.color) + * @sample {highcharts|highstock} highcharts/yaxis/labels-format/ + * Add units to Y axis label * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 4.1.0 - * @product highcharts highstock - * @apioption plotOptions.series.zones.color + * @type {string} + * @default {value} + * @since 3.0 + * @apioption xAxis.labels.format */ /** - * A name for the dash style to use for the graph. + * Callback JavaScript function to format the label. The value + * is given by `this.value`. Additional properties for `this` are + * `axis`, `chart`, `isFirst` and `isLast`. The value of the default + * label formatter can be retrieved by calling + * `this.axis.defaultLabelFormatter.call(this)` within the function. * - * @see [plotOptions.series.dashStyle](#plotOptions.series.dashStyle) + * Defaults to: + * ```js + * function() { + * return this.value; + * } + * ``` * - * @sample {highcharts|highstock} highcharts/series/color-zones-dashstyle-dot/ - * Dashed line indicates prognosis + * @sample {highcharts} highcharts/xaxis/labels-formatter-linked/ + * Linked category names + * @sample {highcharts} highcharts/xaxis/labels-formatter-extended/ + * Modified numeric labels + * @sample {highstock} stock/xaxis/labels-formatter/ + * Added units on Y axis * - * @type {Highcharts.DashStyleValue} - * @since 4.1.0 - * @product highcharts highstock - * @apioption plotOptions.series.zones.dashStyle + * @type {Highcharts.AxisLabelsFormatterCallbackFunction} + * @apioption xAxis.labels.formatter */ /** - * Defines the fill color for the series (in area type series) + * The number of pixels to indent the labels per level in a treegrid + * axis. * - * @see [fillColor](#plotOptions.area.fillColor) + * @sample gantt/treegrid-axis/demo + * Indentation 10px by default. + * @sample gantt/treegrid-axis/indentation-0px + * Indentation set to 0px. * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 4.1.0 - * @product highcharts highstock - * @apioption plotOptions.series.zones.fillColor + * @product gantt */ + indentation: 10, /** - * The value up to where the zone extends, if undefined the zones - * stretches to the last value in the series. + * Horizontal axis only. When `staggerLines` is not set, + * `maxStaggerLines` defines how many lines the axis is allowed to + * add to automatically avoid overlapping X labels. Set to `1` to + * disable overlap detection. * + * @deprecated * @type {number} - * @since 4.1.0 - * @product highcharts highstock - * @apioption plotOptions.series.zones.value + * @default 5 + * @since 1.3.3 + * @apioption xAxis.labels.maxStaggerLines */ /** - * When using dual or multiple color axes, this number defines which - * colorAxis the particular series is connected to. It refers to - * either the - * {@link #colorAxis.id|axis id} - * or the index of the axis in the colorAxis array, with 0 being the - * first. Set this option to false to prevent a series from connecting - * to the default color axis. - * - * Since v7.2.0 the option can also be an axis id or an axis index - * instead of a boolean flag. - * - * @sample highcharts/coloraxis/coloraxis-with-pie/ - * Color axis with pie series - * @sample highcharts/coloraxis/multiple-coloraxis/ - * Multiple color axis - * - * @type {number|string|boolean} - * @default 0 - * @product highcharts highstock highmaps - * @apioption plotOptions.series.colorAxis + * How to handle overflowing labels on horizontal axis. If set to + * `"allow"`, it will not be aligned at all. By default it + * `"justify"` labels inside the chart area. If there is room to + * move it, it will be aligned to the edge, else it will be removed. + * + * @type {string} + * @default justify + * @since 2.2.5 + * @validvalue ["allow", "justify"] + * @apioption xAxis.labels.overflow */ /** - * Determines what data value should be used to calculate point color - * if `colorAxis` is used. Requires to set `min` and `max` if some - * custom point property is used or if approximation for data grouping - * is set to `'sum'`. - * - * @sample highcharts/coloraxis/custom-color-key/ - * Custom color key - * @sample highcharts/coloraxis/changed-default-color-key/ - * Changed default color key + * The pixel padding for axis labels, to ensure white space between + * them. * - * @type {string} - * @default y - * @since 7.2.0 - * @product highcharts highstock highmaps - * @apioption plotOptions.series.colorKey + * @type {number} + * @default 5 + * @product highcharts gantt + * @apioption xAxis.labels.padding */ /** - * Determines whether the series should look for the nearest point - * in both dimensions or just the x-dimension when hovering the series. - * Defaults to `'xy'` for scatter series and `'x'` for most other - * series. If the data has duplicate x-values, it is recommended to - * set this to `'xy'` to allow hovering over all points. + * Whether to reserve space for the labels. By default, space is + * reserved for the labels in these cases: * - * Applies only to series types using nearest neighbor search (not - * direct hover) for tooltip. + * * On all horizontal axes. + * * On vertical axes if `label.align` is `right` on a left-side + * axis or `left` on a right-side axis. + * * On vertical axes if `label.align` is `center`. * - * @sample {highcharts} highcharts/series/findnearestpointby/ - * Different hover behaviors - * @sample {highstock} highcharts/series/findnearestpointby/ - * Different hover behaviors - * @sample {highmaps} highcharts/series/findnearestpointby/ - * Different hover behaviors + * This can be turned off when for example the labels are rendered + * inside the plot area instead of outside. * - * @since 5.0.10 - * @validvalue ["x", "xy"] + * @see [labels.align](#xAxis.labels.align) * - * @private + * @sample {highcharts} highcharts/xaxis/labels-reservespace/ + * No reserved space, labels inside plot + * @sample {highcharts} highcharts/xaxis/labels-reservespace-true/ + * Left-aligned labels on a vertical category axis + * + * @type {boolean} + * @since 4.1.10 + * @product highcharts gantt + * @apioption xAxis.labels.reserveSpace */ - findNearestPointBy: 'x' - }, - /* eslint-disable no-invalid-this, valid-jsdoc */ - /** @lends Highcharts.Series.prototype */ - { - axisTypes: ['xAxis', 'yAxis'], - coll: 'series', - colorCounter: 0, - cropShoulder: 1, - directTouch: false, - isCartesian: true, - // each point's x and y values are stored in this.xData and this.yData - parallelArrays: ['x', 'y'], - pointClass: Point, - requireSorting: true, - sorted: true, - init: function (chart, options) { - fireEvent(this, 'init', { options: options }); - var series = this, - events, - chartSeries = chart.series, - lastSeries; - // A lookup over those events that are added by _options_ (not - // programmatically). These are updated through Series.update() - // (#10861). - this.eventOptions = this.eventOptions || {}; - // The 'eventsToUnbind' property moved from prototype into the - // Series init to avoid reference to the same array between - // the different series and charts. #12959, #13937 - this.eventsToUnbind = []; - /** - * Read only. The chart that the series belongs to. - * - * @name Highcharts.Series#chart - * @type {Highcharts.Chart} - */ - series.chart = chart; - /** - * Read only. The series' type, like "line", "area", "column" etc. - * The type in the series options anc can be altered using - * {@link Series#update}. - * - * @name Highcharts.Series#type - * @type {string} - */ - /** - * Read only. The series' current options. To update, use - * {@link Series#update}. - * - * @name Highcharts.Series#options - * @type {Highcharts.SeriesOptionsType} - */ - series.options = options = series.setOptions(options); - series.linkedSeries = []; - // bind the axes - series.bindAxes(); - // set some variables - extend(series, { - /** - * The series name as given in the options. Defaults to - * "Series {n}". - * - * @name Highcharts.Series#name - * @type {string} - */ - name: options.name, - state: '', - /** - * Read only. The series' visibility state as set by {@link - * Series#show}, {@link Series#hide}, or in the initial - * configuration. - * - * @name Highcharts.Series#visible - * @type {boolean} - */ - visible: options.visible !== false, - /** - * Read only. The series' selected state as set by {@link - * Highcharts.Series#select}. - * - * @name Highcharts.Series#selected - * @type {boolean} - */ - selected: options.selected === true // false by default - }); - // Register event listeners - events = options.events; - objectEach(events, function (event, eventType) { - if (isFunction(event)) { - // If event does not exist, or is changed by Series.update - if (series.eventOptions[eventType] !== event) { - // Remove existing if set by option - if (isFunction(series.eventOptions[eventType])) { - removeEvent(series, eventType, series.eventOptions[eventType]); - } - series.eventOptions[eventType] = event; - addEvent(series, eventType, event); - } - } - }); - if ((events && events.click) || - (options.point && - options.point.events && - options.point.events.click) || - options.allowPointSelect) { - chart.runTrackerClick = true; - } - series.getColor(); - series.getSymbol(); - // Initialize the parallel data arrays - series.parallelArrays.forEach(function (key) { - if (!series[key + 'Data']) { - series[key + 'Data'] = []; - } - }); - // Mark cartesian - if (series.isCartesian) { - chart.hasCartesianSeries = true; - } - // Get the index and register the series in the chart. The index is - // one more than the current latest series index (#5960). - if (chartSeries.length) { - lastSeries = chartSeries[chartSeries.length - 1]; - } - series._i = pick(lastSeries && lastSeries._i, -1) + 1; - series.opacity = series.options.opacity; - // Insert the series and re-order all series above the insertion - // point. - chart.orderSeries(this.insert(chartSeries)); - // Set options for series with sorting and set data later. - if (options.dataSorting && options.dataSorting.enabled) { - series.setDataSortingOptions(); - } - else if (!series.points && !series.data) { - series.setData(options.data, false); - } - fireEvent(this, 'afterInit'); - }, /** - * Check whether the series item is itself or inherits from a certain - * series type. + * Rotation of the labels in degrees. * - * @function Highcharts.Series#is - * @param {string} type The type of series to check for, can be either - * featured or custom series types. For example `column`, `pie`, - * `ohlc` etc. + * @sample {highcharts} highcharts/xaxis/labels-rotation/ + * X axis labels rotated 90° * - * @return {boolean} - * True if this item is or inherits from the given type. + * @type {number} + * @default 0 + * @apioption xAxis.labels.rotation */ - is: function (type) { - return seriesTypes[type] && this instanceof seriesTypes[type]; - }, /** - * Insert the series in a collection with other series, either the chart - * series or yAxis series, in the correct order according to the index - * option. Used internally when adding series. + * Horizontal axes only. The number of lines to spread the labels + * over to make room or tighter labels. * - * @private - * @function Highcharts.Series#insert - * @param {Array<Highcharts.Series>} collection - * A collection of series, like `chart.series` or `xAxis.series`. - * @return {number} - * The index of the series in the collection. - */ - insert: function (collection) { - var indexOption = this.options.index, - i; - // Insert by index option - if (isNumber(indexOption)) { - i = collection.length; - while (i--) { - // Loop down until the interted element has higher index - if (indexOption >= - pick(collection[i].options.index, collection[i]._i)) { - collection.splice(i + 1, 0, this); - break; - } - } - if (i === -1) { - collection.unshift(this); - } - i = i + 1; - // Or just push it to the end - } - else { - collection.push(this); - } - return pick(i, collection.length - 1); - }, - /** - * Set the xAxis and yAxis properties of cartesian series, and register - * the series in the `axis.series` array. + * @sample {highcharts} highcharts/xaxis/labels-staggerlines/ + * Show labels over two lines + * @sample {highstock} stock/xaxis/labels-staggerlines/ + * Show labels over two lines * - * @private - * @function Highcharts.Series#bindAxes - * @return {void} - * @exception 18 + * @type {number} + * @since 2.1 + * @apioption xAxis.labels.staggerLines */ - bindAxes: function () { - var series = this, - seriesOptions = series.options, - chart = series.chart, - axisOptions; - fireEvent(this, 'bindAxes', null, function () { - // repeat for xAxis and yAxis - (series.axisTypes || []).forEach(function (AXIS) { - // loop through the chart's axis objects - chart[AXIS].forEach(function (axis) { - axisOptions = axis.options; - // apply if the series xAxis or yAxis option mathches - // the number of the axis, or if undefined, use the - // first axis - if (seriesOptions[AXIS] === - axisOptions.index || - (typeof seriesOptions[AXIS] !== - 'undefined' && - seriesOptions[AXIS] === axisOptions.id) || - (typeof seriesOptions[AXIS] === - 'undefined' && - axisOptions.index === 0)) { - // register this series in the axis.series lookup - series.insert(axis.series); - // set this series.xAxis or series.yAxis reference - /** - * Read only. The unique xAxis object associated - * with the series. - * - * @name Highcharts.Series#xAxis - * @type {Highcharts.Axis} - */ - /** - * Read only. The unique yAxis object associated - * with the series. - * - * @name Highcharts.Series#yAxis - * @type {Highcharts.Axis} - */ - series[AXIS] = axis; - // mark dirty for redraw - axis.isDirty = true; - } - }); - // The series needs an X and an Y axis - if (!series[AXIS] && - series.optionalAxis !== AXIS) { - error(18, true, chart); - } - }); - }); - fireEvent(this, 'afterBindAxes'); - }, /** - * For simple series types like line and column, the data values are - * held in arrays like xData and yData for quick lookup to find extremes - * and more. For multidimensional series like bubble and map, this can - * be extended with arrays like zData and valueData by adding to the - * `series.parallelArrays` array. + * To show only every _n_'th label on the axis, set the step to _n_. + * Setting the step to 2 shows every other label. * - * @private - * @function Highcharts.Series#updateParallelArrays - * @param {Highcharts.Point} point - * @param {number|string} i - * @return {void} - */ - updateParallelArrays: function (point, i) { - var series = point.series, - args = arguments, - fn = isNumber(i) ? - // Insert the value in the given position - function (key) { - var val = key === 'y' && series.toYData ? - series.toYData(point) : - point[key]; - series[key + 'Data'][i] = val; - } : - // Apply the method specified in i with the following - // arguments as arguments - function (key) { - Array.prototype[i].apply(series[key + 'Data'], Array.prototype.slice.call(args, 2)); - }; - series.parallelArrays.forEach(fn); - }, - /** - * Define hasData functions for series. These return true if there - * are data points on this series within the plot area. + * By default, the step is calculated automatically to avoid + * overlap. To prevent this, set it to 1\. This usually only + * happens on a category axis, and is often a sign that you have + * chosen the wrong axis type. * - * @private - * @function Highcharts.Series#hasData - * @return {boolean} - */ - hasData: function () { - return ((this.visible && - typeof this.dataMax !== 'undefined' && - typeof this.dataMin !== 'undefined') || ( // #3703 - this.visible && - this.yData && - this.yData.length > 0) // #9758 - ); - }, - /** - * Return an auto incremented x value based on the pointStart and - * pointInterval options. This is only used if an x value is not given - * for the point that calls autoIncrement. + * Read more at + * [Axis docs](https://www.highcharts.com/docs/chart-concepts/axes) + * => What axis should I use? * - * @private - * @function Highcharts.Series#autoIncrement - * @return {number} - */ - autoIncrement: function () { - var options = this.options, - xIncrement = this.xIncrement, - date, - pointInterval, - pointIntervalUnit = options.pointIntervalUnit, - time = this.chart.time; - xIncrement = pick(xIncrement, options.pointStart, 0); - this.pointInterval = pointInterval = pick(this.pointInterval, options.pointInterval, 1); - // Added code for pointInterval strings - if (pointIntervalUnit) { - date = new time.Date(xIncrement); - if (pointIntervalUnit === 'day') { - time.set('Date', date, time.get('Date', date) + pointInterval); - } - else if (pointIntervalUnit === 'month') { - time.set('Month', date, time.get('Month', date) + pointInterval); - } - else if (pointIntervalUnit === 'year') { - time.set('FullYear', date, time.get('FullYear', date) + pointInterval); - } - pointInterval = date.getTime() - xIncrement; - } - this.xIncrement = xIncrement + pointInterval; - return xIncrement; - }, - /** - * Internal function to set properties for series if data sorting is - * enabled. + * @sample {highcharts} highcharts/xaxis/labels-step/ + * Showing only every other axis label on a categorized + * x-axis + * @sample {highcharts} highcharts/xaxis/labels-step-auto/ + * Auto steps on a category axis * - * @private - * @function Highcharts.Series#setDataSortingOptions - * @return {void} - */ - setDataSortingOptions: function () { - var options = this.options; - extend(this, { - requireSorting: false, - sorted: false, - enabledDataSorting: true, - allowDG: false - }); - // To allow unsorted data for column series. - if (!defined(options.pointRange)) { - options.pointRange = 1; - } - }, - /** - * Set the series options by merging from the options tree. Called - * internally on initializing and updating series. This function will - * not redraw the series. For API usage, use {@link Series#update}. - * @private - * @function Highcharts.Series#setOptions - * @param {Highcharts.SeriesOptionsType} itemOptions - * The series options. - * @return {Highcharts.SeriesOptionsType} - * @fires Highcharts.Series#event:afterSetOptions + * @type {number} + * @since 2.1 + * @apioption xAxis.labels.step */ - setOptions: function (itemOptions) { - var chart = this.chart, - chartOptions = chart.options, - plotOptions = chartOptions.plotOptions, - userOptions = chart.userOptions || {}, - seriesUserOptions = merge(itemOptions), - options, - zones, - zone, - styledMode = chart.styledMode, - e = { - plotOptions: plotOptions, - userOptions: seriesUserOptions - }; - fireEvent(this, 'setOptions', e); - // These may be modified by the event - var typeOptions = e.plotOptions[this.type], - userPlotOptions = (userOptions.plotOptions || {}); - // use copy to prevent undetected changes (#9762) - /** - * Contains series options by the user without defaults. - * @name Highcharts.Series#userOptions - * @type {Highcharts.SeriesOptionsType} - */ - this.userOptions = e.userOptions; - options = merge(typeOptions, plotOptions.series, - // #3881, chart instance plotOptions[type] should trump - // plotOptions.series - userOptions.plotOptions && - userOptions.plotOptions[this.type], seriesUserOptions); - // The tooltip options are merged between global and series specific - // options. Importance order asscendingly: - // globals: (1)tooltip, (2)plotOptions.series, - // (3)plotOptions[this.type] - // init userOptions with possible later updates: 4-6 like 1-3 and - // (7)this series options - this.tooltipOptions = merge(defaultOptions.tooltip, // 1 - defaultOptions.plotOptions.series && - defaultOptions.plotOptions.series.tooltip, // 2 - defaultOptions.plotOptions[this.type].tooltip, // 3 - chartOptions.tooltip.userOptions, // 4 - plotOptions.series && - plotOptions.series.tooltip, // 5 - plotOptions[this.type].tooltip, // 6 - seriesUserOptions.tooltip // 7 - ); - // When shared tooltip, stickyTracking is true by default, - // unless user says otherwise. - this.stickyTracking = pick(seriesUserOptions.stickyTracking, userPlotOptions[this.type] && - userPlotOptions[this.type].stickyTracking, userPlotOptions.series && userPlotOptions.series.stickyTracking, (this.tooltipOptions.shared && !this.noSharedTooltip ? - true : - options.stickyTracking)); - // Delete marker object if not allowed (#1125) - if (typeOptions.marker === null) { - delete options.marker; - } - // Handle color zones - this.zoneAxis = options.zoneAxis; - zones = this.zones = (options.zones || []).slice(); - if ((options.negativeColor || options.negativeFillColor) && - !options.zones) { - zone = { - value: options[this.zoneAxis + 'Threshold'] || - options.threshold || - 0, - className: 'highcharts-negative' - }; - if (!styledMode) { - zone.color = options.negativeColor; - zone.fillColor = options.negativeFillColor; - } - zones.push(zone); - } - if (zones.length) { // Push one extra zone for the rest - if (defined(zones[zones.length - 1].value)) { - zones.push(styledMode ? {} : { - color: this.color, - fillColor: this.fillColor - }); - } - } - fireEvent(this, 'afterSetOptions', { options: options }); - return options; - }, /** - * Return series name in "Series {Number}" format or the one defined by - * a user. This method can be simply overridden as series name format - * can vary (e.g. technical indicators). + * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the labels. * - * @function Highcharts.Series#getName - * @return {string} - * The series name. + * @type {boolean} + * @default false + * @apioption xAxis.labels.useHTML */ - getName: function () { - // #4119 - return pick(this.options.name, 'Series ' + (this.index + 1)); - }, /** - * @private - * @function Highcharts.Series#getCyclic - * @param {string} prop - * @param {*} [value] - * @param {Highcharts.Dictionary<any>} [defaults] - * @return {void} + * The x position offset of all labels relative to the tick + * positions on the axis. + * + * @sample {highcharts} highcharts/xaxis/labels-x/ + * Y axis labels placed on grid lines */ - getCyclic: function (prop, value, defaults) { - var i, chart = this.chart, userOptions = this.userOptions, indexName = prop + 'Index', counterName = prop + 'Counter', len = defaults ? defaults.length : pick(chart.options.chart[prop + 'Count'], chart[prop + 'Count']), setting; - if (!value) { - // Pick up either the colorIndex option, or the _colorIndex - // after Series.update() - setting = pick(userOptions[indexName], userOptions['_' + indexName]); - if (defined(setting)) { // after Series.update() - i = setting; - } - else { - // #6138 - if (!chart.series.length) { - chart[counterName] = 0; - } - userOptions['_' + indexName] = i = - chart[counterName] % len; - chart[counterName] += 1; - } - if (defaults) { - value = defaults[i]; - } - } - // Set the colorIndex - if (typeof i !== 'undefined') { - this[indexName] = i; - } - this[prop] = value; - }, + x: 0, /** - * Get the series' color based on either the options or pulled from - * global options. + * The y position offset of all labels relative to the tick + * positions on the axis. The default makes it adapt to the font + * size of the bottom axis. * - * @private - * @function Highcharts.Series#getColor - * @return {void} + * @sample {highcharts} highcharts/xaxis/labels-x/ + * Y axis labels placed on grid lines + * + * @type {number} + * @apioption xAxis.labels.y */ - getColor: function () { - if (this.chart.styledMode) { - this.getCyclic('color'); - } - else if (this.options.colorByPoint) { - // #4359, selected slice got series.color even when colorByPoint - // was set. - this.options.color = null; - } - else { - this.getCyclic('color', this.options.color || - defaultOptions.plotOptions[this.type].color, this.chart.options.colors); - } - }, /** - * Get all points' instances created for this series. + * The Z index for the axis labels. * - * @private - * @function Highcharts.Series#getPointsCollection - * @return {Array<Highcharts.Point>} + * @type {number} + * @default 7 + * @apioption xAxis.labels.zIndex */ - getPointsCollection: function () { - return (this.hasGroupedData ? this.points : this.data) || []; - }, /** - * Get the series' symbol based on either the options or pulled from - * global options. + * CSS styles for the label. Use `whiteSpace: 'nowrap'` to prevent + * wrapping of category labels. Use `textOverflow: 'none'` to + * prevent ellipsis (dots). * - * @private - * @function Highcharts.Series#getSymbol - * @return {void} + * In styled mode, the labels are styled with the + * `.highcharts-axis-labels` class. + * + * @sample {highcharts} highcharts/xaxis/labels-style/ + * Red X axis labels + * + * @type {Highcharts.CSSObject} */ - getSymbol: function () { - var seriesMarkerOption = this.options.marker; - this.getCyclic('symbol', seriesMarkerOption.symbol, this.chart.options.symbols); + style: { + /** @internal */ + color: "#666666", + /** @internal */ + cursor: "default", + /** @internal */ + fontSize: "11px", }, - /** - * Finds the index of an existing point that matches the given point - * options. + }, + /** + * The left position as the horizontal axis. If it's a number, it is + * interpreted as pixel position relative to the chart. + * + * Since Highcharts v5.0.13: If it's a percentage string, it is + * interpreted as percentages of the plot width, offset from plot area + * left. + * + * @type {number|string} + * @product highcharts highstock + * @apioption xAxis.left + */ + /** + * The top position as the vertical axis. If it's a number, it is + * interpreted as pixel position relative to the chart. + * + * Since Highcharts 2: If it's a percentage string, it is interpreted + * as percentages of the plot height, offset from plot area top. + * + * @type {number|string} + * @product highcharts highstock + * @apioption xAxis.top + */ + /** + * Index of another axis that this axis is linked to. When an axis is + * linked to a master axis, it will take the same extremes as + * the master, but as assigned by min or max or by setExtremes. + * It can be used to show additional info, or to ease reading the + * chart by duplicating the scales. + * + * @sample {highcharts} highcharts/xaxis/linkedto/ + * Different string formats of the same date + * @sample {highcharts} highcharts/yaxis/linkedto/ + * Y values on both sides + * + * @type {number} + * @since 2.0.2 + * @product highcharts highstock gantt + * @apioption xAxis.linkedTo + */ + /** + * The maximum value of the axis. If `null`, the max value is + * automatically calculated. + * + * If the [endOnTick](#yAxis.endOnTick) option is true, the `max` value + * might be rounded up. + * + * If a [tickAmount](#yAxis.tickAmount) is set, the axis may be extended + * beyond the set max in order to reach the given number of ticks. The + * same may happen in a chart with multiple axes, determined by [chart. + * alignTicks](#chart), where a `tickAmount` is applied internally. + * + * @sample {highcharts} highcharts/yaxis/max-200/ + * Y axis max of 200 + * @sample {highcharts} highcharts/yaxis/max-logarithmic/ + * Y axis max on logarithmic axis + * @sample {highstock} stock/xaxis/min-max/ + * Fixed min and max on X axis + * @sample {highmaps} maps/axis/min-max/ + * Pre-zoomed to a specific area + * + * @type {number|null} + * @apioption xAxis.max + */ + /** + * Padding of the max value relative to the length of the axis. A + * padding of 0.05 will make a 100px axis 5px longer. This is useful + * when you don't want the highest data value to appear on the edge + * of the plot area. When the axis' `max` option is set or a max extreme + * is set using `axis.setExtremes()`, the maxPadding will be ignored. + * + * @sample {highcharts} highcharts/yaxis/maxpadding/ + * Max padding of 0.25 on y axis + * @sample {highstock} stock/xaxis/minpadding-maxpadding/ + * Greater min- and maxPadding + * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/ + * Add some padding + * + * @default {highcharts} 0.01 + * @default {highstock|highmaps} 0 + * @since 1.2.0 + */ + maxPadding: 0.01, + /** + * Deprecated. Use `minRange` instead. + * + * @deprecated + * @type {number} + * @product highcharts highstock + * @apioption xAxis.maxZoom + */ + /** + * The minimum value of the axis. If `null` the min value is + * automatically calculated. + * + * If the [startOnTick](#yAxis.startOnTick) option is true (default), + * the `min` value might be rounded down. + * + * The automatically calculated minimum value is also affected by + * [floor](#yAxis.floor), [softMin](#yAxis.softMin), + * [minPadding](#yAxis.minPadding), [minRange](#yAxis.minRange) + * as well as [series.threshold](#plotOptions.series.threshold) + * and [series.softThreshold](#plotOptions.series.softThreshold). + * + * @sample {highcharts} highcharts/yaxis/min-startontick-false/ + * -50 with startOnTick to false + * @sample {highcharts} highcharts/yaxis/min-startontick-true/ + * -50 with startOnTick true by default + * @sample {highstock} stock/xaxis/min-max/ + * Set min and max on X axis + * @sample {highmaps} maps/axis/min-max/ + * Pre-zoomed to a specific area + * + * @type {number|null} + * @apioption xAxis.min + */ + /** + * The dash or dot style of the minor grid lines. For possible values, + * see [this demonstration](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/). + * + * @sample {highcharts} highcharts/yaxis/minorgridlinedashstyle/ + * Long dashes on minor grid lines + * @sample {highstock} stock/xaxis/minorgridlinedashstyle/ + * Long dashes on minor grid lines + * + * @type {Highcharts.DashStyleValue} + * @default Solid + * @since 1.2 + * @apioption xAxis.minorGridLineDashStyle + */ + /** + * Specific tick interval in axis units for the minor ticks. On a linear + * axis, if `"auto"`, the minor tick interval is calculated as a fifth + * of the tickInterval. If `null` or `undefined`, minor ticks are not + * shown. + * + * On logarithmic axes, the unit is the power of the value. For example, + * setting the minorTickInterval to 1 puts one tick on each of 0.1, 1, + * 10, 100 etc. Setting the minorTickInterval to 0.1 produces 9 ticks + * between 1 and 10, 10 and 100 etc. + * + * If user settings dictate minor ticks to become too dense, they don't + * make sense, and will be ignored to prevent performance problems. + * + * @sample {highcharts} highcharts/yaxis/minortickinterval-null/ + * Null by default + * @sample {highcharts} highcharts/yaxis/minortickinterval-5/ + * 5 units + * @sample {highcharts} highcharts/yaxis/minortickinterval-log-auto/ + * "auto" + * @sample {highcharts} highcharts/yaxis/minortickinterval-log/ + * 0.1 + * @sample {highstock} stock/demo/basic-line/ + * Null by default + * @sample {highstock} stock/xaxis/minortickinterval-auto/ + * "auto" + * + * @type {number|string|null} + * @apioption xAxis.minorTickInterval + */ + /** + * The pixel length of the minor tick marks. + * + * @sample {highcharts} highcharts/yaxis/minorticklength/ + * 10px on Y axis + * @sample {highstock} stock/xaxis/minorticks/ + * 10px on Y axis + */ + minorTickLength: 2, + /** + * The position of the minor tick marks relative to the axis line. + * Can be one of `inside` and `outside`. + * + * @sample {highcharts} highcharts/yaxis/minortickposition-outside/ + * Outside by default + * @sample {highcharts} highcharts/yaxis/minortickposition-inside/ + * Inside + * @sample {highstock} stock/xaxis/minorticks/ + * Inside + * + * @validvalue ["inside", "outside"] + */ + minorTickPosition: "outside", + /** + * Enable or disable minor ticks. Unless + * [minorTickInterval](#xAxis.minorTickInterval) is set, the tick + * interval is calculated as a fifth of the `tickInterval`. + * + * On a logarithmic axis, minor ticks are laid out based on a best + * guess, attempting to enter approximately 5 minor ticks between + * each major tick. + * + * Prior to v6.0.0, ticks were unabled in auto layout by setting + * `minorTickInterval` to `"auto"`. + * + * @productdesc {highcharts} + * On axes using [categories](#xAxis.categories), minor ticks are not + * supported. + * + * @sample {highcharts} highcharts/yaxis/minorticks-true/ + * Enabled on linear Y axis + * + * @type {boolean} + * @default false + * @since 6.0.0 + * @apioption xAxis.minorTicks + */ + /** + * The pixel width of the minor tick mark. + * + * @sample {highcharts} highcharts/yaxis/minortickwidth/ + * 3px width + * @sample {highstock} stock/xaxis/minorticks/ + * 1px width + * + * @type {number} + * @default 0 + * @apioption xAxis.minorTickWidth + */ + /** + * Padding of the min value relative to the length of the axis. A + * padding of 0.05 will make a 100px axis 5px longer. This is useful + * when you don't want the lowest data value to appear on the edge + * of the plot area. When the axis' `min` option is set or a min extreme + * is set using `axis.setExtremes()`, the minPadding will be ignored. + * + * @sample {highcharts} highcharts/yaxis/minpadding/ + * Min padding of 0.2 + * @sample {highstock} stock/xaxis/minpadding-maxpadding/ + * Greater min- and maxPadding + * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/ + * Add some padding + * + * @default {highcharts} 0.01 + * @default {highstock|highmaps} 0 + * @since 1.2.0 + * @product highcharts highstock gantt + */ + minPadding: 0.01, + /** + * The minimum range to display on this axis. The entire axis will not + * be allowed to span over a smaller interval than this. For example, + * for a datetime axis the main unit is milliseconds. If minRange is + * set to 3600000, you can't zoom in more than to one hour. + * + * The default minRange for the x axis is five times the smallest + * interval between any of the data points. + * + * On a logarithmic axis, the unit for the minimum range is the power. + * So a minRange of 1 means that the axis can be zoomed to 10-100, + * 100-1000, 1000-10000 etc. + * + * **Note**: The `minPadding`, `maxPadding`, `startOnTick` and + * `endOnTick` settings also affect how the extremes of the axis + * are computed. + * + * @sample {highcharts} highcharts/xaxis/minrange/ + * Minimum range of 5 + * @sample {highstock} stock/xaxis/minrange/ + * Max zoom of 6 months overrides user selections + * @sample {highmaps} maps/axis/minrange/ + * Minimum range of 1000 + * + * @type {number} + * @apioption xAxis.minRange + */ + /** + * The minimum tick interval allowed in axis values. For example on + * zooming in on an axis with daily data, this can be used to prevent + * the axis from showing hours. Defaults to the closest distance between + * two points on the axis. + * + * @type {number} + * @since 2.3.0 + * @apioption xAxis.minTickInterval + */ + /** + * The distance in pixels from the plot area to the axis line. + * A positive offset moves the axis with it's line, labels and ticks + * away from the plot area. This is typically used when two or more + * axes are displayed on the same side of the plot. With multiple + * axes the offset is dynamically adjusted to avoid collision, this + * can be overridden by setting offset explicitly. + * + * @sample {highcharts} highcharts/yaxis/offset/ + * Y axis offset of 70 + * @sample {highcharts} highcharts/yaxis/offset-centered/ + * Axes positioned in the center of the plot + * @sample {highstock} stock/xaxis/offset/ + * Y axis offset by 70 px + * + * @type {number} + * @default 0 + * @apioption xAxis.offset + */ + /** + * Whether to display the axis on the opposite side of the normal. The + * normal is on the left side for vertical axes and bottom for + * horizontal, so the opposite sides will be right and top respectively. + * This is typically used with dual or multiple axes. + * + * @sample {highcharts} highcharts/yaxis/opposite/ + * Secondary Y axis opposite + * @sample {highstock} stock/xaxis/opposite/ + * Y axis on left side + * + * @type {boolean} + * @default {highcharts|highstock|highmaps} false + * @default {gantt} true + * @apioption xAxis.opposite + */ + /** + * In an ordinal axis, the points are equally spaced in the chart + * regardless of the actual time or x distance between them. This means + * that missing data periods (e.g. nights or weekends for a stock chart) + * will not take up space in the chart. + * Having `ordinal: false` will show any gaps created by the `gapSize` + * setting proportionate to their duration. + * + * In stock charts the X axis is ordinal by default, unless + * the boost module is used and at least one of the series' data length + * exceeds the [boostThreshold](#series.line.boostThreshold). + * + * @sample {highstock} stock/xaxis/ordinal-true/ + * True by default + * @sample {highstock} stock/xaxis/ordinal-false/ + * False + * + * @type {boolean} + * @default true + * @since 1.1 + * @product highstock + * @apioption xAxis.ordinal + */ + /** + * Additional range on the right side of the xAxis. Works similar to + * `xAxis.maxPadding`, but value is set in milliseconds. Can be set for + * both main `xAxis` and the navigator's `xAxis`. + * + * @sample {highstock} stock/xaxis/overscroll/ + * One minute overscroll with live data + * + * @type {number} + * @default 0 + * @since 6.0.0 + * @product highstock + * @apioption xAxis.overscroll + */ + /** + * Refers to the index in the [panes](#panes) array. Used for circular + * gauges and polar charts. When the option is not set then first pane + * will be used. + * + * @sample highcharts/demo/gauge-vu-meter + * Two gauges with different center + * + * @type {number} + * @product highcharts + * @apioption xAxis.pane + */ + /** + * The zoomed range to display when only defining one or none of `min` + * or `max`. For example, to show the latest month, a range of one month + * can be set. + * + * @sample {highstock} stock/xaxis/range/ + * Setting a zoomed range when the rangeSelector is disabled + * + * @type {number} + * @product highstock + * @apioption xAxis.range + */ + /** + * Whether to reverse the axis so that the highest number is closest + * to the origin. If the chart is inverted, the x axis is reversed by + * default. + * + * @sample {highcharts} highcharts/yaxis/reversed/ + * Reversed Y axis + * @sample {highstock} stock/xaxis/reversed/ + * Reversed Y axis + * + * @type {boolean} + * @default false + * @apioption xAxis.reversed + */ + // reversed: false, + /** + * This option determines how stacks should be ordered within a group. + * For example reversed xAxis also reverses stacks, so first series + * comes last in a group. To keep order like for non-reversed xAxis + * enable this option. + * + * @sample {highcharts} highcharts/xaxis/reversedstacks/ + * Reversed stacks comparison + * @sample {highstock} highcharts/xaxis/reversedstacks/ + * Reversed stacks comparison + * + * @type {boolean} + * @default false + * @since 6.1.1 + * @product highcharts highstock + * @apioption xAxis.reversedStacks + */ + /** + * An optional scrollbar to display on the X axis in response to + * limiting the minimum and maximum of the axis values. + * + * In styled mode, all the presentational options for the scrollbar are + * replaced by the classes `.highcharts-scrollbar-thumb`, + * `.highcharts-scrollbar-arrow`, `.highcharts-scrollbar-button`, + * `.highcharts-scrollbar-rifles` and `.highcharts-scrollbar-track`. + * + * @sample {highstock} stock/yaxis/heatmap-scrollbars/ + * Heatmap with both scrollbars + * + * @extends scrollbar + * @since 4.2.6 + * @product highstock + * @apioption xAxis.scrollbar + */ + /** + * Whether to show the axis line and title when the axis has no data. + * + * @sample {highcharts} highcharts/yaxis/showempty/ + * When clicking the legend to hide series, one axis preserves + * line and title, the other doesn't + * @sample {highstock} highcharts/yaxis/showempty/ + * When clicking the legend to hide series, one axis preserves + * line and title, the other doesn't + * + * @since 1.1 + */ + showEmpty: true, + /** + * Whether to show the first tick label. + * + * @sample {highcharts} highcharts/xaxis/showfirstlabel-false/ + * Set to false on X axis + * @sample {highstock} stock/xaxis/showfirstlabel/ + * Labels below plot lines on Y axis + * + * @type {boolean} + * @default true + * @apioption xAxis.showFirstLabel + */ + /** + * Whether to show the last tick label. Defaults to `true` on cartesian + * charts, and `false` on polar charts. + * + * @sample {highcharts} highcharts/xaxis/showlastlabel-true/ + * Set to true on X axis + * @sample {highstock} stock/xaxis/showfirstlabel/ + * Labels below plot lines on Y axis + * + * @type {boolean} + * @default true + * @product highcharts highstock gantt + * @apioption xAxis.showLastLabel + */ + /** + * A soft maximum for the axis. If the series data maximum is less than + * this, the axis will stay at this maximum, but if the series data + * maximum is higher, the axis will flex to show all data. + * + * @sample highcharts/yaxis/softmin-softmax/ + * Soft min and max + * + * @type {number} + * @since 5.0.1 + * @product highcharts highstock gantt + * @apioption xAxis.softMax + */ + /** + * A soft minimum for the axis. If the series data minimum is greater + * than this, the axis will stay at this minimum, but if the series + * data minimum is lower, the axis will flex to show all data. + * + * @sample highcharts/yaxis/softmin-softmax/ + * Soft min and max + * + * @type {number} + * @since 5.0.1 + * @product highcharts highstock gantt + * @apioption xAxis.softMin + */ + /** + * For datetime axes, this decides where to put the tick between weeks. + * 0 = Sunday, 1 = Monday. + * + * @sample {highcharts} highcharts/xaxis/startofweek-monday/ + * Monday by default + * @sample {highcharts} highcharts/xaxis/startofweek-sunday/ + * Sunday + * @sample {highstock} stock/xaxis/startofweek-1 + * Monday by default + * @sample {highstock} stock/xaxis/startofweek-0 + * Sunday + * + * @product highcharts highstock gantt + */ + startOfWeek: 1, + /** + * Whether to force the axis to start on a tick. Use this option with + * the `minPadding` option to control the axis start. + * + * @productdesc {highstock} + * In Highstock, `startOnTick` is always `false` when the navigator + * is enabled, to prevent jumpy scrolling. + * + * @sample {highcharts} highcharts/xaxis/startontick-false/ + * False by default + * @sample {highcharts} highcharts/xaxis/startontick-true/ + * True + * + * @since 1.2.0 + */ + startOnTick: false, + /** + * The amount of ticks to draw on the axis. This opens up for aligning + * the ticks of multiple charts or panes within a chart. This option + * overrides the `tickPixelInterval` option. + * + * This option only has an effect on linear axes. Datetime, logarithmic + * or category axes are not affected. + * + * @sample {highcharts} highcharts/yaxis/tickamount/ + * 8 ticks on Y axis + * @sample {highstock} highcharts/yaxis/tickamount/ + * 8 ticks on Y axis + * + * @type {number} + * @since 4.1.0 + * @product highcharts highstock gantt + * @apioption xAxis.tickAmount + */ + /** + * The interval of the tick marks in axis units. When `undefined`, the + * tick interval is computed to approximately follow the + * [tickPixelInterval](#xAxis.tickPixelInterval) on linear and datetime + * axes. On categorized axes, a `undefined` tickInterval will default to + * 1, one category. Note that datetime axes are based on milliseconds, + * so for example an interval of one day is expressed as + * `24 * 3600 * 1000`. + * + * On logarithmic axes, the tickInterval is based on powers, so a + * tickInterval of 1 means one tick on each of 0.1, 1, 10, 100 etc. A + * tickInterval of 2 means a tick of 0.1, 10, 1000 etc. A tickInterval + * of 0.2 puts a tick on 0.1, 0.2, 0.4, 0.6, 0.8, 1, 2, 4, 6, 8, 10, 20, + * 40 etc. + * + * + * If the tickInterval is too dense for labels to be drawn, Highcharts + * may remove ticks. + * + * If the chart has multiple axes, the [alignTicks](#chart.alignTicks) + * option may interfere with the `tickInterval` setting. + * + * @see [tickPixelInterval](#xAxis.tickPixelInterval) + * @see [tickPositions](#xAxis.tickPositions) + * @see [tickPositioner](#xAxis.tickPositioner) + * + * @sample {highcharts} highcharts/xaxis/tickinterval-5/ + * Tick interval of 5 on a linear axis + * @sample {highstock} stock/xaxis/tickinterval/ + * Tick interval of 0.01 on Y axis + * + * @type {number} + * @apioption xAxis.tickInterval + */ + /** + * The pixel length of the main tick marks. + * + * @sample {highcharts} highcharts/xaxis/ticklength/ + * 20 px tick length on the X axis + * @sample {highstock} stock/xaxis/ticks/ + * Formatted ticks on X axis + */ + tickLength: 10, + /** + * If tickInterval is `null` this option sets the approximate pixel + * interval of the tick marks. Not applicable to categorized axis. + * + * The tick interval is also influenced by the [minTickInterval]( + * #xAxis.minTickInterval) option, that, by default prevents ticks from + * being denser than the data points. + * + * @see [tickInterval](#xAxis.tickInterval) + * @see [tickPositioner](#xAxis.tickPositioner) + * @see [tickPositions](#xAxis.tickPositions) + * + * @sample {highcharts} highcharts/xaxis/tickpixelinterval-50/ + * 50 px on X axis + * @sample {highstock} stock/xaxis/tickpixelinterval/ + * 200 px on X axis + */ + tickPixelInterval: 100, + /** + * For categorized axes only. If `on` the tick mark is placed in the + * center of the category, if `between` the tick mark is placed between + * categories. The default is `between` if the `tickInterval` is 1, else + * `on`. + * + * @sample {highcharts} highcharts/xaxis/tickmarkplacement-between/ + * "between" by default + * @sample {highcharts} highcharts/xaxis/tickmarkplacement-on/ + * "on" + * + * @product highcharts gantt + * @validvalue ["on", "between"] + */ + tickmarkPlacement: "between", + /** + * The position of the major tick marks relative to the axis line. + * Can be one of `inside` and `outside`. + * + * @sample {highcharts} highcharts/xaxis/tickposition-outside/ + * "outside" by default + * @sample {highcharts} highcharts/xaxis/tickposition-inside/ + * "inside" + * @sample {highstock} stock/xaxis/ticks/ + * Formatted ticks on X axis + * + * @validvalue ["inside", "outside"] + */ + tickPosition: "outside", + /** + * A callback function returning array defining where the ticks are + * laid out on the axis. This overrides the default behaviour of + * [tickPixelInterval](#xAxis.tickPixelInterval) and [tickInterval]( + * #xAxis.tickInterval). The automatic tick positions are accessible + * through `this.tickPositions` and can be modified by the callback. + * + * @see [tickPositions](#xAxis.tickPositions) + * + * @sample {highcharts} highcharts/xaxis/tickpositions-tickpositioner/ + * Demo of tickPositions and tickPositioner + * @sample {highstock} highcharts/xaxis/tickpositions-tickpositioner/ + * Demo of tickPositions and tickPositioner + * + * @type {Highcharts.AxisTickPositionerCallbackFunction} + * @apioption xAxis.tickPositioner + */ + /** + * An array defining where the ticks are laid out on the axis. This + * overrides the default behaviour of [tickPixelInterval]( + * #xAxis.tickPixelInterval) and [tickInterval](#xAxis.tickInterval). + * + * @see [tickPositioner](#xAxis.tickPositioner) + * + * @sample {highcharts} highcharts/xaxis/tickpositions-tickpositioner/ + * Demo of tickPositions and tickPositioner + * @sample {highstock} highcharts/xaxis/tickpositions-tickpositioner/ + * Demo of tickPositions and tickPositioner + * + * @type {Array<number>} + * @apioption xAxis.tickPositions + */ + /** + * The pixel width of the major tick marks. Defaults to 0 on category + * axes, otherwise 1. + * + * In styled mode, the stroke width is given in the `.highcharts-tick` + * class, but in order for the element to be generated on category axes, + * the option must be explicitly set to 1. + * + * @sample {highcharts} highcharts/xaxis/tickwidth/ + * 10 px width + * @sample {highcharts} highcharts/css/axis-grid/ + * Styled mode + * @sample {highstock} stock/xaxis/ticks/ + * Formatted ticks on X axis + * @sample {highstock} highcharts/css/axis-grid/ + * Styled mode + * + * @type {undefined|number} + * @default {highstock} 1 + * @default {highmaps} 0 + * @apioption xAxis.tickWidth + */ + /** + * The axis title, showing next to the axis line. + * + * @productdesc {highmaps} + * In Highmaps, the axis is hidden by default, but adding an axis title + * is still possible. X axis and Y axis titles will appear at the bottom + * and left by default. + */ + title: { + /** + * Deprecated. Set the `text` to `null` to disable the title. * - * @private - * @function Highcharts.Series#findPointIndex - * @param {Highcharts.PointOptionsObject} optionsObject - * The options of the point. - * @param {number} fromIndex - * The index to start searching from, used for optimizing - * series with required sorting. - * @returns {number|undefined} - * Returns the index of a matching point, or undefined if no - * match is found. + * @deprecated + * @type {boolean} + * @product highcharts + * @apioption xAxis.title.enabled */ - findPointIndex: function (optionsObject, fromIndex) { - var id = optionsObject.id, - x = optionsObject.x, - oldData = this.points, - matchingPoint, - matchedById, - pointIndex, - matchKey, - dataSorting = this.options.dataSorting; - if (id) { - matchingPoint = this.chart.get(id); - } - else if (this.linkedParent || this.enabledDataSorting) { - matchKey = (dataSorting && dataSorting.matchByName) ? - 'name' : 'index'; - matchingPoint = find(oldData, function (oldPoint) { - return !oldPoint.touched && oldPoint[matchKey] === - optionsObject[matchKey]; - }); - // Add unmatched point as a new point - if (!matchingPoint) { - return void 0; - } - } - if (matchingPoint) { - pointIndex = matchingPoint && matchingPoint.index; - if (typeof pointIndex !== 'undefined') { - matchedById = true; - } - } - // Search for the same X in the existing data set - if (typeof pointIndex === 'undefined' && isNumber(x)) { - pointIndex = this.xData.indexOf(x, fromIndex); - } - // Reduce pointIndex if data is cropped - if (pointIndex !== -1 && - typeof pointIndex !== 'undefined' && - this.cropped) { - pointIndex = (pointIndex >= this.cropStart) ? - pointIndex - this.cropStart : pointIndex; - } - if (!matchedById && - oldData[pointIndex] && oldData[pointIndex].touched) { - pointIndex = void 0; - } - return pointIndex; - }, /** - * @private - * @borrows LegendSymbolMixin.drawLineMarker as Highcharts.Series#drawLegendSymbol + * The pixel distance between the axis labels or line and the title. + * Defaults to 0 for horizontal axes, 10 for vertical + * + * @sample {highcharts} highcharts/xaxis/title-margin/ + * Y axis title margin of 60 + * + * @type {number} + * @apioption xAxis.title.margin */ - drawLegendSymbol: LegendSymbolMixin.drawLineMarker, /** - * Internal function called from setData. If the point count is the same - * as is was, or if there are overlapping X values, just run - * Point.update which is cheaper, allows animation, and keeps references - * to points. This also allows adding or removing points if the X-es - * don't match. + * The distance of the axis title from the axis line. By default, + * this distance is computed from the offset width of the labels, + * the labels' distance from the axis and the title's margin. + * However when the offset option is set, it overrides all this. * - * @private - * @function Highcharts.Series#updateData + * @sample {highcharts} highcharts/yaxis/title-offset/ + * Place the axis title on top of the axis + * @sample {highstock} highcharts/yaxis/title-offset/ + * Place the axis title on top of the Y axis * - * @param {Array<Highcharts.PointOptionsType>} data + * @type {number} + * @since 2.2.0 + * @apioption xAxis.title.offset + */ + /** + * Whether to reserve space for the title when laying out the axis. * - * @return {boolean} + * @type {boolean} + * @default true + * @since 5.0.11 + * @product highcharts highstock gantt + * @apioption xAxis.title.reserveSpace */ - updateData: function (data, animation) { - var options = this.options, - dataSorting = options.dataSorting, - oldData = this.points, - pointsToAdd = [], - hasUpdatedByKey, - i, - point, - lastIndex, - requireSorting = this.requireSorting, - equalLength = data.length === oldData.length, - succeeded = true; - this.xIncrement = null; - // Iterate the new data - data.forEach(function (pointOptions, i) { - var id, - x, - pointIndex, - optionsObject = (defined(pointOptions) && - this.pointClass.prototype.optionsToObject.call({ series: this }, - pointOptions)) || {}; - // Get the x of the new data point - x = optionsObject.x; - id = optionsObject.id; - if (id || isNumber(x)) { - pointIndex = this.findPointIndex(optionsObject, lastIndex); - // Matching X not found - // or used already due to ununique x values (#8995), - // add point (but later) - if (pointIndex === -1 || - typeof pointIndex === 'undefined') { - pointsToAdd.push(pointOptions); - // Matching X found, update - } - else if (oldData[pointIndex] && - pointOptions !== options.data[pointIndex]) { - oldData[pointIndex].update(pointOptions, false, null, false); - // Mark it touched, below we will remove all points that - // are not touched. - oldData[pointIndex].touched = true; - // Speed optimize by only searching after last known - // index. Performs ~20% bettor on large data sets. - if (requireSorting) { - lastIndex = pointIndex + 1; - } - // Point exists, no changes, don't remove it - } - else if (oldData[pointIndex]) { - oldData[pointIndex].touched = true; - } - // If the length is equal and some of the nodes had a - // match in the same position, we don't want to remove - // non-matches. - if (!equalLength || - i !== pointIndex || - (dataSorting && dataSorting.enabled) || - this.hasDerivedData) { - hasUpdatedByKey = true; - } - } - else { - // Gather all points that are not matched - pointsToAdd.push(pointOptions); - } - }, this); - // Remove points that don't exist in the updated data set - if (hasUpdatedByKey) { - i = oldData.length; - while (i--) { - point = oldData[i]; - if (point && !point.touched && point.remove) { - point.remove(false, animation); - } - } - // If we did not find keys (ids or x-values), and the length is the - // same, update one-to-one - } - else if (equalLength && (!dataSorting || !dataSorting.enabled)) { - data.forEach(function (point, i) { - // .update doesn't exist on a linked, hidden series (#3709) - // (#10187) - if (oldData[i].update && point !== oldData[i].y) { - oldData[i].update(point, false, null, false); - } - }); - // Don't add new points since those configs are used above - pointsToAdd.length = 0; - // Did not succeed in updating data - } - else { - succeeded = false; - } - oldData.forEach(function (point) { - if (point) { - point.touched = false; - } - }); - if (!succeeded) { - return false; - } - // Add new points - pointsToAdd.forEach(function (point) { - this.addPoint(point, false, null, null, false); - }, this); - if (this.xIncrement === null && - this.xData && - this.xData.length) { - this.xIncrement = arrayMax(this.xData); - this.autoIncrement(); - } - return true; - }, /** - * Apply a new set of data to the series and optionally redraw it. The - * new data array is passed by reference (except in case of - * `updatePoints`), and may later be mutated when updating the chart - * data. - * - * Note the difference in behaviour when setting the same amount of - * points, or a different amount of points, as handled by the - * `updatePoints` parameter. - * - * @sample highcharts/members/series-setdata/ - * Set new data from a button - * @sample highcharts/members/series-setdata-pie/ - * Set data in a pie - * @sample stock/members/series-setdata/ - * Set new data in Highstock - * @sample maps/members/series-setdata/ - * Set new data in Highmaps - * - * @function Highcharts.Series#setData - * - * @param {Array<Highcharts.PointOptionsType>} data - * Takes an array of data in the same format as described under - * `series.{type}.data` for the given series type, for example a - * line series would take data in the form described under - * [series.line.data](https://api.highcharts.com/highcharts/series.line.data). - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the series is altered. If - * doing more operations on the chart, it is a good idea to set - * redraw to false and call {@link Chart#redraw} after. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * When the updated data is the same length as the existing data, - * points will be updated by default, and animation visualizes - * how the points are changed. Set false to disable animation, or - * a configuration object to set duration or easing. - * - * @param {boolean} [updatePoints=true] - * When this is true, points will be updated instead of replaced - * whenever possible. This occurs a) when the updated data is the - * same length as the existing data, b) when points are matched - * by their id's, or c) when points can be matched by X values. - * This allows updating with animation and performs better. In - * this case, the original array is not passed by reference. Set - * `false` to prevent. - * - * @return {void} + * The rotation of the text in degrees. 0 is horizontal, 270 is + * vertical reading from bottom to top. + * + * @sample {highcharts} highcharts/yaxis/title-offset/ + * Horizontal + * + * @type {number} + * @default 0 + * @apioption xAxis.title.rotation */ - setData: function (data, redraw, animation, updatePoints) { - var series = this, - oldData = series.points, - oldDataLength = (oldData && oldData.length) || 0, - dataLength, - options = series.options, - chart = series.chart, - dataSorting = options.dataSorting, - firstPoint = null, - xAxis = series.xAxis, - i, - turboThreshold = options.turboThreshold, - pt, - xData = this.xData, - yData = this.yData, - pointArrayMap = series.pointArrayMap, - valueCount = pointArrayMap && pointArrayMap.length, - keys = options.keys, - indexOfX = 0, - indexOfY = 1, - updatedData; - data = data || []; - dataLength = data.length; - redraw = pick(redraw, true); - if (dataSorting && dataSorting.enabled) { - data = this.sortData(data); - } - // First try to run Point.update which is cheaper, allows animation, - // and keeps references to points. - if (updatePoints !== false && - dataLength && - oldDataLength && - !series.cropped && - !series.hasGroupedData && - series.visible && - // Soft updating has no benefit in boost, and causes JS error - // (#8355) - !series.isSeriesBoosting) { - updatedData = this.updateData(data, animation); - } - if (!updatedData) { - // Reset properties - series.xIncrement = null; - series.colorCounter = 0; // for series with colorByPoint (#1547) - // Update parallel arrays - this.parallelArrays.forEach(function (key) { - series[key + 'Data'].length = 0; - }); - // In turbo mode, only one- or twodimensional arrays of numbers - // are allowed. The first value is tested, and we assume that - // all the rest are defined the same way. Although the 'for' - // loops are similar, they are repeated inside each if-else - // conditional for max performance. - if (turboThreshold && dataLength > turboThreshold) { - firstPoint = series.getFirstValidPoint(data); - if (isNumber(firstPoint)) { // assume all points are numbers - for (i = 0; i < dataLength; i++) { - xData[i] = this.autoIncrement(); - yData[i] = data[i]; - } - // Assume all points are arrays when first point is - } - else if (isArray(firstPoint)) { - if (valueCount) { // [x, low, high] or [x, o, h, l, c] - for (i = 0; i < dataLength; i++) { - pt = data[i]; - xData[i] = pt[0]; - yData[i] = - pt.slice(1, valueCount + 1); - } - } - else { // [x, y] - if (keys) { - indexOfX = keys.indexOf('x'); - indexOfY = keys.indexOf('y'); - indexOfX = indexOfX >= 0 ? indexOfX : 0; - indexOfY = indexOfY >= 0 ? indexOfY : 1; - } - for (i = 0; i < dataLength; i++) { - pt = data[i]; - xData[i] = pt[indexOfX]; - yData[i] = pt[indexOfY]; - } - } - } - else { - // Highcharts expects configs to be numbers or arrays in - // turbo mode - error(12, false, chart); - } - } - else { - for (i = 0; i < dataLength; i++) { - // stray commas in oldIE: - if (typeof data[i] !== 'undefined') { - pt = { series: series }; - series.pointClass.prototype.applyOptions.apply(pt, [data[i]]); - series.updateParallelArrays(pt, i); - } - } - } - // Forgetting to cast strings to numbers is a common caveat when - // handling CSV or JSON - if (yData && isString(yData[0])) { - error(14, true, chart); - } - series.data = []; - series.options.data = series.userOptions.data = data; - // destroy old points - i = oldDataLength; - while (i--) { - if (oldData[i] && oldData[i].destroy) { - oldData[i].destroy(); - } - } - // reset minRange (#878) - if (xAxis) { - xAxis.minRange = xAxis.userMinRange; - } - // redraw - series.isDirty = chart.isDirtyBox = true; - series.isDirtyData = !!oldData; - animation = false; - } - // Typically for pie series, points need to be processed and - // generated prior to rendering the legend - if (options.legendType === 'point') { - this.processData(); - this.generatePoints(); - } - if (redraw) { - chart.redraw(animation); - } - }, /** - * Internal function to sort series data + * The actual text of the axis title. It can contain basic HTML tags + * like `b`, `i` and `span` with style. * - * @private - * @function Highcharts.Series#sortData - * @param {Array<Highcharts.PointOptionsType>} data - * Force data grouping. - * @return {Array<Highcharts.PointOptionsObject>} + * @sample {highcharts} highcharts/xaxis/title-text/ + * Custom HTML + * @sample {highstock} stock/xaxis/title-text/ + * Titles for both axes + * + * @type {string|null} + * @apioption xAxis.title.text */ - sortData: function (data) { - var series = this, - options = series.options, - dataSorting = options.dataSorting, - sortKey = dataSorting.sortKey || 'y', - sortedData, - getPointOptionsObject = function (series, - pointOptions) { - return (defined(pointOptions) && - series.pointClass.prototype.optionsToObject.call({ - series: series - }, - pointOptions)) || {}; - }; - data.forEach(function (pointOptions, i) { - data[i] = getPointOptionsObject(series, pointOptions); - data[i].index = i; - }, this); - // Sorting - sortedData = data.concat().sort(function (a, b) { - var aValue = getNestedProperty(sortKey, - a); - var bValue = getNestedProperty(sortKey, - b); - return bValue < aValue ? -1 : bValue > aValue ? 1 : 0; - }); - // Set x value depending on the position in the array - sortedData.forEach(function (point, i) { - point.x = i; - }, this); - // Set the same x for linked series points if they don't have their - // own sorting - if (series.linkedSeries) { - series.linkedSeries.forEach(function (linkedSeries) { - var options = linkedSeries.options, - seriesData = options.data; - if ((!options.dataSorting || - !options.dataSorting.enabled) && - seriesData) { - seriesData.forEach(function (pointOptions, i) { - seriesData[i] = getPointOptionsObject(linkedSeries, pointOptions); - if (data[i]) { - seriesData[i].x = data[i].x; - seriesData[i].index = i; - } - }); - linkedSeries.setData(seriesData, false); - } - }); - } - return data; - }, /** - * Internal function to process the data by cropping away unused data - * points if the series is longer than the crop threshold. This saves - * computing time for large series. + * Alignment of the text, can be `"left"`, `"right"` or `"center"`. + * Default alignment depends on the + * [title.align](xAxis.title.align): * - * @private - * @function Highcharts.Series#getProcessedData - * @param {boolean} [forceExtremesFromAll] - * Force getting extremes of a total series data range. - * @return {Highcharts.SeriesProcessedDataObject} + * Horizontal axes: + * - for `align` = `"low"`, `textAlign` is set to `left` + * - for `align` = `"middle"`, `textAlign` is set to `center` + * - for `align` = `"high"`, `textAlign` is set to `right` + * + * Vertical axes: + * - for `align` = `"low"` and `opposite` = `true`, `textAlign` is + * set to `right` + * - for `align` = `"low"` and `opposite` = `false`, `textAlign` is + * set to `left` + * - for `align` = `"middle"`, `textAlign` is set to `center` + * - for `align` = `"high"` and `opposite` = `true` `textAlign` is + * set to `left` + * - for `align` = `"high"` and `opposite` = `false` `textAlign` is + * set to `right` + * + * @type {Highcharts.AlignValue} + * @apioption xAxis.title.textAlign */ - getProcessedData: function (forceExtremesFromAll) { - var series = this, - // copied during slice operation: - processedXData = series.xData, - processedYData = series.yData, - dataLength = processedXData.length, - croppedData, - cropStart = 0, - cropped, - distance, - closestPointRange, - xAxis = series.xAxis, - i, // loop variable - options = series.options, - cropThreshold = options.cropThreshold, - getExtremesFromAll = forceExtremesFromAll || - series.getExtremesFromAll || - options.getExtremesFromAll, // #4599 - isCartesian = series.isCartesian, - xExtremes, - val2lin = xAxis && xAxis.val2lin, - isLog = !!(xAxis && xAxis.logarithmic), - throwOnUnsorted = series.requireSorting, - min, - max; - if (xAxis) { - // corrected for log axis (#3053) - xExtremes = xAxis.getExtremes(); - min = xExtremes.min; - max = xExtremes.max; - } - // optionally filter out points outside the plot area - if (isCartesian && - series.sorted && - !getExtremesFromAll && - (!cropThreshold || - dataLength > cropThreshold || - series.forceCrop)) { - // it's outside current extremes - if (processedXData[dataLength - 1] < min || - processedXData[0] > max) { - processedXData = []; - processedYData = []; - // only crop if it's actually spilling out - } - else if (series.yData && (processedXData[0] < min || - processedXData[dataLength - 1] > max)) { - croppedData = this.cropData(series.xData, series.yData, min, max); - processedXData = croppedData.xData; - processedYData = croppedData.yData; - cropStart = croppedData.start; - cropped = true; - } - } - // Find the closest distance between processed points - i = processedXData.length || 1; - while (--i) { - distance = (isLog ? - (val2lin(processedXData[i]) - - val2lin(processedXData[i - 1])) : - (processedXData[i] - - processedXData[i - 1])); - if (distance > 0 && - (typeof closestPointRange === 'undefined' || - distance < closestPointRange)) { - closestPointRange = distance; - // Unsorted data is not supported by the line tooltip, as well - // as data grouping and navigation in Stock charts (#725) and - // width calculation of columns (#1900) - } - else if (distance < 0 && throwOnUnsorted) { - error(15, false, series.chart); - throwOnUnsorted = false; // Only once - } - } - return { - xData: processedXData, - yData: processedYData, - cropped: cropped, - cropStart: cropStart, - closestPointRange: closestPointRange - }; - }, /** - * Internal function to apply processed data. - * In Highstock, this function is extended to provide data grouping. + * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the axis title. * - * @private - * @function Highcharts.Series#processData - * @param {boolean} [force] - * Force data grouping. - * @return {boolean|undefined} + * @type {boolean} + * @default false + * @product highcharts highstock gantt + * @apioption xAxis.title.useHTML */ - processData: function (force) { - var series = this, - xAxis = series.xAxis, - processedData; - // If the series data or axes haven't changed, don't go through - // this. Return false to pass the message on to override methods - // like in data grouping. - if (series.isCartesian && - !series.isDirty && - !xAxis.isDirty && - !series.yAxis.isDirty && - !force) { - return false; - } - processedData = series.getProcessedData(); - // Record the properties - series.cropped = processedData.cropped; // undefined or true - series.cropStart = processedData.cropStart; - series.processedXData = processedData.xData; - series.processedYData = processedData.yData; - series.closestPointRange = - series.basePointRange = processedData.closestPointRange; - }, /** - * Iterate over xData and crop values between min and max. Returns - * object containing crop start/end cropped xData with corresponding - * part of yData, dataMin and dataMax within the cropped range. + * Horizontal pixel offset of the title position. * - * @private - * @function Highcharts.Series#cropData - * @param {Array<number>} xData - * @param {Array<number>} yData - * @param {number} min - * @param {number} max - * @param {number} [cropShoulder] - * @return {Highcharts.SeriesCropDataObject} + * @type {number} + * @default 0 + * @since 4.1.6 + * @product highcharts highstock gantt + * @apioption xAxis.title.x */ - cropData: function (xData, yData, min, max, cropShoulder) { - var dataLength = xData.length, - cropStart = 0, - cropEnd = dataLength, - i, - j; - // line-type series need one point outside - cropShoulder = pick(cropShoulder, this.cropShoulder); - // iterate up to find slice start - for (i = 0; i < dataLength; i++) { - if (xData[i] >= min) { - cropStart = Math.max(0, i - cropShoulder); - break; - } - } - // proceed to find slice end - for (j = i; j < dataLength; j++) { - if (xData[j] > max) { - cropEnd = j + cropShoulder; - break; - } - } - return { - xData: xData.slice(cropStart, cropEnd), - yData: yData.slice(cropStart, cropEnd), - start: cropStart, - end: cropEnd - }; - }, /** - * Generate the data point after the data has been processed by cropping - * away unused points and optionally grouped in Highcharts Stock. + * Vertical pixel offset of the title position. * - * @private - * @function Highcharts.Series#generatePoints + * @type {number} + * @product highcharts highstock gantt + * @apioption xAxis.title.y */ - generatePoints: function () { - var series = this, - options = series.options, - dataOptions = options.data, - data = series.data, - dataLength, - processedXData = series.processedXData, - processedYData = series.processedYData, - PointClass = series.pointClass, - processedDataLength = processedXData.length, - cropStart = series.cropStart || 0, - cursor, - hasGroupedData = series.hasGroupedData, - keys = options.keys, - point, - points = [], - i; - if (!data && !hasGroupedData) { - var arr = []; - arr.length = dataOptions.length; - data = series.data = arr; - } - if (keys && hasGroupedData) { - // grouped data has already applied keys (#6590) - series.options.keys = false; - } - for (i = 0; i < processedDataLength; i++) { - cursor = cropStart + i; - if (!hasGroupedData) { - point = data[cursor]; - // #970: - if (!point && - typeof dataOptions[cursor] !== 'undefined') { - data[cursor] = point = (new PointClass()).init(series, dataOptions[cursor], processedXData[i]); - } - } - else { - // splat the y data in case of ohlc data array - point = (new PointClass()).init(series, [processedXData[i]].concat(splat(processedYData[i]))); - /** - * Highstock only. If a point object is created by data - * grouping, it doesn't reflect actual points in the raw - * data. In this case, the `dataGroup` property holds - * information that points back to the raw data. - * - * - `dataGroup.start` is the index of the first raw data - * point in the group. - * - * - `dataGroup.length` is the amount of points in the - * group. - * - * @product highstock - * - * @name Highcharts.Point#dataGroup - * @type {Highcharts.DataGroupingInfoObject|undefined} - */ - point.dataGroup = series.groupMap[i]; - if (point.dataGroup.options) { - point.options = point.dataGroup.options; - extend(point, point.dataGroup.options); - // Collision of props and options (#9770) - delete point.dataLabels; - } - } - if (point) { // #6279 - /** - * Contains the point's index in the `Series.points` array. - * - * @name Highcharts.Point#index - * @type {number} - * @readonly - */ - point.index = cursor; // For faster access in Point.update - points[i] = point; - } - } - // restore keys options (#6590) - series.options.keys = keys; - // Hide cropped-away points - this only runs when the number of - // points is above cropThreshold, or when swithching view from - // non-grouped data to grouped data (#637) - if (data && - (processedDataLength !== (dataLength = data.length) || - hasGroupedData)) { - for (i = 0; i < dataLength; i++) { - // when has grouped data, clear all points - if (i === cropStart && !hasGroupedData) { - i += processedDataLength; - } - if (data[i]) { - data[i].destroyElements(); - data[i].plotX = void 0; // #1003 - } - } - } - /** - * Read only. An array containing those values converted to points. - * In case the series data length exceeds the `cropThreshold`, or if - * the data is grouped, `series.data` doesn't contain all the - * points. Also, in case a series is hidden, the `data` array may be - * empty. To access raw values, `series.options.data` will always be - * up to date. `Series.data` only contains the points that have been - * created on demand. To modify the data, use - * {@link Highcharts.Series#setData} or - * {@link Highcharts.Point#update}. - * - * @see Series.points - * - * @name Highcharts.Series#data - * @type {Array<Highcharts.Point>} - */ - series.data = data; - /** - * An array containing all currently visible point objects. In case - * of cropping, the cropped-away points are not part of this array. - * The `series.points` array starts at `series.cropStart` compared - * to `series.data` and `series.options.data`. If however the series - * data is grouped, these can't be correlated one to one. To modify - * the data, use {@link Highcharts.Series#setData} or - * {@link Highcharts.Point#update}. - * - * @name Highcharts.Series#points - * @type {Array<Highcharts.Point>} - */ - series.points = points; - fireEvent(this, 'afterGeneratePoints'); - }, /** - * Get current X extremes for the visible data. + * Alignment of the title relative to the axis values. Possible + * values are "low", "middle" or "high". * - * @private - * @function Highcharts.Series#getXExtremes + * @sample {highcharts} highcharts/xaxis/title-align-low/ + * "low" + * @sample {highcharts} highcharts/xaxis/title-align-center/ + * "middle" by default + * @sample {highcharts} highcharts/xaxis/title-align-high/ + * "high" + * @sample {highcharts} highcharts/yaxis/title-offset/ + * Place the Y axis title on top of the axis + * @sample {highstock} stock/xaxis/title-align/ + * Aligned to "high" value * - * @param {Array<number>} xData - * The data to inspect. Defaults to the current data within the - * visible range. - * @return {Highcharts.RangeObject} + * @type {Highcharts.AxisTitleAlignValue} */ - getXExtremes: function (xData) { - return { - min: arrayMin(xData), - max: arrayMax(xData) - }; - }, + align: "middle", /** - * Calculate Y extremes for the visible data. The result is returned - * as an object with `dataMin` and `dataMax` properties. + * CSS styles for the title. If the title text is longer than the + * axis length, it will wrap to multiple lines by default. This can + * be customized by setting `textOverflow: 'ellipsis'`, by + * setting a specific `width` or by setting `whiteSpace: 'nowrap'`. * - * @private - * @function Highcharts.Series#getExtremes - * @param {Array<number>} [yData] - * The data to inspect. Defaults to the current data within the - * visible range. - * @param {boolean} [forceExtremesFromAll] - * Force getting extremes of a total series data range. - * @return {Highcharts.DataExtremesObject} + * In styled mode, the stroke width is given in the + * `.highcharts-axis-title` class. + * + * @sample {highcharts} highcharts/xaxis/title-style/ + * Red + * @sample {highcharts} highcharts/css/axis/ + * Styled mode + * + * @type {Highcharts.CSSObject} */ - getExtremes: function (yData, forceExtremesFromAll) { - var xAxis = this.xAxis, - yAxis = this.yAxis, - xData = this.processedXData || this.xData, - yDataLength, - activeYData = [], - activeCounter = 0, - // #2117, need to compensate for log X axis - xExtremes, - xMin = 0, - xMax = 0, - validValue, - withinRange, - // Handle X outside the viewed area. This does not work with - // non-sorted data like scatter (#7639). - shoulder = this.requireSorting ? this.cropShoulder : 0, - positiveValuesOnly = yAxis ? yAxis.positiveValuesOnly : false, - x, - y, - i, - j; - yData = yData || this.stackedYData || this.processedYData || []; - yDataLength = yData.length; - if (xAxis) { - xExtremes = xAxis.getExtremes(); - xMin = xExtremes.min; - xMax = xExtremes.max; - } - for (i = 0; i < yDataLength; i++) { - x = xData[i]; - y = yData[i]; - // For points within the visible range, including the first - // point outside the visible range (#7061), consider y extremes. - validValue = ((isNumber(y) || isArray(y)) && - ((y.length || y > 0) || !positiveValuesOnly)); - withinRange = (forceExtremesFromAll || - this.getExtremesFromAll || - this.options.getExtremesFromAll || - this.cropped || - !xAxis || // for colorAxis support - ((xData[i + shoulder] || x) >= xMin && - (xData[i - shoulder] || x) <= xMax)); - if (validValue && withinRange) { - j = y.length; - if (j) { // array, like ohlc or range data - while (j--) { - if (isNumber(y[j])) { // #7380, #11513 - activeYData[activeCounter++] = y[j]; - } - } - } - else { - activeYData[activeCounter++] = y; - } - } - } - var dataExtremes = { - dataMin: arrayMin(activeYData), - dataMax: arrayMax(activeYData) - }; - fireEvent(this, 'afterGetExtremes', { dataExtremes: dataExtremes }); - return dataExtremes; + style: { + /** @internal */ + color: "#666666", }, + }, + /** + * The type of axis. Can be one of `linear`, `logarithmic`, `datetime` + * or `category`. In a datetime axis, the numbers are given in + * milliseconds, and tick marks are placed on appropriate values like + * full hours or days. In a category axis, the + * [point names](#series.line.data.name) of the chart's series are used + * for categories, if not a [categories](#xAxis.categories) array is + * defined. + * + * @sample {highcharts} highcharts/xaxis/type-linear/ + * Linear + * @sample {highcharts} highcharts/yaxis/type-log/ + * Logarithmic + * @sample {highcharts} highcharts/yaxis/type-log-minorgrid/ + * Logarithmic with minor grid lines + * @sample {highcharts} highcharts/xaxis/type-log-both/ + * Logarithmic on two axes + * @sample {highcharts} highcharts/yaxis/type-log-negative/ + * Logarithmic with extension to emulate negative values + * + * @type {Highcharts.AxisTypeValue} + * @product highcharts gantt + */ + type: "linear", + /** + * If there are multiple axes on the same side of the chart, the pixel + * margin between the axes. Defaults to 0 on vertical axes, 15 on + * horizontal axes. + * + * @type {number} + * @since 7.0.3 + * @apioption xAxis.margin + */ + /** + * Applies only when the axis `type` is `category`. When `uniqueNames` + * is true, points are placed on the X axis according to their names. + * If the same point name is repeated in the same or another series, + * the point is placed on the same X position as other points of the + * same name. When `uniqueNames` is false, the points are laid out in + * increasing X positions regardless of their names, and the X axis + * category will take the name of the last point in each position. + * + * @sample {highcharts} highcharts/xaxis/uniquenames-true/ + * True by default + * @sample {highcharts} highcharts/xaxis/uniquenames-false/ + * False + * + * @type {boolean} + * @default true + * @since 4.2.7 + * @product highcharts gantt + * @apioption xAxis.uniqueNames + */ + /** + * Datetime axis only. An array determining what time intervals the + * ticks are allowed to fall on. Each array item is an array where the + * first value is the time unit and the second value another array of + * allowed multiples. + * + * Defaults to: + * ```js + * units: [[ + * 'millisecond', // unit name + * [1, 2, 5, 10, 20, 25, 50, 100, 200, 500] // allowed multiples + * ], [ + * 'second', + * [1, 2, 5, 10, 15, 30] + * ], [ + * 'minute', + * [1, 2, 5, 10, 15, 30] + * ], [ + * 'hour', + * [1, 2, 3, 4, 6, 8, 12] + * ], [ + * 'day', + * [1] + * ], [ + * 'week', + * [1] + * ], [ + * 'month', + * [1, 3, 6] + * ], [ + * 'year', + * null + * ]] + * ``` + * + * @type {Array<Array<string,(Array<number>|null)>>} + * @product highcharts highstock gantt + * @apioption xAxis.units + */ + /** + * Whether axis, including axis title, line, ticks and labels, should + * be visible. + * + * @type {boolean} + * @default true + * @since 4.1.9 + * @product highcharts highstock gantt + * @apioption xAxis.visible + */ + /** + * Color of the minor, secondary grid lines. + * + * In styled mode, the stroke width is given in the + * `.highcharts-minor-grid-line` class. + * + * @sample {highcharts} highcharts/yaxis/minorgridlinecolor/ + * Bright grey lines from Y axis + * @sample {highcharts|highstock} highcharts/css/axis-grid/ + * Styled mode + * @sample {highstock} stock/xaxis/minorgridlinecolor/ + * Bright grey lines from Y axis + * + * @type {Highcharts.ColorType} + * @default #f2f2f2 + */ + minorGridLineColor: "#f2f2f2", + /** + * Width of the minor, secondary grid lines. + * + * In styled mode, the stroke width is given in the + * `.highcharts-grid-line` class. + * + * @sample {highcharts} highcharts/yaxis/minorgridlinewidth/ + * 2px lines from Y axis + * @sample {highcharts|highstock} highcharts/css/axis-grid/ + * Styled mode + * @sample {highstock} stock/xaxis/minorgridlinewidth/ + * 2px lines from Y axis + */ + minorGridLineWidth: 1, + /** + * Color for the minor tick marks. + * + * @sample {highcharts} highcharts/yaxis/minortickcolor/ + * Black tick marks on Y axis + * @sample {highstock} stock/xaxis/minorticks/ + * Black tick marks on Y axis + * + * @type {Highcharts.ColorType} + * @default #999999 + */ + minorTickColor: "#999999", + /** + * The color of the line marking the axis itself. + * + * In styled mode, the line stroke is given in the + * `.highcharts-axis-line` or `.highcharts-xaxis-line` class. + * + * @productdesc {highmaps} + * In Highmaps, the axis line is hidden by default, because the axis is + * not visible by default. + * + * @sample {highcharts} highcharts/yaxis/linecolor/ + * A red line on Y axis + * @sample {highcharts|highstock} highcharts/css/axis/ + * Axes in styled mode + * @sample {highstock} stock/xaxis/linecolor/ + * A red line on X axis + * + * @type {Highcharts.ColorType} + * @default #ccd6eb + */ + lineColor: "#ccd6eb", + /** + * The width of the line marking the axis itself. + * + * In styled mode, the stroke width is given in the + * `.highcharts-axis-line` or `.highcharts-xaxis-line` class. + * + * @sample {highcharts} highcharts/yaxis/linecolor/ + * A 1px line on Y axis + * @sample {highcharts|highstock} highcharts/css/axis/ + * Axes in styled mode + * @sample {highstock} stock/xaxis/linewidth/ + * A 2px line on X axis + * + * @default {highcharts|highstock} 1 + * @default {highmaps} 0 + */ + lineWidth: 1, + /** + * Color of the grid lines extending the ticks across the plot area. + * + * In styled mode, the stroke is given in the `.highcharts-grid-line` + * class. + * + * @productdesc {highmaps} + * In Highmaps, the grid lines are hidden by default. + * + * @sample {highcharts} highcharts/yaxis/gridlinecolor/ + * Green lines + * @sample {highcharts|highstock} highcharts/css/axis-grid/ + * Styled mode + * @sample {highstock} stock/xaxis/gridlinecolor/ + * Green lines + * + * @type {Highcharts.ColorType} + * @default #e6e6e6 + */ + gridLineColor: "#e6e6e6", + // gridLineDashStyle: 'solid', + /** + * The width of the grid lines extending the ticks across the plot area. + * + * In styled mode, the stroke width is given in the + * `.highcharts-grid-line` class. + * + * @sample {highcharts} highcharts/yaxis/gridlinewidth/ + * 2px lines + * @sample {highcharts|highstock} highcharts/css/axis-grid/ + * Styled mode + * @sample {highstock} stock/xaxis/gridlinewidth/ + * 2px lines + * + * @type {number} + * @default 0 + * @apioption xAxis.gridLineWidth + */ + // gridLineWidth: 0, + /** + * The height as the vertical axis. If it's a number, it is + * interpreted as pixels. + * + * Since Highcharts 2: If it's a percentage string, it is interpreted + * as percentages of the total plot height. + * + * @type {number|string} + * @product highcharts highstock + * @apioption xAxis.height + */ + /** + * The width as the horizontal axis. If it's a number, it is interpreted + * as pixels. + * + * Since Highcharts v5.0.13: If it's a percentage string, it is + * interpreted as percentages of the total plot width. + * + * @type {number|string} + * @product highcharts highstock + * @apioption xAxis.width + */ + /** + * Color for the main tick marks. + * + * In styled mode, the stroke is given in the `.highcharts-tick` + * class. + * + * @sample {highcharts} highcharts/xaxis/tickcolor/ + * Red ticks on X axis + * @sample {highcharts|highstock} highcharts/css/axis-grid/ + * Styled mode + * @sample {highstock} stock/xaxis/ticks/ + * Formatted ticks on X axis + * + * @type {Highcharts.ColorType} + * @default #ccd6eb + */ + tickColor: "#ccd6eb", + // tickWidth: 1 + }; + /** + * The Y axis or value axis. Normally this is the vertical axis, + * though if the chart is inverted this is the horizontal axis. + * In case of multiple axes, the yAxis node is an array of + * configuration objects. + * + * See [the Axis object](/class-reference/Highcharts.Axis) for programmatic + * access to the axis. + * + * @type {*|Array<*>} + * @extends xAxis + * @excluding currentDateIndicator,ordinal,overscroll + * @optionparent yAxis + * + * @private + */ + Axis.defaultYAxisOptions = { + /** + * The type of axis. Can be one of `linear`, `logarithmic`, `datetime`, + * `category` or `treegrid`. Defaults to `treegrid` for Gantt charts, + * `linear` for other chart types. + * + * In a datetime axis, the numbers are given in milliseconds, and tick + * marks are placed on appropriate values, like full hours or days. In a + * category or treegrid axis, the [point names](#series.line.data.name) + * of the chart's series are used for categories, if a + * [categories](#xAxis.categories) array is not defined. + * + * @sample {highcharts} highcharts/yaxis/type-log-minorgrid/ + * Logarithmic with minor grid lines + * @sample {highcharts} highcharts/yaxis/type-log-negative/ + * Logarithmic with extension to emulate negative values + * @sample {gantt} gantt/treegrid-axis/demo + * Treegrid axis + * + * @type {Highcharts.AxisTypeValue} + * @default {highcharts} linear + * @default {gantt} treegrid + * @product highcharts gantt + * @apioption yAxis.type + */ + /** + * The height of the Y axis. If it's a number, it is interpreted as + * pixels. + * + * Since Highcharts 2: If it's a percentage string, it is interpreted as + * percentages of the total plot height. + * + * @see [yAxis.top](#yAxis.top) + * + * @sample {highstock} stock/demo/candlestick-and-volume/ + * Percentage height panes + * + * @type {number|string} + * @product highcharts highstock + * @apioption yAxis.height + */ + /** + * Solid gauge only. Unless [stops](#yAxis.stops) are set, the color + * to represent the maximum value of the Y axis. + * + * @sample {highcharts} highcharts/yaxis/mincolor-maxcolor/ + * Min and max colors + * + * @type {Highcharts.ColorType} + * @default #003399 + * @since 4.0 + * @product highcharts + * @apioption yAxis.maxColor + */ + /** + * Solid gauge only. Unless [stops](#yAxis.stops) are set, the color + * to represent the minimum value of the Y axis. + * + * @sample {highcharts} highcharts/yaxis/mincolor-maxcolor/ + * Min and max color + * + * @type {Highcharts.ColorType} + * @default #e6ebf5 + * @since 4.0 + * @product highcharts + * @apioption yAxis.minColor + */ + /** + * Whether to reverse the axis so that the highest number is closest + * to the origin. + * + * @sample {highcharts} highcharts/yaxis/reversed/ + * Reversed Y axis + * @sample {highstock} stock/xaxis/reversed/ + * Reversed Y axis + * + * @type {boolean} + * @default {highcharts} false + * @default {highstock} false + * @default {highmaps} true + * @default {gantt} true + * @apioption yAxis.reversed + */ + /** + * If `true`, the first series in a stack will be drawn on top in a + * positive, non-reversed Y axis. If `false`, the first series is in + * the base of the stack. + * + * @sample {highcharts} highcharts/yaxis/reversedstacks-false/ + * Non-reversed stacks + * @sample {highstock} highcharts/yaxis/reversedstacks-false/ + * Non-reversed stacks + * + * @type {boolean} + * @default true + * @since 3.0.10 + * @product highcharts highstock + * @apioption yAxis.reversedStacks + */ + /** + * Solid gauge series only. Color stops for the solid gauge. Use this + * in cases where a linear gradient between a `minColor` and `maxColor` + * is not sufficient. The stops is an array of tuples, where the first + * item is a float between 0 and 1 assigning the relative position in + * the gradient, and the second item is the color. + * + * For solid gauges, the Y axis also inherits the concept of + * [data classes](https://api.highcharts.com/highmaps#colorAxis.dataClasses) + * from the Highmaps color axis. + * + * @see [minColor](#yAxis.minColor) + * @see [maxColor](#yAxis.maxColor) + * + * @sample {highcharts} highcharts/demo/gauge-solid/ + * True by default + * + * @type {Array<Array<number,Highcharts.ColorType>>} + * @since 4.0 + * @product highcharts + * @apioption yAxis.stops + */ + /** + * The pixel width of the major tick marks. + * + * @sample {highcharts} highcharts/xaxis/tickwidth/ 10 px width + * @sample {highstock} stock/xaxis/ticks/ Formatted ticks on X axis + * + * @type {number} + * @default 0 + * @product highcharts highstock gantt + * @apioption yAxis.tickWidth + */ + /** + * Whether to force the axis to end on a tick. Use this option with + * the `maxPadding` option to control the axis end. + * + * This option is always disabled, when panning type is + * either `y` or `xy`. + * + * @see [type](#chart.panning.type) + * + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * True by default + * @sample {highcharts} highcharts/yaxis/endontick/ + * False + * @sample {highstock} stock/demo/basic-line/ + * True by default + * @sample {highstock} stock/xaxis/endontick/ + * False for Y axis + * + * @since 1.2.0 + */ + endOnTick: true, + /** + * Padding of the max value relative to the length of the axis. A + * padding of 0.05 will make a 100px axis 5px longer. This is useful + * when you don't want the highest data value to appear on the edge + * of the plot area. When the axis' `max` option is set or a max extreme + * is set using `axis.setExtremes()`, the maxPadding will be ignored. + * + * Also the `softThreshold` option takes precedence over `maxPadding`, + * so if the data is tangent to the threshold, `maxPadding` may not + * apply unless `softThreshold` is set to false. + * + * @sample {highcharts} highcharts/yaxis/maxpadding-02/ + * Max padding of 0.2 + * @sample {highstock} stock/xaxis/minpadding-maxpadding/ + * Greater min- and maxPadding + * + * @since 1.2.0 + * @product highcharts highstock gantt + */ + maxPadding: 0.05, + /** + * Padding of the min value relative to the length of the axis. A + * padding of 0.05 will make a 100px axis 5px longer. This is useful + * when you don't want the lowest data value to appear on the edge + * of the plot area. When the axis' `min` option is set or a max extreme + * is set using `axis.setExtremes()`, the maxPadding will be ignored. + * + * Also the `softThreshold` option takes precedence over `minPadding`, + * so if the data is tangent to the threshold, `minPadding` may not + * apply unless `softThreshold` is set to false. + * + * @sample {highcharts} highcharts/yaxis/minpadding/ + * Min padding of 0.2 + * @sample {highstock} stock/xaxis/minpadding-maxpadding/ + * Greater min- and maxPadding + * + * @since 1.2.0 + * @product highcharts highstock gantt + */ + minPadding: 0.05, + /** + * @productdesc {highstock} + * In Highstock 1.x, the Y axis was placed on the left side by default. + * + * @sample {highcharts} highcharts/yaxis/opposite/ + * Secondary Y axis opposite + * @sample {highstock} stock/xaxis/opposite/ + * Y axis on left side + * + * @type {boolean} + * @default {highstock} true + * @default {highcharts} false + * @product highstock highcharts gantt + * @apioption yAxis.opposite + */ + /** + * @see [tickInterval](#xAxis.tickInterval) + * @see [tickPositioner](#xAxis.tickPositioner) + * @see [tickPositions](#xAxis.tickPositions) + */ + tickPixelInterval: 72, + showLastLabel: true, + /** + * @extends xAxis.labels + */ + labels: { + /** + * Angular gauges and solid gauges only. + * The label's pixel distance from the perimeter of the plot area. + * + * Since v7.1.2: If it's a percentage string, it is interpreted the + * same as [series.radius](#plotOptions.gauge.radius), so label can be + * aligned under the gauge's shape. + * + * @sample {highcharts} highcharts/yaxis/labels-distance/ + * Labels centered under the arc + * + * @type {number|string} + * @default -25 + * @product highcharts + * @apioption yAxis.labels.distance + */ /** - * Set the current data extremes as `dataMin` and `dataMax` on the - * Series item. Use this only when the series properties should be - * updated. + * The y position offset of all labels relative to the tick + * positions on the axis. For polar and radial axis consider the use + * of the [distance](#yAxis.labels.distance) option. * - * @private - * @function Highcharts.Series#applyExtremes - * @return {void} + * @sample {highcharts} highcharts/xaxis/labels-x/ + * Y axis labels placed on grid lines + * + * @type {number} + * @default {highcharts} 3 + * @default {highstock} -2 + * @default {highmaps} 3 + * @apioption yAxis.labels.y */ - applyExtremes: function () { - var dataExtremes = this.getExtremes(); - /** - * Contains the minimum value of the series' data point. Some series - * types like `networkgraph` do not support this property as they - * lack a `y`-value. - * @name Highcharts.Series#dataMin - * @type {number|undefined} - * @readonly - */ - this.dataMin = dataExtremes.dataMin; - /** - * Contains the maximum value of the series' data point. Some series - * types like `networkgraph` do not support this property as they - * lack a `y`-value. - * @name Highcharts.Series#dataMax - * @type {number|undefined} - * @readonly - */ - this.dataMax = dataExtremes.dataMax; - return dataExtremes; - }, /** - * Find and return the first non null point in the data + * What part of the string the given position is anchored to. Can + * be one of `"left"`, `"center"` or `"right"`. The exact position + * also depends on the `labels.x` setting. * - * @private - * @function Highcharts.Series.getFirstValidPoint - * @param {Array<Highcharts.PointOptionsType>} data - * Array of options for points + * Angular gauges and solid gauges defaults to `"center"`. + * Solid gauges with two labels have additional option `"auto"` + * for automatic horizontal and vertical alignment. + * + * @see [yAxis.labels.distance](#yAxis.labels.distance) + * + * @sample {highcharts} highcharts/yaxis/labels-align-left/ + * Left + * @sample {highcharts} highcharts/series-solidgauge/labels-auto-aligned/ + * Solid gauge labels auto aligned * - * @return {Highcharts.PointOptionsType} + * @type {Highcharts.AlignValue} + * @default {highcharts|highmaps} right + * @default {highstock} left + * @apioption yAxis.labels.align + */ + /** + * The x position offset of all labels relative to the tick + * positions on the axis. Defaults to -15 for left axis, 15 for + * right axis. + * + * @sample {highcharts} highcharts/xaxis/labels-x/ + * Y axis labels placed on grid lines + */ + x: -8, + }, + /** + * @productdesc {highmaps} + * In Highmaps, the axis line is hidden by default, because the axis is + * not visible by default. + * + * @type {Highcharts.ColorType} + * @apioption yAxis.lineColor + */ + /** + * @sample {highcharts} highcharts/yaxis/max-200/ + * Y axis max of 200 + * @sample {highcharts} highcharts/yaxis/max-logarithmic/ + * Y axis max on logarithmic axis + * @sample {highstock} stock/yaxis/min-max/ + * Fixed min and max on Y axis + * @sample {highmaps} maps/axis/min-max/ + * Pre-zoomed to a specific area + * + * @apioption yAxis.max + */ + /** + * @sample {highcharts} highcharts/yaxis/min-startontick-false/ + * -50 with startOnTick to false + * @sample {highcharts} highcharts/yaxis/min-startontick-true/ + * -50 with startOnTick true by default + * @sample {highstock} stock/yaxis/min-max/ + * Fixed min and max on Y axis + * @sample {highmaps} maps/axis/min-max/ + * Pre-zoomed to a specific area + * + * @apioption yAxis.min + */ + /** + * An optional scrollbar to display on the Y axis in response to + * limiting the minimum an maximum of the axis values. + * + * In styled mode, all the presentational options for the scrollbar + * are replaced by the classes `.highcharts-scrollbar-thumb`, + * `.highcharts-scrollbar-arrow`, `.highcharts-scrollbar-button`, + * `.highcharts-scrollbar-rifles` and `.highcharts-scrollbar-track`. + * + * @sample {highstock} stock/yaxis/scrollbar/ + * Scrollbar on the Y axis + * + * @extends scrollbar + * @since 4.2.6 + * @product highstock + * @excluding height + * @apioption yAxis.scrollbar + */ + /** + * Enable the scrollbar on the Y axis. + * + * @sample {highstock} stock/yaxis/scrollbar/ + * Enabled on Y axis + * + * @type {boolean} + * @default false + * @since 4.2.6 + * @product highstock + * @apioption yAxis.scrollbar.enabled + */ + /** + * Pixel margin between the scrollbar and the axis elements. + * + * @type {number} + * @default 10 + * @since 4.2.6 + * @product highstock + * @apioption yAxis.scrollbar.margin + */ + /** + * Whether to show the scrollbar when it is fully zoomed out at max + * range. Setting it to `false` on the Y axis makes the scrollbar stay + * hidden until the user zooms in, like common in browsers. + * + * @type {boolean} + * @default true + * @since 4.2.6 + * @product highstock + * @apioption yAxis.scrollbar.showFull + */ + /** + * The width of a vertical scrollbar or height of a horizontal + * scrollbar. Defaults to 20 on touch devices. + * + * @type {number} + * @default 14 + * @since 4.2.6 + * @product highstock + * @apioption yAxis.scrollbar.size + */ + /** + * Z index of the scrollbar elements. + * + * @type {number} + * @default 3 + * @since 4.2.6 + * @product highstock + * @apioption yAxis.scrollbar.zIndex + */ + /** + * A soft maximum for the axis. If the series data maximum is less + * than this, the axis will stay at this maximum, but if the series + * data maximum is higher, the axis will flex to show all data. + * + * **Note**: The [series.softThreshold]( + * #plotOptions.series.softThreshold) option takes precedence over this + * option. + * + * @sample highcharts/yaxis/softmin-softmax/ + * Soft min and max + * + * @type {number} + * @since 5.0.1 + * @product highcharts highstock gantt + * @apioption yAxis.softMax + */ + /** + * A soft minimum for the axis. If the series data minimum is greater + * than this, the axis will stay at this minimum, but if the series + * data minimum is lower, the axis will flex to show all data. + * + * **Note**: The [series.softThreshold]( + * #plotOptions.series.softThreshold) option takes precedence over this + * option. + * + * @sample highcharts/yaxis/softmin-softmax/ + * Soft min and max + * + * @type {number} + * @since 5.0.1 + * @product highcharts highstock gantt + * @apioption yAxis.softMin + */ + /** + * Defines the horizontal alignment of the stack total label. Can be one + * of `"left"`, `"center"` or `"right"`. The default value is calculated + * at runtime and depends on orientation and whether the stack is + * positive or negative. + * + * @sample {highcharts} highcharts/yaxis/stacklabels-align-left/ + * Aligned to the left + * @sample {highcharts} highcharts/yaxis/stacklabels-align-center/ + * Aligned in center + * @sample {highcharts} highcharts/yaxis/stacklabels-align-right/ + * Aligned to the right + * + * @type {Highcharts.AlignValue} + * @since 2.1.5 + * @product highcharts + * @apioption yAxis.stackLabels.align + */ + /** + * A format string for the data label. Available variables are the same + * as for `formatter`. + * + * @type {string} + * @default {total} + * @since 3.0.2 + * @product highcharts highstock + * @apioption yAxis.stackLabels.format + */ + /** + * Rotation of the labels in degrees. + * + * @sample {highcharts} highcharts/yaxis/stacklabels-rotation/ + * Labels rotated 45° + * + * @type {number} + * @default 0 + * @since 2.1.5 + * @product highcharts + * @apioption yAxis.stackLabels.rotation + */ + /** + * The text alignment for the label. While `align` determines where the + * texts anchor point is placed with regards to the stack, `textAlign` + * determines how the text is aligned against its anchor point. Possible + * values are `"left"`, `"center"` and `"right"`. The default value is + * calculated at runtime and depends on orientation and whether the + * stack is positive or negative. + * + * @sample {highcharts} highcharts/yaxis/stacklabels-textalign-left/ + * Label in center position but text-aligned left + * + * @type {Highcharts.AlignValue} + * @since 2.1.5 + * @product highcharts + * @apioption yAxis.stackLabels.textAlign + */ + /** + * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the labels. + * + * @type {boolean} + * @default false + * @since 3.0 + * @product highcharts highstock + * @apioption yAxis.stackLabels.useHTML + */ + /** + * Defines the vertical alignment of the stack total label. Can be one + * of `"top"`, `"middle"` or `"bottom"`. The default value is calculated + * at runtime and depends on orientation and whether the stack is + * positive or negative. + * + * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-top/ + * Vertically aligned top + * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-middle/ + * Vertically aligned middle + * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-bottom/ + * Vertically aligned bottom + * + * @type {Highcharts.VerticalAlignValue} + * @since 2.1.5 + * @product highcharts + * @apioption yAxis.stackLabels.verticalAlign + */ + /** + * The x position offset of the label relative to the left of the + * stacked bar. The default value is calculated at runtime and depends + * on orientation and whether the stack is positive or negative. + * + * @sample {highcharts} highcharts/yaxis/stacklabels-x/ + * Stack total labels with x offset + * + * @type {number} + * @since 2.1.5 + * @product highcharts + * @apioption yAxis.stackLabels.x + */ + /** + * The y position offset of the label relative to the tick position + * on the axis. The default value is calculated at runtime and depends + * on orientation and whether the stack is positive or negative. + * + * @sample {highcharts} highcharts/yaxis/stacklabels-y/ + * Stack total labels with y offset + * + * @type {number} + * @since 2.1.5 + * @product highcharts + * @apioption yAxis.stackLabels.y + */ + /** + * Whether to force the axis to start on a tick. Use this option with + * the `maxPadding` option to control the axis start. + * + * This option is always disabled, when panning type is + * either `y` or `xy`. + * + * @see [type](#chart.panning.type) + * + * @sample {highcharts} highcharts/xaxis/startontick-false/ + * False by default + * @sample {highcharts} highcharts/xaxis/startontick-true/ + * True + * @sample {highstock} stock/xaxis/endontick/ + * False for Y axis + * + * @since 1.2.0 + * @product highcharts highstock gantt + */ + startOnTick: true, + title: { + /** + * The pixel distance between the axis labels and the title. + * Positive values are outside the axis line, negative are inside. + * + * @sample {highcharts} highcharts/xaxis/title-margin/ + * Y axis title margin of 60 + * + * @type {number} + * @default 40 + * @apioption yAxis.title.margin */ - getFirstValidPoint: function (data) { - var firstPoint = null, - dataLength = data.length, - i = 0; - while (firstPoint === null && i < dataLength) { - firstPoint = data[i]; - i++; - } - return firstPoint; - }, /** - * Translate data points from raw data values to chart specific - * positioning data needed later in the `drawPoints` and `drawGraph` - * functions. This function can be overridden in plugins and custom - * series type implementations. - * - * @function Highcharts.Series#translate - * @return {void} - * @fires Highcharts.Series#events:translate + * The rotation of the text in degrees. 0 is horizontal, 270 is + * vertical reading from bottom to top. + * + * @sample {highcharts} highcharts/yaxis/title-offset/ + * Horizontal */ - translate: function () { - if (!this.processedXData) { // hidden series - this.processData(); - } - this.generatePoints(); - var series = this, - options = series.options, - stacking = options.stacking, - xAxis = series.xAxis, - categories = xAxis.categories, - enabledDataSorting = series.enabledDataSorting, - yAxis = series.yAxis, - points = series.points, - dataLength = points.length, - hasModifyValue = !!series.modifyValue, - i, - pointPlacement = series.pointPlacementToXValue(), // #7860 - dynamicallyPlaced = Boolean(pointPlacement), - threshold = options.threshold, - stackThreshold = options.startFromThreshold ? threshold : 0, - plotX, - lastPlotX, - stackIndicator, - zoneAxis = this.zoneAxis || 'y', - closestPointRangePx = Number.MAX_VALUE; - /** - * Plotted coordinates need to be within a limited range. Drawing - * too far outside the viewport causes various rendering issues - * (#3201, #3923, #7555). - * @private - */ - function limitedRange(val) { - return clamp(val, -1e5, 1e5); - } - // Translate each point - for (i = 0; i < dataLength; i++) { - var point = points[i], - xValue = point.x, - yValue = point.y, - yBottom = point.low, - stack = stacking && yAxis.stacking && yAxis.stacking.stacks[(series.negStacks && - yValue < - (stackThreshold ? 0 : threshold) ? - '-' : - '') + series.stackKey], - pointStack, - stackValues; - if (yAxis.positiveValuesOnly && !yAxis.validatePositiveValue(yValue) || - xAxis.positiveValuesOnly && !xAxis.validatePositiveValue(xValue)) { - point.isNull = true; - } - // Get the plotX translation - point.plotX = plotX = correctFloat(// #5236 - limitedRange(xAxis.translate(// #3923 - xValue, 0, 0, 0, 1, pointPlacement, this.type === 'flags')) // #3923 - ); - // Calculate the bottom y value for stacked series - if (stacking && - series.visible && - stack && - stack[xValue]) { - stackIndicator = series.getStackIndicator(stackIndicator, xValue, series.index); - if (!point.isNull) { - pointStack = stack[xValue]; - stackValues = - pointStack.points[stackIndicator.key]; - } - } - if (isArray(stackValues)) { - yBottom = stackValues[0]; - yValue = stackValues[1]; - if (yBottom === stackThreshold && - stackIndicator.key === - stack[xValue].base) { - yBottom = pick((isNumber(threshold) && threshold), yAxis.min); - } - // #1200, #1232 - if (yAxis.positiveValuesOnly && yBottom <= 0) { - yBottom = null; - } - point.total = point.stackTotal = pointStack.total; - point.percentage = - pointStack.total && - (point.y / pointStack.total * 100); - point.stackY = yValue; - // Place the stack label - // in case of variwide series (where widths of points are - // different in most cases), stack labels are positioned - // wrongly, so the call of the setOffset is omited here and - // labels are correctly positioned later, at the end of the - // variwide's translate function (#10962) - if (!series.irregularWidths) { - pointStack.setOffset(series.pointXOffset || 0, series.barW || 0); - } - } - // Set translated yBottom or remove it - point.yBottom = defined(yBottom) ? - limitedRange(yAxis.translate(yBottom, 0, 1, 0, 1)) : - null; - // general hook, used for Highstock compare mode - if (hasModifyValue) { - yValue = series.modifyValue(yValue, point); - } - // Set the the plotY value, reset it for redraws - // #3201 - point.plotY = ((typeof yValue === 'number' && yValue !== Infinity) ? - limitedRange(yAxis.translate(yValue, 0, 1, 0, 1)) : - void 0); - point.isInside = this.isPointInside(point); - // Set client related positions for mouse tracking - point.clientX = dynamicallyPlaced ? - correctFloat(xAxis.translate(xValue, 0, 0, 0, 1, pointPlacement)) : - plotX; // #1514, #5383, #5518 - // Negative points. For bubble charts, this means negative z - // values (#9728) - point.negative = point[zoneAxis] < (options[zoneAxis + 'Threshold'] || - threshold || - 0); - // some API data - point.category = (categories && - typeof categories[point.x] !== 'undefined' ? - categories[point.x] : - point.x); - // Determine auto enabling of markers (#3635, #5099) - if (!point.isNull && point.visible !== false) { - if (typeof lastPlotX !== 'undefined') { - closestPointRangePx = Math.min(closestPointRangePx, Math.abs(plotX - lastPlotX)); - } - lastPlotX = plotX; - } - // Find point zone - point.zone = (this.zones.length && point.getZone()); - // Animate new points with data sorting - if (!point.graphic && series.group && enabledDataSorting) { - point.isNew = true; - } - } - series.closestPointRangePx = closestPointRangePx; - fireEvent(this, 'afterTranslate'); - }, + rotation: 270, /** - * Return the series points with null points filtered out. + * The actual text of the axis title. Horizontal texts can contain + * HTML, but rotated texts are painted using vector techniques and + * must be clean text. The Y axis title is disabled by setting the + * `text` option to `undefined`. + * + * @sample {highcharts} highcharts/xaxis/title-text/ + * Custom HTML + * + * @type {string|null} + * @default {highcharts} Values + * @default {highstock} undefined + * @product highcharts highstock gantt + */ + text: "Values", + }, + /** + * The top position of the Y axis. If it's a number, it is interpreted + * as pixel position relative to the chart. + * + * Since Highcharts 2: If it's a percentage string, it is interpreted as + * percentages of the plot height, offset from plot area top. + * + * @see [yAxis.height](#yAxis.height) + * + * @sample {highstock} stock/demo/candlestick-and-volume/ + * Percentage height panes + * + * @type {number|string} + * @product highcharts highstock + * @apioption yAxis.top + */ + /** + * The stack labels show the total value for each bar in a stacked + * column or bar chart. The label will be placed on top of positive + * columns and below negative columns. In case of an inverted column + * chart or a bar chart the label is placed to the right of positive + * bars and to the left of negative bars. + * + * @product highcharts + */ + stackLabels: { + /** + * Enable or disable the initial animation when a series is + * displayed for the `stackLabels`. The animation can also be set as + * a configuration object. Please note that this option only + * applies to the initial animation. + * For other animations, see [chart.animation](#chart.animation) + * and the animation parameter under the API methods. + * The following properties are supported: * - * @function Highcharts.Series#getValidPoints + * - `defer`: The animation delay time in milliseconds. * - * @param {Array<Highcharts.Point>} [points] - * The points to inspect, defaults to {@link Series.points}. + * @sample {highcharts} highcharts/plotoptions/animation-defer/ + * Animation defer settings + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + * @since 8.2.0 + * @apioption yAxis.stackLabels.animation + */ + animation: {}, + /** + * The animation delay time in milliseconds. + * Set to `0` renders stackLabel immediately. + * As `undefined` inherits defer time from the [series.animation.defer](#plotOptions.series.animation.defer). * - * @param {boolean} [insideOnly=false] - * Whether to inspect only the points that are inside the visible - * view. + * @type {number} + * @since 8.2.0 + * @apioption yAxis.stackLabels.animation.defer + */ + /** + * Allow the stack labels to overlap. * - * @param {boolean} [allowNull=false] - * Whether to allow null points to pass as valid points. + * @sample {highcharts} highcharts/yaxis/stacklabels-allowoverlap-false/ + * Default false * - * @return {Array<Highcharts.Point>} - * The valid points. + * @since 5.0.13 + * @product highcharts */ - getValidPoints: function (points, insideOnly, allowNull) { - var chart = this.chart; - // #3916, #5029, #5085 - return (points || this.points || []).filter(function isValidPoint(point) { - if (insideOnly && !chart.isInsidePlot(point.plotX, point.plotY, chart.inverted)) { - return false; - } - return point.visible !== false && - (allowNull || !point.isNull); - }); - }, + allowOverlap: false, /** - * Get the clipping for the series. Could be called for a series to - * initiate animating the clip or to set the final clip (only width - * and x). + * The background color or gradient for the stack label. * - * @private - * @function Highcharts.Series#getClip - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * Initialize the animation. - * @param {boolean} [finalBox] - * Final size for the clip - end state for the animation. - * @return {Highcharts.Dictionary<number>} + * @sample {highcharts} highcharts/yaxis/stacklabels-box/ + * Stack labels box options + * @type {Highcharts.ColorType} + * @since 8.1.0 + * @apioption yAxis.stackLabels.backgroundColor */ - getClipBox: function (animation, finalBox) { - var series = this, - options = series.options, - chart = series.chart, - inverted = chart.inverted, - xAxis = series.xAxis, - yAxis = xAxis && series.yAxis, - clipBox, - scrollablePlotAreaOptions = chart.options.chart.scrollablePlotArea || {}; - if (animation && options.clip === false && yAxis) { - // support for not clipped series animation (#10450) - clipBox = inverted ? { - y: -chart.chartWidth + yAxis.len + yAxis.pos, - height: chart.chartWidth, - width: chart.chartHeight, - x: -chart.chartHeight + xAxis.len + xAxis.pos - } : { - y: -yAxis.pos, - height: chart.chartHeight, - width: chart.chartWidth, - x: -xAxis.pos - }; - // x and width will be changed later when setting for animation - // initial state in Series.setClip - } - else { - clipBox = series.clipBox || chart.clipBox; - if (finalBox) { - clipBox.width = chart.plotSizeX; - clipBox.x = (chart.scrollablePixelsX || 0) * - (scrollablePlotAreaOptions.scrollPositionX || 0); - } - } - return !finalBox ? clipBox : { - width: clipBox.width, - x: clipBox.x - }; - }, /** - * Set the clipping for the series. For animated series it is called - * twice, first to initiate animating the clip then the second time - * without the animation to set the final clip. + * The border color for the stack label. Defaults to `undefined`. * - * @private - * @function Highcharts.Series#setClip - * @param {boolean|Highcharts.AnimationOptionsObject} [animation] + * @sample {highcharts} highcharts/yaxis/stacklabels-box/ + * Stack labels box options + * @type {Highcharts.ColorType} + * @since 8.1.0 + * @apioption yAxis.stackLabels.borderColor */ - setClip: function (animation) { - var chart = this.chart, options = this.options, renderer = chart.renderer, inverted = chart.inverted, seriesClipBox = this.clipBox, clipBox = this.getClipBox(animation), sharedClipKey = this.sharedClipKey || - [ - '_sharedClip', - animation && animation.duration, - animation && animation.easing, - clipBox.height, - options.xAxis, - options.yAxis - ].join(','), // #4526 - clipRect = chart[sharedClipKey], markerClipRect = chart[sharedClipKey + 'm']; - if (animation) { - clipBox.width = 0; - if (inverted) { - clipBox.x = chart.plotHeight + - (options.clip !== false ? 0 : chart.plotTop); - } - } - // If a clipping rectangle with the same properties is currently - // present in the chart, use that. - if (!clipRect) { - // When animation is set, prepare the initial positions - if (animation) { - chart[sharedClipKey + 'm'] = markerClipRect = - renderer.clipRect( - // include the width of the first marker - inverted ? chart.plotSizeX + 99 : -99, inverted ? -chart.plotLeft : -chart.plotTop, 99, inverted ? chart.chartWidth : chart.chartHeight); - } - chart[sharedClipKey] = clipRect = renderer.clipRect(clipBox); - // Create hashmap for series indexes - clipRect.count = { length: 0 }; - // When the series is rendered again before starting animating, in - // compliance to a responsive rule - } - else if (!chart.hasLoaded) { - clipRect.attr(clipBox); - } - if (animation) { - if (!clipRect.count[this.index]) { - clipRect.count[this.index] = true; - clipRect.count.length += 1; - } - } - if (options.clip !== false || animation) { - this.group.clip(animation || seriesClipBox ? clipRect : chart.clipRect); - this.markerGroup.clip(markerClipRect); - this.sharedClipKey = sharedClipKey; - } - // Remove the shared clipping rectangle when all series are shown - if (!animation) { - if (clipRect.count[this.index]) { - delete clipRect.count[this.index]; - clipRect.count.length -= 1; - } - if (clipRect.count.length === 0 && - sharedClipKey && - chart[sharedClipKey]) { - if (!seriesClipBox) { - chart[sharedClipKey] = - chart[sharedClipKey].destroy(); - } - if (chart[sharedClipKey + 'm']) { - chart[sharedClipKey + 'm'] = - chart[sharedClipKey + 'm'].destroy(); - } - } - } - }, /** - * Animate in the series. Called internally twice. First with the `init` - * parameter set to true, which sets up the initial state of the - * animation. Then when ready, it is called with the `init` parameter - * undefined, in order to perform the actual animation. After the - * second run, the function is removed. + * The border radius in pixels for the stack label. * - * @function Highcharts.Series#animate - * - * @param {boolean} [init] - * Initialize the animation. + * @sample {highcharts} highcharts/yaxis/stacklabels-box/ + * Stack labels box options + * @type {number} + * @default 0 + * @since 8.1.0 + * @apioption yAxis.stackLabels.borderRadius */ - animate: function (init) { - var series = this, - chart = series.chart, - animation = animObject(series.options.animation), - clipRect, - sharedClipKey, - finalBox; - // Initialize the animation. Set up the clipping rectangle. - if (!chart.hasRendered) { - if (init) { - series.setClip(animation); - // Run the animation - } - else { - sharedClipKey = this.sharedClipKey; - clipRect = chart[sharedClipKey]; - finalBox = series.getClipBox(animation, true); - if (clipRect) { - clipRect.animate(finalBox, animation); - } - if (chart[sharedClipKey + 'm']) { - chart[sharedClipKey + 'm'].animate({ - width: finalBox.width + 99, - x: finalBox.x - (chart.inverted ? 0 : 99) - }, animation); - } - } - } - }, /** - * This runs after animation to land on the final plot clipping. + * The border width in pixels for the stack label. * - * @private - * @function Highcharts.Series#afterAnimate - * @fires Highcharts.Series#event:afterAnimate + * @sample {highcharts} highcharts/yaxis/stacklabels-box/ + * Stack labels box options + * @type {number} + * @default 0 + * @since 8.1.0 + * @apioption yAxis.stackLabels.borderWidth */ - afterAnimate: function () { - this.setClip(); - fireEvent(this, 'afterAnimate'); - this.finishedAnimating = true; - }, /** - * Draw the markers for line-like series types, and columns or other - * graphical representation for {@link Point} objects for other series - * types. The resulting element is typically stored as - * {@link Point.graphic}, and is created on the first call and updated - * and moved on subsequent calls. + * Enable or disable the stack total labels. + * + * @sample {highcharts} highcharts/yaxis/stacklabels-enabled/ + * Enabled stack total labels + * @sample {highcharts} highcharts/yaxis/stacklabels-enabled-waterfall/ + * Enabled stack labels in waterfall chart * - * @function Highcharts.Series#drawPoints + * @since 2.1.5 + * @product highcharts */ - drawPoints: function () { - var series = this, - points = series.points, - chart = series.chart, - i, - point, - graphic, - verb, - options = series.options, - seriesMarkerOptions = options.marker, - pointMarkerOptions, - hasPointMarker, - markerGroup = (series[series.specialGroup] || - series.markerGroup), - xAxis = series.xAxis, - markerAttribs, - globallyEnabled = pick(seriesMarkerOptions.enabled, !xAxis || xAxis.isRadial ? true : null, - // Use larger or equal as radius is null in bubbles (#6321) - series.closestPointRangePx >= (seriesMarkerOptions.enabledThreshold * - seriesMarkerOptions.radius)); - if (seriesMarkerOptions.enabled !== false || - series._hasPointMarkers) { - for (i = 0; i < points.length; i++) { - point = points[i]; - graphic = point.graphic; - verb = graphic ? 'animate' : 'attr'; - pointMarkerOptions = point.marker || {}; - hasPointMarker = !!point.marker; - var shouldDrawMarker = ((globallyEnabled && - typeof pointMarkerOptions.enabled === 'undefined') || pointMarkerOptions.enabled) && !point.isNull && point.visible !== false; - // only draw the point if y is defined - if (shouldDrawMarker) { - // Shortcuts - var symbol = pick(pointMarkerOptions.symbol, - series.symbol); - markerAttribs = series.markerAttribs(point, (point.selected && 'select')); - // Set starting position for point sliding animation. - if (series.enabledDataSorting) { - point.startXPos = xAxis.reversed ? - -markerAttribs.width : - xAxis.width; - } - var isInside = point.isInside !== false; - if (graphic) { // update - // Since the marker group isn't clipped, each - // individual marker must be toggled - graphic[isInside ? 'show' : 'hide'](isInside) - .animate(markerAttribs); - } - else if (isInside && - (markerAttribs.width > 0 || point.hasImage)) { - /** - * The graphic representation of the point. - * Typically this is a simple shape, like a `rect` - * for column charts or `path` for line markers, but - * for some complex series types like boxplot or 3D - * charts, the graphic may be a `g` element - * containing other shapes. The graphic is generated - * the first time {@link Series#drawPoints} runs, - * and updated and moved on subsequent runs. - * - * @name Point#graphic - * @type {SVGElement} - */ - point.graphic = graphic = chart.renderer - .symbol(symbol, markerAttribs.x, markerAttribs.y, markerAttribs.width, markerAttribs.height, hasPointMarker ? - pointMarkerOptions : - seriesMarkerOptions) - .add(markerGroup); - // Sliding animation for new points - if (series.enabledDataSorting && - chart.hasRendered) { - graphic.attr({ - x: point.startXPos - }); - verb = 'animate'; - } - } - if (graphic && verb === 'animate') { // update - // Since the marker group isn't clipped, each - // individual marker must be toggled - graphic[isInside ? 'show' : 'hide'](isInside) - .animate(markerAttribs); - } - // Presentational attributes - if (graphic && !chart.styledMode) { - graphic[verb](series.pointAttribs(point, (point.selected && 'select'))); - } - if (graphic) { - graphic.addClass(point.getClassName(), true); - } - } - else if (graphic) { - point.graphic = graphic.destroy(); // #1269 - } - } - } - }, + enabled: false, /** - * Get non-presentational attributes for a point. Used internally for - * both styled mode and classic. Can be overridden for different series - * types. + * Whether to hide stack labels that are outside the plot area. + * By default, the stack label is moved + * inside the plot area according to the + * [overflow](/highcharts/#yAxis/stackLabels/overflow) + * option. * - * @see Series#pointAttribs + * @type {boolean} + * @since 7.1.3 + */ + crop: true, + /** + * How to handle stack total labels that flow outside the plot area. + * The default is set to `"justify"`, + * which aligns them inside the plot area. + * For columns and bars, this means it will be moved inside the bar. + * To display stack labels outside the plot area, + * set `crop` to `false` and `overflow` to `"allow"`. * - * @function Highcharts.Series#markerAttribs + * @sample highcharts/yaxis/stacklabels-overflow/ + * Stack labels flows outside the plot area. * - * @param {Highcharts.Point} point - * The Point to inspect. + * @type {Highcharts.DataLabelsOverflowValue} + * @since 7.1.3 + */ + overflow: "justify", + /* eslint-disable valid-jsdoc */ + /** + * Callback JavaScript function to format the label. The value is + * given by `this.total`. * - * @param {string} [state] - * The state, can be either `hover`, `select` or undefined. + * @sample {highcharts} highcharts/yaxis/stacklabels-formatter/ + * Added units to stack total value * - * @return {Highcharts.SVGAttributes} - * A hash containing those attributes that are not settable from - * CSS. + * @type {Highcharts.FormatterCallbackFunction<Highcharts.StackItemObject>} + * @since 2.1.5 + * @product highcharts */ - markerAttribs: function (point, state) { - var seriesOptions = this.options, - seriesMarkerOptions = seriesOptions.marker, - seriesStateOptions, - pointMarkerOptions = point.marker || {}, - symbol = (pointMarkerOptions.symbol || - seriesMarkerOptions.symbol), - pointStateOptions, - radius = pick(pointMarkerOptions.radius, - seriesMarkerOptions.radius), - attribs; - // Handle hover and select states - if (state) { - seriesStateOptions = seriesMarkerOptions.states[state]; - pointStateOptions = pointMarkerOptions.states && - pointMarkerOptions.states[state]; - radius = pick(pointStateOptions && pointStateOptions.radius, seriesStateOptions && seriesStateOptions.radius, radius + (seriesStateOptions && seriesStateOptions.radiusPlus || - 0)); - } - point.hasImage = symbol && symbol.indexOf('url') === 0; - if (point.hasImage) { - radius = 0; // and subsequently width and height is not set - } - attribs = { - // Math.floor for #1843: - x: seriesOptions.crisp ? - Math.floor(point.plotX) - radius : - point.plotX - radius, - y: point.plotY - radius - }; - if (radius) { - attribs.width = attribs.height = 2 * radius; - } - return attribs; + formatter: function () { + var numberFormatter = this.axis.chart.numberFormatter; + /* eslint-enable valid-jsdoc */ + return numberFormatter(this.total, -1); }, /** - * Internal function to get presentational attributes for each point. - * Unlike {@link Series#markerAttribs}, this function should return - * those attributes that can also be set in CSS. In styled mode, - * `pointAttribs` won't be called. - * - * @private - * @function Highcharts.Series#pointAttribs + * CSS styles for the label. * - * @param {Highcharts.Point} [point] - * The point instance to inspect. + * In styled mode, the styles are set in the + * `.highcharts-stack-label` class. * - * @param {string} [state] - * The point state, can be either `hover`, `select` or 'normal'. - * If undefined, normal state is assumed. + * @sample {highcharts} highcharts/yaxis/stacklabels-style/ + * Red stack total labels * - * @return {Highcharts.SVGAttributes} - * The presentational attributes to be set on the point. + * @type {Highcharts.CSSObject} + * @since 2.1.5 + * @product highcharts */ - pointAttribs: function (point, state) { - var seriesMarkerOptions = this.options.marker, - seriesStateOptions, - pointOptions = point && point.options, - pointMarkerOptions = ((pointOptions && pointOptions.marker) || {}), - pointStateOptions, - color = this.color, - pointColorOption = pointOptions && pointOptions.color, - pointColor = point && point.color, - strokeWidth = pick(pointMarkerOptions.lineWidth, - seriesMarkerOptions.lineWidth), - zoneColor = point && point.zone && point.zone.color, - fill, - stroke, - opacity = 1; - color = (pointColorOption || - zoneColor || - pointColor || - color); - fill = (pointMarkerOptions.fillColor || - seriesMarkerOptions.fillColor || - color); - stroke = (pointMarkerOptions.lineColor || - seriesMarkerOptions.lineColor || - color); - // Handle hover and select states - state = state || 'normal'; - if (state) { - seriesStateOptions = seriesMarkerOptions.states[state]; - pointStateOptions = (pointMarkerOptions.states && - pointMarkerOptions.states[state]) || {}; - strokeWidth = pick(pointStateOptions.lineWidth, seriesStateOptions.lineWidth, strokeWidth + pick(pointStateOptions.lineWidthPlus, seriesStateOptions.lineWidthPlus, 0)); - fill = (pointStateOptions.fillColor || - seriesStateOptions.fillColor || - fill); - stroke = (pointStateOptions.lineColor || - seriesStateOptions.lineColor || - stroke); - opacity = pick(pointStateOptions.opacity, seriesStateOptions.opacity, opacity); + style: { + /** @internal */ + color: "#000000", + /** @internal */ + fontSize: "11px", + /** @internal */ + fontWeight: "bold", + /** @internal */ + textOutline: "1px contrast", + }, + }, + gridLineWidth: 1, + lineWidth: 0, + // tickWidth: 0 + }; + /** + * The Z axis or depth axis for 3D plots. + * + * See the [Axis class](/class-reference/Highcharts.Axis) for programmatic + * access to the axis. + * + * @sample {highcharts} highcharts/3d/scatter-zaxis-categories/ + * Z-Axis with Categories + * @sample {highcharts} highcharts/3d/scatter-zaxis-grid/ + * Z-Axis with styling + * + * @type {*|Array<*>} + * @extends xAxis + * @since 5.0.0 + * @product highcharts + * @excluding breaks, crosshair, height, left, lineColor, lineWidth, + * nameToX, showEmpty, top, width + * @apioption zAxis + * + * @private + */ + // This variable extends the defaultOptions for left axes. + Axis.defaultLeftAxisOptions = { + labels: { + x: -15, + }, + title: { + rotation: 270, + }, + }; + // This variable extends the defaultOptions for right axes. + Axis.defaultRightAxisOptions = { + labels: { + x: 15, + }, + title: { + rotation: 90, + }, + }; + // This variable extends the defaultOptions for bottom axes. + Axis.defaultBottomAxisOptions = { + labels: { + autoRotation: [-45], + x: 0, + // overflow: undefined, + // staggerLines: null + }, + margin: 15, + title: { + rotation: 0, + }, + }; + // This variable extends the defaultOptions for top axes. + Axis.defaultTopAxisOptions = { + labels: { + autoRotation: [-45], + x: 0, + // overflow: undefined + // staggerLines: null + }, + margin: 15, + title: { + rotation: 0, + }, + }; + // Properties to survive after destroy, needed for Axis.update (#4317, + // #5773, #5881). + Axis.keepProps = [ + "extKey", + "hcEvents", + "names", + "series", + "userMax", + "userMin", + ]; + return Axis; + })(); + H.Axis = Axis; + + return H.Axis; + } + ); + _registerModule( + _modules, + "Core/Axis/DateTimeAxis.js", + [_modules["Core/Axis/Axis.js"], _modules["Core/Utilities.js"]], + function (Axis, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var addEvent = U.addEvent, + getMagnitude = U.getMagnitude, + normalizeTickInterval = U.normalizeTickInterval, + timeUnits = U.timeUnits; + /* eslint-disable valid-jsdoc */ + var DateTimeAxisAdditions = /** @class */ (function () { + /* * + * + * Constructors + * + * */ + function DateTimeAxisAdditions(axis) { + this.axis = axis; + } + /* * + * + * Functions + * + * */ + /** + * Get a normalized tick interval for dates. Returns a configuration object + * with unit range (interval), count and name. Used to prepare data for + * `getTimeTicks`. Previously this logic was part of getTimeTicks, but as + * `getTimeTicks` now runs of segments in stock charts, the normalizing + * logic was extracted in order to prevent it for running over again for + * each segment having the same interval. #662, #697. + * @private + */ + /** + * Get a normalized tick interval for dates. Returns a configuration object + * with unit range (interval), count and name. Used to prepare data for + * `getTimeTicks`. Previously this logic was part of getTimeTicks, but as + * `getTimeTicks` now runs of segments in stock charts, the normalizing + * logic was extracted in order to prevent it for running over again for + * each segment having the same interval. #662, #697. + * @private + */ + DateTimeAxisAdditions.prototype.normalizeTimeTickInterval = function ( + tickInterval, + unitsOption + ) { + var units = unitsOption || [ + [ + "millisecond", + [1, 2, 5, 10, 20, 25, 50, 100, 200, 500], // allowed multiples + ], + ["second", [1, 2, 5, 10, 15, 30]], + ["minute", [1, 2, 5, 10, 15, 30]], + ["hour", [1, 2, 3, 4, 6, 8, 12]], + ["day", [1, 2]], + ["week", [1, 2]], + ["month", [1, 2, 3, 4, 6]], + ["year", null], + ], + unit = units[units.length - 1], // default unit is years + interval = timeUnits[unit[0]], + multiples = unit[1], + count, + i; + // loop through the units to find the one that best fits the + // tickInterval + for (i = 0; i < units.length; i++) { + unit = units[i]; + interval = timeUnits[unit[0]]; + multiples = unit[1]; + if (units[i + 1]) { + // lessThan is in the middle between the highest multiple and + // the next unit. + var lessThan = + (interval * multiples[multiples.length - 1] + + timeUnits[units[i + 1][0]]) / + 2; + // break and keep the current unit + if (tickInterval <= lessThan) { + break; + } + } + } + // prevent 2.5 years intervals, though 25, 250 etc. are allowed + if (interval === timeUnits.year && tickInterval < 5 * interval) { + multiples = [1, 2, 5]; + } + // get the count + count = normalizeTickInterval( + tickInterval / interval, + multiples, + unit[0] === "year" // #1913, #2360 + ? Math.max(getMagnitude(tickInterval / interval), 1) + : 1 + ); + return { + unitRange: interval, + count: count, + unitName: unit[0], + }; + }; + return DateTimeAxisAdditions; + })(); + /** + * Date and time support for axes. + * + * @private + * @class + */ + var DateTimeAxis = /** @class */ (function () { + function DateTimeAxis() {} + /* * + * + * Static Functions + * + * */ + /** + * Extends axis class with date and time support. + * @private + */ + DateTimeAxis.compose = function (AxisClass) { + AxisClass.keepProps.push("dateTime"); + var axisProto = AxisClass.prototype; + /** + * Set the tick positions to a time unit that makes sense, for example + * on the first of each month or on every Monday. Return an array with + * the time positions. Used in datetime axes as well as for grouping + * data on a datetime axis. + * + * @private + * @function Highcharts.Axis#getTimeTicks + * + * @param {Highcharts.TimeNormalizeObject} normalizedInterval + * The interval in axis values (ms) and thecount. + * + * @param {number} min + * The minimum in axis values. + * + * @param {number} max + * The maximum in axis values. + * + * @param {number} startOfWeek + * + * @return {Highcharts.AxisTickPositionsArray} + */ + axisProto.getTimeTicks = function () { + return this.chart.time.getTimeTicks.apply( + this.chart.time, + arguments + ); + }; + /* eslint-disable no-invalid-this */ + addEvent(AxisClass, "init", function (e) { + var axis = this; + var options = e.userOptions; + if (options.type !== "datetime") { + axis.dateTime = void 0; + return; + } + if (!axis.dateTime) { + axis.dateTime = new DateTimeAxisAdditions(axis); + } + }); + /* eslint-enable no-invalid-this */ + }; + /* * + * + * Static Properties + * + * */ + DateTimeAxis.AdditionsClass = DateTimeAxisAdditions; + return DateTimeAxis; + })(); + DateTimeAxis.compose(Axis); + + return DateTimeAxis; + } + ); + _registerModule( + _modules, + "Core/Axis/LogarithmicAxis.js", + [_modules["Core/Axis/Axis.js"], _modules["Core/Utilities.js"]], + function (Axis, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var addEvent = U.addEvent, + getMagnitude = U.getMagnitude, + normalizeTickInterval = U.normalizeTickInterval, + pick = U.pick; + /* eslint-disable valid-jsdoc */ + /** + * Provides logarithmic support for axes. + * + * @private + * @class + */ + var LogarithmicAxisAdditions = /** @class */ (function () { + /* * + * + * Constructors + * + * */ + function LogarithmicAxisAdditions(axis) { + this.axis = axis; + } + /* * + * + * Functions + * + * */ + /** + * Set the tick positions of a logarithmic axis. + */ + LogarithmicAxisAdditions.prototype.getLogTickPositions = function ( + interval, + min, + max, + minor + ) { + var log = this; + var axis = log.axis; + var axisLength = axis.len; + var options = axis.options; + // Since we use this method for both major and minor ticks, + // use a local variable and return the result + var positions = []; + // Reset + if (!minor) { + log.minorAutoInterval = void 0; + } + // First case: All ticks fall on whole logarithms: 1, 10, 100 etc. + if (interval >= 0.5) { + interval = Math.round(interval); + positions = axis.getLinearTickPositions(interval, min, max); + // Second case: We need intermediary ticks. For example + // 1, 2, 4, 6, 8, 10, 20, 40 etc. + } else if (interval >= 0.08) { + var roundedMin = Math.floor(min), + intermediate, + i, + j, + len, + pos, + lastPos, + break2; + if (interval > 0.3) { + intermediate = [1, 2, 4]; + // 0.2 equals five minor ticks per 1, 10, 100 etc + } else if (interval > 0.15) { + intermediate = [1, 2, 4, 6, 8]; + } else { + // 0.1 equals ten minor ticks per 1, 10, 100 etc + intermediate = [1, 2, 3, 4, 5, 6, 7, 8, 9]; + } + for (i = roundedMin; i < max + 1 && !break2; i++) { + len = intermediate.length; + for (j = 0; j < len && !break2; j++) { + pos = log.log2lin(log.lin2log(i) * intermediate[j]); + // #1670, lastPos is #3113 + if ( + pos > min && + (!minor || lastPos <= max) && + typeof lastPos !== "undefined" + ) { + positions.push(lastPos); } - return { - 'stroke': stroke, - 'stroke-width': strokeWidth, - 'fill': fill, - 'opacity': opacity - }; + if (lastPos > max) { + break2 = true; + } + lastPos = pos; + } + } + // Third case: We are so deep in between whole logarithmic values that + // we might as well handle the tick positions like a linear axis. For + // example 1.01, 1.02, 1.03, 1.04. + } else { + var realMin = log.lin2log(min), + realMax = log.lin2log(max), + tickIntervalOption = minor + ? axis.getMinorTickInterval() + : options.tickInterval, + filteredTickIntervalOption = + tickIntervalOption === "auto" ? null : tickIntervalOption, + tickPixelIntervalOption = + options.tickPixelInterval / (minor ? 5 : 1), + totalPixelLength = minor + ? axisLength / axis.tickPositions.length + : axisLength; + interval = pick( + filteredTickIntervalOption, + log.minorAutoInterval, + ((realMax - realMin) * tickPixelIntervalOption) / + (totalPixelLength || 1) + ); + interval = normalizeTickInterval( + interval, + void 0, + getMagnitude(interval) + ); + positions = axis + .getLinearTickPositions(interval, realMin, realMax) + .map(log.log2lin); + if (!minor) { + log.minorAutoInterval = interval / 5; + } + } + // Set the axis-level tickInterval variable + if (!minor) { + axis.tickInterval = interval; + } + return positions; + }; + LogarithmicAxisAdditions.prototype.lin2log = function (num) { + return Math.pow(10, num); + }; + LogarithmicAxisAdditions.prototype.log2lin = function (num) { + return Math.log(num) / Math.LN10; + }; + return LogarithmicAxisAdditions; + })(); + var LogarithmicAxis = /** @class */ (function () { + function LogarithmicAxis() {} + /** + * Provides logarithmic support for axes. + * + * @private + */ + LogarithmicAxis.compose = function (AxisClass) { + AxisClass.keepProps.push("logarithmic"); + // HC <= 8 backwards compatibility, allow wrapping + // Axis.prototype.lin2log and log2lin + // @todo Remove this in next major + var axisProto = AxisClass.prototype; + var logAxisProto = LogarithmicAxisAdditions.prototype; + axisProto.log2lin = logAxisProto.log2lin; + axisProto.lin2log = logAxisProto.lin2log; + /* eslint-disable no-invalid-this */ + addEvent(AxisClass, "init", function (e) { + var axis = this; + var options = e.userOptions; + var logarithmic = axis.logarithmic; + if (options.type !== "logarithmic") { + axis.logarithmic = void 0; + } else { + if (!logarithmic) { + logarithmic = axis.logarithmic = new LogarithmicAxisAdditions( + axis + ); + } + // HC <= 8 backwards compatibility, allow wrapping + // Axis.prototype.lin2log and log2lin + // @todo Remove this in next major + if (axis.log2lin !== logarithmic.log2lin) { + logarithmic.log2lin = axis.log2lin.bind(axis); + } + if (axis.lin2log !== logarithmic.lin2log) { + logarithmic.lin2log = axis.lin2log.bind(axis); + } + } + }); + addEvent(AxisClass, "afterInit", function () { + var axis = this; + var log = axis.logarithmic; + // extend logarithmic axis + if (log) { + axis.lin2val = function (num) { + return log.lin2log(num); + }; + axis.val2lin = function (num) { + return log.log2lin(num); + }; + } + }); + }; + return LogarithmicAxis; + })(); + LogarithmicAxis.compose(Axis); // @todo move to factory functions + + return LogarithmicAxis; + } + ); + _registerModule( + _modules, + "Core/Axis/PlotLineOrBand.js", + [ + _modules["Core/Axis/Axis.js"], + _modules["Core/Globals.js"], + _modules["Core/Utilities.js"], + ], + function (Axis, H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + /** + * Options for plot bands on axes. + * + * @typedef {Highcharts.XAxisPlotBandsOptions|Highcharts.YAxisPlotBandsOptions|Highcharts.ZAxisPlotBandsOptions} Highcharts.AxisPlotBandsOptions + */ + /** + * Options for plot band labels on axes. + * + * @typedef {Highcharts.XAxisPlotBandsLabelOptions|Highcharts.YAxisPlotBandsLabelOptions|Highcharts.ZAxisPlotBandsLabelOptions} Highcharts.AxisPlotBandsLabelOptions + */ + /** + * Options for plot lines on axes. + * + * @typedef {Highcharts.XAxisPlotLinesOptions|Highcharts.YAxisPlotLinesOptions|Highcharts.ZAxisPlotLinesOptions} Highcharts.AxisPlotLinesOptions + */ + /** + * Options for plot line labels on axes. + * + * @typedef {Highcharts.XAxisPlotLinesLabelOptions|Highcharts.YAxisPlotLinesLabelOptions|Highcharts.ZAxisPlotLinesLabelOptions} Highcharts.AxisPlotLinesLabelOptions + */ + var arrayMax = U.arrayMax, + arrayMin = U.arrayMin, + defined = U.defined, + destroyObjectProperties = U.destroyObjectProperties, + erase = U.erase, + extend = U.extend, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick; + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The object wrapper for plot lines and plot bands + * + * @class + * @name Highcharts.PlotLineOrBand + * + * @param {Highcharts.Axis} axis + * + * @param {Highcharts.AxisPlotLinesOptions|Highcharts.AxisPlotBandsOptions} [options] + */ + var PlotLineOrBand = /** @class */ (function () { + function PlotLineOrBand(axis, options) { + this.axis = axis; + if (options) { + this.options = options; + this.id = options.id; + } + } + /** + * Render the plot line or plot band. If it is already existing, + * move it. + * + * @private + * @function Highcharts.PlotLineOrBand#render + * @return {Highcharts.PlotLineOrBand|undefined} + */ + PlotLineOrBand.prototype.render = function () { + H.fireEvent(this, "render"); + var plotLine = this, + axis = plotLine.axis, + horiz = axis.horiz, + log = axis.logarithmic, + options = plotLine.options, + optionsLabel = options.label, + label = plotLine.label, + to = options.to, + from = options.from, + value = options.value, + isBand = defined(from) && defined(to), + isLine = defined(value), + svgElem = plotLine.svgElem, + isNew = !svgElem, + path = [], + color = options.color, + zIndex = pick(options.zIndex, 0), + events = options.events, + attribs = { + class: + "highcharts-plot-" + + (isBand ? "band " : "line ") + + (options.className || ""), + }, + groupAttribs = {}, + renderer = axis.chart.renderer, + groupName = isBand ? "bands" : "lines", + group; + // logarithmic conversion + if (log) { + from = log.log2lin(from); + to = log.log2lin(to); + value = log.log2lin(value); + } + // Set the presentational attributes + if (!axis.chart.styledMode) { + if (isLine) { + attribs.stroke = color || "#999999"; + attribs["stroke-width"] = pick(options.width, 1); + if (options.dashStyle) { + attribs.dashstyle = options.dashStyle; + } + } else if (isBand) { + // plot band + attribs.fill = color || "#e6ebf5"; + if (options.borderWidth) { + attribs.stroke = options.borderColor; + attribs["stroke-width"] = options.borderWidth; + } + } + } + // Grouping and zIndex + groupAttribs.zIndex = zIndex; + groupName += "-" + zIndex; + group = axis.plotLinesAndBandsGroups[groupName]; + if (!group) { + axis.plotLinesAndBandsGroups[groupName] = group = renderer + .g("plot-" + groupName) + .attr(groupAttribs) + .add(); + } + // Create the path + if (isNew) { + /** + * SVG element of the plot line or band. + * + * @name Highcharts.PlotLineOrBand#svgElement + * @type {Highcharts.SVGElement} + */ + plotLine.svgElem = svgElem = renderer + .path() + .attr(attribs) + .add(group); + } + // Set the path or return + if (isLine) { + path = axis.getPlotLinePath({ + value: value, + lineWidth: svgElem.strokeWidth(), + acrossPanes: options.acrossPanes, + }); + } else if (isBand) { + // plot band + path = axis.getPlotBandPath(from, to, options); + } else { + return; + } + // common for lines and bands + // Add events only if they were not added before. + if (!plotLine.eventsAdded && events) { + objectEach(events, function (event, eventType) { + svgElem.on(eventType, function (e) { + events[eventType].apply(plotLine, [e]); + }); + }); + plotLine.eventsAdded = true; + } + if ((isNew || !svgElem.d) && path && path.length) { + svgElem.attr({ d: path }); + } else if (svgElem) { + if (path) { + svgElem.show(true); + svgElem.animate({ d: path }); + } else if (svgElem.d) { + svgElem.hide(); + if (label) { + plotLine.label = label = label.destroy(); + } + } + } + // the plot band/line label + if ( + optionsLabel && + (defined(optionsLabel.text) || defined(optionsLabel.formatter)) && + path && + path.length && + axis.width > 0 && + axis.height > 0 && + !path.isFlat + ) { + // apply defaults + optionsLabel = merge( + { + align: horiz && isBand && "center", + x: horiz ? !isBand && 4 : 10, + verticalAlign: !horiz && isBand && "middle", + y: horiz ? (isBand ? 16 : 10) : isBand ? 6 : -4, + rotation: horiz && !isBand && 90, + }, + optionsLabel + ); + this.renderLabel(optionsLabel, path, isBand, zIndex); + } else if (label) { + // move out of sight + label.hide(); + } + // chainable + return plotLine; + }; + /** + * Render and align label for plot line or band. + * + * @private + * @function Highcharts.PlotLineOrBand#renderLabel + * @param {Highcharts.AxisPlotLinesLabelOptions|Highcharts.AxisPlotBandsLabelOptions} optionsLabel + * @param {Highcharts.SVGPathArray} path + * @param {boolean} [isBand] + * @param {number} [zIndex] + * @return {void} + */ + PlotLineOrBand.prototype.renderLabel = function ( + optionsLabel, + path, + isBand, + zIndex + ) { + var plotLine = this, + label = plotLine.label, + renderer = plotLine.axis.chart.renderer, + attribs, + xBounds, + yBounds, + x, + y, + labelText; + // add the SVG element + if (!label) { + attribs = { + align: optionsLabel.textAlign || optionsLabel.align, + rotation: optionsLabel.rotation, + class: + "highcharts-plot-" + + (isBand ? "band" : "line") + + "-label " + + (optionsLabel.className || ""), + }; + attribs.zIndex = zIndex; + labelText = this.getLabelText(optionsLabel); + /** + * SVG element of the label. + * + * @name Highcharts.PlotLineOrBand#label + * @type {Highcharts.SVGElement} + */ + plotLine.label = label = renderer + .text(labelText, 0, 0, optionsLabel.useHTML) + .attr(attribs) + .add(); + if (!this.axis.chart.styledMode) { + label.css(optionsLabel.style); + } + } + // get the bounding box and align the label + // #3000 changed to better handle choice between plotband or plotline + xBounds = path.xBounds || [ + path[0][1], + path[1][1], + isBand ? path[2][1] : path[0][1], + ]; + yBounds = path.yBounds || [ + path[0][2], + path[1][2], + isBand ? path[2][2] : path[0][2], + ]; + x = arrayMin(xBounds); + y = arrayMin(yBounds); + label.align(optionsLabel, false, { + x: x, + y: y, + width: arrayMax(xBounds) - x, + height: arrayMax(yBounds) - y, + }); + label.show(true); + }; + /** + * Get label's text content. + * + * @private + * @function Highcharts.PlotLineOrBand#getLabelText + * @param {Highcharts.AxisPlotLinesLabelOptions|Highcharts.AxisPlotBandsLabelOptions} optionsLabel + * @return {string} + */ + PlotLineOrBand.prototype.getLabelText = function (optionsLabel) { + return defined(optionsLabel.formatter) + ? optionsLabel.formatter.call(this) + : optionsLabel.text; + }; + /** + * Remove the plot line or band. + * + * @function Highcharts.PlotLineOrBand#destroy + * @return {void} + */ + PlotLineOrBand.prototype.destroy = function () { + // remove it from the lookup + erase(this.axis.plotLinesAndBands, this); + delete this.axis; + destroyObjectProperties(this); + }; + return PlotLineOrBand; + })(); + /* eslint-enable no-invalid-this, valid-jsdoc */ + // Object with members for extending the Axis prototype + extend( + Axis.prototype, + /** @lends Highcharts.Axis.prototype */ { + /** + * An array of colored bands stretching across the plot area marking an + * interval on the axis. + * + * In styled mode, the plot bands are styled by the `.highcharts-plot-band` + * class in addition to the `className` option. + * + * @productdesc {highcharts} + * In a gauge, a plot band on the Y axis (value axis) will stretch along the + * perimeter of the gauge. + * + * @type {Array<*>} + * @product highcharts highstock gantt + * @apioption xAxis.plotBands + */ + /** + * Flag to decide if plotBand should be rendered across all panes. + * + * @since 7.1.2 + * @product highstock + * @type {boolean} + * @default true + * @apioption xAxis.plotBands.acrossPanes + */ + /** + * Border color for the plot band. Also requires `borderWidth` to be set. + * + * @type {Highcharts.ColorString} + * @apioption xAxis.plotBands.borderColor + */ + /** + * Border width for the plot band. Also requires `borderColor` to be set. + * + * @type {number} + * @default 0 + * @apioption xAxis.plotBands.borderWidth + */ + /** + * A custom class name, in addition to the default `highcharts-plot-band`, + * to apply to each individual band. + * + * @type {string} + * @since 5.0.0 + * @apioption xAxis.plotBands.className + */ + /** + * The color of the plot band. + * + * @sample {highcharts} highcharts/xaxis/plotbands-color/ + * Color band + * @sample {highstock} stock/xaxis/plotbands/ + * Plot band on Y axis + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @default #e6ebf5 + * @apioption xAxis.plotBands.color + */ + /** + * An object defining mouse events for the plot band. Supported properties + * are `click`, `mouseover`, `mouseout`, `mousemove`. + * + * @sample {highcharts} highcharts/xaxis/plotbands-events/ + * Mouse events demonstrated + * + * @since 1.2 + * @apioption xAxis.plotBands.events + */ + /** + * Click event on a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotBands.events.click + */ + /** + * Mouse move event on a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotBands.events.mousemove + */ + /** + * Mouse out event on the corner of a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotBands.events.mouseout + */ + /** + * Mouse over event on a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotBands.events.mouseover + */ + /** + * The start position of the plot band in axis units. + * + * @sample {highcharts} highcharts/xaxis/plotbands-color/ + * Datetime axis + * @sample {highcharts} highcharts/xaxis/plotbands-from/ + * Categorized axis + * @sample {highstock} stock/xaxis/plotbands/ + * Plot band on Y axis + * + * @type {number} + * @apioption xAxis.plotBands.from + */ + /** + * An id used for identifying the plot band in Axis.removePlotBand. + * + * @sample {highcharts} highcharts/xaxis/plotbands-id/ + * Remove plot band by id + * @sample {highstock} highcharts/xaxis/plotbands-id/ + * Remove plot band by id + * + * @type {string} + * @apioption xAxis.plotBands.id + */ + /** + * The end position of the plot band in axis units. + * + * @sample {highcharts} highcharts/xaxis/plotbands-color/ + * Datetime axis + * @sample {highcharts} highcharts/xaxis/plotbands-from/ + * Categorized axis + * @sample {highstock} stock/xaxis/plotbands/ + * Plot band on Y axis + * + * @type {number} + * @apioption xAxis.plotBands.to + */ + /** + * The z index of the plot band within the chart, relative to other + * elements. Using the same z index as another element may give + * unpredictable results, as the last rendered element will be on top. + * Values from 0 to 20 make sense. + * + * @sample {highcharts} highcharts/xaxis/plotbands-color/ + * Behind plot lines by default + * @sample {highcharts} highcharts/xaxis/plotbands-zindex/ + * Above plot lines + * @sample {highcharts} highcharts/xaxis/plotbands-zindex-above-series/ + * Above plot lines and series + * + * @type {number} + * @since 1.2 + * @apioption xAxis.plotBands.zIndex + */ + /** + * Text labels for the plot bands + * + * @product highcharts highstock gantt + * @apioption xAxis.plotBands.label + */ + /** + * Horizontal alignment of the label. Can be one of "left", "center" or + * "right". + * + * @sample {highcharts} highcharts/xaxis/plotbands-label-align/ + * Aligned to the right + * @sample {highstock} stock/xaxis/plotbands-label/ + * Plot band with labels + * + * @type {Highcharts.AlignValue} + * @default center + * @since 2.1 + * @apioption xAxis.plotBands.label.align + */ + /** + * Rotation of the text label in degrees . + * + * @sample {highcharts} highcharts/xaxis/plotbands-label-rotation/ + * Vertical text + * + * @type {number} + * @default 0 + * @since 2.1 + * @apioption xAxis.plotBands.label.rotation + */ + /** + * CSS styles for the text label. + * + * In styled mode, the labels are styled by the + * `.highcharts-plot-band-label` class. + * + * @sample {highcharts} highcharts/xaxis/plotbands-label-style/ + * Blue and bold label + * + * @type {Highcharts.CSSObject} + * @since 2.1 + * @apioption xAxis.plotBands.label.style + */ + /** + * The string text itself. A subset of HTML is supported. + * + * @type {string} + * @since 2.1 + * @apioption xAxis.plotBands.label.text + */ + /** + * The text alignment for the label. While `align` determines where the + * texts anchor point is placed within the plot band, `textAlign` determines + * how the text is aligned against its anchor point. Possible values are + * "left", "center" and "right". Defaults to the same as the `align` option. + * + * @sample {highcharts} highcharts/xaxis/plotbands-label-rotation/ + * Vertical text in center position but text-aligned left + * + * @type {Highcharts.AlignValue} + * @since 2.1 + * @apioption xAxis.plotBands.label.textAlign + */ + /** + * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the labels. + * + * @type {boolean} + * @default false + * @since 3.0.3 + * @apioption xAxis.plotBands.label.useHTML + */ + /** + * Vertical alignment of the label relative to the plot band. Can be one of + * "top", "middle" or "bottom". + * + * @sample {highcharts} highcharts/xaxis/plotbands-label-verticalalign/ + * Vertically centered label + * @sample {highstock} stock/xaxis/plotbands-label/ + * Plot band with labels + * + * @type {Highcharts.VerticalAlignValue} + * @default top + * @since 2.1 + * @apioption xAxis.plotBands.label.verticalAlign + */ + /** + * Horizontal position relative the alignment. Default varies by + * orientation. + * + * @sample {highcharts} highcharts/xaxis/plotbands-label-align/ + * Aligned 10px from the right edge + * @sample {highstock} stock/xaxis/plotbands-label/ + * Plot band with labels + * + * @type {number} + * @since 2.1 + * @apioption xAxis.plotBands.label.x + */ + /** + * Vertical position of the text baseline relative to the alignment. Default + * varies by orientation. + * + * @sample {highcharts} highcharts/xaxis/plotbands-label-y/ + * Label on x axis + * @sample {highstock} stock/xaxis/plotbands-label/ + * Plot band with labels + * + * @type {number} + * @since 2.1 + * @apioption xAxis.plotBands.label.y + */ + /** + * An array of lines stretching across the plot area, marking a specific + * value on one of the axes. + * + * In styled mode, the plot lines are styled by the + * `.highcharts-plot-line` class in addition to the `className` option. + * + * @type {Array<*>} + * @product highcharts highstock gantt + * @sample {highcharts} highcharts/xaxis/plotlines-color/ + * Basic plot line + * @sample {highcharts} highcharts/series-solidgauge/labels-auto-aligned/ + * Solid gauge plot line + * @apioption xAxis.plotLines + */ + /** + * Flag to decide if plotLine should be rendered across all panes. + * + * @sample {highstock} stock/xaxis/plotlines-acrosspanes/ + * Plot lines on different panes + * + * @since 7.1.2 + * @product highstock + * @type {boolean} + * @default true + * @apioption xAxis.plotLines.acrossPanes + */ + /** + * A custom class name, in addition to the default `highcharts-plot-line`, + * to apply to each individual line. + * + * @type {string} + * @since 5.0.0 + * @apioption xAxis.plotLines.className + */ + /** + * The color of the line. + * + * @sample {highcharts} highcharts/xaxis/plotlines-color/ + * A red line from X axis + * @sample {highstock} stock/xaxis/plotlines/ + * Plot line on Y axis + * + * @type {Highcharts.ColorString} + * @default #999999 + * @apioption xAxis.plotLines.color + */ + /** + * The dashing or dot style for the plot line. For possible values see + * [this overview](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/). + * + * @sample {highcharts} highcharts/xaxis/plotlines-dashstyle/ + * Dash and dot pattern + * @sample {highstock} stock/xaxis/plotlines/ + * Plot line on Y axis + * + * @type {Highcharts.DashStyleValue} + * @default Solid + * @since 1.2 + * @apioption xAxis.plotLines.dashStyle + */ + /** + * An object defining mouse events for the plot line. Supported + * properties are `click`, `mouseover`, `mouseout`, `mousemove`. + * + * @sample {highcharts} highcharts/xaxis/plotlines-events/ + * Mouse events demonstrated + * + * @since 1.2 + * @apioption xAxis.plotLines.events + */ + /** + * Click event on a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotLines.events.click + */ + /** + * Mouse move event on a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotLines.events.mousemove + */ + /** + * Mouse out event on the corner of a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotLines.events.mouseout + */ + /** + * Mouse over event on a plot band. + * + * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotLines.events.mouseover + */ + /** + * An id used for identifying the plot line in Axis.removePlotLine. + * + * @sample {highcharts} highcharts/xaxis/plotlines-id/ + * Remove plot line by id + * + * @type {string} + * @apioption xAxis.plotLines.id + */ + /** + * The position of the line in axis units. + * + * @sample {highcharts} highcharts/xaxis/plotlines-color/ + * Between two categories on X axis + * @sample {highstock} stock/xaxis/plotlines/ + * Plot line on Y axis + * + * @type {number} + * @apioption xAxis.plotLines.value + */ + /** + * The width or thickness of the plot line. + * + * @sample {highcharts} highcharts/xaxis/plotlines-color/ + * 2px wide line from X axis + * @sample {highstock} stock/xaxis/plotlines/ + * Plot line on Y axis + * + * @type {number} + * @default 2 + * @apioption xAxis.plotLines.width + */ + /** + * The z index of the plot line within the chart. + * + * @sample {highcharts} highcharts/xaxis/plotlines-zindex-behind/ + * Behind plot lines by default + * @sample {highcharts} highcharts/xaxis/plotlines-zindex-above/ + * Above plot lines + * @sample {highcharts} highcharts/xaxis/plotlines-zindex-above-all/ + * Above plot lines and series + * + * @type {number} + * @since 1.2 + * @apioption xAxis.plotLines.zIndex + */ + /** + * Text labels for the plot bands + * + * @apioption xAxis.plotLines.label + */ + /** + * Horizontal alignment of the label. Can be one of "left", "center" or + * "right". + * + * @sample {highcharts} highcharts/xaxis/plotlines-label-align-right/ + * Aligned to the right + * @sample {highstock} stock/xaxis/plotlines/ + * Plot line on Y axis + * + * @type {Highcharts.AlignValue} + * @default left + * @since 2.1 + * @apioption xAxis.plotLines.label.align + */ + /** + * Callback JavaScript function to format the label. Useful properties like + * the value of plot line or the range of plot band (`from` & `to` + * properties) can be found in `this.options` object. + * + * @sample {highcharts} highcharts/xaxis/plotlines-plotbands-label-formatter + * Label formatters for plot line and plot band. + * @type {Highcharts.FormatterCallbackFunction<Highcharts.PlotLineOrBand>} + * @apioption xAxis.plotLines.label.formatter + */ + /** + * Rotation of the text label in degrees. Defaults to 0 for horizontal plot + * lines and 90 for vertical lines. + * + * @sample {highcharts} highcharts/xaxis/plotlines-label-verticalalign-middle/ + * Slanted text + * + * @type {number} + * @since 2.1 + * @apioption xAxis.plotLines.label.rotation + */ + /** + * CSS styles for the text label. + * + * In styled mode, the labels are styled by the + * `.highcharts-plot-line-label` class. + * + * @sample {highcharts} highcharts/xaxis/plotlines-label-style/ + * Blue and bold label + * + * @type {Highcharts.CSSObject} + * @since 2.1 + * @apioption xAxis.plotLines.label.style + */ + /** + * The text itself. A subset of HTML is supported. + * + * @type {string} + * @since 2.1 + * @apioption xAxis.plotLines.label.text + */ + /** + * The text alignment for the label. While `align` determines where the + * texts anchor point is placed within the plot band, `textAlign` determines + * how the text is aligned against its anchor point. Possible values are + * "left", "center" and "right". Defaults to the same as the `align` option. + * + * @sample {highcharts} highcharts/xaxis/plotlines-label-textalign/ + * Text label in bottom position + * + * @type {Highcharts.AlignValue} + * @since 2.1 + * @apioption xAxis.plotLines.label.textAlign + */ + /** + * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the labels. + * + * @type {boolean} + * @default false + * @since 3.0.3 + * @apioption xAxis.plotLines.label.useHTML + */ + /** + * Vertical alignment of the label relative to the plot line. Can be + * one of "top", "middle" or "bottom". + * + * @sample {highcharts} highcharts/xaxis/plotlines-label-verticalalign-middle/ + * Vertically centered label + * + * @type {Highcharts.VerticalAlignValue} + * @default {highcharts} top + * @default {highstock} top + * @since 2.1 + * @apioption xAxis.plotLines.label.verticalAlign + */ + /** + * Horizontal position relative the alignment. Default varies by + * orientation. + * + * @sample {highcharts} highcharts/xaxis/plotlines-label-align-right/ + * Aligned 10px from the right edge + * @sample {highstock} stock/xaxis/plotlines/ + * Plot line on Y axis + * + * @type {number} + * @since 2.1 + * @apioption xAxis.plotLines.label.x + */ + /** + * Vertical position of the text baseline relative to the alignment. Default + * varies by orientation. + * + * @sample {highcharts} highcharts/xaxis/plotlines-label-y/ + * Label below the plot line + * @sample {highstock} stock/xaxis/plotlines/ + * Plot line on Y axis + * + * @type {number} + * @since 2.1 + * @apioption xAxis.plotLines.label.y + */ + /** + * + * @type {Array<*>} + * @extends xAxis.plotBands + * @apioption yAxis.plotBands + */ + /** + * In a gauge chart, this option determines the inner radius of the + * plot band that stretches along the perimeter. It can be given as + * a percentage string, like `"100%"`, or as a pixel number, like `100`. + * By default, the inner radius is controlled by the [thickness]( + * #yAxis.plotBands.thickness) option. + * + * @sample {highcharts} highcharts/xaxis/plotbands-gauge + * Gauge plot band + * + * @type {number|string} + * @since 2.3 + * @product highcharts + * @apioption yAxis.plotBands.innerRadius + */ + /** + * In a gauge chart, this option determines the outer radius of the + * plot band that stretches along the perimeter. It can be given as + * a percentage string, like `"100%"`, or as a pixel number, like `100`. + * + * @sample {highcharts} highcharts/xaxis/plotbands-gauge + * Gauge plot band + * + * @type {number|string} + * @default 100% + * @since 2.3 + * @product highcharts + * @apioption yAxis.plotBands.outerRadius + */ + /** + * In a gauge chart, this option sets the width of the plot band + * stretching along the perimeter. It can be given as a percentage + * string, like `"10%"`, or as a pixel number, like `10`. The default + * value 10 is the same as the default [tickLength](#yAxis.tickLength), + * thus making the plot band act as a background for the tick markers. + * + * @sample {highcharts} highcharts/xaxis/plotbands-gauge + * Gauge plot band + * + * @type {number|string} + * @default 10 + * @since 2.3 + * @product highcharts + * @apioption yAxis.plotBands.thickness + */ + /** + * @type {Array<*>} + * @extends xAxis.plotLines + * @apioption yAxis.plotLines + */ + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * Internal function to create the SVG path definition for a plot band. + * + * @function Highcharts.Axis#getPlotBandPath + * + * @param {number} from + * The axis value to start from. + * + * @param {number} to + * The axis value to end on. + * + * @param {Highcharts.AxisPlotBandsOptions|Highcharts.AxisPlotLinesOptions} options + * The plotBand or plotLine configuration object. + * + * @return {Highcharts.SVGPathArray} + * The SVG path definition in array form. + */ + getPlotBandPath: function (from, to, options) { + if (options === void 0) { + options = this.options; + } + var toPath = this.getPlotLinePath({ + value: to, + force: true, + acrossPanes: options.acrossPanes, + }), + path = this.getPlotLinePath({ + value: from, + force: true, + acrossPanes: options.acrossPanes, + }), + result = [], + i, + // #4964 check if chart is inverted or plotband is on yAxis + horiz = this.horiz, + plus = 1, + isFlat, + outside = + (from < this.min && to < this.min) || + (from > this.max && to > this.max); + if (path && toPath) { + // Flat paths don't need labels (#3836) + if (outside) { + isFlat = path.toString() === toPath.toString(); + plus = 0; + } + // Go over each subpath - for panes in Highstock + for (i = 0; i < path.length; i += 2) { + var pathStart = path[i], + pathEnd = path[i + 1], + toPathStart = toPath[i], + toPathEnd = toPath[i + 1]; + // Type checking all affected path segments. Consider something + // smarter. + if ( + (pathStart[0] === "M" || pathStart[0] === "L") && + (pathEnd[0] === "M" || pathEnd[0] === "L") && + (toPathStart[0] === "M" || toPathStart[0] === "L") && + (toPathEnd[0] === "M" || toPathEnd[0] === "L") + ) { + // Add 1 pixel when coordinates are the same + if (horiz && toPathStart[1] === pathStart[1]) { + toPathStart[1] += plus; + toPathEnd[1] += plus; + } else if (!horiz && toPathStart[2] === pathStart[2]) { + toPathStart[2] += plus; + toPathEnd[2] += plus; + } + result.push( + ["M", pathStart[1], pathStart[2]], + ["L", pathEnd[1], pathEnd[2]], + ["L", toPathEnd[1], toPathEnd[2]], + ["L", toPathStart[1], toPathStart[2]], + ["Z"] + ); + } + result.isFlat = isFlat; + } + } else { + // outside the axis area + path = null; + } + return result; + }, + /** + * Add a plot band after render time. + * + * @sample highcharts/members/axis-addplotband/ + * Toggle the plot band from a button + * + * @function Highcharts.Axis#addPlotBand + * + * @param {Highcharts.AxisPlotBandsOptions} options + * A configuration object for the plot band, as defined in + * [xAxis.plotBands](https://api.highcharts.com/highcharts/xAxis.plotBands). + * + * @return {Highcharts.PlotLineOrBand|undefined} + * The added plot band. + */ + addPlotBand: function (options) { + return this.addPlotBandOrLine(options, "plotBands"); + }, + /** + * Add a plot line after render time. + * + * @sample highcharts/members/axis-addplotline/ + * Toggle the plot line from a button + * + * @function Highcharts.Axis#addPlotLine + * + * @param {Highcharts.AxisPlotLinesOptions} options + * A configuration object for the plot line, as defined in + * [xAxis.plotLines](https://api.highcharts.com/highcharts/xAxis.plotLines). + * + * @return {Highcharts.PlotLineOrBand|undefined} + * The added plot line. + */ + addPlotLine: function (options) { + return this.addPlotBandOrLine(options, "plotLines"); + }, + /** + * Add a plot band or plot line after render time. Called from addPlotBand + * and addPlotLine internally. + * + * @private + * @function Highcharts.Axis#addPlotBandOrLine + * + * @param {Highcharts.AxisPlotBandsOptions|Highcharts.AxisPlotLinesOptions} options + * The plotBand or plotLine configuration object. + * + * @param {"plotBands"|"plotLines"} [coll] + * + * @return {Highcharts.PlotLineOrBand|undefined} + */ + addPlotBandOrLine: function (options, coll) { + var obj = new H.PlotLineOrBand(this, options), + userOptions = this.userOptions; + if (this.visible) { + obj = obj.render(); + } + if (obj) { + // #2189 + // Add it to the user options for exporting and Axis.update + if (coll) { + // Workaround Microsoft/TypeScript issue #32693 + var updatedOptions = userOptions[coll] || []; + updatedOptions.push(options); + userOptions[coll] = updatedOptions; + } + this.plotLinesAndBands.push(obj); + this._addedPlotLB = true; + } + return obj; + }, + /** + * Remove a plot band or plot line from the chart by id. Called internally + * from `removePlotBand` and `removePlotLine`. + * + * @private + * @function Highcharts.Axis#removePlotBandOrLine + * @param {string} id + * @return {void} + */ + removePlotBandOrLine: function (id) { + var plotLinesAndBands = this.plotLinesAndBands, + options = this.options, + userOptions = this.userOptions, + i = plotLinesAndBands.length; + while (i--) { + if (plotLinesAndBands[i].id === id) { + plotLinesAndBands[i].destroy(); + } + } + [ + options.plotLines || [], + userOptions.plotLines || [], + options.plotBands || [], + userOptions.plotBands || [], + ].forEach(function (arr) { + i = arr.length; + while (i--) { + if ((arr[i] || {}).id === id) { + erase(arr, arr[i]); + } + } + }); + }, + /** + * Remove a plot band by its id. + * + * @sample highcharts/members/axis-removeplotband/ + * Remove plot band by id + * @sample highcharts/members/axis-addplotband/ + * Toggle the plot band from a button + * + * @function Highcharts.Axis#removePlotBand + * + * @param {string} id + * The plot band's `id` as given in the original configuration + * object or in the `addPlotBand` option. + * + * @return {void} + */ + removePlotBand: function (id) { + this.removePlotBandOrLine(id); + }, + /** + * Remove a plot line by its id. + * + * @sample highcharts/xaxis/plotlines-id/ + * Remove plot line by id + * @sample highcharts/members/axis-addplotline/ + * Toggle the plot line from a button + * + * @function Highcharts.Axis#removePlotLine + * + * @param {string} id + * The plot line's `id` as given in the original configuration + * object or in the `addPlotLine` option. + */ + removePlotLine: function (id) { + this.removePlotBandOrLine(id); + }, + } + ); + H.PlotLineOrBand = PlotLineOrBand; + + return H.PlotLineOrBand; + } + ); + _registerModule( + _modules, + "Core/Tooltip.js", + [_modules["Core/Globals.js"], _modules["Core/Utilities.js"]], + function (H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var doc = H.doc; + var clamp = U.clamp, + css = U.css, + defined = U.defined, + discardElement = U.discardElement, + extend = U.extend, + fireEvent = U.fireEvent, + format = U.format, + isNumber = U.isNumber, + isString = U.isString, + merge = U.merge, + pick = U.pick, + splat = U.splat, + syncTimeout = U.syncTimeout, + timeUnits = U.timeUnits; + /** + * Callback function to format the text of the tooltip from scratch. + * + * In case of single or shared tooltips, a string should be be returned. In case + * of splitted tooltips, it should return an array where the first item is the + * header, and subsequent items are mapped to the points. Return `false` to + * disable tooltip for a specific point on series. + * + * @callback Highcharts.TooltipFormatterCallbackFunction + * + * @param {Highcharts.TooltipFormatterContextObject} this + * Context to format + * + * @param {Highcharts.Tooltip} tooltip + * The tooltip instance + * + * @return {false|string|Array<(string|null|undefined)>|null|undefined} + * Formatted text or false + */ + /** + * @interface Highcharts.TooltipFormatterContextObject + */ /** + * @name Highcharts.TooltipFormatterContextObject#color + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + */ /** + * @name Highcharts.TooltipFormatterContextObject#colorIndex + * @type {number|undefined} + */ /** + * @name Highcharts.TooltipFormatterContextObject#key + * @type {number} + */ /** + * @name Highcharts.TooltipFormatterContextObject#percentage + * @type {number|undefined} + */ /** + * @name Highcharts.TooltipFormatterContextObject#point + * @type {Highcharts.Point} + */ /** + * @name Highcharts.TooltipFormatterContextObject#points + * @type {Array<Highcharts.TooltipFormatterContextObject>|undefined} + */ /** + * @name Highcharts.TooltipFormatterContextObject#series + * @type {Highcharts.Series} + */ /** + * @name Highcharts.TooltipFormatterContextObject#total + * @type {number|undefined} + */ /** + * @name Highcharts.TooltipFormatterContextObject#x + * @type {number} + */ /** + * @name Highcharts.TooltipFormatterContextObject#y + * @type {number} + */ + /** + * A callback function to place the tooltip in a specific position. + * + * @callback Highcharts.TooltipPositionerCallbackFunction + * + * @param {Highcharts.Tooltip} this + * Tooltip context of the callback. + * + * @param {number} labelWidth + * Width of the tooltip. + * + * @param {number} labelHeight + * Height of the tooltip. + * + * @param {Highcharts.TooltipPositionerPointObject} point + * Point information for positioning a tooltip. + * + * @return {Highcharts.PositionObject} + * New position for the tooltip. + */ + /** + * Point information for positioning a tooltip. + * + * @interface Highcharts.TooltipPositionerPointObject + * @extends Highcharts.Point + */ /** + * If `tooltip.split` option is enabled and positioner is called for each of the + * boxes separately, this property indicates the call on the xAxis header, which + * is not a point itself. + * @name Highcharts.TooltipPositionerPointObject#isHeader + * @type {boolean} + */ /** + * The reference point relative to the plot area. Add chart.plotLeft to get the + * full coordinates. + * @name Highcharts.TooltipPositionerPointObject#plotX + * @type {number} + */ /** + * The reference point relative to the plot area. Add chart.plotTop to get the + * full coordinates. + * @name Highcharts.TooltipPositionerPointObject#plotY + * @type {number} + */ + /** + * @typedef {"callout"|"circle"|"square"} Highcharts.TooltipShapeValue + */ + (""); // separates doclets above from variables below + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * Tooltip of a chart. + * + * @class + * @name Highcharts.Tooltip + * + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @param {Highcharts.TooltipOptions} options + * Tooltip options. + */ + var Tooltip = /** @class */ (function () { + /* * + * + * Constructors + * + * */ + function Tooltip(chart, options) { + this.container = void 0; + this.crosshairs = []; + this.distance = 0; + this.isHidden = true; + this.isSticky = false; + this.now = {}; + this.options = {}; + this.outside = false; + this.chart = chart; + this.init(chart, options); + } + /* * + * + * Functions + * + * */ + /** + * In styled mode, apply the default filter for the tooltip drop-shadow. It + * needs to have an id specific to the chart, otherwise there will be issues + * when one tooltip adopts the filter of a different chart, specifically one + * where the container is hidden. + * + * @private + * @function Highcharts.Tooltip#applyFilter + */ + Tooltip.prototype.applyFilter = function () { + var chart = this.chart; + chart.renderer.definition({ + tagName: "filter", + id: "drop-shadow-" + chart.index, + opacity: 0.5, + children: [ + { + tagName: "feGaussianBlur", + in: "SourceAlpha", + stdDeviation: 1, + }, + { + tagName: "feOffset", + dx: 1, + dy: 1, + }, + { + tagName: "feComponentTransfer", + children: [ + { + tagName: "feFuncA", + type: "linear", + slope: 0.3, + }, + ], + }, + { + tagName: "feMerge", + children: [ + { + tagName: "feMergeNode", + }, + { + tagName: "feMergeNode", + in: "SourceGraphic", + }, + ], + }, + ], + }); + chart.renderer.definition({ + tagName: "style", + textContent: + ".highcharts-tooltip-" + + chart.index + + "{" + + "filter:url(#drop-shadow-" + + chart.index + + ")" + + "}", + }); + }; + /** + * Build the body (lines) of the tooltip by iterating over the items and + * returning one entry for each item, abstracting this functionality allows + * to easily overwrite and extend it. + * + * @private + * @function Highcharts.Tooltip#bodyFormatter + * @param {Array<(Highcharts.Point|Highcharts.Series)>} items + * @return {Array<string>} + */ + Tooltip.prototype.bodyFormatter = function (items) { + return items.map(function (item) { + var tooltipOptions = item.series.tooltipOptions; + return ( + tooltipOptions[ + (item.point.formatPrefix || "point") + "Formatter" + ] || item.point.tooltipFormatter + ).call( + item.point, + tooltipOptions[(item.point.formatPrefix || "point") + "Format"] || + "" + ); + }); + }; + /** + * Destroy the single tooltips in a split tooltip. + * If the tooltip is active then it is not destroyed, unless forced to. + * + * @private + * @function Highcharts.Tooltip#cleanSplit + * + * @param {boolean} [force] + * Force destroy all tooltips. + */ + Tooltip.prototype.cleanSplit = function (force) { + this.chart.series.forEach(function (series) { + var tt = series && series.tt; + if (tt) { + if (!tt.isActive || force) { + series.tt = tt.destroy(); + } else { + tt.isActive = false; + } + } + }); + }; + /** + * In case no user defined formatter is given, this will be used. Note that + * the context here is an object holding point, series, x, y etc. + * + * @function Highcharts.Tooltip#defaultFormatter + * + * @param {Highcharts.Tooltip} tooltip + * + * @return {Array<string>} + */ + Tooltip.prototype.defaultFormatter = function (tooltip) { + var items = this.points || splat(this), + s; + // Build the header + s = [tooltip.tooltipFooterHeaderFormatter(items[0])]; + // build the values + s = s.concat(tooltip.bodyFormatter(items)); + // footer + s.push(tooltip.tooltipFooterHeaderFormatter(items[0], true)); + return s; + }; + /** + * Removes and destroys the tooltip and its elements. + * + * @function Highcharts.Tooltip#destroy + */ + Tooltip.prototype.destroy = function () { + // Destroy and clear local variables + if (this.label) { + this.label = this.label.destroy(); + } + if (this.split && this.tt) { + this.cleanSplit(this.chart, true); + this.tt = this.tt.destroy(); + } + if (this.renderer) { + this.renderer = this.renderer.destroy(); + discardElement(this.container); + } + U.clearTimeout(this.hideTimer); + U.clearTimeout(this.tooltipTimeout); + }; + /** + * Extendable method to get the anchor position of the tooltip + * from a point or set of points + * + * @private + * @function Highcharts.Tooltip#getAnchor + * + * @param {Highcharts.Point|Array<Highcharts.Point>} points + * + * @param {Highcharts.PointerEventObject} [mouseEvent] + * + * @return {Array<number>} + */ + Tooltip.prototype.getAnchor = function (points, mouseEvent) { + var ret, + chart = this.chart, + pointer = chart.pointer, + inverted = chart.inverted, + plotTop = chart.plotTop, + plotLeft = chart.plotLeft, + plotX = 0, + plotY = 0, + yAxis, + xAxis; + points = splat(points); + // When tooltip follows mouse, relate the position to the mouse + if (this.followPointer && mouseEvent) { + if (typeof mouseEvent.chartX === "undefined") { + mouseEvent = pointer.normalize(mouseEvent); + } + ret = [mouseEvent.chartX - plotLeft, mouseEvent.chartY - plotTop]; + // Some series types use a specificly calculated tooltip position for + // each point + } else if (points[0].tooltipPos) { + ret = points[0].tooltipPos; + // When shared, use the average position + } else { + points.forEach(function (point) { + yAxis = point.series.yAxis; + xAxis = point.series.xAxis; + plotX += + point.plotX + (!inverted && xAxis ? xAxis.left - plotLeft : 0); + plotY += + (point.plotLow + ? (point.plotLow + point.plotHigh) / 2 + : point.plotY) + + (!inverted && yAxis ? yAxis.top - plotTop : 0); // #1151 + }); + plotX /= points.length; + plotY /= points.length; + ret = [ + inverted ? chart.plotWidth - plotY : plotX, + this.shared && !inverted && points.length > 1 && mouseEvent + ? // place shared tooltip next to the mouse (#424) + mouseEvent.chartY - plotTop + : inverted + ? chart.plotHeight - plotX + : plotY, + ]; + } + return ret.map(Math.round); + }; + /** + * Get the optimal date format for a point, based on a range. + * + * @private + * @function Highcharts.Tooltip#getDateFormat + * + * @param {number} range + * The time range + * + * @param {number} date + * The date of the point in question + * + * @param {number} startOfWeek + * An integer representing the first day of the week, where 0 is + * Sunday. + * + * @param {Highcharts.Dictionary<string>} dateTimeLabelFormats + * A map of time units to formats. + * + * @return {string} + * The optimal date format for a point. + */ + Tooltip.prototype.getDateFormat = function ( + range, + date, + startOfWeek, + dateTimeLabelFormats + ) { + var time = this.chart.time, + dateStr = time.dateFormat("%m-%d %H:%M:%S.%L", date), + format, + n, + blank = "01-01 00:00:00.000", + strpos = { + millisecond: 15, + second: 12, + minute: 9, + hour: 6, + day: 3, }, - /** - * Clear DOM objects and free up memory. - * - * @private - * @function Highcharts.Series#destroy - * @param {boolean} [keepEventsForUpdate] - * @return {void} - * @fires Highcharts.Series#event:destroy - */ - destroy: function (keepEventsForUpdate) { - var series = this, - chart = series.chart, - issue134 = /AppleWebKit\/533/.test(win.navigator.userAgent), - destroy, - i, - data = series.data || [], - point, - axis; - // add event hook - fireEvent(series, 'destroy'); - // remove events - this.removeEvents(keepEventsForUpdate); - // erase from axes - (series.axisTypes || []).forEach(function (AXIS) { - axis = series[AXIS]; - if (axis && axis.series) { - erase(axis.series, series); - axis.isDirty = axis.forceRedraw = true; - } - }); - // remove legend items - if (series.legendItem) { - series.chart.legend.destroyItem(series); - } - // destroy all points with their elements - i = data.length; - while (i--) { - point = data[i]; - if (point && point.destroy) { - point.destroy(); - } - } - series.points = null; - // Clear the animation timeout if we are destroying the series - // during initial animation - U.clearTimeout(series.animationTimeout); - // Destroy all SVGElements associated to the series - objectEach(series, function (val, prop) { - // Survive provides a hook for not destroying - if (val instanceof SVGElement && !val.survive) { - // issue 134 workaround - destroy = issue134 && prop === 'group' ? - 'hide' : - 'destroy'; - val[destroy](); - } - }); - // remove from hoverSeries - if (chart.hoverSeries === series) { - chart.hoverSeries = null; - } - erase(chart.series, series); - chart.orderSeries(); - // clear all members - objectEach(series, function (val, prop) { - if (!keepEventsForUpdate || prop !== 'hcEvents') { - delete series[prop]; - } - }); + lastN = "millisecond"; // for sub-millisecond data, #4223 + for (n in timeUnits) { + // eslint-disable-line guard-for-in + // If the range is exactly one week and we're looking at a + // Sunday/Monday, go for the week format + if ( + range === timeUnits.week && + +time.dateFormat("%w", date) === startOfWeek && + dateStr.substr(6) === blank.substr(6) + ) { + n = "week"; + break; + } + // The first format that is too great for the range + if (timeUnits[n] > range) { + n = lastN; + break; + } + // If the point is placed every day at 23:59, we need to show + // the minutes as well. #2637. + if ( + strpos[n] && + dateStr.substr(strpos[n]) !== blank.substr(strpos[n]) + ) { + break; + } + // Weeks are outside the hierarchy, only apply them on + // Mondays/Sundays like in the first condition + if (n !== "week") { + lastN = n; + } + } + if (n) { + format = time.resolveDTLFormat(dateTimeLabelFormats[n]).main; + } + return format; + }; + /** + * Creates the Tooltip label element if it does not exist, then returns it. + * + * @function Highcharts.Tooltip#getLabel + * @return {Highcharts.SVGElement} + */ + Tooltip.prototype.getLabel = function () { + var _a, _b; + var tooltip = this, + renderer = this.chart.renderer, + styledMode = this.chart.styledMode, + options = this.options, + className = + "tooltip" + + (defined(options.className) ? " " + options.className : ""), + pointerEvents = + ((_a = options.style) === null || _a === void 0 + ? void 0 + : _a.pointerEvents) || + (!this.followPointer && options.stickOnContact ? "auto" : "none"), + container, + set, + onMouseEnter = function () { + tooltip.inContact = true; }, - /** - * Get the graph path. - * - * @private - * @function Highcharts.Series#getGraphPath - * @param {Array<Highcharts.Point>} points - * @param {boolean} [nullsAsZeroes] - * @param {boolean} [connectCliffs] - * @return {Highcharts.SVGPathArray} - */ - getGraphPath: function (points, nullsAsZeroes, connectCliffs) { - var series = this, - options = series.options, - step = options.step, - reversed, - graphPath = [], - xMap = [], - gap; - points = points || series.points; - // Bottom of a stack is reversed - reversed = points.reversed; - if (reversed) { - points.reverse(); - } - // Reverse the steps (#5004) - step = { - right: 1, - center: 2 - }[step] || (step && 3); - if (step && reversed) { - step = 4 - step; - } - // Remove invalid points, especially in spline (#5015) - points = this.getValidPoints(points, false, !(options.connectNulls && !nullsAsZeroes && !connectCliffs)); - // Build the line - points.forEach(function (point, i) { - var plotX = point.plotX, - plotY = point.plotY, - lastPoint = points[i - 1], - // the path to this point from the previous - pathToPoint; - if ((point.leftCliff || (lastPoint && lastPoint.rightCliff)) && - !connectCliffs) { - gap = true; // ... and continue - } - // Line series, nullsAsZeroes is not handled - if (point.isNull && !defined(nullsAsZeroes) && i > 0) { - gap = !options.connectNulls; - // Area series, nullsAsZeroes is set - } - else if (point.isNull && !nullsAsZeroes) { - gap = true; - } - else { - if (i === 0 || gap) { - pathToPoint = [[ - 'M', - point.plotX, - point.plotY - ]]; - // Generate the spline as defined in the SplineSeries object - } - else if (series.getPointSpline) { - pathToPoint = [series.getPointSpline(points, point, i)]; - } - else if (step) { - if (step === 1) { // right - pathToPoint = [[ - 'L', - lastPoint.plotX, - plotY - ]]; - } - else if (step === 2) { // center - pathToPoint = [[ - 'L', - (lastPoint.plotX + plotX) / 2, - lastPoint.plotY - ], [ - 'L', - (lastPoint.plotX + plotX) / 2, - plotY - ]]; - } - else { - pathToPoint = [[ - 'L', - plotX, - lastPoint.plotY - ]]; - } - pathToPoint.push([ - 'L', - plotX, - plotY - ]); - } - else { - // normal line to next point - pathToPoint = [[ - 'L', - plotX, - plotY - ]]; - } - // Prepare for animation. When step is enabled, there are - // two path nodes for each x value. - xMap.push(point.x); - if (step) { - xMap.push(point.x); - if (step === 2) { // step = center (#8073) - xMap.push(point.x); - } - } - graphPath.push.apply(graphPath, pathToPoint); - gap = false; - } + onMouseLeave = function () { + var series = tooltip.chart.hoverSeries; + tooltip.inContact = false; + if (series && series.onMouseOut) { + series.onMouseOut(); + } + }; + if (!this.label) { + if (this.outside) { + /** + * Reference to the tooltip's container, when + * [Highcharts.Tooltip#outside] is set to true, otherwise + * it's undefined. + * + * @name Highcharts.Tooltip#container + * @type {Highcharts.HTMLDOMElement|undefined} + */ + this.container = container = H.doc.createElement("div"); + container.className = "highcharts-tooltip-container"; + css(container, { + position: "absolute", + top: "1px", + pointerEvents: pointerEvents, + zIndex: 3, + }); + H.doc.body.appendChild(container); + /** + * Reference to the tooltip's renderer, when + * [Highcharts.Tooltip#outside] is set to true, otherwise + * it's undefined. + * + * @name Highcharts.Tooltip#renderer + * @type {Highcharts.SVGRenderer|undefined} + */ + this.renderer = renderer = new H.Renderer( + container, + 0, + 0, + (_b = this.chart.options.chart) === null || _b === void 0 + ? void 0 + : _b.style, + void 0, + void 0, + renderer.styledMode + ); + } + // Create the label + if (this.split) { + this.label = renderer.g(className); + } else { + this.label = renderer + .label( + "", + 0, + 0, + options.shape || "callout", + null, + null, + options.useHTML, + null, + className + ) + .attr({ + padding: options.padding, + r: options.borderRadius, }); - graphPath.xMap = xMap; - series.graphPath = graphPath; - return graphPath; + if (!styledMode) { + this.label + .attr({ + fill: options.backgroundColor, + "stroke-width": options.borderWidth, + }) + // #2301, #2657 + .css(options.style) + .css({ pointerEvents: pointerEvents }) + .shadow(options.shadow); + } + } + if (styledMode) { + // Apply the drop-shadow filter + this.applyFilter(); + this.label.addClass("highcharts-tooltip-" + this.chart.index); + } + // Split tooltip use updateTooltipContainer to position the tooltip + // container. + if (tooltip.outside && !tooltip.split) { + var label_1 = this.label; + var xSetter_1 = label_1.xSetter, + ySetter_1 = label_1.ySetter; + label_1.xSetter = function (value) { + xSetter_1.call(label_1, tooltip.distance); + container.style.left = value + "px"; + }; + label_1.ySetter = function (value) { + ySetter_1.call(label_1, tooltip.distance); + container.style.top = value + "px"; + }; + } + this.label + .on("mouseenter", onMouseEnter) + .on("mouseleave", onMouseLeave) + .attr({ zIndex: 8 }) + .add(); + } + return this.label; + }; + /** + * Place the tooltip in a chart without spilling over + * and not covering the point it self. + * + * @private + * @function Highcharts.Tooltip#getPosition + * + * @param {number} boxWidth + * + * @param {number} boxHeight + * + * @param {Highcharts.Point} point + * + * @return {Highcharts.PositionObject} + */ + Tooltip.prototype.getPosition = function (boxWidth, boxHeight, point) { + var chart = this.chart, + distance = this.distance, + ret = {}, + // Don't use h if chart isn't inverted (#7242) ??? + h = (chart.inverted && point.h) || 0, // #4117 ??? + swapped, + outside = this.outside, + outerWidth = outside + ? // substract distance to prevent scrollbars + doc.documentElement.clientWidth - 2 * distance + : chart.chartWidth, + outerHeight = outside + ? Math.max( + doc.body.scrollHeight, + doc.documentElement.scrollHeight, + doc.body.offsetHeight, + doc.documentElement.offsetHeight, + doc.documentElement.clientHeight + ) + : chart.chartHeight, + chartPosition = chart.pointer.getChartPosition(), + containerScaling = chart.containerScaling, + scaleX = function (val) { + return ( + // eslint-disable-line no-confusing-arrow + containerScaling ? val * containerScaling.scaleX : val + ); }, - /** - * Draw the graph. Called internally when rendering line-like series - * types. The first time it generates the `series.graph` item and - * optionally other series-wide items like `series.area` for area - * charts. On subsequent calls these items are updated with new - * positions and attributes. - * - * @function Highcharts.Series#drawGraph - */ - drawGraph: function () { - var series = this, - options = this.options, - graphPath = (this.gappedPath || this.getGraphPath).call(this), - styledMode = this.chart.styledMode, - props = [[ - 'graph', - 'highcharts-graph' - ]]; - // Presentational properties - if (!styledMode) { - props[0].push((options.lineColor || - this.color || - '#cccccc' // when colorByPoint = true - ), options.dashStyle); - } - props = series.getZonesGraphs(props); - // Draw the graph - props.forEach(function (prop, i) { - var graphKey = prop[0], - graph = series[graphKey], - verb = graph ? 'animate' : 'attr', - attribs; - if (graph) { - graph.endX = series.preventGraphAnimation ? - null : - graphPath.xMap; - graph.animate({ d: graphPath }); - } - else if (graphPath.length) { // #1487 - /** - * SVG element of area-based charts. Can be used for styling - * purposes. If zones are configured, this element will be - * hidden and replaced by multiple zone areas, accessible - * via `series['zone-area-x']` (where x is a number, - * starting with 0). - * - * @name Highcharts.Series#area - * @type {Highcharts.SVGElement|undefined} - */ - /** - * SVG element of line-based charts. Can be used for styling - * purposes. If zones are configured, this element will be - * hidden and replaced by multiple zone lines, accessible - * via `series['zone-graph-x']` (where x is a number, - * starting with 0). - * - * @name Highcharts.Series#graph - * @type {Highcharts.SVGElement|undefined} - */ - series[graphKey] = graph = series.chart.renderer - .path(graphPath) - .addClass(prop[1]) - .attr({ zIndex: 1 }) // #1069 - .add(series.group); - } - if (graph && !styledMode) { - attribs = { - 'stroke': prop[2], - 'stroke-width': options.lineWidth, - // Polygon series use filled graph - 'fill': (series.fillGraph && series.color) || 'none' - }; - if (prop[3]) { - attribs.dashstyle = prop[3]; - } - else if (options.linecap !== 'square') { - attribs['stroke-linecap'] = - attribs['stroke-linejoin'] = 'round'; - } - graph[verb](attribs) - // Add shadow to normal series (0) or to first - // zone (1) #3932 - .shadow((i < 2) && options.shadow); - } - // Helpers for animation - if (graph) { - graph.startX = graphPath.xMap; - graph.isArea = graphPath.isArea; // For arearange animation - } - }); + scaleY = function (val) { + return ( + // eslint-disable-line no-confusing-arrow + containerScaling ? val * containerScaling.scaleY : val + ); }, - /** - * Get zones properties for building graphs. Extendable by series with - * multiple lines within one series. - * - * @private - * @function Highcharts.Series#getZonesGraphs - * - * @param {Array<Array<string>>} props - * - * @return {Array<Array<string>>} - */ - getZonesGraphs: function (props) { - // Add the zone properties if any - this.zones.forEach(function (zone, i) { - var propset = [ - 'zone-graph-' + i, - 'highcharts-graph highcharts-zone-graph-' + i + ' ' + - (zone.className || '') - ]; - if (!this.chart.styledMode) { - propset.push((zone.color || this.color), (zone.dashStyle || this.options.dashStyle)); - } - props.push(propset); - }, this); - return props; + // Build parameter arrays for firstDimension()/secondDimension() + buildDimensionArray = function (dim) { + var isX = dim === "x"; + return [ + dim, + isX ? outerWidth : outerHeight, + isX ? boxWidth : boxHeight, + ].concat( + outside + ? [ + // If we are using tooltip.outside, we need to scale the + // position to match scaling of the container in case there + // is a transform/zoom on the container. #11329 + isX ? scaleX(boxWidth) : scaleY(boxHeight), + isX + ? chartPosition.left - + distance + + scaleX(point.plotX + chart.plotLeft) + : chartPosition.top - + distance + + scaleY(point.plotY + chart.plotTop), + 0, + isX ? outerWidth : outerHeight, + ] + : [ + // Not outside, no scaling is needed + isX ? boxWidth : boxHeight, + isX + ? point.plotX + chart.plotLeft + : point.plotY + chart.plotTop, + isX ? chart.plotLeft : chart.plotTop, + isX + ? chart.plotLeft + chart.plotWidth + : chart.plotTop + chart.plotHeight, + ] + ); }, - /** - * Clip the graphs into zones for colors and styling. - * - * @private - * @function Highcharts.Series#applyZones - * @return {void} - */ - applyZones: function () { - var series = this, - chart = this.chart, - renderer = chart.renderer, - zones = this.zones, - translatedFrom, - translatedTo, - clips = (this.clips || []), - clipAttr, - graph = this.graph, - area = this.area, - chartSizeMax = Math.max(chart.chartWidth, - chart.chartHeight), - axis = this[(this.zoneAxis || 'y') + 'Axis'], - extremes, - reversed, - inverted = chart.inverted, - horiz, - pxRange, - pxPosMin, - pxPosMax, - ignoreZones = false, - zoneArea, - zoneGraph; - if (zones.length && - (graph || area) && - axis && - typeof axis.min !== 'undefined') { - reversed = axis.reversed; - horiz = axis.horiz; - // The use of the Color Threshold assumes there are no gaps - // so it is safe to hide the original graph and area - // unless it is not waterfall series, then use showLine property - // to set lines between columns to be visible (#7862) - if (graph && !this.showLine) { - graph.hide(); - } - if (area) { - area.hide(); - } - // Create the clips - extremes = axis.getExtremes(); - zones.forEach(function (threshold, i) { - translatedFrom = reversed ? - (horiz ? chart.plotWidth : 0) : - (horiz ? 0 : (axis.toPixels(extremes.min) || 0)); - translatedFrom = clamp(pick(translatedTo, translatedFrom), 0, chartSizeMax); - translatedTo = clamp(Math.round(axis.toPixels(pick(threshold.value, extremes.max), true) || 0), 0, chartSizeMax); - if (ignoreZones) { - translatedFrom = translatedTo = - axis.toPixels(extremes.max); - } - pxRange = Math.abs(translatedFrom - translatedTo); - pxPosMin = Math.min(translatedFrom, translatedTo); - pxPosMax = Math.max(translatedFrom, translatedTo); - if (axis.isXAxis) { - clipAttr = { - x: inverted ? pxPosMax : pxPosMin, - y: 0, - width: pxRange, - height: chartSizeMax - }; - if (!horiz) { - clipAttr.x = chart.plotHeight - clipAttr.x; - } - } - else { - clipAttr = { - x: 0, - y: inverted ? pxPosMax : pxPosMin, - width: chartSizeMax, - height: pxRange - }; - if (horiz) { - clipAttr.y = chart.plotWidth - clipAttr.y; - } - } - // VML SUPPPORT - if (inverted && renderer.isVML) { - if (axis.isXAxis) { - clipAttr = { - x: 0, - y: reversed ? pxPosMin : pxPosMax, - height: clipAttr.width, - width: chart.chartWidth - }; - } - else { - clipAttr = { - x: (clipAttr.y - - chart.plotLeft - - chart.spacingBox.x), - y: 0, - width: clipAttr.height, - height: chart.chartHeight - }; - } - } - // END OF VML SUPPORT - if (clips[i]) { - clips[i].animate(clipAttr); - } - else { - clips[i] = renderer.clipRect(clipAttr); - } - // when no data, graph zone is not applied and after setData - // clip was ignored. As a result, it should be applied each - // time. - zoneArea = series['zone-area-' + i]; - zoneGraph = series['zone-graph-' + i]; - if (graph && zoneGraph) { - zoneGraph.clip(clips[i]); - } - if (area && zoneArea) { - zoneArea.clip(clips[i]); - } - // if this zone extends out of the axis, ignore the others - ignoreZones = threshold.value > extremes.max; - // Clear translatedTo for indicators - if (series.resetZones && translatedTo === 0) { - translatedTo = void 0; - } - }); - this.clips = clips; - } - else if (series.visible) { - // If zones were removed, restore graph and area - if (graph) { - graph.show(true); - } - if (area) { - area.show(true); - } - } + first = buildDimensionArray("y"), + second = buildDimensionArray("x"), + // The far side is right or bottom + preferFarSide = + !this.followPointer && + pick(point.ttBelow, !chart.inverted === !!point.negative), // #4984 + /* + * Handle the preferred dimension. When the preferred dimension is + * tooltip on top or bottom of the point, it will look for space + * there. + * + * @private + */ + firstDimension = function ( + dim, + outerSize, + innerSize, + scaledInnerSize, // #11329 + point, + min, + max + ) { + var scaledDist = + dim === "y" ? scaleY(distance) : scaleX(distance), + scaleDiff = (innerSize - scaledInnerSize) / 2, + roomLeft = scaledInnerSize < point - distance, + roomRight = point + distance + scaledInnerSize < outerSize, + alignedLeft = point - scaledDist - innerSize + scaleDiff, + alignedRight = point + scaledDist - scaleDiff; + if (preferFarSide && roomRight) { + ret[dim] = alignedRight; + } else if (!preferFarSide && roomLeft) { + ret[dim] = alignedLeft; + } else if (roomLeft) { + ret[dim] = Math.min( + max - scaledInnerSize, + alignedLeft - h < 0 ? alignedLeft : alignedLeft - h + ); + } else if (roomRight) { + ret[dim] = Math.max( + min, + alignedRight + h + innerSize > outerSize + ? alignedRight + : alignedRight + h + ); + } else { + return false; + } }, - /** - * Initialize and perform group inversion on series.group and - * series.markerGroup. - * - * @private - * @function Highcharts.Series#invertGroups - * @param {boolean} [inverted] - * @return {void} + /* + * Handle the secondary dimension. If the preferred dimension is + * tooltip on top or bottom of the point, the second dimension is to + * align the tooltip above the point, trying to align center but + * allowing left or right align within the chart box. + * + * @private + */ + secondDimension = function ( + dim, + outerSize, + innerSize, + scaledInnerSize, // #11329 + point + ) { + var retVal; + // Too close to the edge, return false and swap dimensions + if (point < distance || point > outerSize - distance) { + retVal = false; + // Align left/top + } else if (point < innerSize / 2) { + ret[dim] = 1; + // Align right/bottom + } else if (point > outerSize - scaledInnerSize / 2) { + ret[dim] = outerSize - scaledInnerSize - 2; + // Align center + } else { + ret[dim] = point - innerSize / 2; + } + return retVal; + }, + /* + * Swap the dimensions */ - invertGroups: function (inverted) { - var series = this, - chart = series.chart; - /** - * @private - */ - function setInvert() { - ['group', 'markerGroup'].forEach(function (groupName) { - if (series[groupName]) { - // VML/HTML needs explicit attributes for flipping - if (chart.renderer.isVML) { - series[groupName].attr({ - width: series.yAxis.len, - height: series.xAxis.len - }); - } - series[groupName].width = series.yAxis.len; - series[groupName].height = series.xAxis.len; - // If inverted polar, don't invert series group - series[groupName].invert(series.isRadialSeries ? false : inverted); - } - }); - } - // Pie, go away (#1736) - if (!series.xAxis) { - return; + swap = function (count) { + var temp = first; + first = second; + second = temp; + swapped = count; + }, + run = function () { + if (firstDimension.apply(0, first) !== false) { + if (secondDimension.apply(0, second) === false && !swapped) { + swap(true); + run(); + } + } else if (!swapped) { + swap(true); + run(); + } else { + ret.x = ret.y = 0; + } + }; + // Under these conditions, prefer the tooltip on the side of the point + if (chart.inverted || this.len > 1) { + swap(); + } + run(); + return ret; + }; + /** + * Get the best X date format based on the closest point range on the axis. + * + * @private + * @function Highcharts.Tooltip#getXDateFormat + * + * @param {Highcharts.Point} point + * + * @param {Highcharts.TooltipOptions} options + * + * @param {Highcharts.Axis} xAxis + * + * @return {string} + */ + Tooltip.prototype.getXDateFormat = function (point, options, xAxis) { + var xDateFormat, + dateTimeLabelFormats = options.dateTimeLabelFormats, + closestPointRange = xAxis && xAxis.closestPointRange; + if (closestPointRange) { + xDateFormat = this.getDateFormat( + closestPointRange, + point.x, + xAxis.options.startOfWeek, + dateTimeLabelFormats + ); + } else { + xDateFormat = dateTimeLabelFormats.day; + } + return xDateFormat || dateTimeLabelFormats.year; // #2546, 2581 + }; + /** + * Hides the tooltip with a fade out animation. + * + * @function Highcharts.Tooltip#hide + * + * @param {number} [delay] + * The fade out in milliseconds. If no value is provided the value + * of the tooltip.hideDelay option is used. A value of 0 disables + * the fade out animation. + */ + Tooltip.prototype.hide = function (delay) { + var tooltip = this; + // disallow duplicate timers (#1728, #1766) + U.clearTimeout(this.hideTimer); + delay = pick(delay, this.options.hideDelay, 500); + if (!this.isHidden) { + this.hideTimer = syncTimeout(function () { + // If there is a delay, do fadeOut with the default duration. If + // the hideDelay is 0, we assume no animation is wanted, so we + // pass 0 duration. #12994. + tooltip.getLabel().fadeOut(delay ? void 0 : delay); + tooltip.isHidden = true; + }, delay); + } + }; + /** + * @private + * @function Highcharts.Tooltip#init + * + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @param {Highcharts.TooltipOptions} options + * Tooltip options. + */ + Tooltip.prototype.init = function (chart, options) { + /** + * Chart of the tooltip. + * + * @readonly + * @name Highcharts.Tooltip#chart + * @type {Highcharts.Chart} + */ + this.chart = chart; + /** + * Used tooltip options. + * + * @readonly + * @name Highcharts.Tooltip#options + * @type {Highcharts.TooltipOptions} + */ + this.options = options; + /** + * List of crosshairs. + * + * @private + * @readonly + * @name Highcharts.Tooltip#crosshairs + * @type {Array<null>} + */ + this.crosshairs = []; + /** + * Current values of x and y when animating. + * + * @private + * @readonly + * @name Highcharts.Tooltip#now + * @type {Highcharts.PositionObject} + */ + this.now = { x: 0, y: 0 }; + /** + * Tooltips are initially hidden. + * + * @private + * @readonly + * @name Highcharts.Tooltip#isHidden + * @type {boolean} + */ + this.isHidden = true; + /** + * True, if the tooltip is split into one label per series, with the + * header close to the axis. + * + * @readonly + * @name Highcharts.Tooltip#split + * @type {boolean|undefined} + */ + this.split = options.split && !chart.inverted && !chart.polar; + /** + * When the tooltip is shared, the entire plot area will capture mouse + * movement or touch events. + * + * @readonly + * @name Highcharts.Tooltip#shared + * @type {boolean|undefined} + */ + this.shared = options.shared || this.split; + /** + * Whether to allow the tooltip to render outside the chart's SVG + * element box. By default (false), the tooltip is rendered within the + * chart's SVG element, which results in the tooltip being aligned + * inside the chart area. + * + * @readonly + * @name Highcharts.Tooltip#outside + * @type {boolean} + * + * @todo + * Split tooltip does not support outside in the first iteration. Should + * not be too complicated to implement. + */ + this.outside = pick( + options.outside, + Boolean(chart.scrollablePixelsX || chart.scrollablePixelsY) + ); + }; + /** + * Returns true, if the pointer is in contact with the tooltip tracker. + */ + Tooltip.prototype.isStickyOnContact = function () { + return !!( + !this.followPointer && + this.options.stickOnContact && + this.inContact + ); + }; + /** + * Moves the tooltip with a soft animation to a new position. + * + * @private + * @function Highcharts.Tooltip#move + * + * @param {number} x + * + * @param {number} y + * + * @param {number} anchorX + * + * @param {number} anchorY + */ + Tooltip.prototype.move = function (x, y, anchorX, anchorY) { + var tooltip = this, + now = tooltip.now, + animate = + tooltip.options.animation !== false && + !tooltip.isHidden && + // When we get close to the target position, abort animation and + // land on the right place (#3056) + (Math.abs(x - now.x) > 1 || Math.abs(y - now.y) > 1), + skipAnchor = tooltip.followPointer || tooltip.len > 1; + // Get intermediate values for animation + extend(now, { + x: animate ? (2 * now.x + x) / 3 : x, + y: animate ? (now.y + y) / 2 : y, + anchorX: skipAnchor + ? void 0 + : animate + ? (2 * now.anchorX + anchorX) / 3 + : anchorX, + anchorY: skipAnchor + ? void 0 + : animate + ? (now.anchorY + anchorY) / 2 + : anchorY, + }); + // Move to the intermediate value + tooltip.getLabel().attr(now); + tooltip.drawTracker(); + // Run on next tick of the mouse tracker + if (animate) { + // Never allow two timeouts + U.clearTimeout(this.tooltipTimeout); + // Set the fixed interval ticking for the smooth tooltip + this.tooltipTimeout = setTimeout(function () { + // The interval function may still be running during destroy, + // so check that the chart is really there before calling. + if (tooltip) { + tooltip.move(x, y, anchorX, anchorY); + } + }, 32); + } + }; + /** + * Refresh the tooltip's text and position. + * + * @function Highcharts.Tooltip#refresh + * + * @param {Highcharts.Point|Array<Highcharts.Point>} pointOrPoints + * Either a point or an array of points. + * + * @param {Highcharts.PointerEventObject} [mouseEvent] + * Mouse event, that is responsible for the refresh and should be + * used for the tooltip update. + */ + Tooltip.prototype.refresh = function (pointOrPoints, mouseEvent) { + var tooltip = this, + chart = this.chart, + options = tooltip.options, + x, + y, + point = pointOrPoints, + anchor, + textConfig = {}, + text, + pointConfig = [], + formatter = options.formatter || tooltip.defaultFormatter, + shared = tooltip.shared, + currentSeries, + styledMode = chart.styledMode; + if (!options.enabled) { + return; + } + U.clearTimeout(this.hideTimer); + // get the reference point coordinates (pie charts use tooltipPos) + tooltip.followPointer = + splat(point)[0].series.tooltipOptions.followPointer; + anchor = tooltip.getAnchor(point, mouseEvent); + x = anchor[0]; + y = anchor[1]; + // shared tooltip, array is sent over + if (shared && !(point.series && point.series.noSharedTooltip)) { + chart.pointer.applyInactiveState(point); + // Now set hover state for the choosen ones: + point.forEach(function (item) { + item.setState("hover"); + pointConfig.push(item.getLabelConfig()); + }); + textConfig = { + x: point[0].category, + y: point[0].y, + }; + textConfig.points = pointConfig; + point = point[0]; + // single point tooltip + } else { + textConfig = point.getLabelConfig(); + } + this.len = pointConfig.length; // #6128 + text = formatter.call(textConfig, tooltip); + // register the current series + currentSeries = point.series; + this.distance = pick(currentSeries.tooltipOptions.distance, 16); + // update the inner HTML + if (text === false) { + this.hide(); + } else { + // update text + if (tooltip.split) { + this.renderSplit(text, splat(pointOrPoints)); + } else { + var label = tooltip.getLabel(); + // Prevent the tooltip from flowing over the chart box (#6659) + if (!options.style.width || styledMode) { + label.css({ + width: this.chart.spacingBox.width + "px", + }); + } + label.attr({ + text: text && text.join ? text.join("") : text, + }); + // Set the stroke color of the box to reflect the point + label + .removeClass(/highcharts-color-[\d]+/g) + .addClass( + "highcharts-color-" + + pick(point.colorIndex, currentSeries.colorIndex) + ); + if (!styledMode) { + label.attr({ + stroke: + options.borderColor || + point.color || + currentSeries.color || + "#666666", + }); + } + tooltip.updatePosition({ + plotX: x, + plotY: y, + negative: point.negative, + ttBelow: point.ttBelow, + h: anchor[2] || 0, + }); + } + // show it + if (tooltip.isHidden && tooltip.label) { + tooltip.label + .attr({ + opacity: 1, + }) + .show(); + } + tooltip.isHidden = false; + } + fireEvent(this, "refresh"); + }; + /** + * Render the split tooltip. Loops over each point's text and adds + * a label next to the point, then uses the distribute function to + * find best non-overlapping positions. + * + * @private + * @function Highcharts.Tooltip#renderSplit + * + * @param {string|Array<(boolean|string)>} labels + * + * @param {Array<Highcharts.Point>} points + */ + Tooltip.prototype.renderSplit = function (labels, points) { + var tooltip = this; + var chart = tooltip.chart, + _a = tooltip.chart, + chartWidth = _a.chartWidth, + chartHeight = _a.chartHeight, + plotHeight = _a.plotHeight, + plotLeft = _a.plotLeft, + plotTop = _a.plotTop, + pointer = _a.pointer, + ren = _a.renderer, + _b = _a.scrollablePixelsY, + scrollablePixelsY = _b === void 0 ? 0 : _b, + _c = _a.scrollingContainer, + _d = _c === void 0 ? { scrollLeft: 0, scrollTop: 0 } : _c, + scrollLeft = _d.scrollLeft, + scrollTop = _d.scrollTop, + styledMode = _a.styledMode, + distance = tooltip.distance, + options = tooltip.options, + positioner = tooltip.options.positioner; + // The area which the tooltip should be limited to. Limit to scrollable + // plot area if enabled, otherwise limit to the chart container. + var bounds = { + left: scrollLeft, + right: scrollLeft + chartWidth, + top: scrollTop, + bottom: scrollTop + chartHeight, + }; + var tooltipLabel = tooltip.getLabel(); + var headerTop = Boolean(chart.xAxis[0] && chart.xAxis[0].opposite); + var distributionBoxTop = plotTop + scrollTop; + var headerHeight = 0; + var adjustedPlotHeight = plotHeight - scrollablePixelsY; + /** + * Calculates the anchor position for the partial tooltip + * + * @private + * @param {Highcharts.Point} point The point related to the tooltip + * @return {object} Returns an object with anchorX and anchorY + */ + function getAnchor(point) { + var isHeader = point.isHeader, + _a = point.plotX, + plotX = _a === void 0 ? 0 : _a, + _b = point.plotY, + plotY = _b === void 0 ? 0 : _b, + series = point.series; + var anchorX; + var anchorY; + if (isHeader) { + // Set anchorX to plotX + anchorX = plotLeft + plotX; + // Set anchorY to center of visible plot area. + anchorY = plotTop + plotHeight / 2; + } else { + var xAxis = series.xAxis, + yAxis = series.yAxis; + // Set anchorX to plotX. Limit to within xAxis. + anchorX = + xAxis.pos + clamp(plotX, -distance, xAxis.len + distance); + // Set anchorY, limit to the scrollable plot area + if ( + yAxis.pos + plotY >= scrollTop + plotTop && + yAxis.pos + plotY <= + scrollTop + plotTop + plotHeight - scrollablePixelsY + ) { + anchorY = yAxis.pos + plotY; + } + } + // Limit values to plot area + anchorX = clamp( + anchorX, + bounds.left - distance, + bounds.right + distance + ); + return { anchorX: anchorX, anchorY: anchorY }; + } + /** + * Calculates the position of the partial tooltip + * + * @private + * @param {number} anchorX The partial tooltip anchor x position + * @param {number} anchorY The partial tooltip anchor y position + * @param {boolean} isHeader Whether the partial tooltip is a header + * @param {number} boxWidth Width of the partial tooltip + * @return {Highcharts.PositionObject} Returns the partial tooltip x and + * y position + */ + function defaultPositioner( + anchorX, + anchorY, + isHeader, + boxWidth, + alignedLeft + ) { + if (alignedLeft === void 0) { + alignedLeft = true; + } + var y; + var x; + if (isHeader) { + y = headerTop ? 0 : adjustedPlotHeight; + x = clamp( + anchorX - boxWidth / 2, + bounds.left, + bounds.right - boxWidth + ); + } else { + y = anchorY - distributionBoxTop; + x = alignedLeft + ? anchorX - boxWidth - distance + : anchorX + distance; + x = clamp(x, alignedLeft ? x : bounds.left, bounds.right); + } + // NOTE: y is relative to distributionBoxTop + return { x: x, y: y }; + } + /** + * Updates the attributes and styling of the partial tooltip. Creates a + * new partial tooltip if it does not exists. + * + * @private + * @param {Highcharts.SVGElement|undefined} partialTooltip + * The partial tooltip to update + * @param {Highcharts.Point} point + * The point related to the partial tooltip + * @param {boolean|string} str The text for the partial tooltip + * @return {Highcharts.SVGElement} Returns the updated partial tooltip + */ + function updatePartialTooltip(partialTooltip, point, str) { + var tt = partialTooltip; + var isHeader = point.isHeader, + series = point.series; + var colorClass = + "highcharts-color-" + + pick(point.colorIndex, series.colorIndex, "none"); + if (!tt) { + var attribs = { + padding: options.padding, + r: options.borderRadius, + }; + if (!styledMode) { + attribs.fill = options.backgroundColor; + attribs["stroke-width"] = options.borderWidth; + } + tt = ren + .label( + "", + 0, + 0, + options[isHeader ? "headerShape" : "shape"] || "callout", + void 0, + void 0, + options.useHTML + ) + .addClass( + (isHeader ? "highcharts-tooltip-header " : "") + + "highcharts-tooltip-box " + + colorClass + ) + .attr(attribs) + .add(tooltipLabel); + } + tt.isActive = true; + tt.attr({ + text: str, + }); + if (!styledMode) { + tt.css(options.style) + .shadow(options.shadow) + .attr({ + stroke: + options.borderColor || + point.color || + series.color || + "#333333", + }); + } + return tt; + } + // Graceful degradation for legacy formatters + if (isString(labels)) { + labels = [false, labels]; + } + // Create the individual labels for header and points, ignore footer + var boxes = labels + .slice(0, points.length + 1) + .reduce(function (boxes, str, i) { + if (str !== false && str !== "") { + var point = points[i - 1] || { + // Item 0 is the header. Instead of this, we could also + // use the crosshair label + isHeader: true, + plotX: points[0].plotX, + plotY: plotHeight, + series: {}, + }; + var isHeader = point.isHeader; + // Store the tooltip label referance on the series + var owner = isHeader ? tooltip : point.series; + var tt = (owner.tt = updatePartialTooltip( + owner.tt, + point, + str + )); + // Get X position now, so we can move all to the other side in + // case of overflow + var bBox = tt.getBBox(); + var boxWidth = bBox.width + tt.strokeWidth(); + if (isHeader) { + headerHeight = bBox.height; + adjustedPlotHeight += headerHeight; + if (headerTop) { + distributionBoxTop -= headerHeight; + } + } + var _a = getAnchor(point), + anchorX = _a.anchorX, + anchorY = _a.anchorY; + if (typeof anchorY === "number") { + var size = bBox.height + 1; + var boxPosition = positioner + ? positioner.call(tooltip, boxWidth, size, point) + : defaultPositioner(anchorX, anchorY, isHeader, boxWidth); + boxes.push({ + // 0-align to the top, 1-align to the bottom + align: positioner ? 0 : void 0, + anchorX: anchorX, + anchorY: anchorY, + boxWidth: boxWidth, + point: point, + rank: pick(boxPosition.rank, isHeader ? 1 : 0), + size: size, + target: boxPosition.y, + tt: tt, + x: boxPosition.x, + }); + } else { + // Hide tooltips which anchorY is outside the visible plot + // area + tt.isActive = false; + } + } + return boxes; + }, []); + // If overflow left then align all labels to the right + if ( + !positioner && + boxes.some(function (box) { + return box.x < bounds.left; + }) + ) { + boxes = boxes.map(function (box) { + var _a = defaultPositioner( + box.anchorX, + box.anchorY, + box.point.isHeader, + box.boxWidth, + false + ), + x = _a.x, + y = _a.y; + return extend(box, { + target: y, + x: x, + }); + }); + } + // Clean previous run (for missing points) + tooltip.cleanSplit(); + // Distribute and put in place + H.distribute(boxes, adjustedPlotHeight); + boxes.forEach(function (box) { + var anchorX = box.anchorX, + anchorY = box.anchorY, + pos = box.pos, + x = box.x; + // Put the label in place + box.tt.attr({ + visibility: typeof pos === "undefined" ? "hidden" : "inherit", + x: x, + /* NOTE: y should equal pos to be consistent with !split + * tooltip, but is currently relative to plotTop. Is left as is + * to avoid breaking change. Remove distributionBoxTop to make + * it consistent. + */ + y: pos + distributionBoxTop, + anchorX: anchorX, + anchorY: anchorY, + }); + }); + /* If we have a seperate tooltip container, then update the necessary + * container properties. + * Test that tooltip has its own container and renderer before executing + * the operation. + */ + var container = tooltip.container, + outside = tooltip.outside, + renderer = tooltip.renderer; + if (outside && container && renderer) { + // Set container size to fit the tooltip + var _e = tooltipLabel.getBBox(), + width = _e.width, + height = _e.height, + x = _e.x, + y = _e.y; + renderer.setSize(width + x, height + y, false); + // Position the tooltip container to the chart container + var chartPosition = pointer.getChartPosition(); + container.style.left = chartPosition.left + "px"; + container.style.top = chartPosition.top + "px"; + } + }; + /** + * If the `stickOnContact` option is active, this will add a tracker shape. + * + * @private + * @function Highcharts.Tooltip#drawTracker + */ + Tooltip.prototype.drawTracker = function () { + var tooltip = this; + if (tooltip.followPointer || !tooltip.options.stickOnContact) { + if (tooltip.tracker) { + tooltip.tracker.destroy(); + } + return; + } + var chart = tooltip.chart; + var label = tooltip.label; + var point = chart.hoverPoint; + if (!label || !point) { + return; + } + var box = { + x: 0, + y: 0, + width: 0, + height: 0, + }; + // Combine anchor and tooltip + var anchorPos = this.getAnchor(point); + var labelBBox = label.getBBox(); + anchorPos[0] += chart.plotLeft - label.translateX; + anchorPos[1] += chart.plotTop - label.translateY; + // When the mouse pointer is between the anchor point and the label, + // the label should stick. + box.x = Math.min(0, anchorPos[0]); + box.y = Math.min(0, anchorPos[1]); + box.width = + anchorPos[0] < 0 + ? Math.max(Math.abs(anchorPos[0]), labelBBox.width - anchorPos[0]) + : Math.max(Math.abs(anchorPos[0]), labelBBox.width); + box.height = + anchorPos[1] < 0 + ? Math.max( + Math.abs(anchorPos[1]), + labelBBox.height - Math.abs(anchorPos[1]) + ) + : Math.max(Math.abs(anchorPos[1]), labelBBox.height); + if (tooltip.tracker) { + tooltip.tracker.attr(box); + } else { + tooltip.tracker = label.renderer + .rect(box) + .addClass("highcharts-tracker") + .add(label); + if (!chart.styledMode) { + tooltip.tracker.attr({ + fill: "rgba(0,0,0,0)", + }); + } + } + }; + /** + * @private + */ + Tooltip.prototype.styledModeFormat = function (formatString) { + return formatString + .replace('style="font-size: 10px"', 'class="highcharts-header"') + .replace( + /style="color:{(point|series)\.color}"/g, + 'class="highcharts-color-{$1.colorIndex}"' + ); + }; + /** + * Format the footer/header of the tooltip + * #3397: abstraction to enable formatting of footer and header + * + * @private + * @function Highcharts.Tooltip#tooltipFooterHeaderFormatter + * @param {Highcharts.PointLabelObject} labelConfig + * @param {boolean} [isFooter] + * @return {string} + */ + Tooltip.prototype.tooltipFooterHeaderFormatter = function ( + labelConfig, + isFooter + ) { + var footOrHead = isFooter ? "footer" : "header", + series = labelConfig.series, + tooltipOptions = series.tooltipOptions, + xDateFormat = tooltipOptions.xDateFormat, + xAxis = series.xAxis, + isDateTime = + xAxis && + xAxis.options.type === "datetime" && + isNumber(labelConfig.key), + formatString = tooltipOptions[footOrHead + "Format"], + e = { + isFooter: isFooter, + labelConfig: labelConfig, + }; + fireEvent(this, "headerFormatter", e, function (e) { + // Guess the best date format based on the closest point distance + // (#568, #3418) + if (isDateTime && !xDateFormat) { + xDateFormat = this.getXDateFormat( + labelConfig, + tooltipOptions, + xAxis + ); + } + // Insert the footer date format if any + if (isDateTime && xDateFormat) { + ( + (labelConfig.point && labelConfig.point.tooltipDateKeys) || [ + "key", + ] + ).forEach(function (key) { + formatString = formatString.replace( + "{point." + key + "}", + "{point." + key + ":" + xDateFormat + "}" + ); + }); + } + // Replace default header style with class name + if (series.chart.styledMode) { + formatString = this.styledModeFormat(formatString); + } + e.text = format( + formatString, + { + point: labelConfig, + series: series, + }, + this.chart + ); + }); + return e.text; + }; + /** + * Updates the tooltip with the provided tooltip options. + * + * @function Highcharts.Tooltip#update + * + * @param {Highcharts.TooltipOptions} options + * The tooltip options to update. + */ + Tooltip.prototype.update = function (options) { + this.destroy(); + // Update user options (#6218) + merge(true, this.chart.options.tooltip.userOptions, options); + this.init(this.chart, merge(true, this.options, options)); + }; + /** + * Find the new position and perform the move + * + * @private + * @function Highcharts.Tooltip#updatePosition + * + * @param {Highcharts.Point} point + */ + Tooltip.prototype.updatePosition = function (point) { + var chart = this.chart, + pointer = chart.pointer, + label = this.getLabel(), + pos, + anchorX = point.plotX + chart.plotLeft, + anchorY = point.plotY + chart.plotTop, + pad; + // Needed for outside: true (#11688) + var chartPosition = pointer.getChartPosition(); + pos = (this.options.positioner || this.getPosition).call( + this, + label.width, + label.height, + point + ); + // Set the renderer size dynamically to prevent document size to change + if (this.outside) { + pad = (this.options.borderWidth || 0) + 2 * this.distance; + this.renderer.setSize(label.width + pad, label.height + pad, false); + // Anchor and tooltip container need scaling if chart container has + // scale transform/css zoom. #11329. + var containerScaling = chart.containerScaling; + if (containerScaling) { + css(this.container, { + transform: + "scale(" + + containerScaling.scaleX + + ", " + + containerScaling.scaleY + + ")", + }); + anchorX *= containerScaling.scaleX; + anchorY *= containerScaling.scaleY; + } + anchorX += chartPosition.left - pos.x; + anchorY += chartPosition.top - pos.y; + } + // do the move + this.move( + Math.round(pos.x), + Math.round(pos.y || 0), // can be undefined (#3977) + anchorX, + anchorY + ); + }; + return Tooltip; + })(); + H.Tooltip = Tooltip; + + return H.Tooltip; + } + ); + _registerModule( + _modules, + "Core/Pointer.js", + [ + _modules["Core/Color/Color.js"], + _modules["Core/Globals.js"], + _modules["Core/Tooltip.js"], + _modules["Core/Utilities.js"], + ], + function (Color, H, Tooltip, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var color = Color.parse; + var charts = H.charts, + noop = H.noop; + var addEvent = U.addEvent, + attr = U.attr, + css = U.css, + defined = U.defined, + extend = U.extend, + find = U.find, + fireEvent = U.fireEvent, + isNumber = U.isNumber, + isObject = U.isObject, + objectEach = U.objectEach, + offset = U.offset, + pick = U.pick, + splat = U.splat; + /** + * One position in relation to an axis. + * + * @interface Highcharts.PointerAxisCoordinateObject + */ /** + * Related axis. + * + * @name Highcharts.PointerAxisCoordinateObject#axis + * @type {Highcharts.Axis} + */ /** + * Axis value. + * + * @name Highcharts.PointerAxisCoordinateObject#value + * @type {number} + */ + /** + * Positions in terms of axis values. + * + * @interface Highcharts.PointerAxisCoordinatesObject + */ /** + * Positions on the x-axis. + * @name Highcharts.PointerAxisCoordinatesObject#xAxis + * @type {Array<Highcharts.PointerAxisCoordinateObject>} + */ /** + * Positions on the y-axis. + * @name Highcharts.PointerAxisCoordinatesObject#yAxis + * @type {Array<Highcharts.PointerAxisCoordinateObject>} + */ + /** + * Pointer coordinates. + * + * @interface Highcharts.PointerCoordinatesObject + */ /** + * @name Highcharts.PointerCoordinatesObject#chartX + * @type {number} + */ /** + * @name Highcharts.PointerCoordinatesObject#chartY + * @type {number} + */ + /** + * A native browser mouse or touch event, extended with position information + * relative to the {@link Chart.container}. + * + * @interface Highcharts.PointerEventObject + * @extends global.PointerEvent + */ /** + * The X coordinate of the pointer interaction relative to the chart. + * + * @name Highcharts.PointerEventObject#chartX + * @type {number} + */ /** + * The Y coordinate of the pointer interaction relative to the chart. + * + * @name Highcharts.PointerEventObject#chartY + * @type {number} + */ + /** + * Axis-specific data of a selection. + * + * @interface Highcharts.SelectDataObject + */ /** + * @name Highcharts.SelectDataObject#axis + * @type {Highcharts.Axis} + */ /** + * @name Highcharts.SelectDataObject#max + * @type {number} + */ /** + * @name Highcharts.SelectDataObject#min + * @type {number} + */ + /** + * Object for select events. + * + * @interface Highcharts.SelectEventObject + */ /** + * @name Highcharts.SelectEventObject#originalEvent + * @type {global.Event} + */ /** + * @name Highcharts.SelectEventObject#xAxis + * @type {Array<Highcharts.SelectDataObject>} + */ /** + * @name Highcharts.SelectEventObject#yAxis + * @type {Array<Highcharts.SelectDataObject>} + */ + (""); // detach doclets above + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The mouse and touch tracker object. Each {@link Chart} item has one + * assosiated Pointer item that can be accessed from the {@link Chart.pointer} + * property. + * + * @class + * @name Highcharts.Pointer + * + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @param {Highcharts.Options} options + * The root options object. The pointer uses options from the chart and + * tooltip structures. + */ + var Pointer = /** @class */ (function () { + /* * + * + * Constructors + * + * */ + function Pointer(chart, options) { + this.lastValidTouch = {}; + this.pinchDown = []; + this.runChartClick = false; + this.chart = chart; + this.hasDragged = false; + this.options = options; + this.unbindContainerMouseLeave = function () {}; + this.unbindContainerMouseEnter = function () {}; + this.init(chart, options); + } + /* * + * + * Functions + * + * */ + /** + * Set inactive state to all series that are not currently hovered, + * or, if `inactiveOtherPoints` is set to true, set inactive state to + * all points within that series. + * + * @private + * @function Highcharts.Pointer#applyInactiveState + * @param {Array<Highcharts.Point>} points + * Currently hovered points + */ + Pointer.prototype.applyInactiveState = function (points) { + var activeSeries = [], + series; + // Get all active series from the hovered points + (points || []).forEach(function (item) { + series = item.series; + // Include itself + activeSeries.push(series); + // Include parent series + if (series.linkedParent) { + activeSeries.push(series.linkedParent); + } + // Include all child series + if (series.linkedSeries) { + activeSeries = activeSeries.concat(series.linkedSeries); + } + // Include navigator series + if (series.navigatorSeries) { + activeSeries.push(series.navigatorSeries); + } + }); + // Now loop over all series, filtering out active series + this.chart.series.forEach(function (inactiveSeries) { + if (activeSeries.indexOf(inactiveSeries) === -1) { + // Inactive series + inactiveSeries.setState("inactive", true); + } else if (inactiveSeries.options.inactiveOtherPoints) { + // Active series, but other points should be inactivated + inactiveSeries.setAllPointsToState("inactive"); + } + }); + }; + /** + * Destroys the Pointer object and disconnects DOM events. + * + * @function Highcharts.Pointer#destroy + */ + Pointer.prototype.destroy = function () { + var pointer = this; + if (typeof pointer.unDocMouseMove !== "undefined") { + pointer.unDocMouseMove(); + } + this.unbindContainerMouseLeave(); + if (!H.chartCount) { + if (H.unbindDocumentMouseUp) { + H.unbindDocumentMouseUp = H.unbindDocumentMouseUp(); + } + if (H.unbindDocumentTouchEnd) { + H.unbindDocumentTouchEnd = H.unbindDocumentTouchEnd(); + } + } + // memory and CPU leak + clearInterval(pointer.tooltipTimeout); + objectEach(pointer, function (_val, prop) { + pointer[prop] = void 0; + }); + }; + /** + * Perform a drag operation in response to a mousemove event while the mouse + * is down. + * + * @private + * @function Highcharts.Pointer#drag + * + * @param {Highcharts.PointerEventObject} e + * + * @return {void} + */ + Pointer.prototype.drag = function (e) { + var chart = this.chart, + chartOptions = chart.options.chart, + chartX = e.chartX, + chartY = e.chartY, + zoomHor = this.zoomHor, + zoomVert = this.zoomVert, + plotLeft = chart.plotLeft, + plotTop = chart.plotTop, + plotWidth = chart.plotWidth, + plotHeight = chart.plotHeight, + clickedInside, + size, + selectionMarker = this.selectionMarker, + mouseDownX = this.mouseDownX || 0, + mouseDownY = this.mouseDownY || 0, + panningEnabled = isObject(chartOptions.panning) + ? chartOptions.panning && chartOptions.panning.enabled + : chartOptions.panning, + panKey = chartOptions.panKey && e[chartOptions.panKey + "Key"]; + // If the device supports both touch and mouse (like IE11), and we are + // touch-dragging inside the plot area, don't handle the mouse event. + // #4339. + if (selectionMarker && selectionMarker.touch) { + return; + } + // If the mouse is outside the plot area, adjust to cooordinates + // inside to prevent the selection marker from going outside + if (chartX < plotLeft) { + chartX = plotLeft; + } else if (chartX > plotLeft + plotWidth) { + chartX = plotLeft + plotWidth; + } + if (chartY < plotTop) { + chartY = plotTop; + } else if (chartY > plotTop + plotHeight) { + chartY = plotTop + plotHeight; + } + // determine if the mouse has moved more than 10px + this.hasDragged = Math.sqrt( + Math.pow(mouseDownX - chartX, 2) + Math.pow(mouseDownY - chartY, 2) + ); + if (this.hasDragged > 10) { + clickedInside = chart.isInsidePlot( + mouseDownX - plotLeft, + mouseDownY - plotTop + ); + // make a selection + if ( + chart.hasCartesianSeries && + (this.zoomX || this.zoomY) && + clickedInside && + !panKey + ) { + if (!selectionMarker) { + this.selectionMarker = selectionMarker = chart.renderer + .rect( + plotLeft, + plotTop, + zoomHor ? 1 : plotWidth, + zoomVert ? 1 : plotHeight, + 0 + ) + .attr({ + class: "highcharts-selection-marker", + zIndex: 7, + }) + .add(); + if (!chart.styledMode) { + selectionMarker.attr({ + fill: + chartOptions.selectionMarkerFill || + color("#335cad").setOpacity(0.25).get(), + }); } - // A fixed size is needed for inversion to work - series.eventsToUnbind.push(addEvent(chart, 'resize', setInvert)); - // Do it now - setInvert(); - // On subsequent render and redraw, just do setInvert without - // setting up events again - series.invertGroups = setInvert; - }, - /** - * General abstraction for creating plot groups like series.group, - * series.dataLabelsGroup and series.markerGroup. On subsequent calls, - * the group will only be adjusted to the updated plot size. - * - * @private - * @function Highcharts.Series#plotGroup - * @param {string} prop - * @param {string} name - * @param {string} visibility - * @param {number} [zIndex] - * @param {Highcharts.SVGElement} [parent] - * @return {Highcharts.SVGElement} - */ - plotGroup: function (prop, name, visibility, zIndex, parent) { - var group = this[prop], - isNew = !group, - attrs = { - visibility: visibility, - zIndex: zIndex || 0.1 // IE8 and pointer logic use this - }; - // Avoid setting undefined opacity, or in styled mode - if (typeof this.opacity !== 'undefined' && - !this.chart.styledMode && this.state !== 'inactive' // #13719 + } + } + // adjust the width of the selection marker + if (selectionMarker && zoomHor) { + size = chartX - mouseDownX; + selectionMarker.attr({ + width: Math.abs(size), + x: (size > 0 ? 0 : size) + mouseDownX, + }); + } + // adjust the height of the selection marker + if (selectionMarker && zoomVert) { + size = chartY - mouseDownY; + selectionMarker.attr({ + height: Math.abs(size), + y: (size > 0 ? 0 : size) + mouseDownY, + }); + } + // panning + if (clickedInside && !selectionMarker && panningEnabled) { + chart.pan(e, chartOptions.panning); + } + } + }; + /** + * Start a drag operation. + * + * @private + * @function Highcharts.Pointer#dragStart + * + * @param {Highcharts.PointerEventObject} e + * + * @return {void} + */ + Pointer.prototype.dragStart = function (e) { + var chart = this.chart; + // Record the start position + chart.mouseIsDown = e.type; + chart.cancelClick = false; + chart.mouseDownX = this.mouseDownX = e.chartX; + chart.mouseDownY = this.mouseDownY = e.chartY; + }; + /** + * On mouse up or touch end across the entire document, drop the selection. + * + * @private + * @function Highcharts.Pointer#drop + * + * @param {global.Event} e + */ + Pointer.prototype.drop = function (e) { + var pointer = this, + chart = this.chart, + hasPinched = this.hasPinched; + if (this.selectionMarker) { + var selectionData = { + originalEvent: e, + xAxis: [], + yAxis: [], + }, + selectionBox = this.selectionMarker, + selectionLeft = selectionBox.attr + ? selectionBox.attr("x") + : selectionBox.x, + selectionTop = selectionBox.attr + ? selectionBox.attr("y") + : selectionBox.y, + selectionWidth = selectionBox.attr + ? selectionBox.attr("width") + : selectionBox.width, + selectionHeight = selectionBox.attr + ? selectionBox.attr("height") + : selectionBox.height, + runZoom; + // a selection has been made + if (this.hasDragged || hasPinched) { + // record each axis' min and max + chart.axes.forEach(function (axis) { + if ( + axis.zoomEnabled && + defined(axis.min) && + (hasPinched || + pointer[ + { + xAxis: "zoomX", + yAxis: "zoomY", + }[axis.coll] + ]) && + isNumber(selectionLeft) && + isNumber(selectionTop) ) { - attrs.opacity = this.opacity; - } - // Generate it on first call - if (isNew) { - this[prop] = group = this.chart.renderer - .g() - .add(parent); - } - // Add the class names, and replace existing ones as response to - // Series.update (#6660) - group.addClass(('highcharts-' + name + - ' highcharts-series-' + this.index + - ' highcharts-' + this.type + '-series ' + - (defined(this.colorIndex) ? - 'highcharts-color-' + this.colorIndex + ' ' : - '') + - (this.options.className || '') + - (group.hasClass('highcharts-tracker') ? - ' highcharts-tracker' : - '')), true); - // Place it on first and subsequent (redraw) calls - group.attr(attrs)[isNew ? 'attr' : 'animate'](this.getPlotBox()); - return group; - }, - /** - * Get the translation and scale for the plot area of this series. - * - * @function Highcharts.Series#getPlotBox - * - * @return {Highcharts.SeriesPlotBoxObject} - */ - getPlotBox: function () { - var chart = this.chart, - xAxis = this.xAxis, - yAxis = this.yAxis; - // Swap axes for inverted (#2339) - if (chart.inverted) { - xAxis = yAxis; - yAxis = this.xAxis; - } - return { - translateX: xAxis ? xAxis.left : chart.plotLeft, - translateY: yAxis ? yAxis.top : chart.plotTop, - scaleX: 1, - scaleY: 1 + // #859, #3569 + var horiz = axis.horiz, + minPixelPadding = + e.type === "touchend" ? axis.minPixelPadding : 0, // #1207, #3075 + selectionMin = axis.toValue( + (horiz ? selectionLeft : selectionTop) + minPixelPadding + ), + selectionMax = axis.toValue( + (horiz + ? selectionLeft + selectionWidth + : selectionTop + selectionHeight) - minPixelPadding + ); + selectionData[axis.coll].push({ + axis: axis, + // Min/max for reversed axes + min: Math.min(selectionMin, selectionMax), + max: Math.max(selectionMin, selectionMax), + }); + runZoom = true; + } + }); + if (runZoom) { + fireEvent(chart, "selection", selectionData, function (args) { + chart.zoom( + extend(args, hasPinched ? { animation: false } : null) + ); + }); + } + } + if (isNumber(chart.index)) { + this.selectionMarker = this.selectionMarker.destroy(); + } + // Reset scaling preview + if (hasPinched) { + this.scaleGroups(); + } + } + // Reset all. Check isNumber because it may be destroyed on mouse up + // (#877) + if (chart && isNumber(chart.index)) { + css(chart.container, { cursor: chart._cursor }); + chart.cancelClick = this.hasDragged > 10; // #370 + chart.mouseIsDown = this.hasDragged = this.hasPinched = false; + this.pinchDown = []; + } + }; + /** + * Finds the closest point to a set of coordinates, using the k-d-tree + * algorithm. + * + * @function Highcharts.Pointer#findNearestKDPoint + * + * @param {Array<Highcharts.Series>} series + * All the series to search in. + * + * @param {boolean|undefined} shared + * Whether it is a shared tooltip or not. + * + * @param {Highcharts.PointerEventObject} e + * The pointer event object, containing chart coordinates of the + * pointer. + * + * @return {Highcharts.Point|undefined} + * The point closest to given coordinates. + */ + Pointer.prototype.findNearestKDPoint = function (series, shared, e) { + var chart = this.chart; + var hoverPoint = chart.hoverPoint; + var tooltip = chart.tooltip; + if (hoverPoint && tooltip && tooltip.isStickyOnContact()) { + return hoverPoint; + } + var closest; + /** @private */ + function sort(p1, p2) { + var isCloserX = p1.distX - p2.distX, + isCloser = p1.dist - p2.dist, + isAbove = + (p2.series.group && p2.series.group.zIndex) - + (p1.series.group && p1.series.group.zIndex), + result; + // We have two points which are not in the same place on xAxis + // and shared tooltip: + if (isCloserX !== 0 && shared) { + // #5721 + result = isCloserX; + // Points are not exactly in the same place on x/yAxis: + } else if (isCloser !== 0) { + result = isCloser; + // The same xAxis and yAxis position, sort by z-index: + } else if (isAbove !== 0) { + result = isAbove; + // The same zIndex, sort by array index: + } else { + result = p1.series.index > p2.series.index ? -1 : 1; + } + return result; + } + series.forEach(function (s) { + var noSharedTooltip = s.noSharedTooltip && shared, + compareX = + !noSharedTooltip && + s.options.findNearestPointBy.indexOf("y") < 0, + point = s.searchPoint(e, compareX); + if ( + // Check that we actually found a point on the series. + isObject(point, true) && + // Use the new point if it is closer. + (!isObject(closest, true) || sort(closest, point) > 0) + ) { + closest = point; + } + }); + return closest; + }; + /** + * @private + * @function Highcharts.Pointer#getChartCoordinatesFromPoint + * @param {Highcharts.Point} point + * @param {boolean} [inverted] + * @return {Highcharts.PointerCoordinatesObject|undefined} + */ + Pointer.prototype.getChartCoordinatesFromPoint = function ( + point, + inverted + ) { + var series = point.series, + xAxis = series.xAxis, + yAxis = series.yAxis, + plotX = pick(point.clientX, point.plotX), + shapeArgs = point.shapeArgs; + if (xAxis && yAxis) { + return inverted + ? { + chartX: xAxis.len + xAxis.pos - plotX, + chartY: yAxis.len + yAxis.pos - point.plotY, + } + : { + chartX: plotX + xAxis.pos, + chartY: point.plotY + yAxis.pos, }; + } + if (shapeArgs && shapeArgs.x && shapeArgs.y) { + // E.g. pies do not have axes + return { + chartX: shapeArgs.x, + chartY: shapeArgs.y, + }; + } + }; + /** + * Return the cached chartPosition if it is available on the Pointer, + * otherwise find it. Running offset is quite expensive, so it should be + * avoided when we know the chart hasn't moved. + * + * @function Highcharts.Pointer#getChartPosition + * + * @return {Highcharts.OffsetObject} + * The offset of the chart container within the page + */ + Pointer.prototype.getChartPosition = function () { + return ( + this.chartPosition || + (this.chartPosition = offset(this.chart.container)) + ); + }; + /** + * Get the click position in terms of axis values. + * + * @function Highcharts.Pointer#getCoordinates + * + * @param {Highcharts.PointerEventObject} e + * Pointer event, extended with `chartX` and `chartY` properties. + * + * @return {Highcharts.PointerAxisCoordinatesObject} + */ + Pointer.prototype.getCoordinates = function (e) { + var coordinates = { + xAxis: [], + yAxis: [], + }; + this.chart.axes.forEach(function (axis) { + coordinates[axis.isXAxis ? "xAxis" : "yAxis"].push({ + axis: axis, + value: axis.toValue(e[axis.horiz ? "chartX" : "chartY"]), + }); + }); + return coordinates; + }; + /** + * Calculates what is the current hovered point/points and series. + * + * @private + * @function Highcharts.Pointer#getHoverData + * + * @param {Highcharts.Point|undefined} existingHoverPoint + * The point currrently beeing hovered. + * + * @param {Highcharts.Series|undefined} existingHoverSeries + * The series currently beeing hovered. + * + * @param {Array<Highcharts.Series>} series + * All the series in the chart. + * + * @param {boolean} isDirectTouch + * Is the pointer directly hovering the point. + * + * @param {boolean|undefined} shared + * Whether it is a shared tooltip or not. + * + * @param {Highcharts.PointerEventObject} [e] + * The triggering event, containing chart coordinates of the pointer. + * + * @return {object} + * Object containing resulting hover data: hoverPoint, hoverSeries, + * and hoverPoints. + */ + Pointer.prototype.getHoverData = function ( + existingHoverPoint, + existingHoverSeries, + series, + isDirectTouch, + shared, + e + ) { + var hoverPoint, + hoverPoints = [], + hoverSeries = existingHoverSeries, + useExisting = !!(isDirectTouch && existingHoverPoint), + notSticky = hoverSeries && !hoverSeries.stickyTracking, + // Which series to look in for the hover point + searchSeries, + // Parameters needed for beforeGetHoverData event. + eventArgs = { + chartX: e ? e.chartX : void 0, + chartY: e ? e.chartY : void 0, + shared: shared, }, - /** - * Removes the event handlers attached previously with addEvents. - * - * @private - * @function Highcharts.Series#removeEvents - * @param {boolean} [keepEventsForUpdate] - * @return {void} - */ - removeEvents: function (keepEventsForUpdate) { - var series = this; - if (!keepEventsForUpdate) { - // remove all events - removeEvent(series); - } - else if (series.eventsToUnbind.length) { - // remove only internal events for proper update - // #12355 - solves problem with multiple destroy events - series.eventsToUnbind.forEach(function (unbind) { - unbind(); - }); - series.eventsToUnbind.length = 0; - } - }, - /** - * Render the graph and markers. Called internally when first rendering - * and later when redrawing the chart. This function can be extended in - * plugins, but normally shouldn't be called directly. - * - * @function Highcharts.Series#render - * - * @return {void} - * - * @fires Highcharts.Series#event:afterRender - */ - render: function () { - var series = this, - chart = series.chart, - group, - options = series.options, - animOptions = animObject(options.animation), - // Animation doesn't work in IE8 quirks when the group div is - // hidden, and looks bad in other oldIE - animDuration = (!series.finishedAnimating && - chart.renderer.isSVG && - animOptions.duration), - visibility = series.visible ? 'inherit' : 'hidden', // #2597 - zIndex = options.zIndex, - hasRendered = series.hasRendered, - chartSeriesGroup = chart.seriesGroup, - inverted = chart.inverted; - fireEvent(this, 'render'); - // the group - group = series.plotGroup('group', 'series', visibility, zIndex, chartSeriesGroup); - series.markerGroup = series.plotGroup('markerGroup', 'markers', visibility, zIndex, chartSeriesGroup); - // initiate the animation - if (animDuration && series.animate) { - series.animate(true); - } - // SVGRenderer needs to know this before drawing elements (#1089, - // #1795) - group.inverted = series.isCartesian || series.invertable ? - inverted : false; - // Draw the graph if any - if (series.drawGraph) { - series.drawGraph(); - series.applyZones(); - } - // Draw the points - if (series.visible) { - series.drawPoints(); - } - /* series.points.forEach(function (point) { - if (point.redraw) { - point.redraw(); - } - }); */ - // Draw the data labels - if (series.drawDataLabels) { - series.drawDataLabels(); - } - // In pie charts, slices are added to the DOM, but actual rendering - // is postponed until labels reserved their space - if (series.redrawPoints) { - series.redrawPoints(); - } - // draw the mouse tracking area - if (series.drawTracker && - series.options.enableMouseTracking !== false) { - series.drawTracker(); - } - // Handle inverted series and tracker groups - series.invertGroups(inverted); - // Initial clipping, must be defined after inverting groups for VML. - // Applies to columns etc. (#3839). - if (options.clip !== false && - !series.sharedClipKey && - !hasRendered) { - group.clip(chart.clipRect); - } - // Run the animation - if (animDuration && series.animate) { - series.animate(); - } - // Call the afterAnimate function on animation complete (but don't - // overwrite the animation.complete option which should be available - // to the user). - if (!hasRendered) { - // Additional time if defer is defined before afterAnimate - // will be triggered - if (animDuration && animOptions.defer) { - animDuration += animOptions.defer; - } - series.animationTimeout = syncTimeout(function () { - series.afterAnimate(); - }, animDuration || 0); - } - // Means data is in accordance with what you see - series.isDirty = false; - // (See #322) series.isDirty = series.isDirtyData = false; // means - // data is in accordance with what you see - series.hasRendered = true; - fireEvent(series, 'afterRender'); - }, - /** - * Redraw the series. This function is called internally from - * `chart.redraw` and normally shouldn't be called directly. - * - * @private - * @function Highcharts.Series#redraw - * @return {void} - */ - redraw: function () { - var series = this, - chart = series.chart, - // cache it here as it is set to false in render, but used after - wasDirty = series.isDirty || series.isDirtyData, - group = series.group, - xAxis = series.xAxis, - yAxis = series.yAxis; - // reposition on resize - if (group) { - if (chart.inverted) { - group.attr({ - width: chart.plotWidth, - height: chart.plotHeight - }); - } - group.animate({ - translateX: pick(xAxis && xAxis.left, chart.plotLeft), - translateY: pick(yAxis && yAxis.top, chart.plotTop) - }); - } - series.translate(); - series.render(); - if (wasDirty) { // #3868, #3945 - delete this.kdTree; - } - }, - kdAxisArray: ['clientX', 'plotY'], - /** - * @private - * @function Highcharts.Series#searchPoint - * @param {Highcharts.PointerEventObject} e - * @param {boolean} [compareX] - * @return {Highcharts.Point} - */ - searchPoint: function (e, compareX) { - var series = this, - xAxis = series.xAxis, - yAxis = series.yAxis, - inverted = series.chart.inverted; - return this.searchKDTree({ - clientX: inverted ? - xAxis.len - e.chartY + xAxis.pos : - e.chartX - xAxis.pos, - plotY: inverted ? - yAxis.len - e.chartX + yAxis.pos : - e.chartY - yAxis.pos - }, compareX, e); - }, - /** - * Build the k-d-tree that is used by mouse and touch interaction to get - * the closest point. Line-like series typically have a one-dimensional - * tree where points are searched along the X axis, while scatter-like - * series typically search in two dimensions, X and Y. - * - * @private - * @function Highcharts.Series#buildKDTree - * @param {Highcharts.PointerEventObject} [e] - * @return {void} - */ - buildKDTree: function (e) { - // Prevent multiple k-d-trees from being built simultaneously - // (#6235) - this.buildingKdTree = true; - var series = this, - dimensions = series.options.findNearestPointBy - .indexOf('y') > -1 ? 2 : 1; - /** - * Internal function - * @private - */ - function _kdtree(points, depth, dimensions) { - var axis, - median, - length = points && points.length; - if (length) { - // alternate between the axis - axis = series.kdAxisArray[depth % dimensions]; - // sort point array - points.sort(function (a, b) { - return a[axis] - b[axis]; - }); - median = Math.floor(length / 2); - // build and return nod - return { - point: points[median], - left: _kdtree(points.slice(0, median), depth + 1, dimensions), - right: _kdtree(points.slice(median + 1), depth + 1, dimensions) - }; - } - } - /** - * Start the recursive build process with a clone of the points - * array and null points filtered out. (#3873) - * @private - */ - function startRecursive() { - series.kdTree = _kdtree(series.getValidPoints(null, - // For line-type series restrict to plot area, but - // column-type series not (#3916, #4511) - !series.directTouch), dimensions, dimensions); - series.buildingKdTree = false; - } - delete series.kdTree; - // For testing tooltips, don't build async. Also if touchstart, we - // may be dealing with click events on mobile, so don't delay - // (#6817). - syncTimeout(startRecursive, series.options.kdNow || (e && e.type === 'touchstart') ? 0 : 1); - }, - /** - * @private - * @function Highcharts.Series#searchKDTree - * @param {Highcharts.KDPointSearchObject} point - * @param {boolean} [compareX] - * @param {Highcharts.PointerEventObject} [e] - * @return {Highcharts.Point|undefined} - */ - searchKDTree: function (point, compareX, e) { - var series = this, - kdX = this.kdAxisArray[0], - kdY = this.kdAxisArray[1], - kdComparer = compareX ? 'distX' : 'dist', - kdDimensions = series.options.findNearestPointBy - .indexOf('y') > -1 ? 2 : 1; - /** - * Set the one and two dimensional distance on the point object. - * @private - */ - function setDistance(p1, p2) { - var x = (defined(p1[kdX]) && - defined(p2[kdX])) ? - Math.pow(p1[kdX] - p2[kdX], 2) : - null, - y = (defined(p1[kdY]) && - defined(p2[kdY])) ? - Math.pow(p1[kdY] - p2[kdY], 2) : - null, - r = (x || 0) + (y || 0); - p2.dist = defined(r) ? Math.sqrt(r) : Number.MAX_VALUE; - p2.distX = defined(x) ? Math.sqrt(x) : Number.MAX_VALUE; - } - /** - * @private - */ - function _search(search, tree, depth, dimensions) { - var point = tree.point, - axis = series.kdAxisArray[depth % dimensions], - tdist, - sideA, - sideB, - ret = point, - nPoint1, - nPoint2; - setDistance(search, point); - // Pick side based on distance to splitting point - tdist = search[axis] - point[axis]; - sideA = tdist < 0 ? 'left' : 'right'; - sideB = tdist < 0 ? 'right' : 'left'; - // End of tree - if (tree[sideA]) { - nPoint1 = _search(search, tree[sideA], depth + 1, dimensions); - ret = (nPoint1[kdComparer] < - ret[kdComparer] ? - nPoint1 : - point); - } - if (tree[sideB]) { - // compare distance to current best to splitting point to - // decide wether to check side B or not - if (Math.sqrt(tdist * tdist) < ret[kdComparer]) { - nPoint2 = _search(search, tree[sideB], depth + 1, dimensions); - ret = (nPoint2[kdComparer] < - ret[kdComparer] ? - nPoint2 : - ret); - } - } - return ret; - } - if (!this.kdTree && !this.buildingKdTree) { - this.buildKDTree(e); - } - if (this.kdTree) { - return _search(point, this.kdTree, kdDimensions, kdDimensions); - } - }, - /** - * @private - * @function Highcharts.Series#pointPlacementToXValue - * @return {number} - */ - pointPlacementToXValue: function () { - var _a = this, - _b = _a.options, - pointPlacement = _b.pointPlacement, - pointRange = _b.pointRange, - axis = _a.xAxis; - var factor = pointPlacement; - // Point placement is relative to each series pointRange (#5889) - if (factor === 'between') { - factor = axis.reversed ? -0.5 : 0.5; // #11955 - } - return isNumber(factor) ? - factor * pick(pointRange, axis.pointRange) : - 0; - }, - /** - * @private - * @function Highcharts.Series#isPointInside - * @param {Highcharts.Point} point - * @return {boolean} - */ - isPointInside: function (point) { - var isInside = typeof point.plotY !== 'undefined' && - typeof point.plotX !== 'undefined' && - point.plotY >= 0 && - point.plotY <= this.yAxis.len && // #3519 - point.plotX >= 0 && - point.plotX <= this.xAxis.len; - return isInside; - } - }); // end Series prototype - /** - * A line series displays information as a series of data points connected by - * straight line segments. - * - * @sample {highcharts} highcharts/demo/line-basic/ - * Line chart - * @sample {highstock} stock/demo/basic-line/ - * Line chart - * - * @extends plotOptions.series - * @product highcharts highstock - * @apioption plotOptions.line + filter = function (s) { + return ( + s.visible && + !(!shared && s.directTouch) && // #3821 + pick(s.options.enableMouseTracking, true) + ); + }; + // Find chart.hoverPane and update filter method in polar. + fireEvent(this, "beforeGetHoverData", eventArgs); + searchSeries = notSticky + ? // Only search on hovered series if it has stickyTracking false + [hoverSeries] + : // Filter what series to look in. + series.filter(function (s) { + return eventArgs.filter + ? eventArgs.filter(s) + : filter(s) && s.stickyTracking; + }); + // Use existing hovered point or find the one closest to coordinates. + hoverPoint = + useExisting || !e + ? existingHoverPoint + : this.findNearestKDPoint(searchSeries, shared, e); + // Assign hover series + hoverSeries = hoverPoint && hoverPoint.series; + // If we have a hoverPoint, assign hoverPoints. + if (hoverPoint) { + // When tooltip is shared, it displays more than one point + if (shared && !hoverSeries.noSharedTooltip) { + searchSeries = series.filter(function (s) { + return eventArgs.filter + ? eventArgs.filter(s) + : filter(s) && !s.noSharedTooltip; + }); + // Get all points with the same x value as the hoverPoint + searchSeries.forEach(function (s) { + var point = find(s.points, function (p) { + return p.x === hoverPoint.x && !p.isNull; + }); + if (isObject(point)) { + /* + * Boost returns a minimal point. Convert it to a usable + * point for tooltip and states. + */ + if (s.chart.isBoosting) { + point = s.getPoint(point); + } + hoverPoints.push(point); + } + }); + } else { + hoverPoints.push(hoverPoint); + } + } + // Check whether the hoverPoint is inside pane we are hovering over. + eventArgs = { hoverPoint: hoverPoint }; + fireEvent(this, "afterGetHoverData", eventArgs); + return { + hoverPoint: eventArgs.hoverPoint, + hoverSeries: hoverSeries, + hoverPoints: hoverPoints, + }; + }; + /** + * @private + * @function Highcharts.Pointer#getPointFromEvent + * + * @param {global.Event} e + * + * @return {Highcharts.Point|undefined} + */ + Pointer.prototype.getPointFromEvent = function (e) { + var target = e.target, + point; + while (target && !point) { + point = target.point; + target = target.parentNode; + } + return point; + }; + /** + * @private + * @function Highcharts.Pointer#onTrackerMouseOut + * + * @param {Highcharts.PointerEventObject} e + * + * @return {void} + */ + Pointer.prototype.onTrackerMouseOut = function (e) { + var chart = this.chart; + var relatedTarget = e.relatedTarget || e.toElement; + var series = chart.hoverSeries; + this.isDirectTouch = false; + if ( + series && + relatedTarget && + !series.stickyTracking && + !this.inClass(relatedTarget, "highcharts-tooltip") && + (!this.inClass( + relatedTarget, + "highcharts-series-" + series.index + ) || // #2499, #4465, #5553 + !this.inClass(relatedTarget, "highcharts-tracker")) + ) { + series.onMouseOut(); + } + }; + /** + * Utility to detect whether an element has, or has a parent with, a + * specificclass name. Used on detection of tracker objects and on deciding + * whether hovering the tooltip should cause the active series to mouse out. + * + * @function Highcharts.Pointer#inClass + * + * @param {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} element + * The element to investigate. + * + * @param {string} className + * The class name to look for. + * + * @return {boolean|undefined} + * True if either the element or one of its parents has the given + * class name. + */ + Pointer.prototype.inClass = function (element, className) { + var elemClassName; + while (element) { + elemClassName = attr(element, "class"); + if (elemClassName) { + if (elemClassName.indexOf(className) !== -1) { + return true; + } + if (elemClassName.indexOf("highcharts-container") !== -1) { + return false; + } + } + element = element.parentNode; + } + }; + /** + * Initialize the Pointer. + * + * @private + * @function Highcharts.Pointer#init + * + * @param {Highcharts.Chart} chart + * The Chart instance. + * + * @param {Highcharts.Options} options + * The root options object. The pointer uses options from the chart + * and tooltip structures. + * + * @return {void} + */ + Pointer.prototype.init = function (chart, options) { + // Store references + this.options = options; + this.chart = chart; + // Do we need to handle click on a touch device? + this.runChartClick = + options.chart.events && !!options.chart.events.click; + this.pinchDown = []; + this.lastValidTouch = {}; + if (Tooltip) { + /** + * Tooltip object for points of series. + * + * @name Highcharts.Chart#tooltip + * @type {Highcharts.Tooltip} + */ + chart.tooltip = new Tooltip(chart, options.tooltip); + this.followTouchMove = pick(options.tooltip.followTouchMove, true); + } + this.setDOMEvents(); + }; + /** + * Takes a browser event object and extends it with custom Highcharts + * properties `chartX` and `chartY` in order to work on the internal + * coordinate system. + * + * @function Highcharts.Pointer#normalize + * + * @param {global.MouseEvent|global.PointerEvent|global.TouchEvent} e + * Event object in standard browsers. + * + * @param {Highcharts.OffsetObject} [chartPosition] + * Additional chart offset. + * + * @return {Highcharts.PointerEventObject} + * A browser event with extended properties `chartX` and `chartY`. + */ + Pointer.prototype.normalize = function (e, chartPosition) { + var touches = e.touches; + // iOS (#2757) + var ePos = touches + ? touches.length + ? touches.item(0) + : pick( + // #13534 + touches.changedTouches, + e.changedTouches + )[0] + : e; + // Get mouse position + if (!chartPosition) { + chartPosition = this.getChartPosition(); + } + var chartX = ePos.pageX - chartPosition.left, + chartY = ePos.pageY - chartPosition.top; + // #11329 - when there is scaling on a parent element, we need to take + // this into account + var containerScaling = this.chart.containerScaling; + if (containerScaling) { + chartX /= containerScaling.scaleX; + chartY /= containerScaling.scaleY; + } + return extend(e, { + chartX: Math.round(chartX), + chartY: Math.round(chartY), + }); + }; + /** + * @private + * @function Highcharts.Pointer#onContainerClick + */ + Pointer.prototype.onContainerClick = function (e) { + var chart = this.chart; + var hoverPoint = chart.hoverPoint; + var pEvt = this.normalize(e); + var plotLeft = chart.plotLeft; + var plotTop = chart.plotTop; + if (!chart.cancelClick) { + // On tracker click, fire the series and point events. #783, #1583 + if (hoverPoint && this.inClass(pEvt.target, "highcharts-tracker")) { + // the series click event + fireEvent( + hoverPoint.series, + "click", + extend(pEvt, { + point: hoverPoint, + }) + ); + // the point click event + if (chart.hoverPoint) { + // it may be destroyed (#1844) + hoverPoint.firePointEvent("click", pEvt); + } + // When clicking outside a tracker, fire a chart event + } else { + extend(pEvt, this.getCoordinates(pEvt)); + // fire a click event in the chart + if ( + chart.isInsidePlot( + pEvt.chartX - plotLeft, + pEvt.chartY - plotTop + ) + ) { + fireEvent(chart, "click", pEvt); + } + } + } + }; + /** + * @private + * @function Highcharts.Pointer#onContainerMouseDown + * + * @param {global.MouseEvent} e + */ + Pointer.prototype.onContainerMouseDown = function (e) { + var isPrimaryButton = ((e.buttons || e.button) & 1) === 1; + // Normalize before the 'if' for the legacy IE (#7850) + e = this.normalize(e); + // #11635, Firefox does not reliable fire move event after click scroll + if (H.isFirefox && e.button !== 0) { + this.onContainerMouseMove(e); + } + // #11635, limiting to primary button (incl. IE 8 support) + if (typeof e.button === "undefined" || isPrimaryButton) { + this.zoomOption(e); + // #295, #13737 solve conflict between container drag and chart zoom + if (isPrimaryButton && e.preventDefault) { + e.preventDefault(); + } + this.dragStart(e); + } + }; + /** + * When mouse leaves the container, hide the tooltip. + * + * @private + * @function Highcharts.Pointer#onContainerMouseLeave + * + * @param {global.MouseEvent} e + * + * @return {void} + */ + Pointer.prototype.onContainerMouseLeave = function (e) { + var chart = charts[pick(H.hoverChartIndex, -1)]; + var tooltip = this.chart.tooltip; + e = this.normalize(e); + // #4886, MS Touch end fires mouseleave but with no related target + if (chart && (e.relatedTarget || e.toElement)) { + chart.pointer.reset(); + // Also reset the chart position, used in #149 fix + chart.pointer.chartPosition = void 0; + } + if ( + // #11635, Firefox wheel scroll does not fire out events consistently + tooltip && + !tooltip.isHidden + ) { + this.reset(); + } + }; + /** + * When mouse enters the container, delete pointer's chartPosition. + * + * @private + * @function Highcharts.Pointer#onContainerMouseEnter + * + * @param {global.MouseEvent} e + * + * @return {void} + */ + Pointer.prototype.onContainerMouseEnter = function (e) { + delete this.chartPosition; + }; + /** + * The mousemove, touchmove and touchstart event handler + * + * @private + * @function Highcharts.Pointer#onContainerMouseMove + * + * @param {global.MouseEvent} e + * + * @return {void} + */ + Pointer.prototype.onContainerMouseMove = function (e) { + var chart = this.chart; + var pEvt = this.normalize(e); + this.setHoverChartIndex(); + // In IE8 we apparently need this returnValue set to false in order to + // avoid text being selected. But in Chrome, e.returnValue is prevented, + // plus we don't need to run e.preventDefault to prevent selected text + // in modern browsers. So we set it conditionally. Remove it when IE8 is + // no longer needed. #2251, #3224. + if (!pEvt.preventDefault) { + pEvt.returnValue = false; + } + if (chart.mouseIsDown === "mousedown") { + this.drag(pEvt); + } + // Show the tooltip and run mouse over events (#977) + if ( + !chart.openMenu && + (this.inClass(pEvt.target, "highcharts-tracker") || + chart.isInsidePlot( + pEvt.chartX - chart.plotLeft, + pEvt.chartY - chart.plotTop + )) + ) { + this.runPointActions(pEvt); + } + }; + /** + * @private + * @function Highcharts.Pointer#onDocumentTouchEnd + * + * @param {Highcharts.PointerEventObject} e + * + * @return {void} + */ + Pointer.prototype.onDocumentTouchEnd = function (e) { + if (charts[H.hoverChartIndex]) { + charts[H.hoverChartIndex].pointer.drop(e); + } + }; + /** + * @private + * @function Highcharts.Pointer#onContainerTouchMove + * + * @param {Highcharts.PointerEventObject} e + * + * @return {void} */ + Pointer.prototype.onContainerTouchMove = function (e) { + this.touch(e); + }; + /** + * @private + * @function Highcharts.Pointer#onContainerTouchStart + * + * @param {Highcharts.PointerEventObject} e + * + * @return {void} + */ + Pointer.prototype.onContainerTouchStart = function (e) { + this.zoomOption(e); + this.touch(e, true); + }; + /** + * Special handler for mouse move that will hide the tooltip when the mouse + * leaves the plotarea. Issue #149 workaround. The mouseleave event does not + * always fire. + * + * @private + * @function Highcharts.Pointer#onDocumentMouseMove + * + * @param {global.MouseEvent} e + * + * @return {void} + */ + Pointer.prototype.onDocumentMouseMove = function (e) { + var chart = this.chart; + var chartPosition = this.chartPosition; + var pEvt = this.normalize(e, chartPosition); + var tooltip = chart.tooltip; + // If we're outside, hide the tooltip + if ( + chartPosition && + (!tooltip || !tooltip.isStickyOnContact()) && + !chart.isInsidePlot( + pEvt.chartX - chart.plotLeft, + pEvt.chartY - chart.plotTop + ) && + !this.inClass(pEvt.target, "highcharts-tracker") + ) { + this.reset(); + } + }; /** - * The SVG value used for the `stroke-linecap` and `stroke-linejoin` - * of a line graph. Round means that lines are rounded in the ends and - * bends. + * @private + * @function Highcharts.Pointer#onDocumentMouseUp + * + * @param {global.MouseEvent} e * - * @type {Highcharts.SeriesLinecapValue} - * @default round - * @since 3.0.7 - * @apioption plotOptions.line.linecap + * @return {void} */ + Pointer.prototype.onDocumentMouseUp = function (e) { + var chart = charts[pick(H.hoverChartIndex, -1)]; + if (chart) { + chart.pointer.drop(e); + } + }; /** - * A `line` series. If the [type](#series.line.type) option is not - * specified, it is inherited from [chart.type](#chart.type). + * Handle touch events with two touches * - * @extends series,plotOptions.line - * @excluding dataParser,dataURL - * @product highcharts highstock - * @apioption series.line - */ - /** - * An array of data points for the series. For the `line` series type, - * points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. The `x` values will be automatically - * calculated, either starting at 0 and incremented by 1, or from - * `pointStart` and `pointInterval` given in the series options. If the axis - * has categories, these will be used. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 2. An array of arrays with 2 values. In this case, the values correspond to - * `x,y`. If the first value is a string, it is applied as the name of the - * point, and the `x` value is inferred. - * ```js - * data: [ - * [0, 1], - * [1, 2], - * [2, 8] - * ] - * ``` - * - * 3. 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.line.turboThreshold), - * this option is not available. - * ```js - * data: [{ - * x: 1, - * y: 9, - * name: "Point2", - * color: "#00FF00" - * }, { - * x: 1, - * y: 6, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * **Note:** In TypeScript you have to extend `PointOptionsObject` with an - * additional declaration to allow custom data types: - * ```ts - * declare module `highcharts` { - * interface PointOptionsObject { - * custom: Record<string, (boolean|number|string)>; - * } - * } - * ``` + * @private + * @function Highcharts.Pointer#pinch * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects + * @param {Highcharts.PointerEventObject} e * - * @declare Highcharts.PointOptionsObject - * @type {Array<number|Array<(number|string),(number|null)>|null|*>} - * @apioption series.line.data + * @return {void} */ + Pointer.prototype.pinch = function (e) { + var self = this, + chart = self.chart, + pinchDown = self.pinchDown, + touches = e.touches || [], + touchesLength = touches.length, + lastValidTouch = self.lastValidTouch, + hasZoom = self.hasZoom, + selectionMarker = self.selectionMarker, + transform = {}, + fireClickEvent = + touchesLength === 1 && + ((self.inClass(e.target, "highcharts-tracker") && + chart.runTrackerClick) || + self.runChartClick), + clip = {}; + // Don't initiate panning until the user has pinched. This prevents us + // from blocking page scrolling as users scroll down a long page + // (#4210). + if (touchesLength > 1) { + self.initiated = true; + } + // On touch devices, only proceed to trigger click if a handler is + // defined + if ( + hasZoom && + self.initiated && + !fireClickEvent && + e.cancelable !== false + ) { + e.preventDefault(); + } + // Normalize each touch + [].map.call(touches, function (e) { + return self.normalize(e); + }); + // Register the touch start position + if (e.type === "touchstart") { + [].forEach.call(touches, function (e, i) { + pinchDown[i] = { chartX: e.chartX, chartY: e.chartY }; + }); + lastValidTouch.x = [ + pinchDown[0].chartX, + pinchDown[1] && pinchDown[1].chartX, + ]; + lastValidTouch.y = [ + pinchDown[0].chartY, + pinchDown[1] && pinchDown[1].chartY, + ]; + // Identify the data bounds in pixels + chart.axes.forEach(function (axis) { + if (axis.zoomEnabled) { + var bounds = chart.bounds[axis.horiz ? "h" : "v"], + minPixelPadding = axis.minPixelPadding, + min = axis.toPixels( + Math.min(pick(axis.options.min, axis.dataMin), axis.dataMin) + ), + max = axis.toPixels( + Math.max(pick(axis.options.max, axis.dataMax), axis.dataMax) + ), + absMin = Math.min(min, max), + absMax = Math.max(min, max); + // Store the bounds for use in the touchmove handler + bounds.min = Math.min(axis.pos, absMin - minPixelPadding); + bounds.max = Math.max( + axis.pos + axis.len, + absMax + minPixelPadding + ); + } + }); + self.res = true; // reset on next move + // Optionally move the tooltip on touchmove + } else if (self.followTouchMove && touchesLength === 1) { + this.runPointActions(self.normalize(e)); + // Event type is touchmove, handle panning and pinching + } else if (pinchDown.length) { + // can be 0 when releasing, if touchend + // fires first + // Set the marker + if (!selectionMarker) { + self.selectionMarker = selectionMarker = extend( + { + destroy: noop, + touch: true, + }, + chart.plotBox + ); + } + self.pinchTranslate( + pinchDown, + touches, + transform, + selectionMarker, + clip, + lastValidTouch + ); + self.hasPinched = hasZoom; + // Scale and translate the groups to provide visual feedback during + // pinching + self.scaleGroups(transform, clip); + if (self.res) { + self.res = false; + this.reset(false, 0); + } + } + }; /** - * An additional, individual class name for the data point's graphic - * representation. + * Run translation operations * - * @type {string} - * @since 5.0.0 - * @product highcharts gantt - * @apioption series.line.data.className - */ - /** - * Individual color for the point. By default the color is pulled from - * the global `colors` array. + * @private + * @function Highcharts.Pointer#pinchTranslate + * + * @param {Array<*>} pinchDown * - * In styled mode, the `color` option doesn't take effect. Instead, use - * `colorIndex`. + * @param {Array<Highcharts.PointerEventObject>} touches * - * @sample {highcharts} highcharts/point/color/ - * Mark the highest point + * @param {*} transform * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @product highcharts highstock gantt - * @apioption series.line.data.color + * @param {*} selectionMarker + * + * @param {*} clip + * + * @param {*} lastValidTouch + * + * @return {void} */ + Pointer.prototype.pinchTranslate = function ( + pinchDown, + touches, + transform, + selectionMarker, + clip, + lastValidTouch + ) { + if (this.zoomHor) { + this.pinchTranslateDirection( + true, + pinchDown, + touches, + transform, + selectionMarker, + clip, + lastValidTouch + ); + } + if (this.zoomVert) { + this.pinchTranslateDirection( + false, + pinchDown, + touches, + transform, + selectionMarker, + clip, + lastValidTouch + ); + } + }; /** - * A specific color index to use for the point, so its graphic representations - * are given the class name `highcharts-color-{n}`. In styled mode this will - * change the color of the graphic. In non-styled mode, the color by is set by - * the `fill` attribute, so the change in class name won't have a visual effect - * by default. + * Run translation operations for each direction (horizontal and vertical) + * independently. * - * @type {number} - * @since 5.0.0 - * @product highcharts gantt - * @apioption series.line.data.colorIndex + * @private + * @function Highcharts.Pointer#pinchTranslateDirection + * + * @param {boolean} horiz + * + * @param {Array<*>} pinchDown + * + * @param {Array<Highcharts.PointerEventObject>} touches + * + * @param {*} transform + * + * @param {*} selectionMarker + * + * @param {*} clip + * + * @param {*} lastValidTouch + * + * @param {number|undefined} [forcedScale=1] + * + * @return {void} */ + Pointer.prototype.pinchTranslateDirection = function ( + horiz, + pinchDown, + touches, + transform, + selectionMarker, + clip, + lastValidTouch, + forcedScale + ) { + var chart = this.chart, + xy = horiz ? "x" : "y", + XY = horiz ? "X" : "Y", + sChartXY = "chart" + XY, + wh = horiz ? "width" : "height", + plotLeftTop = chart["plot" + (horiz ? "Left" : "Top")], + selectionWH, + selectionXY, + clipXY, + scale = forcedScale || 1, + inverted = chart.inverted, + bounds = chart.bounds[horiz ? "h" : "v"], + singleTouch = pinchDown.length === 1, + touch0Start = pinchDown[0][sChartXY], + touch0Now = touches[0][sChartXY], + touch1Start = !singleTouch && pinchDown[1][sChartXY], + touch1Now = !singleTouch && touches[1][sChartXY], + outOfBounds, + transformScale, + scaleKey, + setScale = function () { + // Don't zoom if fingers are too close on this axis + if ( + typeof touch1Now === "number" && + Math.abs(touch0Start - touch1Start) > 20 + ) { + scale = + forcedScale || + Math.abs(touch0Now - touch1Now) / + Math.abs(touch0Start - touch1Start); + } + clipXY = (plotLeftTop - touch0Now) / scale + touch0Start; + selectionWH = + chart["plot" + (horiz ? "Width" : "Height")] / scale; + }; + // Set the scale, first pass + setScale(); + // The clip position (x or y) is altered if out of bounds, the selection + // position is not + selectionXY = clipXY; + // Out of bounds + if (selectionXY < bounds.min) { + selectionXY = bounds.min; + outOfBounds = true; + } else if (selectionXY + selectionWH > bounds.max) { + selectionXY = bounds.max - selectionWH; + outOfBounds = true; + } + // Is the chart dragged off its bounds, determined by dataMin and + // dataMax? + if (outOfBounds) { + // Modify the touchNow position in order to create an elastic drag + // movement. This indicates to the user that the chart is responsive + // but can't be dragged further. + touch0Now -= 0.8 * (touch0Now - lastValidTouch[xy][0]); + if (typeof touch1Now === "number") { + touch1Now -= 0.8 * (touch1Now - lastValidTouch[xy][1]); + } + // Set the scale, second pass to adapt to the modified touchNow + // positions + setScale(); + } else { + lastValidTouch[xy] = [touch0Now, touch1Now]; + } + // Set geometry for clipping, selection and transformation + if (!inverted) { + clip[xy] = clipXY - plotLeftTop; + clip[wh] = selectionWH; + } + scaleKey = inverted ? (horiz ? "scaleY" : "scaleX") : "scale" + XY; + transformScale = inverted ? 1 / scale : scale; + selectionMarker[wh] = selectionWH; + selectionMarker[xy] = selectionXY; + transform[scaleKey] = scale; + transform["translate" + XY] = + transformScale * plotLeftTop + + (touch0Now - transformScale * touch0Start); + }; /** - * A reserved subspace to store options and values for customized functionality. - * Here you can add additional data for your own event callbacks and formatter - * callbacks. + * Reset the tracking by hiding the tooltip, the hover series state and the + * hover point + * + * @function Highcharts.Pointer#reset + * + * @param {boolean} [allowMove] + * Instead of destroying the tooltip altogether, allow moving it if + * possible. * - * @sample {highcharts} highcharts/point/custom/ - * Point and series with custom data + * @param {number} [delay] * - * @type {Highcharts.Dictionary<*>} - * @apioption series.line.data.custom + * @return {void} */ + Pointer.prototype.reset = function (allowMove, delay) { + var pointer = this, + chart = pointer.chart, + hoverSeries = chart.hoverSeries, + hoverPoint = chart.hoverPoint, + hoverPoints = chart.hoverPoints, + tooltip = chart.tooltip, + tooltipPoints = + tooltip && tooltip.shared ? hoverPoints : hoverPoint; + // Check if the points have moved outside the plot area (#1003, #4736, + // #5101) + if (allowMove && tooltipPoints) { + splat(tooltipPoints).forEach(function (point) { + if ( + point.series.isCartesian && + typeof point.plotX === "undefined" + ) { + allowMove = false; + } + }); + } + // Just move the tooltip, #349 + if (allowMove) { + if (tooltip && tooltipPoints && splat(tooltipPoints).length) { + tooltip.refresh(tooltipPoints); + if (tooltip.shared && hoverPoints) { + // #8284 + hoverPoints.forEach(function (point) { + point.setState(point.state, true); + if (point.series.isCartesian) { + if (point.series.xAxis.crosshair) { + point.series.xAxis.drawCrosshair(null, point); + } + if (point.series.yAxis.crosshair) { + point.series.yAxis.drawCrosshair(null, point); + } + } + }); + } else if (hoverPoint) { + // #2500 + hoverPoint.setState(hoverPoint.state, true); + chart.axes.forEach(function (axis) { + if (axis.crosshair && hoverPoint.series[axis.coll] === axis) { + axis.drawCrosshair(null, hoverPoint); + } + }); + } + } + // Full reset + } else { + if (hoverPoint) { + hoverPoint.onMouseOut(); + } + if (hoverPoints) { + hoverPoints.forEach(function (point) { + point.setState(); + }); + } + if (hoverSeries) { + hoverSeries.onMouseOut(); + } + if (tooltip) { + tooltip.hide(delay); + } + if (pointer.unDocMouseMove) { + pointer.unDocMouseMove = pointer.unDocMouseMove(); + } + // Remove crosshairs + chart.axes.forEach(function (axis) { + axis.hideCrosshair(); + }); + pointer.hoverX = chart.hoverPoints = chart.hoverPoint = null; + } + }; /** - * Individual data label for each point. The options are the same as - * the ones for [plotOptions.series.dataLabels]( - * #plotOptions.series.dataLabels). + * With line type charts with a single tracker, get the point closest to the + * mouse. Run Point.onMouseOver and display tooltip for the point or points. + * + * @private + * @function Highcharts.Pointer#runPointActions * - * @sample highcharts/point/datalabels/ - * Show a label for the last value + * @param {global.Event} e * - * @declare Highcharts.DataLabelsOptions - * @extends plotOptions.line.dataLabels - * @product highcharts highstock gantt - * @apioption series.line.data.dataLabels - */ + * @param {Highcharts.PointerEventObject} [p] + * + * @return {void} + * + * @fires Highcharts.Point#event:mouseOut + * @fires Highcharts.Point#event:mouseOver + */ + Pointer.prototype.runPointActions = function (e, p) { + var pointer = this, + chart = pointer.chart, + series = chart.series, + tooltip = + chart.tooltip && chart.tooltip.options.enabled + ? chart.tooltip + : void 0, + shared = tooltip ? tooltip.shared : false, + hoverPoint = p || chart.hoverPoint, + hoverSeries = + (hoverPoint && hoverPoint.series) || chart.hoverSeries, + // onMouseOver or already hovering a series with directTouch + isDirectTouch = + (!e || e.type !== "touchmove") && + (!!p || + (hoverSeries && + hoverSeries.directTouch && + pointer.isDirectTouch)), + hoverData = this.getHoverData( + hoverPoint, + hoverSeries, + series, + isDirectTouch, + shared, + e + ), + useSharedTooltip, + followPointer, + anchor, + points; + // Update variables from hoverData. + hoverPoint = hoverData.hoverPoint; + points = hoverData.hoverPoints; + hoverSeries = hoverData.hoverSeries; + followPointer = + hoverSeries && hoverSeries.tooltipOptions.followPointer; + useSharedTooltip = + shared && hoverSeries && !hoverSeries.noSharedTooltip; + // Refresh tooltip for kdpoint if new hover point or tooltip was hidden + // #3926, #4200 + if ( + hoverPoint && + // !(hoverSeries && hoverSeries.directTouch) && + (hoverPoint !== chart.hoverPoint || (tooltip && tooltip.isHidden)) + ) { + (chart.hoverPoints || []).forEach(function (p) { + if (points.indexOf(p) === -1) { + p.setState(); + } + }); + // Set normal state to previous series + if (chart.hoverSeries !== hoverSeries) { + hoverSeries.onMouseOver(); + } + pointer.applyInactiveState(points); + // Do mouseover on all points (#3919, #3985, #4410, #5622) + (points || []).forEach(function (p) { + p.setState("hover"); + }); + // If tracking is on series in stead of on each point, + // fire mouseOver on hover point. // #4448 + if (chart.hoverPoint) { + chart.hoverPoint.firePointEvent("mouseOut"); + } + // Hover point may have been destroyed in the event handlers (#7127) + if (!hoverPoint.series) { + return; + } + /** + * Contains all hovered points. + * + * @name Highcharts.Chart#hoverPoints + * @type {Array<Highcharts.Point>|null} + */ + chart.hoverPoints = points; + /** + * Contains the original hovered point. + * + * @name Highcharts.Chart#hoverPoint + * @type {Highcharts.Point|null} + */ + chart.hoverPoint = hoverPoint; + /** + * Hover state should not be lost when axis is updated (#12569) + * Axis.update runs pointer.reset which uses chart.hoverPoint.state + * to apply state which does not exist in hoverPoint yet. + * The mouseOver event should be triggered when hoverPoint + * is correct. + */ + hoverPoint.firePointEvent("mouseOver"); + // Draw tooltip if necessary + if (tooltip) { + tooltip.refresh(useSharedTooltip ? points : hoverPoint, e); + } + // Update positions (regardless of kdpoint or hoverPoint) + } else if (followPointer && tooltip && !tooltip.isHidden) { + anchor = tooltip.getAnchor([{}], e); + tooltip.updatePosition({ plotX: anchor[0], plotY: anchor[1] }); + } + // Start the event listener to pick up the tooltip and crosshairs + if (!pointer.unDocMouseMove) { + pointer.unDocMouseMove = addEvent( + chart.container.ownerDocument, + "mousemove", + function (e) { + var chart = charts[H.hoverChartIndex]; + if (chart) { + chart.pointer.onDocumentMouseMove(e); + } + } + ); + } + // Issues related to crosshair #4927, #5269 #5066, #5658 + chart.axes.forEach(function drawAxisCrosshair(axis) { + var snap = pick((axis.crosshair || {}).snap, true); + var point; + if (snap) { + point = chart.hoverPoint; // #13002 + if (!point || point.series[axis.coll] !== axis) { + point = find(points, function (p) { + return p.series[axis.coll] === axis; + }); + } + } + // Axis has snapping crosshairs, and one of the hover points belongs + // to axis. Always call drawCrosshair when it is not snap. + if (point || !snap) { + axis.drawCrosshair(e, point); + // Axis has snapping crosshairs, but no hover point belongs to axis + } else { + axis.hideCrosshair(); + } + }); + }; /** - * A description of the point to add to the screen reader information - * about the point. + * Scale series groups to a certain scale and translation. * - * @type {string} - * @since 5.0.0 - * @requires modules/accessibility - * @apioption series.line.data.description + * @private + * @function Highcharts.Pointer#scaleGroups + * + * @param {Highcharts.SeriesPlotBoxObject} [attribs] + * + * @param {boolean} [clip] + * + * @return {void} */ + Pointer.prototype.scaleGroups = function (attribs, clip) { + var chart = this.chart, + seriesAttribs; + // Scale each series + chart.series.forEach(function (series) { + seriesAttribs = attribs || series.getPlotBox(); // #1701 + if (series.xAxis && series.xAxis.zoomEnabled && series.group) { + series.group.attr(seriesAttribs); + if (series.markerGroup) { + series.markerGroup.attr(seriesAttribs); + series.markerGroup.clip(clip ? chart.clipRect : null); + } + if (series.dataLabelsGroup) { + series.dataLabelsGroup.attr(seriesAttribs); + } + } + }); + // Clip + chart.clipRect.attr(clip || chart.clipBox); + }; /** - * An id for the point. This can be used after render time to get a - * pointer to the point object through `chart.get()`. + * Set the JS DOM events on the container and document. This method should + * contain a one-to-one assignment between methods and their handlers. Any + * advanced logic should be moved to the handler reflecting the event's + * name. * - * @sample {highcharts} highcharts/point/id/ - * Remove an id'd point + * @private + * @function Highcharts.Pointer#setDOMEvents * - * @type {string} - * @since 1.2.0 - * @product highcharts highstock gantt - * @apioption series.line.data.id + * @return {void} */ + Pointer.prototype.setDOMEvents = function () { + var container = this.chart.container, + ownerDoc = container.ownerDocument; + container.onmousedown = this.onContainerMouseDown.bind(this); + container.onmousemove = this.onContainerMouseMove.bind(this); + container.onclick = this.onContainerClick.bind(this); + this.unbindContainerMouseEnter = addEvent( + container, + "mouseenter", + this.onContainerMouseEnter.bind(this) + ); + this.unbindContainerMouseLeave = addEvent( + container, + "mouseleave", + this.onContainerMouseLeave.bind(this) + ); + if (!H.unbindDocumentMouseUp) { + H.unbindDocumentMouseUp = addEvent( + ownerDoc, + "mouseup", + this.onDocumentMouseUp.bind(this) + ); + } + if (H.hasTouch) { + addEvent( + container, + "touchstart", + this.onContainerTouchStart.bind(this) + ); + addEvent( + container, + "touchmove", + this.onContainerTouchMove.bind(this) + ); + if (!H.unbindDocumentTouchEnd) { + H.unbindDocumentTouchEnd = addEvent( + ownerDoc, + "touchend", + this.onDocumentTouchEnd.bind(this) + ); + } + } + }; /** - * The rank for this point's data label in case of collision. If two - * data labels are about to overlap, only the one with the highest `labelrank` - * will be drawn. + * Sets the index of the hovered chart and leaves the previous hovered + * chart, to reset states like tooltip. * - * @type {number} - * @apioption series.line.data.labelrank - */ + * @private + * @function Highcharts.Pointer#setHoverChartIndex + */ + Pointer.prototype.setHoverChartIndex = function () { + var chart = this.chart; + var hoverChart = H.charts[pick(H.hoverChartIndex, -1)]; + if (hoverChart && hoverChart !== chart) { + hoverChart.pointer.onContainerMouseLeave({ relatedTarget: true }); + } + if (!hoverChart || !hoverChart.mouseIsDown) { + H.hoverChartIndex = chart.index; + } + }; /** - * The name of the point as shown in the legend, tooltip, dataLabels, etc. + * General touch handler shared by touchstart and touchmove. * - * @see [xAxis.uniqueNames](#xAxis.uniqueNames) + * @private + * @function Highcharts.Pointer#touch * - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Point names + * @param {Highcharts.PointerEventObject} e * - * @type {string} - * @apioption series.line.data.name + * @param {boolean} [start] + * + * @return {void} */ + Pointer.prototype.touch = function (e, start) { + var chart = this.chart, + hasMoved, + pinchDown, + isInside; + this.setHoverChartIndex(); + if (e.touches.length === 1) { + e = this.normalize(e); + isInside = chart.isInsidePlot( + e.chartX - chart.plotLeft, + e.chartY - chart.plotTop + ); + if (isInside && !chart.openMenu) { + // Run mouse events and display tooltip etc + if (start) { + this.runPointActions(e); + } + // Android fires touchmove events after the touchstart even if + // the finger hasn't moved, or moved only a pixel or two. In iOS + // however, the touchmove doesn't fire unless the finger moves + // more than ~4px. So we emulate this behaviour in Android by + // checking how much it moved, and cancelling on small + // distances. #3450. + if (e.type === "touchmove") { + pinchDown = this.pinchDown; + hasMoved = pinchDown[0] + ? Math.sqrt( + // #5266 + Math.pow(pinchDown[0].chartX - e.chartX, 2) + + Math.pow(pinchDown[0].chartY - e.chartY, 2) + ) >= 4 + : false; + } + if (pick(hasMoved, true)) { + this.pinch(e); + } + } else if (start) { + // Hide the tooltip on touching outside the plot area (#1203) + this.reset(); + } + } else if (e.touches.length === 2) { + this.pinch(e); + } + }; /** - * Whether the data point is selected initially. + * Resolve the zoomType option, this is reset on all touch start and mouse + * down events. * - * @type {boolean} - * @default false - * @product highcharts highstock gantt - * @apioption series.line.data.selected - */ + * @private + * @function Highcharts.Pointer#zoomOption + * + * @param {global.Event} e + * Event object. + * + * @param {void} + */ + Pointer.prototype.zoomOption = function (e) { + var chart = this.chart, + options = chart.options.chart, + zoomType = options.zoomType || "", + inverted = chart.inverted, + zoomX, + zoomY; + // Look for the pinchType option + if (/touch/.test(e.type)) { + zoomType = pick(options.pinchType, zoomType); + } + this.zoomX = zoomX = /x/.test(zoomType); + this.zoomY = zoomY = /y/.test(zoomType); + this.zoomHor = (zoomX && !inverted) || (zoomY && inverted); + this.zoomVert = (zoomY && !inverted) || (zoomX && inverted); + this.hasZoom = zoomX || zoomY; + }; + return Pointer; + })(); + H.Pointer = Pointer; + + return Pointer; + } + ); + _registerModule( + _modules, + "Core/MSPointer.js", + [ + _modules["Core/Globals.js"], + _modules["Core/Pointer.js"], + _modules["Core/Utilities.js"], + ], + function (H, Pointer, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var __extends = + (this && this.__extends) || + (function () { + var extendStatics = function (d, b) { + extendStatics = + Object.setPrototypeOf || + ({ __proto__: [] } instanceof Array && + function (d, b) { + d.__proto__ = b; + }) || + function (d, b) { + for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; + }; + return extendStatics(d, b); + }; + return function (d, b) { + extendStatics(d, b); + function __() { + this.constructor = d; + } + d.prototype = + b === null + ? Object.create(b) + : ((__.prototype = b.prototype), new __()); + }; + })(); + var charts = H.charts, + doc = H.doc, + noop = H.noop, + win = H.win; + var addEvent = U.addEvent, + css = U.css, + objectEach = U.objectEach, + removeEvent = U.removeEvent; + /* globals MSPointerEvent, PointerEvent */ + // The touches object keeps track of the points being touched at all times + var touches = {}; + var hasPointerEvent = !!win.PointerEvent; + /* eslint-disable valid-jsdoc */ + /** @private */ + function getWebkitTouches() { + var fake = []; + fake.item = function (i) { + return this[i]; + }; + objectEach(touches, function (touch) { + fake.push({ + pageX: touch.pageX, + pageY: touch.pageY, + target: touch.target, + }); + }); + return fake; + } + /** @private */ + function translateMSPointer(e, method, wktype, func) { + var p; + if ( + (e.pointerType === "touch" || + e.pointerType === e.MSPOINTER_TYPE_TOUCH) && + charts[H.hoverChartIndex] + ) { + func(e); + p = charts[H.hoverChartIndex].pointer; + p[method]({ + type: wktype, + target: e.currentTarget, + preventDefault: noop, + touches: getWebkitTouches(), + }); + } + } + /** @private */ + var MSPointer = /** @class */ (function (_super) { + __extends(MSPointer, _super); + function MSPointer() { + return (_super !== null && _super.apply(this, arguments)) || this; + } + /* * + * + * Functions + * + * */ /** - * The x value of the point. For datetime axes, the X value is the timestamp - * in milliseconds since 1970. + * Add or remove the MS Pointer specific events * - * @type {number} - * @product highcharts highstock - * @apioption series.line.data.x + * @private + * @function Highcharts.Pointer#batchMSEvents + * + * @param {Function} fn + * + * @return {void} */ + MSPointer.prototype.batchMSEvents = function (fn) { + fn( + this.chart.container, + hasPointerEvent ? "pointerdown" : "MSPointerDown", + this.onContainerPointerDown + ); + fn( + this.chart.container, + hasPointerEvent ? "pointermove" : "MSPointerMove", + this.onContainerPointerMove + ); + fn( + doc, + hasPointerEvent ? "pointerup" : "MSPointerUp", + this.onDocumentPointerUp + ); + }; + // Destroy MS events also + MSPointer.prototype.destroy = function () { + this.batchMSEvents(removeEvent); + _super.prototype.destroy.call(this); + }; + // Disable default IE actions for pinch and such on chart element + MSPointer.prototype.init = function (chart, options) { + _super.prototype.init.call(this, chart, options); + if (this.hasZoom) { + // #4014 + css(chart.container, { + "-ms-touch-action": "none", + "touch-action": "none", + }); + } + }; /** - * The y value of the point. + * @private + * @function Highcharts.Pointer#onContainerPointerDown * - * @type {number|null} - * @product highcharts highstock - * @apioption series.line.data.y + * @param {Highcharts.PointerEventObject} e + * + * @return {void} */ + MSPointer.prototype.onContainerPointerDown = function (e) { + translateMSPointer( + e, + "onContainerTouchStart", + "touchstart", + function (e) { + touches[e.pointerId] = { + pageX: e.pageX, + pageY: e.pageY, + target: e.currentTarget, + }; + } + ); + }; /** - * The individual point events. + * @private + * @function Highcharts.Pointer#onContainerPointerMove + * + * @param {Highcharts.PointerEventObject} e * - * @extends plotOptions.series.point.events - * @product highcharts highstock gantt - * @apioption series.line.data.events + * @return {void} */ + MSPointer.prototype.onContainerPointerMove = function (e) { + translateMSPointer( + e, + "onContainerTouchMove", + "touchmove", + function (e) { + touches[e.pointerId] = { pageX: e.pageX, pageY: e.pageY }; + if (!touches[e.pointerId].target) { + touches[e.pointerId].target = e.currentTarget; + } + } + ); + }; /** - * Options for the point markers of line-like series. + * @private + * @function Highcharts.Pointer#onDocumentPointerUp * - * @declare Highcharts.PointMarkerOptionsObject - * @extends plotOptions.series.marker - * @product highcharts highstock - * @apioption series.line.data.marker + * @param {Highcharts.PointerEventObject} e + * + * @return {void} */ - ''; // include precedent doclets in transpilat + MSPointer.prototype.onDocumentPointerUp = function (e) { + translateMSPointer(e, "onDocumentTouchEnd", "touchend", function (e) { + delete touches[e.pointerId]; + }); + }; + // Add IE specific touch events to chart + MSPointer.prototype.setDOMEvents = function () { + _super.prototype.setDOMEvents.call(this); + if (this.hasZoom || this.followTouchMove) { + this.batchMSEvents(addEvent); + } + }; + return MSPointer; + })(Pointer); - return CartesianSeries; - }); - _registerModule(_modules, 'Series/LineSeries.js', [_modules['Core/Series/CartesianSeries.js'], _modules['Core/Globals.js']], function (CartesianSeries, H) { + return MSPointer; + } + ); + _registerModule( + _modules, + "Core/Legend.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Globals.js"], + _modules["Core/Utilities.js"], + ], + function (A, H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animObject = A.animObject, + setAnimation = A.setAnimation; + /** + * Gets fired when the legend item belonging to a point is clicked. The default + * action is to toggle the visibility of the point. This can be prevented by + * returning `false` or calling `event.preventDefault()`. + * + * @callback Highcharts.PointLegendItemClickCallbackFunction + * + * @param {Highcharts.Point} this + * The point on which the event occured. + * + * @param {Highcharts.PointLegendItemClickEventObject} event + * The event that occured. + */ + /** + * Information about the legend click event. + * + * @interface Highcharts.PointLegendItemClickEventObject + */ /** + * Related browser event. + * @name Highcharts.PointLegendItemClickEventObject#browserEvent + * @type {Highcharts.PointerEvent} + */ /** + * Prevent the default action of toggle the visibility of the point. + * @name Highcharts.PointLegendItemClickEventObject#preventDefault + * @type {Function} + */ /** + * Related point. + * @name Highcharts.PointLegendItemClickEventObject#target + * @type {Highcharts.Point} + */ /** + * Event type. + * @name Highcharts.PointLegendItemClickEventObject#type + * @type {"legendItemClick"} + */ + /** + * Gets fired when the legend item belonging to a series is clicked. The default + * action is to toggle the visibility of the series. This can be prevented by + * returning `false` or calling `event.preventDefault()`. + * + * @callback Highcharts.SeriesLegendItemClickCallbackFunction + * + * @param {Highcharts.Series} this + * The series where the event occured. + * + * @param {Highcharts.SeriesLegendItemClickEventObject} event + * The event that occured. + */ + /** + * Information about the legend click event. + * + * @interface Highcharts.SeriesLegendItemClickEventObject + */ /** + * Related browser event. + * @name Highcharts.SeriesLegendItemClickEventObject#browserEvent + * @type {Highcharts.PointerEvent} + */ /** + * Prevent the default action of toggle the visibility of the series. + * @name Highcharts.SeriesLegendItemClickEventObject#preventDefault + * @type {Function} + */ /** + * Related series. + * @name Highcharts.SeriesLegendItemClickEventObject#target + * @type {Highcharts.Series} + */ /** + * Event type. + * @name Highcharts.SeriesLegendItemClickEventObject#type + * @type {"legendItemClick"} + */ + var addEvent = U.addEvent, + css = U.css, + defined = U.defined, + discardElement = U.discardElement, + find = U.find, + fireEvent = U.fireEvent, + format = U.format, + isNumber = U.isNumber, + merge = U.merge, + pick = U.pick, + relativeLength = U.relativeLength, + stableSort = U.stableSort, + syncTimeout = U.syncTimeout, + wrap = U.wrap; + var isFirefox = H.isFirefox, + marginNames = H.marginNames, + win = H.win; + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The overview of the chart's series. The legend object is instanciated + * internally in the chart constructor, and is available from the `chart.legend` + * property. Each chart has only one legend. + * + * @class + * @name Highcharts.Legend + * + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @param {Highcharts.LegendOptions} options + * Legend options. + */ + var Legend = /** @class */ (function () { /* * * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * Constructors * * */ - H.Series = CartesianSeries; // backwards compatibility - - return H.Series; - }); - _registerModule(_modules, 'Extensions/Stacking.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Axis/StackingAxis.js'], _modules['Core/Utilities.js']], function (Axis, Chart, H, StackingAxis, U) { + function Legend(chart, options) { + /* * + * + * Properties + * + * */ + this.allItems = []; + this.box = void 0; + this.contentGroup = void 0; + this.display = false; + this.group = void 0; + this.initialItemY = 0; + this.itemHeight = 0; + this.itemMarginBottom = 0; + this.itemMarginTop = 0; + this.itemX = 0; + this.itemY = 0; + this.lastItemY = 0; + this.lastLineHeight = 0; + this.legendHeight = 0; + this.legendWidth = 0; + this.maxItemWidth = 0; + this.maxLegendWidth = 0; + this.offsetWidth = 0; + this.options = {}; + this.padding = 0; + this.pages = []; + this.proximate = false; + this.scrollGroup = void 0; + this.symbolHeight = 0; + this.symbolWidth = 0; + this.titleHeight = 0; + this.totalItemWidth = 0; + this.widthOption = 0; + this.chart = chart; + this.init(chart, options); + } /* * * - * (c) 2010-2020 Torstein Honsi + * Functions * - * License: www.highcharts.com/license + * */ + /** + * Initialize the legend. * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @private + * @function Highcharts.Legend#init * - * */ - var correctFloat = U.correctFloat, - defined = U.defined, - destroyObjectProperties = U.destroyObjectProperties, - format = U.format, - isNumber = U.isNumber, - pick = U.pick; + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @param {Highcharts.LegendOptions} options + * Legend options. + */ + Legend.prototype.init = function (chart, options) { + /** + * Chart of this legend. + * + * @readonly + * @name Highcharts.Legend#chart + * @type {Highcharts.Chart} + */ + this.chart = chart; + this.setOptions(options); + if (options.enabled) { + // Render it + this.render(); + // move checkboxes + addEvent(this.chart, "endResize", function () { + this.legend.positionCheckboxes(); + }); + if (this.proximate) { + this.unchartrender = addEvent(this.chart, "render", function () { + this.legend.proximatePositions(); + this.legend.positionItems(); + }); + } else if (this.unchartrender) { + this.unchartrender(); + } + } + }; /** - * Stack of data points + * @private + * @function Highcharts.Legend#setOptions + * @param {Highcharts.LegendOptions} options + */ + Legend.prototype.setOptions = function (options) { + var padding = pick(options.padding, 8); + /** + * Legend options. + * + * @readonly + * @name Highcharts.Legend#options + * @type {Highcharts.LegendOptions} + */ + this.options = options; + if (!this.chart.styledMode) { + this.itemStyle = options.itemStyle; + this.itemHiddenStyle = merge( + this.itemStyle, + options.itemHiddenStyle + ); + } + this.itemMarginTop = options.itemMarginTop || 0; + this.itemMarginBottom = options.itemMarginBottom || 0; + this.padding = padding; + this.initialItemY = padding - 5; // 5 is pixels above the text + this.symbolWidth = pick(options.symbolWidth, 16); + this.pages = []; + this.proximate = + options.layout === "proximate" && !this.chart.inverted; + this.baseline = void 0; // #12705: baseline has to be reset on every update + }; + /** + * Update the legend with new options. Equivalent to running `chart.update` + * with a legend configuration option. * - * @product highcharts + * @sample highcharts/legend/legend-update/ + * Legend update * - * @interface Highcharts.StackItemObject - */ /** - * Alignment settings - * @name Highcharts.StackItemObject#alignOptions - * @type {Highcharts.AlignObject} - */ /** - * Related axis - * @name Highcharts.StackItemObject#axis - * @type {Highcharts.Axis} - */ /** - * Cumulative value of the stacked data points - * @name Highcharts.StackItemObject#cumulative - * @type {number} - */ /** - * True if on the negative side - * @name Highcharts.StackItemObject#isNegative - * @type {boolean} - */ /** - * Related SVG element - * @name Highcharts.StackItemObject#label - * @type {Highcharts.SVGElement} - */ /** - * Related stack options - * @name Highcharts.StackItemObject#options - * @type {Highcharts.YAxisStackLabelsOptions} - */ /** - * Total value of the stacked data points - * @name Highcharts.StackItemObject#total - * @type {number} - */ /** - * Shared x value of the stack - * @name Highcharts.StackItemObject#x - * @type {number} - */ - ''; // detached doclets above - var Series = H.Series; - /* eslint-disable no-invalid-this, valid-jsdoc */ + * @function Highcharts.Legend#update + * + * @param {Highcharts.LegendOptions} options + * Legend options. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the axis is altered. If doing more + * operations on the chart, it is a good idea to set redraw to false and + * call {@link Chart#redraw} after. Whether to redraw the chart. + * + * @fires Highcharts.Legends#event:afterUpdate + */ + Legend.prototype.update = function (options, redraw) { + var chart = this.chart; + this.setOptions(merge(true, this.options, options)); + this.destroy(); + chart.isDirtyLegend = chart.isDirtyBox = true; + if (pick(redraw, true)) { + chart.redraw(); + } + fireEvent(this, "afterUpdate"); + }; /** - * The class for stacks. Each stack, on a specific X value and either negative - * or positive, has its own stack item. + * Set the colors for the legend item. * * @private - * @class - * @name Highcharts.StackItem - * @param {Highcharts.Axis} axis - * @param {Highcharts.YAxisStackLabelsOptions} options - * @param {boolean} isNegative - * @param {number} x - * @param {Highcharts.OptionsStackingValue} [stackOption] - */ - var StackItem = /** @class */ (function () { - function StackItem(axis, options, isNegative, x, stackOption) { - var inverted = axis.chart.inverted; - this.axis = axis; - // Tells if the stack is negative - this.isNegative = isNegative; - // Save the options to be able to style the label - this.options = options = options || {}; - // Save the x value to be able to position the label later - this.x = x; - // Initialize total value - this.total = null; - // This will keep each points' extremes stored by series.index and point - // index - this.points = {}; - this.hasValidPoints = false; - // Save the stack option on the series configuration object, - // and whether to treat it as percent - this.stack = stackOption; - this.leftCliff = 0; - this.rightCliff = 0; - // The align options and text align varies on whether the stack is - // negative and if the chart is inverted or not. - // First test the user supplied value, then use the dynamic. - this.alignOptions = { - align: options.align || - (inverted ? (isNegative ? 'left' : 'right') : 'center'), - verticalAlign: options.verticalAlign || - (inverted ? 'middle' : (isNegative ? 'bottom' : 'top')), - y: options.y, - x: options.x - }; - this.textAlign = options.textAlign || - (inverted ? (isNegative ? 'right' : 'left') : 'center'); + * @function Highcharts.Legend#colorizeItem + * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item + * A Series or Point instance + * @param {boolean} [visible=false] + * Dimmed or colored + * + * @todo + * Make events official: Fires the event `afterColorizeItem`. + */ + Legend.prototype.colorizeItem = function (item, visible) { + item.legendGroup[visible ? "removeClass" : "addClass"]( + "highcharts-legend-item-hidden" + ); + if (!this.chart.styledMode) { + var legend = this, + options = legend.options, + legendItem = item.legendItem, + legendLine = item.legendLine, + legendSymbol = item.legendSymbol, + hiddenColor = legend.itemHiddenStyle.color, + textColor = visible ? options.itemStyle.color : hiddenColor, + symbolColor = visible ? item.color || hiddenColor : hiddenColor, + markerOptions = item.options && item.options.marker, + symbolAttr = { fill: symbolColor }; + if (legendItem) { + legendItem.css({ + fill: textColor, + color: textColor, // #1553, oldIE + }); } - /** - * @private - * @function Highcharts.StackItem#destroy - */ - StackItem.prototype.destroy = function () { - destroyObjectProperties(this, this.axis); - }; - /** - * Renders the stack total label and adds it to the stack label group. - * - * @private - * @function Highcharts.StackItem#render - * @param {Highcharts.SVGElement} group - */ - StackItem.prototype.render = function (group) { - var chart = this.axis.chart, - options = this.options, - formatOption = options.format, - attr = {}, - str = formatOption ? // format the text in the label - format(formatOption, - this, - chart) : - options.formatter.call(this); - // Change the text to reflect the new total and set visibility to hidden - // in case the serie is hidden - if (this.label) { - this.label.attr({ text: str, visibility: 'hidden' }); - } - else { - // Create new label - this.label = chart.renderer - .label(str, null, null, options.shape, null, null, options.useHTML, false, 'stack-labels'); - attr = { - r: options.borderRadius || 0, - text: str, - rotation: options.rotation, - padding: pick(options.padding, 5), - visibility: 'hidden' // hidden until setOffset is called - }; - if (!chart.styledMode) { - attr.fill = options.backgroundColor; - attr.stroke = options.borderColor; - attr['stroke-width'] = options.borderWidth; - this.label.css(options.style); - } - this.label.attr(attr); - if (!this.label.added) { - this.label.add(group); // add to the labels-group - } - } - // Rank it higher than data labels (#8742) - this.label.labelrank = chart.plotHeight; - }; - /** - * Sets the offset that the stack has from the x value and repositions the - * label. - * - * @private - * @function Highcarts.StackItem#setOffset - * @param {number} xOffset - * @param {number} xWidth - * @param {number} [boxBottom] - * @param {number} [boxTop] - * @param {number} [defaultX] - */ - StackItem.prototype.setOffset = function (xOffset, xWidth, boxBottom, boxTop, defaultX) { - var stackItem = this, - axis = stackItem.axis, - chart = axis.chart, - // stack value translated mapped to chart coordinates - y = axis.translate(axis.stacking.usePercentage ? - 100 : - (boxTop ? - boxTop : - stackItem.total), 0, 0, 0, 1), - yZero = axis.translate(boxBottom ? boxBottom : 0), // stack origin - // stack height: - h = defined(y) && Math.abs(y - yZero), - // x position: - x = pick(defaultX, - chart.xAxis[0].translate(stackItem.x)) + - xOffset, - stackBox = defined(y) && stackItem.getStackBox(chart, - stackItem, - x, - y, - xWidth, - h, - axis), - label = stackItem.label, - isNegative = stackItem.isNegative, - isJustify = pick(stackItem.options.overflow, 'justify') === 'justify', - textAlign = stackItem.textAlign, - visible; - if (label && stackBox) { - var bBox = label.getBBox(), - padding = label.padding, - boxOffsetX, - boxOffsetY; - if (textAlign === 'left') { - boxOffsetX = chart.inverted ? -padding : padding; - } - else if (textAlign === 'right') { - boxOffsetX = bBox.width; - } - else { - if (chart.inverted && textAlign === 'center') { - boxOffsetX = bBox.width / 2; - } - else { - boxOffsetX = chart.inverted ? - (isNegative ? bBox.width + padding : -padding) : bBox.width / 2; - } - } - boxOffsetY = chart.inverted ? - bBox.height / 2 : (isNegative ? -padding : bBox.height); - // Reset alignOptions property after justify #12337 - stackItem.alignOptions.x = pick(stackItem.options.x, 0); - stackItem.alignOptions.y = pick(stackItem.options.y, 0); - // Set the stackBox position - stackBox.x -= boxOffsetX; - stackBox.y -= boxOffsetY; - // Align the label to the box - label.align(stackItem.alignOptions, null, stackBox); - // Check if label is inside the plotArea #12294 - if (chart.isInsidePlot(label.alignAttr.x + boxOffsetX - stackItem.alignOptions.x, label.alignAttr.y + boxOffsetY - stackItem.alignOptions.y)) { - label.show(); - } - else { - // Move label away to avoid the overlapping issues - label.alignAttr.y = -9999; - isJustify = false; - } - if (isJustify) { - // Justify stackLabel into the stackBox - Series.prototype.justifyDataLabel.call(this.axis, label, stackItem.alignOptions, label.alignAttr, bBox, stackBox); - } - label.attr({ - x: label.alignAttr.x, - y: label.alignAttr.y - }); - if (pick(!isJustify && stackItem.options.crop, true)) { - visible = - isNumber(label.x) && - isNumber(label.y) && - chart.isInsidePlot(label.x - padding + label.width, label.y) && - chart.isInsidePlot(label.x + padding, label.y); - if (!visible) { - label.hide(); - } - } - } - }; - /** - * @private - * @function Highcharts.StackItem#getStackBox - * - * @param {Highcharts.Chart} chart - * - * @param {Highcharts.StackItem} stackItem - * - * @param {number} x - * - * @param {number} y - * - * @param {number} xWidth - * - * @param {number} h - * - * @param {Highcharts.Axis} axis - * - * @return {Highcharts.BBoxObject} - */ - StackItem.prototype.getStackBox = function (chart, stackItem, x, y, xWidth, h, axis) { - var reversed = stackItem.axis.reversed, - inverted = chart.inverted, - axisPos = axis.height + axis.pos - - (inverted ? chart.plotLeft : chart.plotTop), - neg = (stackItem.isNegative && !reversed) || - (!stackItem.isNegative && reversed); // #4056 - return { - x: inverted ? (neg ? y - axis.right : y - h + axis.pos - chart.plotLeft) : - x + chart.xAxis[0].transB - chart.plotLeft, - y: inverted ? - axis.height - x - xWidth : - (neg ? - (axisPos - y - h) : - axisPos - y), - width: inverted ? h : xWidth, - height: inverted ? xWidth : h - }; - }; - return StackItem; - }()); + if (legendLine) { + legendLine.attr({ stroke: symbolColor }); + } + if (legendSymbol) { + // Apply marker options + if (markerOptions && legendSymbol.isMarker) { + // #585 + symbolAttr = item.pointAttribs(); + if (!visible) { + // #6769 + symbolAttr.stroke = symbolAttr.fill = hiddenColor; + } + } + legendSymbol.attr(symbolAttr); + } + } + fireEvent(this, "afterColorizeItem", { + item: item, + visible: visible, + }); + }; + /** + * @private + * @function Highcharts.Legend#positionItems + */ + Legend.prototype.positionItems = function () { + // Now that the legend width and height are established, put the items + // in the final position + this.allItems.forEach(this.positionItem, this); + if (!this.chart.isResizing) { + this.positionCheckboxes(); + } + }; + /** + * Position the legend item. + * + * @private + * @function Highcharts.Legend#positionItem + * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item + * The item to position + */ + Legend.prototype.positionItem = function (item) { + var _this = this; + var legend = this, + options = legend.options, + symbolPadding = options.symbolPadding, + ltr = !options.rtl, + legendItemPos = item._legendItemPos, + itemX = legendItemPos[0], + itemY = legendItemPos[1], + checkbox = item.checkbox, + legendGroup = item.legendGroup; + if (legendGroup && legendGroup.element) { + var attribs = { + translateX: ltr + ? itemX + : legend.legendWidth - itemX - 2 * symbolPadding - 4, + translateY: itemY, + }; + var complete = function () { + fireEvent(_this, "afterPositionItem", { item: item }); + }; + if (defined(legendGroup.translateY)) { + legendGroup.animate(attribs, void 0, complete); + } else { + legendGroup.attr(attribs); + complete(); + } + } + if (checkbox) { + checkbox.x = itemX; + checkbox.y = itemY; + } + }; /** - * Generate stacks for each series and calculate stacks total values + * Destroy a single legend item, used internally on removing series items. * * @private - * @function Highcharts.Chart#getStacks - */ - Chart.prototype.getStacks = function () { - var chart = this, - inverted = chart.inverted; - // reset stacks for each yAxis - chart.yAxis.forEach(function (axis) { - if (axis.stacking && axis.stacking.stacks && axis.hasVisibleSeries) { - axis.stacking.oldStacks = axis.stacking.stacks; - } - }); - chart.series.forEach(function (series) { - var xAxisOptions = series.xAxis && series.xAxis.options || {}; - if (series.options.stacking && - (series.visible === true || - chart.options.chart.ignoreHiddenSeries === false)) { - series.stackKey = [ - series.type, - pick(series.options.stack, ''), - inverted ? xAxisOptions.top : xAxisOptions.left, - inverted ? xAxisOptions.height : xAxisOptions.width - ].join(','); - } - }); + * @function Highcharts.Legend#destroyItem + * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item + * The item to remove + */ + Legend.prototype.destroyItem = function (item) { + var checkbox = item.checkbox; + // destroy SVG elements + ["legendItem", "legendLine", "legendSymbol", "legendGroup"].forEach( + function (key) { + if (item[key]) { + item[key] = item[key].destroy(); + } + } + ); + if (checkbox) { + discardElement(item.checkbox); + } }; - // Stacking methods defined on the Axis prototype - StackingAxis.compose(Axis); - // Stacking methods defined for Series prototype /** - * Set grouped points in a stack-like object. When `centerInCategory` is true, - * and `stacking` is not enabled, we need a pseudo (horizontal) stack in order - * to handle grouping of points within the same category. + * Destroy the legend. Used internally. To reflow objects, `chart.redraw` + * must be called after destruction. * * @private - * @function Highcharts.Series#setStackedPoints - * @return {void} - */ - Series.prototype.setGroupedPoints = function () { - if (this.options.centerInCategory && - (this.is('column') || this.is('columnrange')) && - // With stacking enabled, we already have stacks that we can compute - // from - !this.options.stacking && - // With only one series, we don't need to consider centerInCategory - this.chart.series.length > 1) { - Series.prototype.setStackedPoints.call(this, 'group'); + * @function Highcharts.Legend#destroy + */ + Legend.prototype.destroy = function () { + /** + * @private + * @param {string} key + * @return {void} + */ + function destroyItems(key) { + if (this[key]) { + this[key] = this[key].destroy(); } + } + // Destroy items + this.getAllItems().forEach(function (item) { + ["legendItem", "legendGroup"].forEach(destroyItems, item); + }); + // Destroy legend elements + [ + "clipRect", + "up", + "down", + "pager", + "nav", + "box", + "title", + "group", + ].forEach(destroyItems, this); + this.display = null; // Reset in .render on update. + }; + /** + * Position the checkboxes after the width is determined. + * + * @private + * @function Highcharts.Legend#positionCheckboxes + */ + Legend.prototype.positionCheckboxes = function () { + var alignAttr = this.group && this.group.alignAttr, + translateY, + clipHeight = this.clipHeight || this.legendHeight, + titleHeight = this.titleHeight; + if (alignAttr) { + translateY = alignAttr.translateY; + this.allItems.forEach(function (item) { + var checkbox = item.checkbox, + top; + if (checkbox) { + top = + translateY + + titleHeight + + checkbox.y + + (this.scrollOffset || 0) + + 3; + css(checkbox, { + left: + alignAttr.translateX + + item.checkboxOffset + + checkbox.x - + 20 + + "px", + top: top + "px", + display: + this.proximate || + (top > translateY - 6 && top < translateY + clipHeight - 6) + ? "" + : "none", + }); + } + }, this); + } }; /** - * Adds series' points value to corresponding stack + * Render the legend title on top of the legend. * * @private - * @function Highcharts.Series#setStackedPoints + * @function Highcharts.Legend#renderTitle + */ + Legend.prototype.renderTitle = function () { + var options = this.options, + padding = this.padding, + titleOptions = options.title, + titleHeight = 0, + bBox; + if (titleOptions.text) { + if (!this.title) { + /** + * SVG element of the legend title. + * + * @readonly + * @name Highcharts.Legend#title + * @type {Highcharts.SVGElement} + */ + this.title = this.chart.renderer + .label( + titleOptions.text, + padding - 3, + padding - 4, + null, + null, + null, + options.useHTML, + null, + "legend-title" + ) + .attr({ zIndex: 1 }); + if (!this.chart.styledMode) { + this.title.css(titleOptions.style); + } + this.title.add(this.group); + } + // Set the max title width (#7253) + if (!titleOptions.width) { + this.title.css({ + width: this.maxLegendWidth + "px", + }); + } + bBox = this.title.getBBox(); + titleHeight = bBox.height; + this.offsetWidth = bBox.width; // #1717 + this.contentGroup.attr({ translateY: titleHeight }); + } + this.titleHeight = titleHeight; + }; + /** + * Set the legend item text. + * + * @function Highcharts.Legend#setText + * @param {Highcharts.Point|Highcharts.Series} item + * The item for which to update the text in the legend. */ - Series.prototype.setStackedPoints = function (stackingParam) { - var stacking = stackingParam || this.options.stacking; - if (!stacking || - (this.visible !== true && - this.chart.options.chart.ignoreHiddenSeries !== false)) { - return; + Legend.prototype.setText = function (item) { + var options = this.options; + item.legendItem.attr({ + text: options.labelFormat + ? format(options.labelFormat, item, this.chart) + : options.labelFormatter.call(item), + }); + }; + /** + * Render a single specific legend item. Called internally from the `render` + * function. + * + * @private + * @function Highcharts.Legend#renderItem + * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item + * The item to render. + */ + Legend.prototype.renderItem = function (item) { + var legend = this, + chart = legend.chart, + renderer = chart.renderer, + options = legend.options, + horizontal = options.layout === "horizontal", + symbolWidth = legend.symbolWidth, + symbolPadding = options.symbolPadding, + itemStyle = legend.itemStyle, + itemHiddenStyle = legend.itemHiddenStyle, + itemDistance = horizontal ? pick(options.itemDistance, 20) : 0, + ltr = !options.rtl, + bBox, + li = item.legendItem, + isSeries = !item.series, + series = + !isSeries && item.series.drawLegendSymbol ? item.series : item, + seriesOptions = series.options, + showCheckbox = + legend.createCheckboxForItem && + seriesOptions && + seriesOptions.showCheckbox, + // full width minus text width + itemExtraWidth = + symbolWidth + + symbolPadding + + itemDistance + + (showCheckbox ? 20 : 0), + useHTML = options.useHTML, + itemClassName = item.options.className; + if (!li) { + // generate it once, later move it + // Generate the group box, a group to hold the symbol and text. Text + // is to be appended in Legend class. + item.legendGroup = renderer + .g("legend-item") + .addClass( + "highcharts-" + + series.type + + "-series " + + "highcharts-color-" + + item.colorIndex + + (itemClassName ? " " + itemClassName : "") + + (isSeries ? " highcharts-series-" + item.index : "") + ) + .attr({ zIndex: 1 }) + .add(legend.scrollGroup); + // Generate the list item text and add it to the group + item.legendItem = li = renderer.text( + "", + ltr ? symbolWidth + symbolPadding : -symbolPadding, + legend.baseline || 0, + useHTML + ); + if (!chart.styledMode) { + // merge to prevent modifying original (#1021) + li.css(merge(item.visible ? itemStyle : itemHiddenStyle)); } - var series = this, xData = series.processedXData, yData = series.processedYData, stackedYData = [], yDataLength = yData.length, seriesOptions = series.options, threshold = seriesOptions.threshold, stackThreshold = pick(seriesOptions.startFromThreshold && threshold, 0), stackOption = seriesOptions.stack, stackKey = stackingParam ? series.type + "," + stacking : series.stackKey, negKey = '-' + stackKey, negStacks = series.negStacks, yAxis = series.yAxis, stacks = yAxis.stacking.stacks, oldStacks = yAxis.stacking.oldStacks, stackIndicator, isNegative, stack, other, key, pointKey, i, x, y; - yAxis.stacking.stacksTouched += 1; - // loop over the non-null y values and read them into a local array - for (i = 0; i < yDataLength; i++) { - x = xData[i]; - y = yData[i]; - stackIndicator = series.getStackIndicator(stackIndicator, x, series.index); - pointKey = stackIndicator.key; - // Read stacked values into a stack based on the x value, - // the sign of y and the stack key. Stacking is also handled for null - // values (#739) - isNegative = negStacks && y < (stackThreshold ? 0 : threshold); - key = isNegative ? negKey : stackKey; - // Create empty object for this stack if it doesn't exist yet - if (!stacks[key]) { - stacks[key] = - {}; - } - // Initialize StackItem for this x - if (!stacks[key][x]) { - if (oldStacks[key] && - oldStacks[key][x]) { - stacks[key][x] = oldStacks[key][x]; - stacks[key][x].total = null; - } - else { - stacks[key][x] = new StackItem(yAxis, yAxis.options.stackLabels, isNegative, x, stackOption); - } - } - // If the StackItem doesn't exist, create it first - stack = stacks[key][x]; - if (y !== null) { - stack.points[pointKey] = stack.points[series.index] = - [pick(stack.cumulative, stackThreshold)]; - // Record the base of the stack - if (!defined(stack.cumulative)) { - stack.base = pointKey; - } - stack.touched = yAxis.stacking.stacksTouched; - // In area charts, if there are multiple points on the same X value, - // let the area fill the full span of those points - if (stackIndicator.index > 0 && series.singleStacks === false) { - stack.points[pointKey][0] = - stack.points[series.index + ',' + x + ',0'][0]; - } - // When updating to null, reset the point stack (#7493) - } - else { - stack.points[pointKey] = stack.points[series.index] = - null; - } - // Add value to the stack total - if (stacking === 'percent') { - // Percent stacked column, totals are the same for the positive and - // negative stacks - other = isNegative ? stackKey : negKey; - if (negStacks && stacks[other] && stacks[other][x]) { - other = stacks[other][x]; - stack.total = other.total = - Math.max(other.total, stack.total) + - Math.abs(y) || - 0; - // Percent stacked areas - } - else { - stack.total = - correctFloat(stack.total + (Math.abs(y) || 0)); - } - } - else if (stacking === 'group') { - // In this stack, the total is the number of valid points - if (y !== null) { - stack.total = (stack.total || 0) + 1; - } - } - else { - stack.total = correctFloat(stack.total + (y || 0)); - } - if (stacking === 'group') { - // This point's index within the stack, pushed to stack.points[1] - stack.cumulative = (stack.total || 1) - 1; - } - else { - stack.cumulative = - pick(stack.cumulative, stackThreshold) + (y || 0); - } - if (y !== null) { - stack.points[pointKey].push(stack.cumulative); - stackedYData[i] = stack.cumulative; - stack.hasValidPoints = true; - } + li.attr({ + align: ltr ? "left" : "right", + zIndex: 2, + }).add(item.legendGroup); + // Get the baseline for the first item - the font size is equal for + // all + if (!legend.baseline) { + legend.fontMetrics = renderer.fontMetrics( + chart.styledMode ? 12 : itemStyle.fontSize, + li + ); + legend.baseline = legend.fontMetrics.f + 3 + legend.itemMarginTop; + li.attr("y", legend.baseline); } - if (stacking === 'percent') { - yAxis.stacking.usePercentage = true; + // Draw the legend symbol inside the group box + legend.symbolHeight = options.symbolHeight || legend.fontMetrics.f; + series.drawLegendSymbol(legend, item); + if (legend.setItemEvents) { + legend.setItemEvents(item, li, useHTML); } - if (stacking !== 'group') { - this.stackedYData = stackedYData; // To be used in getExtremes + } + // Add the HTML checkbox on top + if (showCheckbox && !item.checkbox && legend.createCheckboxForItem) { + legend.createCheckboxForItem(item); + } + // Colorize the items + legend.colorizeItem(item, item.visible); + // Take care of max width and text overflow (#6659) + if (chart.styledMode || !itemStyle.width) { + li.css({ + width: + (options.itemWidth || + legend.widthOption || + chart.spacingBox.width) - + itemExtraWidth + + "px", + }); + } + // Always update the text + legend.setText(item); + // calculate the positions for the next line + bBox = li.getBBox(); + item.itemWidth = item.checkboxOffset = + options.itemWidth || + item.legendItemWidth || + bBox.width + itemExtraWidth; + legend.maxItemWidth = Math.max(legend.maxItemWidth, item.itemWidth); + legend.totalItemWidth += item.itemWidth; + legend.itemHeight = item.itemHeight = Math.round( + item.legendItemHeight || bBox.height || legend.symbolHeight + ); + }; + /** + * Get the position of the item in the layout. We now know the + * maxItemWidth from the previous loop. + * + * @private + * @function Highcharts.Legend#layoutItem + * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item + */ + Legend.prototype.layoutItem = function (item) { + var options = this.options, + padding = this.padding, + horizontal = options.layout === "horizontal", + itemHeight = item.itemHeight, + itemMarginBottom = this.itemMarginBottom, + itemMarginTop = this.itemMarginTop, + itemDistance = horizontal ? pick(options.itemDistance, 20) : 0, + maxLegendWidth = this.maxLegendWidth, + itemWidth = + options.alignColumns && this.totalItemWidth > maxLegendWidth + ? this.maxItemWidth + : item.itemWidth; + // If the item exceeds the width, start a new line + if (horizontal && this.itemX - padding + itemWidth > maxLegendWidth) { + this.itemX = padding; + if (this.lastLineHeight) { + // Not for the first line (#10167) + this.itemY += + itemMarginTop + this.lastLineHeight + itemMarginBottom; } - // Reset old stacks - yAxis.stacking.oldStacks = {}; + this.lastLineHeight = 0; // reset for next line (#915, #3976) + } + // Set the edge positions + this.lastItemY = itemMarginTop + this.itemY + itemMarginBottom; + this.lastLineHeight = Math.max( + // #915 + itemHeight, + this.lastLineHeight + ); + // cache the position of the newly generated or reordered items + item._legendItemPos = [this.itemX, this.itemY]; + // advance + if (horizontal) { + this.itemX += itemWidth; + } else { + this.itemY += itemMarginTop + itemHeight + itemMarginBottom; + this.lastLineHeight = itemHeight; + } + // the width of the widest item + this.offsetWidth = + this.widthOption || + Math.max( + (horizontal + ? this.itemX - + padding - + (item.checkbox + ? // decrease by itemDistance only when no checkbox #4853 + 0 + : itemDistance) + : itemWidth) + padding, + this.offsetWidth + ); }; /** - * Iterate over all stacks and compute the absolute values to percent + * Get all items, which is one item per series for most series and one + * item per point for pie series and its derivatives. Fires the event + * `afterGetAllItems`. * * @private - * @function Highcharts.Series#modifyStacks - */ - Series.prototype.modifyStacks = function () { - var series = this, - yAxis = series.yAxis, - stackKey = series.stackKey, - stacks = yAxis.stacking.stacks, - processedXData = series.processedXData, - stackIndicator, - stacking = series.options.stacking; - if (series[stacking + 'Stacker']) { // Modifier function exists - [stackKey, '-' + stackKey].forEach(function (key) { - var i = processedXData.length, - x, - stack, - pointExtremes; - while (i--) { - x = processedXData[i]; - stackIndicator = series.getStackIndicator(stackIndicator, x, series.index, key); - stack = stacks[key] && stacks[key][x]; - pointExtremes = - stack && stack.points[stackIndicator.key]; - if (pointExtremes) { - series[stacking + 'Stacker'](pointExtremes, stack, i); - } - } - }); + * @function Highcharts.Legend#getAllItems + * @return {Array<(Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series)>} + * The current items in the legend. + * @fires Highcharts.Legend#event:afterGetAllItems + */ + Legend.prototype.getAllItems = function () { + var allItems = []; + this.chart.series.forEach(function (series) { + var seriesOptions = series && series.options; + // Handle showInLegend. If the series is linked to another series, + // defaults to false. + if ( + series && + pick( + seriesOptions.showInLegend, + !defined(seriesOptions.linkedTo) ? void 0 : false, + true + ) + ) { + // Use points or series for the legend item depending on + // legendType + allItems = allItems.concat( + series.legendItems || + (seriesOptions.legendType === "point" ? series.data : series) + ); } + }); + fireEvent(this, "afterGetAllItems", { allItems: allItems }); + return allItems; }; /** - * Modifier function for percent stacks. Blows up the stack to 100%. + * Get a short, three letter string reflecting the alignment and layout. * * @private - * @function Highcharts.Series#percentStacker - * @param {Array<number>} pointExtremes - * @param {Highcharts.StackItem} stack - * @param {number} i + * @function Highcharts.Legend#getAlignment + * @return {string} + * The alignment, empty string if floating + */ + Legend.prototype.getAlignment = function () { + var options = this.options; + // Use the first letter of each alignment option in order to detect + // the side. (#4189 - use charAt(x) notation instead of [x] for IE7) + if (this.proximate) { + return options.align.charAt(0) + "tv"; + } + return options.floating + ? "" + : options.align.charAt(0) + + options.verticalAlign.charAt(0) + + options.layout.charAt(0); + }; + /** + * Adjust the chart margins by reserving space for the legend on only one + * side of the chart. If the position is set to a corner, top or bottom is + * reserved for horizontal legends and left or right for vertical ones. + * + * @private + * @function Highcharts.Legend#adjustMargins + * @param {Array<number>} margin + * @param {Array<number>} spacing + */ + Legend.prototype.adjustMargins = function (margin, spacing) { + var chart = this.chart, + options = this.options, + alignment = this.getAlignment(); + if (alignment) { + [ + /(lth|ct|rth)/, + /(rtv|rm|rbv)/, + /(rbh|cb|lbh)/, + /(lbv|lm|ltv)/, + ].forEach(function (alignments, side) { + if (alignments.test(alignment) && !defined(margin[side])) { + // Now we have detected on which side of the chart we should + // reserve space for the legend + chart[marginNames[side]] = Math.max( + chart[marginNames[side]], + chart.legend[ + (side + 1) % 2 ? "legendHeight" : "legendWidth" + ] + + [1, -1, -1, 1][side] * options[side % 2 ? "x" : "y"] + + pick(options.margin, 12) + + spacing[side] + + (chart.titleOffset[side] || 0) + ); + } + }); + } + }; + /** + * @private + * @function Highcharts.Legend#proximatePositions + */ + Legend.prototype.proximatePositions = function () { + var chart = this.chart, + boxes = [], + alignLeft = this.options.align === "left"; + this.allItems.forEach(function (item) { + var lastPoint, + height, + useFirstPoint = alignLeft, + target, + top; + if (item.yAxis) { + if (item.xAxis.options.reversed) { + useFirstPoint = !useFirstPoint; + } + if (item.points) { + lastPoint = find( + useFirstPoint ? item.points : item.points.slice(0).reverse(), + function (item) { + return isNumber(item.plotY); + } + ); + } + height = + this.itemMarginTop + + item.legendItem.getBBox().height + + this.itemMarginBottom; + top = item.yAxis.top - chart.plotTop; + if (item.visible) { + target = lastPoint ? lastPoint.plotY : item.yAxis.height; + target += top - 0.3 * height; + } else { + target = top + item.yAxis.height; + } + boxes.push({ + target: target, + size: height, + item: item, + }); + } + }, this); + H.distribute(boxes, chart.plotHeight); + boxes.forEach(function (box) { + box.item._legendItemPos[1] = + chart.plotTop - chart.spacing[0] + box.pos; + }); + }; + /** + * Render the legend. This method can be called both before and after + * `chart.render`. If called after, it will only rearrange items instead + * of creating new ones. Called internally on initial render and after + * redraws. + * + * @private + * @function Highcharts.Legend#render + */ + Legend.prototype.render = function () { + var legend = this, + chart = legend.chart, + renderer = chart.renderer, + legendGroup = legend.group, + allItems, + display, + legendWidth, + legendHeight, + box = legend.box, + options = legend.options, + padding = legend.padding, + allowedWidth; + legend.itemX = padding; + legend.itemY = legend.initialItemY; + legend.offsetWidth = 0; + legend.lastItemY = 0; + legend.widthOption = relativeLength( + options.width, + chart.spacingBox.width - padding + ); + // Compute how wide the legend is allowed to be + allowedWidth = chart.spacingBox.width - 2 * padding - options.x; + if ( + ["rm", "lm"].indexOf(legend.getAlignment().substring(0, 2)) > -1 + ) { + allowedWidth /= 2; + } + legend.maxLegendWidth = legend.widthOption || allowedWidth; + if (!legendGroup) { + /** + * SVG group of the legend. + * + * @readonly + * @name Highcharts.Legend#group + * @type {Highcharts.SVGElement} + */ + legend.group = legendGroup = renderer + .g("legend") + .attr({ zIndex: 7 }) + .add(); + legend.contentGroup = renderer + .g() + .attr({ zIndex: 1 }) // above background + .add(legendGroup); + legend.scrollGroup = renderer.g().add(legend.contentGroup); + } + legend.renderTitle(); + // add each series or point + allItems = legend.getAllItems(); + // sort by legendIndex + stableSort(allItems, function (a, b) { + return ( + ((a.options && a.options.legendIndex) || 0) - + ((b.options && b.options.legendIndex) || 0) + ); + }); + // reversed legend + if (options.reversed) { + allItems.reverse(); + } + /** + * All items for the legend, which is an array of series for most series + * and an array of points for pie series and its derivatives. + * + * @readonly + * @name Highcharts.Legend#allItems + * @type {Array<(Highcharts.Point|Highcharts.Series)>} + */ + legend.allItems = allItems; + legend.display = display = !!allItems.length; + // Render the items. First we run a loop to set the text and properties + // and read all the bounding boxes. The next loop computes the item + // positions based on the bounding boxes. + legend.lastLineHeight = 0; + legend.maxItemWidth = 0; + legend.totalItemWidth = 0; + legend.itemHeight = 0; + allItems.forEach(legend.renderItem, legend); + allItems.forEach(legend.layoutItem, legend); + // Get the box + legendWidth = (legend.widthOption || legend.offsetWidth) + padding; + legendHeight = + legend.lastItemY + legend.lastLineHeight + legend.titleHeight; + legendHeight = legend.handleOverflow(legendHeight); + legendHeight += padding; + // Draw the border and/or background + if (!box) { + /** + * SVG element of the legend box. + * + * @readonly + * @name Highcharts.Legend#box + * @type {Highcharts.SVGElement} + */ + legend.box = box = renderer + .rect() + .addClass("highcharts-legend-box") + .attr({ + r: options.borderRadius, + }) + .add(legendGroup); + box.isNew = true; + } + // Presentational + if (!chart.styledMode) { + box + .attr({ + stroke: options.borderColor, + "stroke-width": options.borderWidth || 0, + fill: options.backgroundColor || "none", + }) + .shadow(options.shadow); + } + if (legendWidth > 0 && legendHeight > 0) { + box[box.isNew ? "attr" : "animate"]( + box.crisp.call( + {}, + { + x: 0, + y: 0, + width: legendWidth, + height: legendHeight, + }, + box.strokeWidth() + ) + ); + box.isNew = false; + } + // hide the border if no items + box[display ? "show" : "hide"](); + // Open for responsiveness + if (chart.styledMode && legendGroup.getStyle("display") === "none") { + legendWidth = legendHeight = 0; + } + legend.legendWidth = legendWidth; + legend.legendHeight = legendHeight; + if (display) { + legend.align(); + } + if (!this.proximate) { + this.positionItems(); + } + fireEvent(this, "afterRender"); + }; + /** + * Align the legend to chart's box. + * + * @private + * @function Highcharts.align + * @param {Highcharts.BBoxObject} alignTo + * @return {void} */ - Series.prototype.percentStacker = function (pointExtremes, stack, i) { - var totalFactor = stack.total ? 100 / stack.total : 0; - // Y bottom value - pointExtremes[0] = correctFloat(pointExtremes[0] * totalFactor); - // Y value - pointExtremes[1] = correctFloat(pointExtremes[1] * totalFactor); - this.stackedYData[i] = pointExtremes[1]; + Legend.prototype.align = function (alignTo) { + if (alignTo === void 0) { + alignTo = this.chart.spacingBox; + } + var chart = this.chart, + options = this.options; + // If aligning to the top and the layout is horizontal, adjust for + // the title (#7428) + var y = alignTo.y; + if ( + /(lth|ct|rth)/.test(this.getAlignment()) && + chart.titleOffset[0] > 0 + ) { + y += chart.titleOffset[0]; + } else if ( + /(lbh|cb|rbh)/.test(this.getAlignment()) && + chart.titleOffset[2] > 0 + ) { + y -= chart.titleOffset[2]; + } + if (y !== alignTo.y) { + alignTo = merge(alignTo, { y: y }); + } + this.group.align( + merge(options, { + width: this.legendWidth, + height: this.legendHeight, + verticalAlign: this.proximate ? "top" : options.verticalAlign, + }), + true, + alignTo + ); }; /** - * Get stack indicator, according to it's x-value, to determine points with the - * same x-value + * Set up the overflow handling by adding navigation with up and down arrows + * below the legend. * * @private - * @function Highcharts.Series#getStackIndicator - * @param {Highcharts.StackItemIndicatorObject|undefined} stackIndicator - * @param {number} x - * @param {number} index - * @param {string} [key] - * @return {Highcharts.StackItemIndicatorObject} - */ - Series.prototype.getStackIndicator = function (stackIndicator, x, index, key) { - // Update stack indicator, when: - // first point in a stack || x changed || stack type (negative vs positive) - // changed: - if (!defined(stackIndicator) || - stackIndicator.x !== x || - (key && stackIndicator.key !== key)) { - stackIndicator = { - x: x, - index: 0, - key: key - }; + * @function Highcharts.Legend#handleOverflow + * @param {number} legendHeight + * @return {number} + */ + Legend.prototype.handleOverflow = function (legendHeight) { + var legend = this, + chart = this.chart, + renderer = chart.renderer, + options = this.options, + optionsY = options.y, + alignTop = options.verticalAlign === "top", + padding = this.padding, + spaceHeight = + chart.spacingBox.height + + (alignTop ? -optionsY : optionsY) - + padding, + maxHeight = options.maxHeight, + clipHeight, + clipRect = this.clipRect, + navOptions = options.navigation, + animation = pick(navOptions.animation, true), + arrowSize = navOptions.arrowSize || 12, + nav = this.nav, + pages = this.pages, + lastY, + allItems = this.allItems, + clipToHeight = function (height) { + if (typeof height === "number") { + clipRect.attr({ + height: height, + }); + } else if (clipRect) { + // Reset (#5912) + legend.clipRect = clipRect.destroy(); + legend.contentGroup.clip(); + } + // useHTML + if (legend.contentGroup.div) { + legend.contentGroup.div.style.clip = height + ? "rect(" + + padding + + "px,9999px," + + (padding + height) + + "px,0)" + : "auto"; + } + }, + addTracker = function (key) { + legend[key] = renderer + .circle(0, 0, arrowSize * 1.3) + .translate(arrowSize / 2, arrowSize / 2) + .add(nav); + if (!chart.styledMode) { + legend[key].attr("fill", "rgba(0,0,0,0.0001)"); + } + return legend[key]; + }; + // Adjust the height + if ( + options.layout === "horizontal" && + options.verticalAlign !== "middle" && + !options.floating + ) { + spaceHeight /= 2; + } + if (maxHeight) { + spaceHeight = Math.min(spaceHeight, maxHeight); + } + // Reset the legend height and adjust the clipping rectangle + pages.length = 0; + if (legendHeight > spaceHeight && navOptions.enabled !== false) { + this.clipHeight = clipHeight = Math.max( + spaceHeight - 20 - this.titleHeight - padding, + 0 + ); + this.currentPage = pick(this.currentPage, 1); + this.fullHeight = legendHeight; + // Fill pages with Y positions so that the top of each a legend item + // defines the scroll top for each page (#2098) + allItems.forEach(function (item, i) { + var y = item._legendItemPos[1], + h = Math.round(item.legendItem.getBBox().height), + len = pages.length; + if ( + !len || + (y - pages[len - 1] > clipHeight && + (lastY || y) !== pages[len - 1]) + ) { + pages.push(lastY || y); + len++; + } + // Keep track of which page each item is on + item.pageIx = len - 1; + if (lastY) { + allItems[i - 1].pageIx = len - 1; + } + if ( + i === allItems.length - 1 && + y + h - pages[len - 1] > clipHeight && + y !== lastY // #2617 + ) { + pages.push(y); + item.pageIx = len; + } + if (y !== lastY) { + lastY = y; + } + }); + // Only apply clipping if needed. Clipping causes blurred legend in + // PDF export (#1787) + if (!clipRect) { + clipRect = legend.clipRect = renderer.clipRect( + 0, + padding, + 9999, + 0 + ); + legend.contentGroup.clip(clipRect); } - else { - (stackIndicator).index++; + clipToHeight(clipHeight); + // Add navigation elements + if (!nav) { + this.nav = nav = renderer.g().attr({ zIndex: 1 }).add(this.group); + this.up = renderer + .symbol("triangle", 0, 0, arrowSize, arrowSize) + .add(nav); + addTracker("upTracker").on("click", function () { + legend.scroll(-1, animation); + }); + this.pager = renderer + .text("", 15, 10) + .addClass("highcharts-legend-navigation"); + if (!chart.styledMode) { + this.pager.css(navOptions.style); + } + this.pager.add(nav); + this.down = renderer + .symbol("triangle-down", 0, 0, arrowSize, arrowSize) + .add(nav); + addTracker("downTracker").on("click", function () { + legend.scroll(1, animation); + }); } - stackIndicator.key = - [index, x, stackIndicator.index].join(','); - return stackIndicator; + // Set initial position + legend.scroll(0); + legendHeight = spaceHeight; + // Reset + } else if (nav) { + clipToHeight(); + this.nav = nav.destroy(); // #6322 + this.scrollGroup.attr({ + translateY: 1, + }); + this.clipHeight = 0; // #1379 + } + return legendHeight; }; - H.StackItem = StackItem; - - return H.StackItem; - }); - _registerModule(_modules, 'Core/Dynamics.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Axis/Axis.js'], _modules['Core/Series/Series.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Series/LineSeries.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Time.js'], _modules['Core/Utilities.js']], function (A, Axis, BaseSeries, Chart, H, LineSeries, O, Point, Time, U) { - /* * + /** + * Scroll the legend by a number of pages. * - * (c) 2010-2020 Torstein Honsi + * @private + * @function Highcharts.Legend#scroll * - * License: www.highcharts.com/license + * @param {number} scrollBy + * The number of pages to scroll. * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * Whether and how to apply animation. * - * */ - var animate = A.animate, - setAnimation = A.setAnimation; - var seriesTypes = BaseSeries.seriesTypes; - var time = O.time; - var addEvent = U.addEvent, - createElement = U.createElement, - css = U.css, - defined = U.defined, - erase = U.erase, - error = U.error, - extend = U.extend, - fireEvent = U.fireEvent, - isArray = U.isArray, - isNumber = U.isNumber, - isObject = U.isObject, - isString = U.isString, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick, - relativeLength = U.relativeLength, - splat = U.splat; - /* eslint-disable valid-jsdoc */ - /** - * Remove settings that have not changed, to avoid unnecessary rendering or - * computing (#9197). - * @private + * @return {void} */ - H.cleanRecursively = function (newer, older) { - var result = {}; - objectEach(newer, function (val, key) { - var ob; - // Dive into objects (except DOM nodes) - if (isObject(newer[key], true) && - !newer.nodeType && // #10044 - older[key]) { - ob = H.cleanRecursively(newer[key], older[key]); - if (Object.keys(ob).length) { - result[key] = ob; - } - // Arrays, primitives and DOM nodes are copied directly - } - else if (isObject(newer[key]) || - newer[key] !== older[key]) { - result[key] = newer[key]; - } + Legend.prototype.scroll = function (scrollBy, animation) { + var _this = this; + var chart = this.chart, + pages = this.pages, + pageCount = pages.length, + currentPage = this.currentPage + scrollBy, + clipHeight = this.clipHeight, + navOptions = this.options.navigation, + pager = this.pager, + padding = this.padding; + // When resizing while looking at the last page + if (currentPage > pageCount) { + currentPage = pageCount; + } + if (currentPage > 0) { + if (typeof animation !== "undefined") { + setAnimation(animation, chart); + } + this.nav.attr({ + translateX: padding, + translateY: clipHeight + this.padding + 7 + this.titleHeight, + visibility: "visible", }); - return result; - }; - // Extend the Chart prototype for dynamic methods - extend(Chart.prototype, /** @lends Highcharts.Chart.prototype */ { - /** - * Add a series to the chart after render time. Note that this method should - * never be used when adding data synchronously at chart render time, as it - * adds expense to the calculations and rendering. When adding data at the - * same time as the chart is initialized, add the series as a configuration - * option instead. With multiple axes, the `offset` is dynamically adjusted. - * - * @sample highcharts/members/chart-addseries/ - * Add a series from a button - * @sample stock/members/chart-addseries/ - * Add a series in Highstock - * - * @function Highcharts.Chart#addSeries - * - * @param {Highcharts.SeriesOptionsType} options - * The config options for the series. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after adding. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * Whether to apply animation, and optionally animation - * configuration. - * - * @return {Highcharts.Series} - * The newly created series object. - * - * @fires Highcharts.Chart#event:addSeries - * @fires Highcharts.Chart#event:afterAddSeries - */ - addSeries: function (options, redraw, animation) { - var series, - chart = this; - if (options) { // <- not necessary - redraw = pick(redraw, true); // defaults to true - fireEvent(chart, 'addSeries', { options: options }, function () { - series = chart.initSeries(options); - chart.isDirtyLegend = true; - chart.linkSeries(); - if (series.enabledDataSorting) { - // We need to call `setData` after `linkSeries` - series.setData(options.data, false); - } - fireEvent(chart, 'afterAddSeries', { series: series }); - if (redraw) { - chart.redraw(animation); - } - }); - } - return series; - }, - /** - * Add an axis to the chart after render time. Note that this method should - * never be used when adding data synchronously at chart render time, as it - * adds expense to the calculations and rendering. When adding data at the - * same time as the chart is initialized, add the axis as a configuration - * option instead. - * - * @sample highcharts/members/chart-addaxis/ - * Add and remove axes - * - * @function Highcharts.Chart#addAxis - * - * @param {Highcharts.AxisOptions} options - * The axis options. - * - * @param {boolean} [isX=false] - * Whether it is an X axis or a value axis. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after adding. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] - * Whether and how to apply animation in the redraw. - * - * @return {Highcharts.Axis} - * The newly generated Axis object. - */ - addAxis: function (options, isX, redraw, animation) { - return this.createAxis(isX ? 'xAxis' : 'yAxis', { axis: options, redraw: redraw, animation: animation }); - }, - /** - * Add a color axis to the chart after render time. Note that this method - * should never be used when adding data synchronously at chart render time, - * as it adds expense to the calculations and rendering. When adding data at - * the same time as the chart is initialized, add the axis as a - * configuration option instead. - * - * @sample highcharts/members/chart-addaxis/ - * Add and remove axes - * - * @function Highcharts.Chart#addColorAxis - * - * @param {Highcharts.ColorAxisOptions} options - * The axis options. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after adding. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] - * Whether and how to apply animation in the redraw. - * - * @return {Highcharts.ColorAxis} - * The newly generated Axis object. - */ - addColorAxis: function (options, redraw, animation) { - return this.createAxis('colorAxis', { axis: options, redraw: redraw, animation: animation }); - }, - /** - * Factory for creating different axis types. - * - * @private - * @function Highcharts.Chart#createAxis - * - * @param {string} type - * An axis type. - * - * @param {...Array<*>} arguments - * All arguments for the constructor. - * - * @return {Highcharts.Axis | Highcharts.ColorAxis} - * The newly generated Axis object. - */ - createAxis: function (type, options) { - var chartOptions = this.options, - isColorAxis = type === 'colorAxis', - axisOptions = options.axis, - redraw = options.redraw, - animation = options.animation, - userOptions = merge(axisOptions, { - index: this[type].length, - isX: type === 'xAxis' - }), - axis; - if (isColorAxis) { - axis = new H.ColorAxis(this, userOptions); - } - else { - axis = new Axis(this, userOptions); - } - // Push the new axis options to the chart options - chartOptions[type] = splat(chartOptions[type] || {}); - chartOptions[type].push(userOptions); - if (isColorAxis) { - this.isDirtyLegend = true; - // Clear before 'bindAxes' (#11924) - this.axes.forEach(function (axis) { - axis.series = []; - }); - this.series.forEach(function (series) { - series.bindAxes(); - series.isDirtyData = true; - }); - } - if (pick(redraw, true)) { - this.redraw(animation); - } - return axis; - }, - /** - * Dim the chart and show a loading text or symbol. Options for the loading - * screen are defined in {@link - * https://api.highcharts.com/highcharts/loading|the loading options}. - * - * @sample highcharts/members/chart-hideloading/ - * Show and hide loading from a button - * @sample highcharts/members/chart-showloading/ - * Apply different text labels - * @sample stock/members/chart-show-hide-loading/ - * Toggle loading in Highstock - * - * @function Highcharts.Chart#showLoading - * - * @param {string} [str] - * An optional text to show in the loading label instead of the - * default one. The default text is set in - * [lang.loading](https://api.highcharts.com/highcharts/lang.loading). - */ - showLoading: function (str) { - var chart = this, - options = chart.options, - loadingDiv = chart.loadingDiv, - loadingOptions = options.loading, - setLoadingSize = function () { - if (loadingDiv) { - css(loadingDiv, { - left: chart.plotLeft + 'px', - top: chart.plotTop + 'px', - width: chart.plotWidth + 'px', - height: chart.plotHeight + 'px' - }); - } - }; - // create the layer at the first call - if (!loadingDiv) { - chart.loadingDiv = loadingDiv = createElement('div', { - className: 'highcharts-loading highcharts-loading-hidden' - }, null, chart.container); - chart.loadingSpan = createElement('span', { className: 'highcharts-loading-inner' }, null, loadingDiv); - addEvent(chart, 'redraw', setLoadingSize); // #1080 - } - loadingDiv.className = 'highcharts-loading'; - // Update text - chart.loadingSpan.innerHTML = - pick(str, options.lang.loading, ''); - if (!chart.styledMode) { - // Update visuals - css(loadingDiv, extend(loadingOptions.style, { - zIndex: 10 - })); - css(chart.loadingSpan, loadingOptions.labelStyle); - // Show it - if (!chart.loadingShown) { - css(loadingDiv, { - opacity: 0, - display: '' - }); - animate(loadingDiv, { - opacity: loadingOptions.style.opacity || 0.5 - }, { - duration: loadingOptions.showDuration || 0 - }); - } - } - chart.loadingShown = true; - setLoadingSize(); - }, - /** - * Hide the loading layer. - * - * @see Highcharts.Chart#showLoading - * - * @sample highcharts/members/chart-hideloading/ - * Show and hide loading from a button - * @sample stock/members/chart-show-hide-loading/ - * Toggle loading in Highstock - * - * @function Highcharts.Chart#hideLoading - */ - hideLoading: function () { - var options = this.options, - loadingDiv = this.loadingDiv; - if (loadingDiv) { - loadingDiv.className = - 'highcharts-loading highcharts-loading-hidden'; - if (!this.styledMode) { - animate(loadingDiv, { - opacity: 0 - }, { - duration: options.loading.hideDuration || 100, - complete: function () { - css(loadingDiv, { display: 'none' }); - } - }); - } - } - this.loadingShown = false; - }, - /** - * These properties cause isDirtyBox to be set to true when updating. Can be - * extended from plugins. - */ - propsRequireDirtyBox: [ - 'backgroundColor', - 'borderColor', - 'borderWidth', - 'borderRadius', - 'plotBackgroundColor', - 'plotBackgroundImage', - 'plotBorderColor', - 'plotBorderWidth', - 'plotShadow', - 'shadow' - ], - /** - * These properties require a full reflow of chart elements, best - * implemented through running `Chart.setSize` internally (#8190). - * @type {Array} - */ - propsRequireReflow: [ - 'margin', - 'marginTop', - 'marginRight', - 'marginBottom', - 'marginLeft', - 'spacing', - 'spacingTop', - 'spacingRight', - 'spacingBottom', - 'spacingLeft' - ], - /** - * These properties cause all series to be updated when updating. Can be - * extended from plugins. - */ - propsRequireUpdateSeries: [ - 'chart.inverted', - 'chart.polar', - 'chart.ignoreHiddenSeries', - 'chart.type', - 'colors', - 'plotOptions', - 'time', - 'tooltip' - ], - /** - * These collections (arrays) implement update() methods with support for - * one-to-one option. - */ - collectionsWithUpdate: [ - 'xAxis', - 'yAxis', - 'zAxis', - 'series' - ], - /** - * A generic function to update any element of the chart. Elements can be - * enabled and disabled, moved, re-styled, re-formatted etc. - * - * A special case is configuration objects that take arrays, for example - * [xAxis](https://api.highcharts.com/highcharts/xAxis), - * [yAxis](https://api.highcharts.com/highcharts/yAxis) or - * [series](https://api.highcharts.com/highcharts/series). For these - * collections, an `id` option is used to map the new option set to an - * existing object. If an existing object of the same id is not found, the - * corresponding item is updated. So for example, running `chart.update` - * with a series item without an id, will cause the existing chart's series - * with the same index in the series array to be updated. When the - * `oneToOne` parameter is true, `chart.update` will also take care of - * adding and removing items from the collection. Read more under the - * parameter description below. - * - * Note that when changing series data, `chart.update` may mutate the passed - * data options. - * - * See also the - * [responsive option set](https://api.highcharts.com/highcharts/responsive). - * Switching between `responsive.rules` basically runs `chart.update` under - * the hood. - * - * @sample highcharts/members/chart-update/ - * Update chart geometry - * - * @function Highcharts.Chart#update - * - * @param {Highcharts.Options} options - * A configuration object for the new chart options. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart. - * - * @param {boolean} [oneToOne=false] - * When `true`, the `series`, `xAxis`, `yAxis` and `annotations` - * collections will be updated one to one, and items will be either - * added or removed to match the new updated options. For example, - * if the chart has two series and we call `chart.update` with a - * configuration containing three series, one will be added. If we - * call `chart.update` with one series, one will be removed. Setting - * an empty `series` array will remove all series, but leaving out - * the`series` property will leave all series untouched. If the - * series have id's, the new series options will be matched by id, - * and the remaining ones removed. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] - * Whether to apply animation, and optionally animation - * configuration. - * - * @fires Highcharts.Chart#event:update - * @fires Highcharts.Chart#event:afterUpdate - */ - update: function (options, redraw, oneToOne, animation) { - var chart = this, - adders = { - credits: 'addCredits', - title: 'setTitle', - subtitle: 'setSubtitle', - caption: 'setCaption' - }, - optionsChart, - updateAllAxes, - updateAllSeries, - newWidth, - newHeight, - runSetSize, - isResponsiveOptions = options.isResponsiveOptions, - itemsForRemoval = []; - fireEvent(chart, 'update', { options: options }); - // If there are responsive rules in action, undo the responsive rules - // before we apply the updated options and replay the responsive rules - // on top from the chart.redraw function (#9617). - if (!isResponsiveOptions) { - chart.setResponsive(false, true); - } - options = H.cleanRecursively(options, chart.options); - merge(true, chart.userOptions, options); - // If the top-level chart option is present, some special updates are - // required - optionsChart = options.chart; - if (optionsChart) { - merge(true, chart.options.chart, optionsChart); - // Setter function - if ('className' in optionsChart) { - chart.setClassName(optionsChart.className); - } - if ('reflow' in optionsChart) { - chart.setReflow(optionsChart.reflow); - } - if ('inverted' in optionsChart || - 'polar' in optionsChart || - 'type' in optionsChart) { - // Parse options.chart.inverted and options.chart.polar together - // with the available series. - chart.propFromSeries(); - updateAllAxes = true; - } - if ('alignTicks' in optionsChart) { // #6452 - updateAllAxes = true; - } - objectEach(optionsChart, function (val, key) { - if (chart.propsRequireUpdateSeries.indexOf('chart.' + key) !== - -1) { - updateAllSeries = true; - } - // Only dirty box - if (chart.propsRequireDirtyBox.indexOf(key) !== -1) { - chart.isDirtyBox = true; - } - // Chart setSize - if (chart.propsRequireReflow.indexOf(key) !== -1) { - if (isResponsiveOptions) { - chart.isDirtyBox = true; - } - else { - runSetSize = true; - } - } - }); - if (!chart.styledMode && 'style' in optionsChart) { - chart.renderer.setStyle(optionsChart.style); - } - } - // Moved up, because tooltip needs updated plotOptions (#6218) - if (!chart.styledMode && options.colors) { - this.options.colors = options.colors; - } - if (options.time) { - // Maintaining legacy global time. If the chart is instanciated - // first with global time, then updated with time options, we need - // to create a new Time instance to avoid mutating the global time - // (#10536). - if (this.time === time) { - this.time = new Time(options.time); - } - // If we're updating, the time class is different from other chart - // classes (chart.legend, chart.tooltip etc) in that it doesn't know - // about the chart. The other chart[something].update functions also - // set the chart.options[something]. For the time class however we - // need to update the chart options separately. #14230. - merge(true, chart.options.time, options.time); - } - // Some option stuctures correspond one-to-one to chart objects that - // have update methods, for example - // options.credits => chart.credits - // options.legend => chart.legend - // options.title => chart.title - // options.tooltip => chart.tooltip - // options.subtitle => chart.subtitle - // options.mapNavigation => chart.mapNavigation - // options.navigator => chart.navigator - // options.scrollbar => chart.scrollbar - objectEach(options, function (val, key) { - if (chart[key] && - typeof chart[key].update === 'function') { - chart[key].update(val, false); - // If a one-to-one object does not exist, look for an adder function - } - else if (typeof chart[adders[key]] === 'function') { - chart[adders[key]](val); - // Else, just merge the options. For nodes like loading, noData, - // plotOptions - } - else if (key !== 'color' && - chart.collectionsWithUpdate.indexOf(key) === -1) { - merge(true, chart.options[key], options[key]); - } - if (key !== 'chart' && - chart.propsRequireUpdateSeries.indexOf(key) !== -1) { - updateAllSeries = true; - } - }); - // Setters for collections. For axes and series, each item is referred - // by an id. If the id is not found, it defaults to the corresponding - // item in the collection, so setting one series without an id, will - // update the first series in the chart. Setting two series without - // an id will update the first and the second respectively (#6019) - // chart.update and responsive. - this.collectionsWithUpdate.forEach(function (coll) { - var indexMap; - if (options[coll]) { - // In stock charts, the navigator series are also part of the - // chart.series array, but those series should not be handled - // here (#8196). - if (coll === 'series') { - indexMap = []; - chart[coll].forEach(function (s, i) { - if (!s.options.isInternal) { - indexMap.push(pick(s.options.index, i)); - } - }); - } - splat(options[coll]).forEach(function (newOptions, i) { - var hasId = defined(newOptions.id); - var item; - // Match by id - if (hasId) { - item = chart.get(newOptions.id); - } - // No match by id found, match by index instead - if (!item) { - item = chart[coll][indexMap ? indexMap[i] : i]; - // Check if we grabbed an item with an exising but - // different id (#13541) - if (item && hasId && defined(item.options.id)) { - item = void 0; - } - } - if (item && item.coll === coll) { - item.update(newOptions, false); - if (oneToOne) { - item.touched = true; - } - } - // If oneToOne and no matching item is found, add one - if (!item && oneToOne && chart.collectionsWithInit[coll]) { - chart.collectionsWithInit[coll][0].apply(chart, - // [newOptions, ...extraArguments, redraw=false] - [ - newOptions - ].concat( - // Not all initializers require extra args - chart.collectionsWithInit[coll][1] || []).concat([ - false - ])).touched = true; - } - }); - // Add items for removal - if (oneToOne) { - chart[coll].forEach(function (item) { - if (!item.touched && !item.options.isInternal) { - itemsForRemoval.push(item); - } - else { - delete item.touched; - } - }); - } - } - }); - itemsForRemoval.forEach(function (item) { - if (item.remove) { - item.remove(false); - } - }); - if (updateAllAxes) { - chart.axes.forEach(function (axis) { - axis.update({}, false); - }); - } - // Certain options require the whole series structure to be thrown away - // and rebuilt - if (updateAllSeries) { - chart.getSeriesOrderByLinks().forEach(function (series) { - // Avoid removed navigator series - if (series.chart) { - series.update({}, false); - } - }, this); - } - // Update size. Redraw is forced. - newWidth = optionsChart && optionsChart.width; - newHeight = optionsChart && optionsChart.height; - if (isString(newHeight)) { - newHeight = relativeLength(newHeight, newWidth || chart.chartWidth); - } - if ( - // In this case, run chart.setSize with newWidth and newHeight which - // are undefined, only for reflowing chart elements because margin - // or spacing has been set (#8190) - runSetSize || - // In this case, the size is actually set - (isNumber(newWidth) && newWidth !== chart.chartWidth) || - (isNumber(newHeight) && newHeight !== chart.chartHeight)) { - chart.setSize(newWidth, newHeight, animation); - } - else if (pick(redraw, true)) { - chart.redraw(animation); - } - fireEvent(chart, 'afterUpdate', { - options: options, - redraw: redraw, - animation: animation - }); - }, - /** - * Shortcut to set the subtitle options. This can also be done from {@link - * Chart#update} or {@link Chart#setTitle}. - * - * @function Highcharts.Chart#setSubtitle - * - * @param {Highcharts.SubtitleOptions} options - * New subtitle options. The subtitle text itself is set by the - * `options.text` property. - */ - setSubtitle: function (options, redraw) { - this.applyDescription('subtitle', options); - this.layOutTitles(redraw); - }, - /** - * Set the caption options. This can also be done from {@link - * Chart#update}. - * - * @function Highcharts.Chart#setCaption - * - * @param {Highcharts.CaptionOptions} options - * New caption options. The caption text itself is set by the - * `options.text` property. - */ - setCaption: function (options, redraw) { - this.applyDescription('caption', options); - this.layOutTitles(redraw); + [this.up, this.upTracker].forEach(function (elem) { + elem.attr({ + class: + currentPage === 1 + ? "highcharts-legend-nav-inactive" + : "highcharts-legend-nav-active", + }); + }); + pager.attr({ + text: currentPage + "/" + pageCount, + }); + [this.down, this.downTracker].forEach(function (elem) { + elem.attr({ + // adjust to text width + x: 18 + this.pager.getBBox().width, + class: + currentPage === pageCount + ? "highcharts-legend-nav-inactive" + : "highcharts-legend-nav-active", + }); + }, this); + if (!chart.styledMode) { + this.up.attr({ + fill: + currentPage === 1 + ? navOptions.inactiveColor + : navOptions.activeColor, + }); + this.upTracker.css({ + cursor: currentPage === 1 ? "default" : "pointer", + }); + this.down.attr({ + fill: + currentPage === pageCount + ? navOptions.inactiveColor + : navOptions.activeColor, + }); + this.downTracker.css({ + cursor: currentPage === pageCount ? "default" : "pointer", + }); } + this.scrollOffset = -pages[currentPage - 1] + this.initialItemY; + this.scrollGroup.animate({ + translateY: this.scrollOffset, + }); + this.currentPage = currentPage; + this.positionCheckboxes(); + // Fire event after scroll animation is complete + var animOptions = animObject( + pick(animation, chart.renderer.globalAnimation, true) + ); + syncTimeout(function () { + fireEvent(_this, "afterScroll", { currentPage: currentPage }); + }, animOptions.duration); + } + }; + return Legend; + })(); + // Workaround for #2030, horizontal legend items not displaying in IE11 Preview, + // and for #2580, a similar drawing flaw in Firefox 26. + // Explore if there's a general cause for this. The problem may be related + // to nested group elements, as the legend item texts are within 4 group + // elements. + if ( + /Trident\/7\.0/.test(win.navigator && win.navigator.userAgent) || + isFirefox + ) { + wrap(Legend.prototype, "positionItem", function (proceed, item) { + var legend = this, + // If chart destroyed in sync, this is undefined (#2030) + runPositionItem = function () { + if (item._legendItemPos) { + proceed.call(legend, item); + } + }; + // Do it now, for export and to get checkbox placement + runPositionItem(); + // Do it after to work around the core issue + if (!legend.bubbleLegend) { + setTimeout(runPositionItem); + } }); + } + H.Legend = Legend; + + return H.Legend; + } + ); + _registerModule( + _modules, + "Core/Series/Point.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Globals.js"], + _modules["Core/Utilities.js"], + ], + function (A, H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animObject = A.animObject; + var defined = U.defined, + erase = U.erase, + extend = U.extend, + fireEvent = U.fireEvent, + format = U.format, + getNestedProperty = U.getNestedProperty, + isArray = U.isArray, + isNumber = U.isNumber, + isObject = U.isObject, + syncTimeout = U.syncTimeout, + pick = U.pick, + removeEvent = U.removeEvent, + uniqueKey = U.uniqueKey; + /** + * Function callback when a series point is clicked. Return false to cancel the + * action. + * + * @callback Highcharts.PointClickCallbackFunction + * + * @param {Highcharts.Point} this + * The point where the event occured. + * + * @param {Highcharts.PointClickEventObject} event + * Event arguments. + */ + /** + * Common information for a click event on a series point. + * + * @interface Highcharts.PointClickEventObject + * @extends Highcharts.PointerEventObject + */ /** + * Clicked point. + * @name Highcharts.PointClickEventObject#point + * @type {Highcharts.Point} + */ + /** + * Configuration hash for the data label and tooltip formatters. + * + * @interface Highcharts.PointLabelObject + */ /** + * The point's current color. + * @name Highcharts.PointLabelObject#color + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} + */ /** + * The point's current color index, used in styled mode instead of `color`. The + * color index is inserted in class names used for styling. + * @name Highcharts.PointLabelObject#colorIndex + * @type {number} + */ /** + * The name of the related point. + * @name Highcharts.PointLabelObject#key + * @type {string|undefined} + */ /** + * The percentage for related points in a stacked series or pies. + * @name Highcharts.PointLabelObject#percentage + * @type {number} + */ /** + * The related point. The point name, if defined, is available through + * `this.point.name`. + * @name Highcharts.PointLabelObject#point + * @type {Highcharts.Point} + */ /** + * The related series. The series name is available through `this.series.name`. + * @name Highcharts.PointLabelObject#series + * @type {Highcharts.Series} + */ /** + * The total of values in either a stack for stacked series, or a pie in a pie + * series. + * @name Highcharts.PointLabelObject#total + * @type {number|undefined} + */ /** + * For categorized axes this property holds the category name for the point. For + * other axes it holds the X value. + * @name Highcharts.PointLabelObject#x + * @type {number|string|undefined} + */ /** + * The y value of the point. + * @name Highcharts.PointLabelObject#y + * @type {number|undefined} + */ + /** + * Gets fired when the mouse leaves the area close to the point. + * + * @callback Highcharts.PointMouseOutCallbackFunction + * + * @param {Highcharts.Point} this + * Point where the event occured. + * + * @param {global.PointerEvent} event + * Event that occured. + */ + /** + * Gets fired when the mouse enters the area close to the point. + * + * @callback Highcharts.PointMouseOverCallbackFunction + * + * @param {Highcharts.Point} this + * Point where the event occured. + * + * @param {global.Event} event + * Event that occured. + */ + /** + * The generic point options for all series. + * + * In TypeScript you have to extend `PointOptionsObject` with an additional + * declaration to allow custom data options: + * + * ``` + * declare interface PointOptionsObject { + * customProperty: string; + * } + * ``` + * + * @interface Highcharts.PointOptionsObject + */ + /** + * Possible option types for a data point. Use `null` to indicate a gap. + * + * @typedef {number|string|Highcharts.PointOptionsObject|Array<(number|string|null)>|null} Highcharts.PointOptionsType + */ + /** + * Gets fired when the point is removed using the `.remove()` method. + * + * @callback Highcharts.PointRemoveCallbackFunction + * + * @param {Highcharts.Point} this + * Point where the event occured. + * + * @param {global.Event} event + * Event that occured. + */ + /** + * Possible key values for the point state options. + * + * @typedef {"hover"|"inactive"|"normal"|"select"} Highcharts.PointStateValue + */ + /** + * Gets fired when the point is updated programmatically through the `.update()` + * method. + * + * @callback Highcharts.PointUpdateCallbackFunction + * + * @param {Highcharts.Point} this + * Point where the event occured. + * + * @param {Highcharts.PointUpdateEventObject} event + * Event that occured. + */ + /** + * Information about the update event. + * + * @interface Highcharts.PointUpdateEventObject + * @extends global.Event + */ /** + * Options data of the update event. + * @name Highcharts.PointUpdateEventObject#options + * @type {Highcharts.PointOptionsType} + */ + (""); // detach doclet above + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The Point object. The point objects are generated from the `series.data` + * configuration objects or raw numbers. They can be accessed from the + * `Series.points` array. Other ways to instantiate points are through {@link + * Highcharts.Series#addPoint} or {@link Highcharts.Series#setData}. + * + * @class + * @name Highcharts.Point + */ + var Point = /** @class */ (function () { + function Point() { + /* * + * + * Properties + * + * */ + /** + * For categorized axes this property holds the category name for the + * point. For other axes it holds the X value. + * + * @name Highcharts.Point#category + * @type {string} + */ + this.category = void 0; + /** + * The point's current color index, used in styled mode instead of + * `color`. The color index is inserted in class names used for styling. + * + * @name Highcharts.Point#colorIndex + * @type {number} + */ + this.colorIndex = void 0; + this.formatPrefix = "point"; + this.id = void 0; + this.isNull = false; + /** + * The name of the point. The name can be given as the first position of the + * point configuration array, or as a `name` property in the configuration: + * + * @example + * // Array config + * data: [ + * ['John', 1], + * ['Jane', 2] + * ] + * + * // Object config + * data: [{ + * name: 'John', + * y: 1 + * }, { + * name: 'Jane', + * y: 2 + * }] + * + * @name Highcharts.Point#name + * @type {string} + */ + this.name = void 0; + /** + * The point's options as applied in the initial configuration, or + * extended through `Point.update`. + * + * In TypeScript you have to extend `PointOptionsObject` via an + * additional interface to allow custom data options: + * + * ``` + * declare interface PointOptionsObject { + * customProperty: string; + * } + * ``` + * + * @name Highcharts.Point#options + * @type {Highcharts.PointOptionsObject} + */ + this.options = void 0; + /** + * The percentage for points in a stacked series or pies. + * + * @name Highcharts.Point#percentage + * @type {number|undefined} + */ + this.percentage = void 0; + this.selected = false; + /** + * The series object associated with the point. + * + * @name Highcharts.Point#series + * @type {Highcharts.Series} + */ + this.series = void 0; + /** + * The total of values in either a stack for stacked series, or a pie in a + * pie series. + * + * @name Highcharts.Point#total + * @type {number|undefined} + */ + this.total = void 0; + /** + * For certain series types, like pie charts, where individual points can + * be shown or hidden. + * + * @name Highcharts.Point#visible + * @type {boolean} + * @default true + */ + this.visible = true; + this.x = void 0; + } + /* * + * + * Functions + * + * */ + /** + * Animate SVG elements associated with the point. + * + * @private + * @function Highcharts.Point#animateBeforeDestroy + */ + Point.prototype.animateBeforeDestroy = function () { + var point = this, + animateParams = { x: point.startXPos, opacity: 0 }, + isDataLabel, + graphicalProps = point.getGraphicalProps(); + graphicalProps.singular.forEach(function (prop) { + isDataLabel = prop === "dataLabel"; + point[prop] = point[prop].animate( + isDataLabel + ? { + x: point[prop].startXPos, + y: point[prop].startYPos, + opacity: 0, + } + : animateParams + ); + }); + graphicalProps.plural.forEach(function (plural) { + point[plural].forEach(function (item) { + if (item.element) { + item.animate( + extend( + { x: point.startXPos }, + item.startYPos + ? { + x: item.startXPos, + y: item.startYPos, + } + : {} + ) + ); + } + }); + }); + }; + /** + * Apply the options containing the x and y data and possible some extra + * properties. Called on point init or from point.update. + * + * @private + * @function Highcharts.Point#applyOptions + * + * @param {Highcharts.PointOptionsType} options + * The point options as defined in series.data. + * + * @param {number} [x] + * Optionally, the x value. + * + * @return {Highcharts.Point} + * The Point instance. + */ + Point.prototype.applyOptions = function (options, x) { + var point = this, + series = point.series, + pointValKey = series.options.pointValKey || series.pointValKey; + options = Point.prototype.optionsToObject.call(this, options); + // copy options directly to point + extend(point, options); + point.options = point.options + ? extend(point.options, options) + : options; + // Since options are copied into the Point instance, some accidental + // options must be shielded (#5681) + if (options.group) { + delete point.group; + } + if (options.dataLabels) { + delete point.dataLabels; + } + /** + * The y value of the point. + * @name Highcharts.Point#y + * @type {number|undefined} + */ + // For higher dimension series types. For instance, for ranges, point.y + // is mapped to point.low. + if (pointValKey) { + point.y = Point.prototype.getNestedProperty.call( + point, + pointValKey + ); + } + point.isNull = pick( + point.isValid && !point.isValid(), + point.x === null || !isNumber(point.y) + ); // #3571, check for NaN + point.formatPrefix = point.isNull ? "null" : "point"; // #9233, #10874 + // The point is initially selected by options (#5777) + if (point.selected) { + point.state = "select"; + } + /** + * The x value of the point. + * @name Highcharts.Point#x + * @type {number} + */ + // If no x is set by now, get auto incremented value. All points must + // have an x value, however the y value can be null to create a gap in + // the series + if ( + "name" in point && + typeof x === "undefined" && + series.xAxis && + series.xAxis.hasNames + ) { + point.x = series.xAxis.nameToX(point); + } + if (typeof point.x === "undefined" && series) { + if (typeof x === "undefined") { + point.x = series.autoIncrement(point); + } else { + point.x = x; + } + } + return point; + }; /** - * These collections (arrays) implement `Chart.addSomethig` method used in - * chart.update() to create new object in the collection. Equivalent for - * deleting is resolved by simple `Somethig.remove()`. + * Destroy a point to clear memory. Its reference still stays in + * `series.data`. * - * Note: We need to define these references after initializers are bound to - * chart's prototype. + * @private + * @function Highcharts.Point#destroy */ - Chart.prototype.collectionsWithInit = { - // collectionName: [ initializingMethod, [extraArguments] ] - xAxis: [Chart.prototype.addAxis, [true]], - yAxis: [Chart.prototype.addAxis, [false]], - series: [Chart.prototype.addSeries] + Point.prototype.destroy = function () { + var point = this, + series = point.series, + chart = series.chart, + dataSorting = series.options.dataSorting, + hoverPoints = chart.hoverPoints, + globalAnimation = point.series.chart.renderer.globalAnimation, + animation = animObject(globalAnimation), + prop; + /** + * Allow to call after animation. + * @private + */ + function destroyPoint() { + // Remove all events and elements + if (point.graphic || point.dataLabel || point.dataLabels) { + removeEvent(point); + point.destroyElements(); + } + for (prop in point) { + // eslint-disable-line guard-for-in + point[prop] = null; + } + } + if (point.legendItem) { + // pies have legend items + chart.legend.destroyItem(point); + } + if (hoverPoints) { + point.setState(); + erase(hoverPoints, point); + if (!hoverPoints.length) { + chart.hoverPoints = null; + } + } + if (point === chart.hoverPoint) { + point.onMouseOut(); + } + // Remove properties after animation + if (!dataSorting || !dataSorting.enabled) { + destroyPoint(); + } else { + this.animateBeforeDestroy(); + syncTimeout(destroyPoint, animation.duration); + } + chart.pointCount--; }; - // extend the Point prototype for dynamic methods - extend(Point.prototype, /** @lends Highcharts.Point.prototype */ { - /** - * Update point with new options (typically x/y data) and optionally redraw - * the series. - * - * @sample highcharts/members/point-update-column/ - * Update column value - * @sample highcharts/members/point-update-pie/ - * Update pie slice - * @sample maps/members/point-update/ - * Update map area value in Highmaps - * - * @function Highcharts.Point#update - * - * @param {Highcharts.PointOptionsType} options - * The point options. Point options are handled as described under - * the `series.type.data` item for each series type. For example - * for a line series, if options is a single number, the point will - * be given that number as the marin y value. If it is an array, it - * will be interpreted as x and y values respectively. If it is an - * object, advanced options are applied. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the point is updated. If doing - * more operations on the chart, it is best practice to set - * `redraw` to false and call `chart.redraw()` after. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] - * Whether to apply animation, and optionally animation - * configuration. - * - * @return {void} - * - * @fires Highcharts.Point#event:update - */ - update: function (options, redraw, animation, runEvent) { - var point = this, - series = point.series, - graphic = point.graphic, - i, - chart = series.chart, - seriesOptions = series.options; - redraw = pick(redraw, true); - /** - * @private - */ - function update() { - point.applyOptions(options); - // Update visuals, #4146 - // Handle dummy graphic elements for a11y, #12718 - var hasDummyGraphic = graphic && point.hasDummyGraphic; - var shouldDestroyGraphic = point.y === null ? !hasDummyGraphic : hasDummyGraphic; - if (graphic && shouldDestroyGraphic) { - point.graphic = graphic.destroy(); - delete point.hasDummyGraphic; - } - if (isObject(options, true)) { - // Destroy so we can get new elements - if (graphic && graphic.element) { - // "null" is also a valid symbol - if (options && - options.marker && - typeof options.marker.symbol !== 'undefined') { - point.graphic = graphic.destroy(); - } - } - if (options && options.dataLabels && point.dataLabel) { - point.dataLabel = point.dataLabel.destroy(); // #2468 - } - if (point.connector) { - point.connector = point.connector.destroy(); // #7243 - } - } - // record changes in the parallel arrays - i = point.index; - series.updateParallelArrays(point, i); - // Record the options to options.data. If the old or the new config - // is an object, use point options, otherwise use raw options - // (#4701, #4916). - seriesOptions.data[i] = (isObject(seriesOptions.data[i], true) || - isObject(options, true)) ? - point.options : - pick(options, seriesOptions.data[i]); - // redraw - series.isDirty = series.isDirtyData = true; - if (!series.fixedBox && series.hasCartesianSeries) { // #1906, #2320 - chart.isDirtyBox = true; - } - if (seriesOptions.legendType === 'point') { // #1831, #1885 - chart.isDirtyLegend = true; - } - if (redraw) { - chart.redraw(animation); - } - } - // Fire the event with a default handler of doing the update - if (runEvent === false) { // When called from setData - update(); - } - else { - point.firePointEvent('update', { options: options }, update); - } - }, - /** - * Remove a point and optionally redraw the series and if necessary the axes - * - * @sample highcharts/plotoptions/series-point-events-remove/ - * Remove point and confirm - * @sample highcharts/members/point-remove/ - * Remove pie slice - * @sample maps/members/point-remove/ - * Remove selected points in Highmaps - * - * @function Highcharts.Point#remove - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart or wait for an explicit call. When - * doing more operations on the chart, for example running - * `point.remove()` in a loop, it is best practice to set `redraw` - * to false and call `chart.redraw()` after. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=false] - * Whether to apply animation, and optionally animation - * configuration. - * - * @return {void} - */ - remove: function (redraw, animation) { - this.series.removePoint(this.series.data.indexOf(this), redraw, animation); + /** + * Destroy SVG elements associated with the point. + * + * @private + * @function Highcharts.Point#destroyElements + * @param {Highcharts.Dictionary<number>} [kinds] + */ + Point.prototype.destroyElements = function (kinds) { + var point = this, + props = point.getGraphicalProps(kinds); + props.singular.forEach(function (prop) { + point[prop] = point[prop].destroy(); + }); + props.plural.forEach(function (plural) { + point[plural].forEach(function (item) { + if (item.element) { + item.destroy(); + } + }); + delete point[plural]; + }); + }; + /** + * Fire an event on the Point object. + * + * @private + * @function Highcharts.Point#firePointEvent + * + * @param {string} eventType + * Type of the event. + * + * @param {Highcharts.Dictionary<any>|Event} [eventArgs] + * Additional event arguments. + * + * @param {Highcharts.EventCallbackFunction<Highcharts.Point>|Function} [defaultFunction] + * Default event handler. + * + * @fires Highcharts.Point#event:* + */ + Point.prototype.firePointEvent = function ( + eventType, + eventArgs, + defaultFunction + ) { + var point = this, + series = this.series, + seriesOptions = series.options; + // load event handlers on demand to save time on mouseover/out + if ( + seriesOptions.point.events[eventType] || + (point.options && + point.options.events && + point.options.events[eventType]) + ) { + point.importEvents(); + } + // add default handler if in selection mode + if (eventType === "click" && seriesOptions.allowPointSelect) { + defaultFunction = function (event) { + // Control key is for Windows, meta (= Cmd key) for Mac, Shift + // for Opera. + if (point.select) { + // #2911 + point.select( + null, + event.ctrlKey || event.metaKey || event.shiftKey + ); + } + }; + } + fireEvent(point, eventType, eventArgs, defaultFunction); + }; + /** + * Get the CSS class names for individual points. Used internally where the + * returned value is set on every point. + * + * @function Highcharts.Point#getClassName + * + * @return {string} + * The class names. + */ + Point.prototype.getClassName = function () { + var point = this; + return ( + "highcharts-point" + + (point.selected ? " highcharts-point-select" : "") + + (point.negative ? " highcharts-negative" : "") + + (point.isNull ? " highcharts-null-point" : "") + + (typeof point.colorIndex !== "undefined" + ? " highcharts-color-" + point.colorIndex + : "") + + (point.options.className ? " " + point.options.className : "") + + (point.zone && point.zone.className + ? " " + point.zone.className.replace("highcharts-negative", "") + : "") + ); + }; + /** + * Get props of all existing graphical point elements. + * + * @private + * @function Highcharts.Point#getGraphicalProps + * @param {Highcharts.Dictionary<number>} [kinds] + * @return {Highcharts.PointGraphicalProps} + */ + Point.prototype.getGraphicalProps = function (kinds) { + var point = this, + props = [], + prop, + i, + graphicalProps = { singular: [], plural: [] }; + kinds = kinds || { graphic: 1, dataLabel: 1 }; + if (kinds.graphic) { + props.push("graphic", "shadowGroup"); + } + if (kinds.dataLabel) { + props.push("dataLabel", "dataLabelUpper", "connector"); + } + i = props.length; + while (i--) { + prop = props[i]; + if (point[prop]) { + graphicalProps.singular.push(prop); } - }); - // Extend the series prototype for dynamic methods - extend(LineSeries.prototype, /** @lends Series.prototype */ { - /** - * Add a point to the series after render time. The point can be added at - * the end, or by giving it an X value, to the start or in the middle of the - * series. - * - * @sample highcharts/members/series-addpoint-append/ - * Append point - * @sample highcharts/members/series-addpoint-append-and-shift/ - * Append and shift - * @sample highcharts/members/series-addpoint-x-and-y/ - * Both X and Y values given - * @sample highcharts/members/series-addpoint-pie/ - * Append pie slice - * @sample stock/members/series-addpoint/ - * Append 100 points in Highstock - * @sample stock/members/series-addpoint-shift/ - * Append and shift in Highstock - * @sample maps/members/series-addpoint/ - * Add a point in Highmaps - * - * @function Highcharts.Series#addPoint - * - * @param {Highcharts.PointOptionsType} options - * The point options. If options is a single number, a point with - * that y value is appended to the series. If it is an array, it will - * be interpreted as x and y values respectively. If it is an - * object, advanced options as outlined under `series.data` are - * applied. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the point is added. When adding - * more than one point, it is highly recommended that the redraw - * option be set to false, and instead {@link Chart#redraw} is - * explicitly called after the adding of points is finished. - * Otherwise, the chart will redraw after adding each point. - * - * @param {boolean} [shift=false] - * If true, a point is shifted off the start of the series as one is - * appended to the end. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * Whether to apply animation, and optionally animation - * configuration. - * - * @param {boolean} [withEvent=true] - * Used internally, whether to fire the series `addPoint` event. - * - * @return {void} - * - * @fires Highcharts.Series#event:addPoint - */ - addPoint: function (options, redraw, shift, animation, withEvent) { - var series = this, - seriesOptions = series.options, - data = series.data, - chart = series.chart, - xAxis = series.xAxis, - names = xAxis && xAxis.hasNames && xAxis.names, - dataOptions = seriesOptions.data, - point, - xData = series.xData, - isInTheMiddle, - i, - x; - // Optional redraw, defaults to true - redraw = pick(redraw, true); - // Get options and push the point to xData, yData and series.options. In - // series.generatePoints the Point instance will be created on demand - // and pushed to the series.data array. - point = { series: series }; - series.pointClass.prototype.applyOptions.apply(point, [options]); - x = point.x; - // Get the insertion point - i = xData.length; - if (series.requireSorting && x < xData[i - 1]) { - isInTheMiddle = true; - while (i && xData[i - 1] > x) { - i--; - } - } - // Insert undefined item - series.updateParallelArrays(point, 'splice', i, 0, 0); - // Update it - series.updateParallelArrays(point, i); - if (names && point.name) { - names[x] = point.name; - } - dataOptions.splice(i, 0, options); - if (isInTheMiddle) { - series.data.splice(i, 0, null); - series.processData(); - } - // Generate points to be added to the legend (#1329) - if (seriesOptions.legendType === 'point') { - series.generatePoints(); - } - // Shift the first point off the parallel arrays - if (shift) { - if (data[0] && data[0].remove) { - data[0].remove(false); - } - else { - data.shift(); - series.updateParallelArrays(point, 'shift'); - dataOptions.shift(); - } - } - // Fire event - if (withEvent !== false) { - fireEvent(series, 'addPoint', { point: point }); - } - // redraw - series.isDirty = true; - series.isDirtyData = true; - if (redraw) { - chart.redraw(animation); // Animation is set anyway on redraw, #5665 - } - }, - /** - * Remove a point from the series. Unlike the - * {@link Highcharts.Point#remove} method, this can also be done on a point - * that is not instanciated because it is outside the view or subject to - * Highstock data grouping. - * - * @sample highcharts/members/series-removepoint/ - * Remove cropped point - * - * @function Highcharts.Series#removePoint - * - * @param {number} i - * The index of the point in the {@link Highcharts.Series.data|data} - * array. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the point is added. When - * removing more than one point, it is highly recommended that the - * `redraw` option be set to `false`, and instead {@link - * Highcharts.Chart#redraw} is explicitly called after the adding of - * points is finished. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * Whether and optionally how the series should be animated. - * - * @return {void} - * - * @fires Highcharts.Point#event:remove - */ - removePoint: function (i, redraw, animation) { - var series = this, - data = series.data, - point = data[i], - points = series.points, - chart = series.chart, - remove = function () { - if (points && points.length === data.length) { // #4935 - points.splice(i, 1); - } - data.splice(i, 1); - series.options.data.splice(i, 1); - series.updateParallelArrays(point || { series: series }, 'splice', i, 1); - if (point) { - point.destroy(); - } - // redraw - series.isDirty = true; - series.isDirtyData = true; - if (redraw) { - chart.redraw(); - } - }; - setAnimation(animation, chart); - redraw = pick(redraw, true); - // Fire the event with a default handler of removing the point - if (point) { - point.firePointEvent('remove', null, remove); - } - else { - remove(); - } - }, - /** - * Remove a series and optionally redraw the chart. - * - * @sample highcharts/members/series-remove/ - * Remove first series from a button - * - * @function Highcharts.Series#remove - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart or wait for an explicit call to - * {@link Highcharts.Chart#redraw}. - * - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] - * Whether to apply animation, and optionally animation - * configuration. - * - * @param {boolean} [withEvent=true] - * Used internally, whether to fire the series `remove` event. - * - * @return {void} - * - * @fires Highcharts.Series#event:remove - */ - remove: function (redraw, animation, withEvent, keepEvents) { - var series = this, - chart = series.chart; - /** - * @private - */ - function remove() { - // Destroy elements - series.destroy(keepEvents); - series.remove = null; // Prevent from doing again (#9097) - // Redraw - chart.isDirtyLegend = chart.isDirtyBox = true; - chart.linkSeries(); - if (pick(redraw, true)) { - chart.redraw(animation); - } - } - // Fire the event with a default handler of removing the point - if (withEvent !== false) { - fireEvent(series, 'remove', null, remove); - } - else { - remove(); - } - }, - /** - * Update the series with a new set of options. For a clean and precise - * handling of new options, all methods and elements from the series are - * removed, and it is initialized from scratch. Therefore, this method is - * more performance expensive than some other utility methods like {@link - * Series#setData} or {@link Series#setVisible}. - * - * Note that `Series.update` may mutate the passed `data` options. - * - * @sample highcharts/members/series-update/ - * Updating series options - * @sample maps/members/series-update/ - * Update series options in Highmaps - * - * @function Highcharts.Series#update - * - * @param {Highcharts.SeriesOptionsType} options - * New options that will be merged with the series' existing options. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the series is altered. If doing - * more operations on the chart, it is a good idea to set redraw to - * false and call {@link Chart#redraw} after. - * - * @return {void} - * - * @fires Highcharts.Series#event:update - * @fires Highcharts.Series#event:afterUpdate - */ - update: function (options, redraw) { - options = H.cleanRecursively(options, this.userOptions); - fireEvent(this, 'update', { options: options }); - var series = this, - chart = series.chart, - // must use user options when changing type because series.options - // is merged in with type specific plotOptions - oldOptions = series.userOptions, - seriesOptions, - initialType = series.initialType || series.type, - plotOptions = chart.options.plotOptions, - newType = (options.type || - oldOptions.type || - chart.options.chart.type), - keepPoints = !( - // Indicators, histograms etc recalculate the data. It should be - // possible to omit this. - this.hasDerivedData || - // New type requires new point classes - (newType && newType !== this.type) || - // New options affecting how the data points are built - typeof options.pointStart !== 'undefined' || - typeof options.pointInterval !== 'undefined' || - // Changes to data grouping requires new points in new group - series.hasOptionChanged('dataGrouping') || - series.hasOptionChanged('pointStart') || - series.hasOptionChanged('pointInterval') || - series.hasOptionChanged('pointIntervalUnit') || - series.hasOptionChanged('keys')), - initialSeriesProto = seriesTypes[initialType].prototype, - n, - groups = [ - 'group', - 'markerGroup', - 'dataLabelsGroup', - 'transformGroup' - ], - preserve = [ - 'eventOptions', - 'navigatorSeries', - 'baseSeries' - ], - // Animation must be enabled when calling update before the initial - // animation has first run. This happens when calling update - // directly after chart initialization, or when applying responsive - // rules (#6912). - animation = series.finishedAnimating && { animation: false }, - kinds = {}; - if (keepPoints) { - preserve.push('data', 'isDirtyData', 'points', 'processedXData', 'processedYData', 'xIncrement', 'cropped', '_hasPointMarkers', '_hasPointLabels', - // Map specific, consider moving it to series-specific preserve- - // properties (#10617) - 'mapMap', 'mapData', 'minY', 'maxY', 'minX', 'maxX'); - if (options.visible !== false) { - preserve.push('area', 'graph'); - } - series.parallelArrays.forEach(function (key) { - preserve.push(key + 'Data'); - }); - if (options.data) { - // setData uses dataSorting options so we need to update them - // earlier - if (options.dataSorting) { - extend(series.options.dataSorting, options.dataSorting); - } - this.setData(options.data, false); - } - } - // Do the merge, with some forced options - options = merge(oldOptions, animation, { - // When oldOptions.index is null it should't be cleared. - // Otherwise navigator series will have wrong indexes (#10193). - index: typeof oldOptions.index === 'undefined' ? - series.index : oldOptions.index, - pointStart: pick( - // when updating from blank (#7933) - plotOptions && plotOptions.series && plotOptions.series.pointStart, oldOptions.pointStart, - // when updating after addPoint - series.xData[0]) - }, (!keepPoints && { data: series.options.data }), options); - // Merge does not merge arrays, but replaces them. Since points were - // updated, `series.options.data` has correct merged options, use it: - if (keepPoints && options.data) { - options.data = series.options.data; - } - // Make sure preserved properties are not destroyed (#3094) - preserve = groups.concat(preserve); - preserve.forEach(function (prop) { - preserve[prop] = series[prop]; - delete series[prop]; - }); - // Destroy the series and delete all properties. Reinsert all - // methods and properties from the new type prototype (#2270, - // #3719). - series.remove(false, null, false, true); - for (n in initialSeriesProto) { // eslint-disable-line guard-for-in - series[n] = void 0; - } - if (seriesTypes[newType || initialType]) { - extend(series, seriesTypes[newType || initialType].prototype); - } - else { - error(17, true, chart, { missingModuleFor: (newType || initialType) }); - } - // Re-register groups (#3094) and other preserved properties - preserve.forEach(function (prop) { - series[prop] = preserve[prop]; - }); - series.init(chart, options); - // Remove particular elements of the points. Check `series.options` - // because we need to consider the options being set on plotOptions as - // well. - if (keepPoints && this.points) { - seriesOptions = series.options; - // What kind of elements to destroy - if (seriesOptions.visible === false) { - kinds.graphic = 1; - kinds.dataLabel = 1; - } - else if (!series._hasPointLabels) { - var marker = seriesOptions.marker, - dataLabels = seriesOptions.dataLabels; - if (marker && (marker.enabled === false || - 'symbol' in marker // #10870 - )) { - kinds.graphic = 1; - } - if (dataLabels && - dataLabels.enabled === false) { - kinds.dataLabel = 1; - } - } - this.points.forEach(function (point) { - if (point && point.series) { - point.resolveColor(); - // Destroy elements in order to recreate based on updated - // series options. - if (Object.keys(kinds).length) { - point.destroyElements(kinds); - } - if (seriesOptions.showInLegend === false && - point.legendItem) { - chart.legend.destroyItem(point); - } - } - }, this); - } - series.initialType = initialType; - chart.linkSeries(); // Links are lost in series.remove (#3028) - fireEvent(this, 'afterUpdate'); - if (pick(redraw, true)) { - chart.redraw(keepPoints ? void 0 : false); - } - }, - /** - * Used from within series.update - * - * @private - * @function Highcharts.Series#setName - * - * @param {string} name - * - * @return {void} - */ - setName: function (name) { - this.name = this.options.name = this.userOptions.name = name; - this.chart.isDirtyLegend = true; - }, - /** - * Check if the option has changed. - * - * @private - * @function Highcharts.Series#hasOptionChanged - * - * @param {string} option - * - * @return {boolean} - */ - hasOptionChanged: function (optionName) { - var chart = this.chart, - option = this.options[optionName], - plotOptions = chart.options.plotOptions, - oldOption = this.userOptions[optionName]; - if (oldOption) { - return option !== oldOption; - } - return option !== - pick(plotOptions && plotOptions[this.type] && plotOptions[this.type][optionName], plotOptions && plotOptions.series && plotOptions.series[optionName], option); + } + ["dataLabel", "connector"].forEach(function (prop) { + var plural = prop + "s"; + if (kinds[prop] && point[plural]) { + graphicalProps.plural.push(plural); } - }); - // Extend the Axis.prototype for dynamic methods - extend(Axis.prototype, /** @lends Highcharts.Axis.prototype */ { - /** - * Update an axis object with a new set of options. The options are merged - * with the existing options, so only new or altered options need to be - * specified. - * - * @sample highcharts/members/axis-update/ - * Axis update demo - * - * @function Highcharts.Axis#update - * - * @param {Highcharts.AxisOptions} options - * The new options that will be merged in with existing options on - * the axis. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the axis is altered. If doing - * more operations on the chart, it is a good idea to set redraw to - * false and call {@link Chart#redraw} after. - * - * @return {void} - */ - update: function (options, redraw) { - var chart = this.chart, - newEvents = ((options && options.events) || {}); - options = merge(this.userOptions, options); - // Color Axis is not an array, - // This change is applied in the ColorAxis wrapper - if (chart.options[this.coll].indexOf) { - // Don't use this.options.index, - // StockChart has Axes in navigator too - chart.options[this.coll][chart.options[this.coll].indexOf(this.userOptions)] = options; - } - // Remove old events, if no new exist (#8161) - objectEach(chart.options[this.coll].events, function (fn, ev) { - if (typeof newEvents[ev] === 'undefined') { - newEvents[ev] = void 0; - } - }); - this.destroy(true); - this.init(chart, extend(options, { events: newEvents })); - chart.isDirtyBox = true; - if (pick(redraw, true)) { - chart.redraw(); - } - }, - /** - * Remove the axis from the chart. - * - * @sample highcharts/members/chart-addaxis/ - * Add and remove axes - * - * @function Highcharts.Axis#remove - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart following the remove. - * - * @return {void} - */ - remove: function (redraw) { - var chart = this.chart, - key = this.coll, // xAxis or yAxis - axisSeries = this.series, - i = axisSeries.length; - // Remove associated series (#2687) - while (i--) { - if (axisSeries[i]) { - axisSeries[i].remove(false); - } - } - // Remove the axis - erase(chart.axes, this); - erase(chart[key], this); - if (isArray(chart.options[key])) { - chart.options[key].splice(this.options.index, 1); - } - else { // color axis, #6488 - delete chart.options[key]; - } - chart[key].forEach(function (axis, i) { - // Re-index, #1706, #8075 - axis.options.index = axis.userOptions.index = i; - }); - this.destroy(); - chart.isDirtyBox = true; - if (pick(redraw, true)) { - chart.redraw(); - } - }, - /** - * Update the axis title by options after render time. - * - * @sample highcharts/members/axis-settitle/ - * Set a new Y axis title - * - * @function Highcharts.Axis#setTitle - * - * @param {Highcharts.AxisTitleOptions} titleOptions - * The additional title options. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after setting the title. - * - * @return {void} - */ - setTitle: function (titleOptions, redraw) { - this.update({ title: titleOptions }, redraw); - }, - /** - * Set new axis categories and optionally redraw. - * - * @sample highcharts/members/axis-setcategories/ - * Set categories by click on a button - * - * @function Highcharts.Axis#setCategories - * - * @param {Array<string>} categories - * The new categories. - * - * @param {boolean} [redraw=true] - * Whether to redraw the chart. - * - * @return {void} - */ - setCategories: function (categories, redraw) { - this.update({ categories: categories }, redraw); + }); + return graphicalProps; + }; + /** + * Return the configuration hash needed for the data label and tooltip + * formatters. + * + * @function Highcharts.Point#getLabelConfig + * + * @return {Highcharts.PointLabelObject} + * Abstract object used in formatters and formats. + */ + Point.prototype.getLabelConfig = function () { + return { + x: this.category, + y: this.y, + color: this.color, + colorIndex: this.colorIndex, + key: this.name || this.category, + series: this.series, + point: this, + percentage: this.percentage, + total: this.total || this.stackTotal, + }; + }; + /** + * Returns the value of the point property for a given value. + * @private + */ + Point.prototype.getNestedProperty = function (key) { + if (!key) { + return; + } + if (key.indexOf("custom.") === 0) { + return getNestedProperty(key, this.options); + } + return this[key]; + }; + /** + * In a series with `zones`, return the zone that the point belongs to. + * + * @function Highcharts.Point#getZone + * + * @return {Highcharts.SeriesZonesOptionsObject} + * The zone item. + */ + Point.prototype.getZone = function () { + var series = this.series, + zones = series.zones, + zoneAxis = series.zoneAxis || "y", + i = 0, + zone; + zone = zones[i]; + while (this[zoneAxis] >= zone.value) { + zone = zones[++i]; + } + // For resetting or reusing the point (#8100) + if (!this.nonZonedColor) { + this.nonZonedColor = this.color; + } + if (zone && zone.color && !this.options.color) { + this.color = zone.color; + } else { + this.color = this.nonZonedColor; + } + return zone; + }; + /** + * Utility to check if point has new shape type. Used in column series and + * all others that are based on column series. + * + * @return boolean|undefined + */ + Point.prototype.hasNewShapeType = function () { + var point = this; + var oldShapeType = + point.graphic && + (point.graphic.symbolName || point.graphic.element.nodeName); + return oldShapeType !== this.shapeType; + }; + /** + * Initialize the point. Called internally based on the `series.data` + * option. + * + * @function Highcharts.Point#init + * + * @param {Highcharts.Series} series + * The series object containing this point. + * + * @param {Highcharts.PointOptionsType} options + * The data in either number, array or object format. + * + * @param {number} [x] + * Optionally, the X value of the point. + * + * @return {Highcharts.Point} + * The Point instance. + * + * @fires Highcharts.Point#event:afterInit + */ + Point.prototype.init = function (series, options, x) { + this.series = series; + this.applyOptions(options, x); + // Add a unique ID to the point if none is assigned + this.id = defined(this.id) ? this.id : uniqueKey(); + this.resolveColor(); + series.chart.pointCount++; + fireEvent(this, "afterInit"); + return this; + }; + /** + * Transform number or array configs into objects. Also called for object + * configs. Used internally to unify the different configuration formats for + * points. For example, a simple number `10` in a line series will be + * transformed to `{ y: 10 }`, and an array config like `[1, 10]` in a + * scatter series will be transformed to `{ x: 1, y: 10 }`. + * + * @function Highcharts.Point#optionsToObject + * + * @param {Highcharts.PointOptionsType} options + * The input option. + * + * @return {Highcharts.Dictionary<*>} + * Transformed options. + */ + Point.prototype.optionsToObject = function (options) { + var ret = {}, + series = this.series, + keys = series.options.keys, + pointArrayMap = keys || series.pointArrayMap || ["y"], + valueCount = pointArrayMap.length, + firstItemType, + i = 0, + j = 0; + if (isNumber(options) || options === null) { + ret[pointArrayMap[0]] = options; + } else if (isArray(options)) { + // with leading x value + if (!keys && options.length > valueCount) { + firstItemType = typeof options[0]; + if (firstItemType === "string") { + ret.name = options[0]; + } else if (firstItemType === "number") { + ret.x = options[0]; + } + i++; + } + while (j < valueCount) { + // Skip undefined positions for keys + if (!keys || typeof options[i] !== "undefined") { + if (pointArrayMap[j].indexOf(".") > 0) { + // Handle nested keys, e.g. ['color.pattern.image'] + // Avoid function call unless necessary. + Point.prototype.setNestedProperty( + ret, + options[i], + pointArrayMap[j] + ); + } else { + ret[pointArrayMap[j]] = options[i]; + } + } + i++; + j++; + } + } else if (typeof options === "object") { + ret = options; + // This is the fastest way to detect if there are individual point + // dataLabels that need to be considered in drawDataLabels. These + // can only occur in object configs. + if (options.dataLabels) { + series._hasPointLabels = true; + } + // Same approach as above for markers + if (options.marker) { + series._hasPointMarkers = true; + } + } + return ret; + }; + /** + * @private + * @function Highcharts.Point#resolveColor + * @return {void} + */ + Point.prototype.resolveColor = function () { + var series = this.series, + colors, + optionsChart = series.chart.options.chart, + colorCount = optionsChart.colorCount, + styledMode = series.chart.styledMode, + colorIndex; + // remove points nonZonedColor for later recalculation + delete this.nonZonedColor; + /** + * The point's current color. + * + * @name Highcharts.Point#color + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined} + */ + if (!styledMode && !this.options.color) { + this.color = series.color; // #3445 + } + if (series.options.colorByPoint) { + if (!styledMode) { + colors = series.options.colors || series.chart.options.colors; + this.color = this.color || colors[series.colorCounter]; + colorCount = colors.length; + } + colorIndex = series.colorCounter; + series.colorCounter++; + // loop back to zero + if (series.colorCounter === colorCount) { + series.colorCounter = 0; + } + } else { + colorIndex = series.colorIndex; + } + this.colorIndex = pick(this.colorIndex, colorIndex); + }; + /** + * Set a value in an object, on the property defined by key. The key + * supports nested properties using dot notation. The function modifies the + * input object and does not make a copy. + * + * @function Highcharts.Point#setNestedProperty<T> + * + * @param {T} object + * The object to set the value on. + * + * @param {*} value + * The value to set. + * + * @param {string} key + * Key to the property to set. + * + * @return {T} + * The modified object. + */ + Point.prototype.setNestedProperty = function (object, value, key) { + var nestedKeys = key.split("."); + nestedKeys.reduce(function (result, key, i, arr) { + var isLastKey = arr.length - 1 === i; + result[key] = isLastKey + ? value + : isObject(result[key], true) + ? result[key] + : {}; + return result[key]; + }, object); + return object; + }; + /** + * Extendable method for formatting each point's tooltip line. + * + * @function Highcharts.Point#tooltipFormatter + * + * @param {string} pointFormat + * The point format. + * + * @return {string} + * A string to be concatenated in to the common tooltip text. + */ + Point.prototype.tooltipFormatter = function (pointFormat) { + // Insert options for valueDecimals, valuePrefix, and valueSuffix + var series = this.series, + seriesTooltipOptions = series.tooltipOptions, + valueDecimals = pick(seriesTooltipOptions.valueDecimals, ""), + valuePrefix = seriesTooltipOptions.valuePrefix || "", + valueSuffix = seriesTooltipOptions.valueSuffix || ""; + // Replace default point style with class name + if (series.chart.styledMode) { + pointFormat = series.chart.tooltip.styledModeFormat(pointFormat); + } + // Loop over the point array map and replace unformatted values with + // sprintf formatting markup + (series.pointArrayMap || ["y"]).forEach(function (key) { + key = "{point." + key; // without the closing bracket + if (valuePrefix || valueSuffix) { + pointFormat = pointFormat.replace( + RegExp(key + "}", "g"), + valuePrefix + key + "}" + valueSuffix + ); } - }); + pointFormat = pointFormat.replace( + RegExp(key + "}", "g"), + key + ":,." + valueDecimals + "f}" + ); + }); + return format( + pointFormat, + { + point: this, + series: this.series, + }, + series.chart + ); + }; + return Point; + })(); + H.Point = Point; - }); - _registerModule(_modules, 'Series/AreaSeries.js', [_modules['Core/Series/Series.js'], _modules['Core/Color/Color.js'], _modules['Core/Globals.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Utilities.js']], function (BaseSeries, Color, H, LegendSymbolMixin, U) { + return Point; + } + ); + _registerModule( + _modules, + "Core/Series/Series.js", + [ + _modules["Core/Globals.js"], + _modules["Core/Series/Point.js"], + _modules["Core/Utilities.js"], + ], + function (H, Point, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var error = U.error, + extendClass = U.extendClass, + fireEvent = U.fireEvent, + getOptions = U.getOptions, + isObject = U.isObject, + merge = U.merge, + objectEach = U.objectEach; + /** + * @class + * @name Highcharts.Series + */ + var BaseSeries = /** @class */ (function () { /* * * - * (c) 2010-2020 Torstein Honsi + * Constructor * - * License: www.highcharts.com/license + * */ + function BaseSeries(chart, options) { + var mergedOptions = merge(BaseSeries.defaultOptions, options); + this.chart = chart; + this._i = chart.series.length; + chart.series.push(this); + this.options = mergedOptions; + this.userOptions = merge(options); + } + /* * * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * Static Functions * * */ - var color = Color.parse; - var objectEach = U.objectEach, - pick = U.pick; - var Series = H.Series; + BaseSeries.addSeries = function (seriesName, seriesType) { + BaseSeries.seriesTypes[seriesName] = seriesType; + }; + BaseSeries.cleanRecursively = function (toClean, reference) { + var clean = {}; + objectEach(toClean, function (_val, key) { + var ob; + // Dive into objects (except DOM nodes) + if ( + isObject(toClean[key], true) && + !toClean.nodeType && // #10044 + reference[key] + ) { + ob = BaseSeries.cleanRecursively(toClean[key], reference[key]); + if (Object.keys(ob).length) { + clean[key] = ob; + } + // Arrays, primitives and DOM nodes are copied directly + } else if ( + isObject(toClean[key]) || + toClean[key] !== reference[key] + ) { + clean[key] = toClean[key]; + } + }); + return clean; + }; + // eslint-disable-next-line valid-jsdoc /** - * Area series type. - * + * Internal function to initialize an individual series. * @private - * @class - * @name Highcharts.seriesTypes.area - * - * @augments Highcharts.Series */ - BaseSeries.seriesType('area', 'line', + BaseSeries.getSeries = function (chart, options) { + if (options === void 0) { + options = {}; + } + var optionsChart = chart.options.chart, + type = + options.type || + optionsChart.type || + optionsChart.defaultSeriesType || + "", + Series = BaseSeries.seriesTypes[type]; + // No such series type + if (!Series) { + error(17, true, chart, { missingModuleFor: type }); + } + return new Series(chart, options); + }; /** - * The area series type. - * - * @sample {highcharts} highcharts/demo/area-basic/ - * Area chart - * @sample {highstock} stock/demo/area/ - * Area chart + * Factory to create new series prototypes. * - * @extends plotOptions.line - * @excluding useOhlcData - * @product highcharts highstock - * @optionparent plotOptions.area - */ - { - /** - * @see [fillColor](#plotOptions.area.fillColor) - * @see [fillOpacity](#plotOptions.area.fillOpacity) - * - * @apioption plotOptions.area.color - */ - /** - * Fill color or gradient for the area. When `null`, the series' `color` - * is used with the series' `fillOpacity`. - * - * In styled mode, the fill color can be set with the `.highcharts-area` - * class name. - * - * @see [color](#plotOptions.area.color) - * @see [fillOpacity](#plotOptions.area.fillOpacity) - * - * @sample {highcharts} highcharts/plotoptions/area-fillcolor-default/ - * Null by default - * @sample {highcharts} highcharts/plotoptions/area-fillcolor-gradient/ - * Gradient - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @product highcharts highstock - * @apioption plotOptions.area.fillColor - */ - /** - * Fill opacity for the area. When you set an explicit `fillColor`, - * the `fillOpacity` is not applied. Instead, you should define the - * opacity in the `fillColor` with an rgba color definition. The - * `fillOpacity` setting, also the default setting, overrides the alpha - * component of the `color` setting. - * - * In styled mode, the fill opacity can be set with the - * `.highcharts-area` class name. - * - * @see [color](#plotOptions.area.color) - * @see [fillColor](#plotOptions.area.fillColor) - * - * @sample {highcharts} highcharts/plotoptions/area-fillopacity/ - * Automatic fill color and fill opacity of 0.1 - * - * @type {number} - * @default {highcharts} 0.75 - * @default {highstock} 0.75 - * @product highcharts highstock - * @apioption plotOptions.area.fillOpacity - */ - /** - * A separate color for the graph line. By default the line takes the - * `color` of the series, but the lineColor setting allows setting a - * separate color for the line without altering the `fillColor`. - * - * In styled mode, the line stroke can be set with the - * `.highcharts-graph` class name. - * - * @sample {highcharts} highcharts/plotoptions/area-linecolor/ - * Dark gray line - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @product highcharts highstock - * @apioption plotOptions.area.lineColor - */ - /** - * A separate color for the negative part of the area. - * - * In styled mode, a negative color is set with the - * `.highcharts-negative` class name. - * - * @see [negativeColor](#plotOptions.area.negativeColor) - * - * @sample {highcharts} highcharts/css/series-negative-color/ - * Negative color in styled mode - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 3.0 - * @product highcharts - * @apioption plotOptions.area.negativeFillColor - */ - /** - * Whether the whole area or just the line should respond to mouseover - * tooltips and other mouse or touch events. - * - * @sample {highcharts|highstock} highcharts/plotoptions/area-trackbyarea/ - * Display the tooltip when the area is hovered - * - * @type {boolean} - * @default false - * @since 1.1.6 - * @product highcharts highstock - * @apioption plotOptions.area.trackByArea - */ - /** - * The Y axis value to serve as the base for the area, for - * distinguishing between values above and below a threshold. The area - * between the graph and the threshold is filled. - * - * * If a number is given, the Y axis will scale to the threshold. - * * If `null`, the scaling behaves like a line series with fill between - * the graph and the Y axis minimum. - * * If `Infinity` or `-Infinity`, the area between the graph and the - * corresponding Y axis extreme is filled (since v6.1.0). - * - * @sample {highcharts} highcharts/plotoptions/area-threshold/ - * A threshold of 100 - * @sample {highcharts} highcharts/plotoptions/area-threshold-infinity/ - * A threshold of Infinity - * - * @type {number|null} - * @since 2.0 - * @product highcharts highstock - */ - threshold: 0 - }, - /* eslint-disable valid-jsdoc */ - /** - * @lends seriesTypes.area.prototype - */ - { - singleStacks: false, - /** - * Return an array of stacked points, where null and missing points are - * replaced by dummy points in order for gaps to be drawn correctly in - * stacks. - * @private - */ - getStackPoints: function (points) { - var series = this, - segment = [], - keys = [], - xAxis = this.xAxis, - yAxis = this.yAxis, - stack = yAxis.stacking.stacks[this.stackKey], - pointMap = {}, - seriesIndex = series.index, - yAxisSeries = yAxis.series, - seriesLength = yAxisSeries.length, - visibleSeries, - upOrDown = pick(yAxis.options.reversedStacks, - true) ? 1 : -1, - i; - points = points || this.points; - if (this.options.stacking) { - for (i = 0; i < points.length; i++) { - // Reset after point update (#7326) - points[i].leftNull = points[i].rightNull = void 0; - // Create a map where we can quickly look up the points by - // their X values. - pointMap[points[i].x] = points[i]; - } - // Sort the keys (#1651) - objectEach(stack, function (stackX, x) { - // nulled after switching between - // grouping and not (#1651, #2336) - if (stackX.total !== null) { - keys.push(x); - } - }); - keys.sort(function (a, b) { - return a - b; - }); - visibleSeries = yAxisSeries.map(function (s) { - return s.visible; - }); - keys.forEach(function (x, idx) { - var y = 0, - stackPoint, - stackedValues; - if (pointMap[x] && !pointMap[x].isNull) { - segment.push(pointMap[x]); - // Find left and right cliff. -1 goes left, 1 goes - // right. - [-1, 1].forEach(function (direction) { - var nullName = direction === 1 ? - 'rightNull' : - 'leftNull', - cliffName = direction === 1 ? - 'rightCliff' : - 'leftCliff', - cliff = 0, - otherStack = stack[keys[idx + direction]]; - // If there is a stack next to this one, - // to the left or to the right... - if (otherStack) { - i = seriesIndex; - // Can go either up or down, - // depending on reversedStacks - while (i >= 0 && i < seriesLength) { - stackPoint = otherStack.points[i]; - if (!stackPoint) { - // If the next point in this series - // is missing, mark the point - // with point.leftNull or - // point.rightNull = true. - if (i === seriesIndex) { - pointMap[x][nullName] = - true; - // If there are missing points in - // the next stack in any of the - // series below this one, we need - // to substract the missing values - // and add a hiatus to the left or - // right. - } - else if (visibleSeries[i]) { - stackedValues = - stack[x].points[i]; - if (stackedValues) { - cliff -= - stackedValues[1] - - stackedValues[0]; - } - } - } - // When reversedStacks is true, loop up, - // else loop down - i += upOrDown; - } - } - pointMap[x][cliffName] = cliff; - }); - // There is no point for this X value in this series, so we - // insert a dummy point in order for the areas to be drawn - // correctly. - } - else { - // Loop down the stack to find the series below this - // one that has a value (#1991) - i = seriesIndex; - while (i >= 0 && i < seriesLength) { - stackPoint = stack[x].points[i]; - if (stackPoint) { - y = stackPoint[1]; - break; - } - // When reversedStacks is true, loop up, else loop - // down - i += upOrDown; - } - y = yAxis.translate(// #6272 - y, 0, 1, 0, 1); - segment.push({ - isNull: true, - plotX: xAxis.translate(// #6272 - x, 0, 0, 0, 1), - x: x, - plotY: y, - yBottom: y - }); - } - }); - } - return segment; - }, - /** - * @private - */ - getGraphPath: function (points) { - var getGraphPath = Series.prototype.getGraphPath, graphPath, options = this.options, stacking = options.stacking, yAxis = this.yAxis, topPath, bottomPath, bottomPoints = [], graphPoints = [], seriesIndex = this.index, i, areaPath, plotX, stacks = yAxis.stacking.stacks[this.stackKey], threshold = options.threshold, translatedThreshold = Math.round(// #10909 - yAxis.getThreshold(options.threshold)), isNull, yBottom, connectNulls = pick(// #10574 - options.connectNulls, stacking === 'percent'), - // To display null points in underlying stacked series, this - // series graph must be broken, and the area also fall down to - // fill the gap left by the null point. #2069 - addDummyPoints = function (i, otherI, side) { - var point = points[i], stackedValues = stacking && - stacks[point.x].points[seriesIndex], nullVal = point[side + 'Null'] || 0, cliffVal = point[side + 'Cliff'] || 0, top, bottom, isNull = true; - if (cliffVal || nullVal) { - top = (nullVal ? - stackedValues[0] : - stackedValues[1]) + cliffVal; - bottom = stackedValues[0] + cliffVal; - isNull = !!nullVal; - } - else if (!stacking && - points[otherI] && - points[otherI].isNull) { - top = bottom = threshold; - } - // Add to the top and bottom line of the area - if (typeof top !== 'undefined') { - graphPoints.push({ - plotX: plotX, - plotY: top === null ? - translatedThreshold : - yAxis.getThreshold(top), - isNull: isNull, - isCliff: true - }); - bottomPoints.push({ - plotX: plotX, - plotY: bottom === null ? - translatedThreshold : - yAxis.getThreshold(bottom), - doCurve: false // #1041, gaps in areaspline areas - }); - } - }; - // Find what points to use - points = points || this.points; - // Fill in missing points - if (stacking) { - points = this.getStackPoints(points); - } - for (i = 0; i < points.length; i++) { - // Reset after series.update of stacking property (#12033) - if (!stacking) { - points[i].leftCliff = points[i].rightCliff = - points[i].leftNull = points[i].rightNull = void 0; - } - isNull = points[i].isNull; - plotX = pick(points[i].rectPlotX, points[i].plotX); - yBottom = stacking ? points[i].yBottom : translatedThreshold; - if (!isNull || connectNulls) { - if (!connectNulls) { - addDummyPoints(i, i - 1, 'left'); - } - // Skip null point when stacking is false and connectNulls - // true - if (!(isNull && !stacking && connectNulls)) { - graphPoints.push(points[i]); - bottomPoints.push({ - x: i, - plotX: plotX, - plotY: yBottom - }); - } - if (!connectNulls) { - addDummyPoints(i, i + 1, 'right'); - } - } - } - topPath = getGraphPath.call(this, graphPoints, true, true); - bottomPoints.reversed = true; - bottomPath = getGraphPath.call(this, bottomPoints, true, true); - var firstBottomPoint = bottomPath[0]; - if (firstBottomPoint && firstBottomPoint[0] === 'M') { - bottomPath[0] = ['L', firstBottomPoint[1], firstBottomPoint[2]]; - } - areaPath = topPath.concat(bottomPath); - // TODO: don't set leftCliff and rightCliff when connectNulls? - graphPath = getGraphPath - .call(this, graphPoints, false, connectNulls); - areaPath.xMap = topPath.xMap; - this.areaPath = areaPath; - return graphPath; - }, - /** - * Draw the graph and the underlying area. This method calls the Series - * base function and adds the area. The areaPath is calculated in the - * getSegmentPath method called from Series.prototype.drawGraph. - * @private - */ - drawGraph: function () { - // Define or reset areaPath - this.areaPath = []; - // Call the base method - Series.prototype.drawGraph.apply(this); - // Define local variables - var series = this, - areaPath = this.areaPath, - options = this.options, - zones = this.zones, - props = [[ - 'area', - 'highcharts-area', - this.color, - options.fillColor - ]]; // area name, main color, fill color - zones.forEach(function (zone, - i) { - props.push([ - 'zone-area-' + i, - 'highcharts-area highcharts-zone-area-' + i + ' ' + - zone.className, - zone.color || series.color, - zone.fillColor || options.fillColor - ]); - }); - props.forEach(function (prop) { - var areaKey = prop[0], - area = series[areaKey], - verb = area ? 'animate' : 'attr', - attribs = {}; - // Create or update the area - if (area) { // update - area.endX = series.preventGraphAnimation ? - null : - areaPath.xMap; - area.animate({ d: areaPath }); - } - else { // create - attribs.zIndex = 0; // #1069 - area = series[areaKey] = series.chart.renderer - .path(areaPath) - .addClass(prop[1]) - .add(series.group); - area.isArea = true; - } - if (!series.chart.styledMode) { - attribs.fill = pick(prop[3], color(prop[2]) - .setOpacity(pick(options.fillOpacity, 0.75)) - .get()); - } - area[verb](attribs); - area.startX = areaPath.xMap; - area.shiftUnit = options.step ? 2 : 1; - }); - }, - drawLegendSymbol: LegendSymbolMixin.drawRectangle - }); - /* eslint-enable valid-jsdoc */ - /** - * A `area` series. If the [type](#series.area.type) option is not - * specified, it is inherited from [chart.type](#chart.type). + * @function Highcharts.seriesType * - * @extends series,plotOptions.area - * @excluding dataParser, dataURL, useOhlcData - * @product highcharts highstock - * @apioption series.area - */ - /** - * @see [fillColor](#series.area.fillColor) - * @see [fillOpacity](#series.area.fillOpacity) - * - * @apioption series.area.color - */ - /** - * An array of data points for the series. For the `area` series type, - * points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. The `x` values will be automatically - * calculated, either starting at 0 and incremented by 1, or from - * `pointStart` * and `pointInterval` given in the series options. If the - * axis has categories, these will be used. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 2. An array of arrays with 2 values. In this case, the values correspond to - * `x,y`. If the first value is a string, it is applied as the name of the - * point, and the `x` value is inferred. - * ```js - * data: [ - * [0, 9], - * [1, 7], - * [2, 6] - * ] - * ``` - * - * 3. 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.area.turboThreshold), this option is not - * available. - * ```js - * data: [{ - * x: 1, - * y: 9, - * name: "Point2", - * color: "#00FF00" - * }, { - * x: 1, - * y: 6, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects - * - * @type {Array<number|Array<(number|string),(number|null)>|null|*>} - * @extends series.line.data - * @product highcharts highstock - * @apioption series.area.data - */ - /** - * @see [color](#series.area.color) - * @see [fillOpacity](#series.area.fillOpacity) + * @param {string} type + * The series type name. + * + * @param {string} parent + * The parent series type name. Use `line` to inherit from the basic + * {@link Series} object. + * + * @param {Highcharts.SeriesOptionsType|Highcharts.Dictionary<*>} options + * The additional default options that are merged with the parent's options. + * + * @param {Highcharts.Dictionary<*>} [props] + * The properties (functions and primitives) to set on the new prototype. + * + * @param {Highcharts.Dictionary<*>} [pointProps] + * Members for a series-specific extension of the {@link Point} prototype if + * needed. + * + * @return {Highcharts.Series} + * The newly created prototype as extended from {@link Series} or its + * derivatives. + */ + // docs: add to API + extending Highcharts + BaseSeries.seriesType = function ( + type, + parent, + options, + seriesProto, + pointProto + ) { + var defaultOptions = getOptions().plotOptions || {}, + seriesTypes = BaseSeries.seriesTypes; + parent = parent || ""; + // Merge the options + defaultOptions[type] = merge(defaultOptions[parent], options); + // Create the class + BaseSeries.addSeries( + type, + extendClass(seriesTypes[parent] || function () {}, seriesProto) + ); + seriesTypes[type].prototype.type = type; + // Create the point class if needed + if (pointProto) { + seriesTypes[type].prototype.pointClass = extendClass( + Point, + pointProto + ); + } + return seriesTypes[type]; + }; + BaseSeries.prototype.update = function (newOptions, redraw) { + if (redraw === void 0) { + redraw = true; + } + var series = this; + newOptions = BaseSeries.cleanRecursively( + newOptions, + this.userOptions + ); + var newType = newOptions.type; + if (typeof newType !== "undefined" && newType !== series.type) { + series = BaseSeries.getSeries(series.chart, newOptions); + } + fireEvent(series, "update", { newOptions: newOptions }); + series.userOptions = merge(newOptions); + fireEvent(series, "afterUpdate", { newOptions: newOptions }); + if (redraw) { + series.chart.redraw(); + } + return series; + }; + /* * * - * @apioption series.area.fillColor - */ - /** - * @see [color](#series.area.color) - * @see [fillColor](#series.area.fillColor) + * Static Properties * - * @default {highcharts} 0.75 - * @default {highstock} 0.75 - * @apioption series.area.fillOpacity - */ - ''; // adds doclets above to transpilat + * */ + BaseSeries.defaultOptions = { + type: "base", + }; + BaseSeries.seriesTypes = {}; + return BaseSeries; + })(); + BaseSeries.prototype.pointClass = Point; + // backwards compatibility + H.seriesType = BaseSeries.seriesType; + H.seriesTypes = BaseSeries.seriesTypes; + /* * + * + * Export + * + * */ - }); - _registerModule(_modules, 'Series/SplineSeries.js', [_modules['Core/Series/Series.js'], _modules['Core/Utilities.js']], function (BaseSeries, U) { + return BaseSeries; + } + ); + _registerModule( + _modules, + "Core/Chart/Chart.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Axis/Axis.js"], + _modules["Core/Series/Series.js"], + _modules["Core/Globals.js"], + _modules["Core/Legend.js"], + _modules["Core/MSPointer.js"], + _modules["Core/Options.js"], + _modules["Core/Pointer.js"], + _modules["Core/Time.js"], + _modules["Core/Utilities.js"], + ], + function (A, Axis, BaseSeries, H, Legend, MSPointer, O, Pointer, Time, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animate = A.animate, + animObject = A.animObject, + setAnimation = A.setAnimation; + var charts = H.charts, + doc = H.doc, + win = H.win; + var defaultOptions = O.defaultOptions; + var addEvent = U.addEvent, + attr = U.attr, + createElement = U.createElement, + css = U.css, + defined = U.defined, + discardElement = U.discardElement, + erase = U.erase, + error = U.error, + extend = U.extend, + find = U.find, + fireEvent = U.fireEvent, + getStyle = U.getStyle, + isArray = U.isArray, + isFunction = U.isFunction, + isNumber = U.isNumber, + isObject = U.isObject, + isString = U.isString, + merge = U.merge, + numberFormat = U.numberFormat, + objectEach = U.objectEach, + pick = U.pick, + pInt = U.pInt, + relativeLength = U.relativeLength, + removeEvent = U.removeEvent, + splat = U.splat, + syncTimeout = U.syncTimeout, + uniqueKey = U.uniqueKey; + /** + * Callback for chart constructors. + * + * @callback Highcharts.ChartCallbackFunction + * + * @param {Highcharts.Chart} chart + * Created chart. + */ + /** + * Format a number and return a string based on input settings. + * + * @callback Highcharts.NumberFormatterCallbackFunction + * + * @param {number} number + * The input number to format. + * + * @param {number} decimals + * The amount of decimals. A value of -1 preserves the amount in the + * input number. + * + * @param {string} [decimalPoint] + * The decimal point, defaults to the one given in the lang options, or + * a dot. + * + * @param {string} [thousandsSep] + * The thousands separator, defaults to the one given in the lang + * options, or a space character. + * + * @return {string} The formatted number. + */ + /** + * The chart title. The title has an `update` method that allows modifying the + * options directly or indirectly via `chart.update`. + * + * @interface Highcharts.TitleObject + * @extends Highcharts.SVGElement + */ /** + * Modify options for the title. + * + * @function Highcharts.TitleObject#update + * + * @param {Highcharts.TitleOptions} titleOptions + * Options to modify. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the title is altered. If doing more + * operations on the chart, it is a good idea to set redraw to false and + * call {@link Chart#redraw} after. + */ + /** + * The chart subtitle. The subtitle has an `update` method that + * allows modifying the options directly or indirectly via + * `chart.update`. + * + * @interface Highcharts.SubtitleObject + * @extends Highcharts.SVGElement + */ /** + * Modify options for the subtitle. + * + * @function Highcharts.SubtitleObject#update + * + * @param {Highcharts.SubtitleOptions} subtitleOptions + * Options to modify. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the subtitle is altered. If doing + * more operations on the chart, it is a good idea to set redraw to false + * and call {@link Chart#redraw} after. + */ + /** + * The chart caption. The caption has an `update` method that + * allows modifying the options directly or indirectly via + * `chart.update`. + * + * @interface Highcharts.CaptionObject + * @extends Highcharts.SVGElement + */ /** + * Modify options for the caption. + * + * @function Highcharts.CaptionObject#update + * + * @param {Highcharts.CaptionOptions} captionOptions + * Options to modify. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the caption is altered. If doing + * more operations on the chart, it is a good idea to set redraw to false + * and call {@link Chart#redraw} after. + */ + var marginNames = H.marginNames; + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The Chart class. The recommended constructor is {@link Highcharts#chart}. + * + * @example + * var chart = Highcharts.chart('container', { + * title: { + * text: 'My chart' + * }, + * series: [{ + * data: [1, 3, 2, 4] + * }] + * }) + * + * @class + * @name Highcharts.Chart + * + * @param {string|Highcharts.HTMLDOMElement} [renderTo] + * The DOM element to render to, or its id. + * + * @param {Highcharts.Options} options + * The chart options structure. + * + * @param {Highcharts.ChartCallbackFunction} [callback] + * Function to run when the chart has loaded and and all external images + * are loaded. Defining a + * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load) + * handler is equivalent. + */ + var Chart = /** @class */ (function () { + function Chart(a, b, c) { + this.axes = void 0; + this.axisOffset = void 0; + this.bounds = void 0; + this.chartHeight = void 0; + this.chartWidth = void 0; + this.clipBox = void 0; + this.colorCounter = void 0; + this.container = void 0; + this.index = void 0; + this.isResizing = void 0; + this.labelCollectors = void 0; + this.legend = void 0; + this.margin = void 0; + this.numberFormatter = void 0; + this.options = void 0; + this.plotBox = void 0; + this.plotHeight = void 0; + this.plotLeft = void 0; + this.plotTop = void 0; + this.plotWidth = void 0; + this.pointCount = void 0; + this.pointer = void 0; + this.renderer = void 0; + this.renderTo = void 0; + this.series = void 0; + this.spacing = void 0; + this.spacingBox = void 0; + this.symbolCounter = void 0; + this.time = void 0; + this.titleOffset = void 0; + this.userOptions = void 0; + this.xAxis = void 0; + this.yAxis = void 0; + this.getArgs(a, b, c); + } /* * * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * Functions * * */ - var pick = U.pick; /** - * Spline series type. + * Handle the arguments passed to the constructor. * * @private - * @class - * @name Highcharts.seriesTypes.spline - * - * @augments Highcarts.Series - */ - BaseSeries.seriesType('spline', 'line', + * @function Highcharts.Chart#getArgs + * + * @param {...Array<*>} arguments + * All arguments for the constructor. + * + * @fires Highcharts.Chart#event:init + * @fires Highcharts.Chart#event:afterInit + */ + Chart.prototype.getArgs = function (a, b, c) { + // Remove the optional first argument, renderTo, and + // set it on this. + if (isString(a) || a.nodeName) { + this.renderTo = a; + this.init(b, c); + } else { + this.init(a, b); + } + }; /** - * A spline series is a special type of line series, where the segments - * between the data points are smoothed. + * Overridable function that initializes the chart. The constructor's + * arguments are passed on directly. * - * @sample {highcharts} highcharts/demo/spline-irregular-time/ - * Spline chart - * @sample {highstock} stock/demo/spline/ - * Spline chart + * @function Highcharts.Chart#init * - * @extends plotOptions.series - * @excluding step, boostThreshold, boostBlending - * @product highcharts highstock - * @optionparent plotOptions.spline - */ - {}, - /** - * @lends seriesTypes.spline.prototype - */ - { - /* eslint-disable valid-jsdoc */ + * @param {Highcharts.Options} userOptions + * Custom options. + * + * @param {Function} [callback] + * Function to run when the chart has loaded and and all external + * images are loaded. + * + * @return {void} + * + * @fires Highcharts.Chart#event:init + * @fires Highcharts.Chart#event:afterInit + */ + Chart.prototype.init = function (userOptions, callback) { + // Handle regular options + var options, + // skip merging data points to increase performance + seriesOptions = userOptions.series, + userPlotOptions = userOptions.plotOptions || {}; + // Fire the event with a default function + fireEvent(this, "init", { args: arguments }, function () { + userOptions.series = null; + options = merge(defaultOptions, userOptions); // do the merge + var optionsChart = options.chart || {}; + // Override (by copy of user options) or clear tooltip options + // in chart.options.plotOptions (#6218) + objectEach(options.plotOptions, function (typeOptions, type) { + if (isObject(typeOptions)) { + // #8766 + typeOptions.tooltip = + (userPlotOptions[type] && // override by copy: + merge(userPlotOptions[type].tooltip)) || + void 0; // or clear + } + }); + // User options have higher priority than default options + // (#6218). In case of exporting: path is changed + options.tooltip.userOptions = + (userOptions.chart && + userOptions.chart.forExport && + userOptions.tooltip.userOptions) || + userOptions.tooltip; + // set back the series data + options.series = userOptions.series = seriesOptions; /** - * Get the spline segment from a given point's previous neighbour to the - * given point. + * The original options given to the constructor or a chart factory + * like {@link Highcharts.chart} and {@link Highcharts.stockChart}. * - * @private - * @function Highcharts.seriesTypes.spline#getPointSpline + * @name Highcharts.Chart#userOptions + * @type {Highcharts.Options} + */ + this.userOptions = userOptions; + var chartEvents = optionsChart.events; + this.margin = []; + this.spacing = []; + // Pixel data bounds for touch zoom + this.bounds = { h: {}, v: {} }; + // An array of functions that returns labels that should be + // considered for anti-collision + this.labelCollectors = []; + this.callback = callback; + this.isResizing = 0; + /** + * The options structure for the chart after merging + * {@link #defaultOptions} and {@link #userOptions}. It contains + * members for the sub elements like series, legend, tooltip etc. + * + * @name Highcharts.Chart#options + * @type {Highcharts.Options} + */ + this.options = options; + /** + * All the axes in the chart. * - * @param {Array<Highcharts.Point>} + * @see Highcharts.Chart.xAxis + * @see Highcharts.Chart.yAxis * - * @param {Highcharts.Point} point + * @name Highcharts.Chart#axes + * @type {Array<Highcharts.Axis>} + */ + this.axes = []; + /** + * All the current series in the chart. * - * @param {number} i + * @name Highcharts.Chart#series + * @type {Array<Highcharts.Series>} + */ + this.series = []; + /** + * The `Time` object associated with the chart. Since v6.0.5, + * time settings can be applied individually for each chart. If + * no individual settings apply, the `Time` object is shared by + * all instances. * - * @return {Highcharts.SVGPathArray} + * @name Highcharts.Chart#time + * @type {Highcharts.Time} */ - getPointSpline: function (points, point, i) { - var - // 1 means control points midway between points, 2 means 1/3 - // from the point, 3 is 1/4 etc - smoothing = 1.5, - denom = smoothing + 1, - plotX = point.plotX || 0, - plotY = point.plotY || 0, - lastPoint = points[i - 1], - nextPoint = points[i + 1], - leftContX, - leftContY, - rightContX, - rightContY, - ret; - /** - * @private - */ - function doCurve(otherPoint) { - return otherPoint && - !otherPoint.isNull && - otherPoint.doCurve !== false && - // #6387, area splines next to null: - !point.isCliff; - } - // Find control points - if (doCurve(lastPoint) && doCurve(nextPoint)) { - var lastX = lastPoint.plotX || 0, - lastY = lastPoint.plotY || 0, - nextX = nextPoint.plotX || 0, - nextY = nextPoint.plotY || 0, - correction = 0; - leftContX = (smoothing * plotX + lastX) / denom; - leftContY = (smoothing * plotY + lastY) / denom; - rightContX = (smoothing * plotX + nextX) / denom; - rightContY = (smoothing * plotY + nextY) / denom; - // Have the two control points make a straight line through main - // point - if (rightContX !== leftContX) { // #5016, division by zero - correction = (((rightContY - leftContY) * - (rightContX - plotX)) / - (rightContX - leftContX) + plotY - rightContY); - } - leftContY += correction; - rightContY += correction; - // to prevent false extremes, check that control points are - // between neighbouring points' y values - if (leftContY > lastY && leftContY > plotY) { - leftContY = Math.max(lastY, plotY); - // mirror of left control point - rightContY = 2 * plotY - leftContY; - } - else if (leftContY < lastY && leftContY < plotY) { - leftContY = Math.min(lastY, plotY); - rightContY = 2 * plotY - leftContY; - } - if (rightContY > nextY && rightContY > plotY) { - rightContY = Math.max(nextY, plotY); - leftContY = 2 * plotY - rightContY; - } - else if (rightContY < nextY && rightContY < plotY) { - rightContY = Math.min(nextY, plotY); - leftContY = 2 * plotY - rightContY; - } - // record for drawing in next point - point.rightContX = rightContX; - point.rightContY = rightContY; - } - // Visualize control points for debugging - /* - if (leftContX) { - this.chart.renderer.circle( - leftContX + this.chart.plotLeft, - leftContY + this.chart.plotTop, - 2 - ) - .attr({ - stroke: 'red', - 'stroke-width': 2, - fill: 'none', - zIndex: 9 - }) - .add(); - this.chart.renderer.path(['M', leftContX + this.chart.plotLeft, - leftContY + this.chart.plotTop, - 'L', plotX + this.chart.plotLeft, plotY + this.chart.plotTop]) - .attr({ - stroke: 'red', - 'stroke-width': 2, - zIndex: 9 - }) - .add(); - } - if (rightContX) { - this.chart.renderer.circle( - rightContX + this.chart.plotLeft, - rightContY + this.chart.plotTop, - 2 - ) - .attr({ - stroke: 'green', - 'stroke-width': 2, - fill: 'none', - zIndex: 9 - }) - .add(); - this.chart.renderer.path(['M', rightContX + this.chart.plotLeft, - rightContY + this.chart.plotTop, - 'L', plotX + this.chart.plotLeft, plotY + this.chart.plotTop]) - .attr({ - stroke: 'green', - 'stroke-width': 2, - zIndex: 9 - }) - .add(); + this.time = + userOptions.time && Object.keys(userOptions.time).length + ? new Time(userOptions.time) + : H.time; + /** + * Callback function to override the default function that formats + * all the numbers in the chart. Returns a string with the formatted + * number. + * + * @name Highcharts.Chart#numberFormatter + * @type {Highcharts.NumberFormatterCallbackFunction} + */ + this.numberFormatter = optionsChart.numberFormatter || numberFormat; + /** + * Whether the chart is in styled mode, meaning all presentatinoal + * attributes are avoided. + * + * @name Highcharts.Chart#styledMode + * @type {boolean} + */ + this.styledMode = optionsChart.styledMode; + this.hasCartesianSeries = optionsChart.showAxes; + var chart = this; + /** + * Index position of the chart in the {@link Highcharts#charts} + * property. + * + * @name Highcharts.Chart#index + * @type {number} + * @readonly + */ + chart.index = charts.length; // Add the chart to the global lookup + charts.push(chart); + H.chartCount++; + // Chart event handlers + if (chartEvents) { + objectEach(chartEvents, function (event, eventType) { + if (isFunction(event)) { + addEvent(chart, eventType, event); + } + }); } - // */ - ret = [ - 'C', - pick(lastPoint.rightContX, lastPoint.plotX, 0), - pick(lastPoint.rightContY, lastPoint.plotY, 0), - pick(leftContX, plotX, 0), - pick(leftContY, plotY, 0), - plotX, - plotY - ]; - // reset for updating series later - lastPoint.rightContX = lastPoint.rightContY = void 0; - return ret; + /** + * A collection of the X axes in the chart. + * + * @name Highcharts.Chart#xAxis + * @type {Array<Highcharts.Axis>} + */ + chart.xAxis = []; + /** + * A collection of the Y axes in the chart. + * + * @name Highcharts.Chart#yAxis + * @type {Array<Highcharts.Axis>} + * + * @todo + * Make events official: Fire the event `afterInit`. + */ + chart.yAxis = []; + chart.pointCount = chart.colorCounter = chart.symbolCounter = 0; + // Fire after init but before first render, before axes and series + // have been initialized. + fireEvent(chart, "afterInit"); + chart.firstRender(); + }); + }; + /** + * Internal function to unitialize an individual series. + * + * @private + * @function Highcharts.Chart#initSeries + */ + Chart.prototype.initSeries = function (options) { + var chart = this, + optionsChart = chart.options.chart, + type = + options.type || + optionsChart.type || + optionsChart.defaultSeriesType, + series, + Constr = BaseSeries.seriesTypes[type]; + // No such series type + if (!Constr) { + error(17, true, chart, { missingModuleFor: type }); + } + series = new Constr(chart, options); + if (typeof series.init === "function") { + series.init(this, options); + } + return series; + }; + /** + * Internal function to set data for all series with enabled sorting. + * + * @private + * @function Highcharts.Chart#setSeriesData + */ + Chart.prototype.setSeriesData = function () { + this.getSeriesOrderByLinks().forEach(function (series) { + // We need to set data for series with sorting after series init + if (!series.points && !series.data && series.enabledDataSorting) { + series.setData(series.options.data, false); } - /* eslint-enable valid-jsdoc */ - }); + }); + }; /** - * A `spline` series. If the [type](#series.spline.type) option is - * not specified, it is inherited from [chart.type](#chart.type). + * Sort and return chart series in order depending on the number of linked + * series. * - * @extends series,plotOptions.spline - * @excluding dataParser, dataURL, step, boostThreshold, boostBlending - * @product highcharts highstock - * @apioption series.spline - */ - /** - * An array of data points for the series. For the `spline` series type, - * points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. The `x` values will be automatically - * calculated, either starting at 0 and incremented by 1, or from - * `pointStart` and `pointInterval` given in the series options. If the axis - * has categories, these will be used. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 2. An array of arrays with 2 values. In this case, the values correspond to - * `x,y`. If the first value is a string, it is applied as the name of the - * point, and the `x` value is inferred. - * ```js - * data: [ - * [0, 9], - * [1, 2], - * [2, 8] - * ] - * ``` - * - * 3. 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.spline.turboThreshold), - * this option is not available. - * ```js - * data: [{ - * x: 1, - * y: 9, - * name: "Point2", - * color: "#00FF00" - * }, { - * x: 1, - * y: 0, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects - * - * @type {Array<number|Array<(number|string),(number|null)>|null|*>} - * @extends series.line.data - * @product highcharts highstock - * @apioption series.spline.data + * @private + * @function Highcharts.Series#getSeriesOrderByLinks + * @return {Array<Highcharts.Series>} */ - ''; // adds doclets above intro transpilat - - }); - _registerModule(_modules, 'Series/AreaSplineSeries.js', [_modules['Core/Series/Series.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Options.js']], function (BaseSeries, LegendSymbolMixin, O) { - /* * + Chart.prototype.getSeriesOrderByLinks = function () { + return this.series.concat().sort(function (a, b) { + if (a.linkedSeries.length || b.linkedSeries.length) { + return b.linkedSeries.length - a.linkedSeries.length; + } + return 0; + }); + }; + /** + * Order all series above a given index. When series are added and ordered + * by configuration, only the last series is handled (#248, #1123, #2456, + * #6112). This function is called on series initialization and destroy. * - * (c) 2010-2020 Torstein Honsi + * @private + * @function Highcharts.Series#orderSeries + * @param {number} [fromIndex] + * If this is given, only the series above this index are handled. + */ + Chart.prototype.orderSeries = function (fromIndex) { + var series = this.series, + i = fromIndex || 0; + for (; i < series.length; i++) { + if (series[i]) { + /** + * Contains the series' index in the `Chart.series` array. + * + * @name Highcharts.Series#index + * @type {number} + * @readonly + */ + series[i].index = i; + series[i].name = series[i].getName(); + } + } + }; + /** + * Check whether a given point is within the plot area. * - * License: www.highcharts.com/license + * @function Highcharts.Chart#isInsidePlot * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @param {number} plotX + * Pixel x relative to the plot area. * - * */ - var defaultOptions = O.defaultOptions; - var areaProto = BaseSeries.seriesTypes.area.prototype; - /** - * AreaSpline series type. + * @param {number} plotY + * Pixel y relative to the plot area. * - * @private - * @class - * @name Highcharts.seriesTypes.areaspline + * @param {boolean} [inverted] + * Whether the chart is inverted. * - * @augments Highcharts.Series - */ - BaseSeries.seriesType('areaspline', 'spline', + * @return {boolean} + * Returns true if the given point is inside the plot area. + */ + Chart.prototype.isInsidePlot = function (plotX, plotY, inverted) { + var x = inverted ? plotY : plotX, + y = inverted ? plotX : plotY, + e = { + x: x, + y: y, + isInsidePlot: + x >= 0 && x <= this.plotWidth && y >= 0 && y <= this.plotHeight, + }; + fireEvent(this, "afterIsInsidePlot", e); + return e.isInsidePlot; + }; /** - * The area spline series is an area series where the graph between the - * points is smoothed into a spline. + * Redraw the chart after changes have been done to the data, axis extremes + * chart size or chart elements. All methods for updating axes, series or + * points have a parameter for redrawing the chart. This is `true` by + * default. But in many cases you want to do more than one operation on the + * chart before redrawing, for example add a number of points. In those + * cases it is a waste of resources to redraw the chart for each new point + * added. So you add the points and call `chart.redraw()` after. + * + * @function Highcharts.Chart#redraw + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * If or how to apply animation to the redraw. + * + * @fires Highcharts.Chart#event:afterSetExtremes + * @fires Highcharts.Chart#event:beforeRedraw + * @fires Highcharts.Chart#event:predraw + * @fires Highcharts.Chart#event:redraw + * @fires Highcharts.Chart#event:render + * @fires Highcharts.Chart#event:updatedData + */ + Chart.prototype.redraw = function (animation) { + fireEvent(this, "beforeRedraw"); + var chart = this, + axes = chart.axes, + series = chart.series, + pointer = chart.pointer, + legend = chart.legend, + legendUserOptions = chart.userOptions.legend, + redrawLegend = chart.isDirtyLegend, + hasStackedSeries, + hasDirtyStacks, + hasCartesianSeries = chart.hasCartesianSeries, + isDirtyBox = chart.isDirtyBox, + i, + serie, + renderer = chart.renderer, + isHiddenChart = renderer.isHidden(), + afterRedraw = []; + // Handle responsive rules, not only on resize (#6130) + if (chart.setResponsive) { + chart.setResponsive(false); + } + // Set the global animation. When chart.hasRendered is not true, the + // redraw call comes from a responsive rule and animation should not + // occur. + setAnimation(chart.hasRendered ? animation : false, chart); + if (isHiddenChart) { + chart.temporaryDisplay(); + } + // Adjust title layout (reflow multiline text) + chart.layOutTitles(); + // link stacked series + i = series.length; + while (i--) { + serie = series[i]; + if (serie.options.stacking) { + hasStackedSeries = true; + if (serie.isDirty) { + hasDirtyStacks = true; + break; + } + } + } + if (hasDirtyStacks) { + // mark others as dirty + i = series.length; + while (i--) { + serie = series[i]; + if (serie.options.stacking) { + serie.isDirty = true; + } + } + } + // Handle updated data in the series + series.forEach(function (serie) { + if (serie.isDirty) { + if (serie.options.legendType === "point") { + if (typeof serie.updateTotals === "function") { + serie.updateTotals(); + } + redrawLegend = true; + } else if ( + legendUserOptions && + (legendUserOptions.labelFormatter || + legendUserOptions.labelFormat) + ) { + redrawLegend = true; // #2165 + } + } + if (serie.isDirtyData) { + fireEvent(serie, "updatedData"); + } + }); + // handle added or removed series + if (redrawLegend && legend && legend.options.enabled) { + // draw legend graphics + legend.render(); + chart.isDirtyLegend = false; + } + // reset stacks + if (hasStackedSeries) { + chart.getStacks(); + } + if (hasCartesianSeries) { + // set axes scales + axes.forEach(function (axis) { + // Don't do setScale again if we're only resizing. Regression + // #13507. But we need it after chart.update (responsive), as + // axis is initialized again (#12137). + if (!chart.isResizing || !isNumber(axis.min)) { + axis.updateNames(); + axis.setScale(); + } + }); + } + chart.getMargins(); // #3098 + if (hasCartesianSeries) { + // If one axis is dirty, all axes must be redrawn (#792, #2169) + axes.forEach(function (axis) { + if (axis.isDirty) { + isDirtyBox = true; + } + }); + // redraw axes + axes.forEach(function (axis) { + // Fire 'afterSetExtremes' only if extremes are set + var key = axis.min + "," + axis.max; + if (axis.extKey !== key) { + // #821, #4452 + axis.extKey = key; + // prevent a recursive call to chart.redraw() (#1119) + afterRedraw.push(function () { + fireEvent( + axis, + "afterSetExtremes", + extend(axis.eventArgs, axis.getExtremes()) + ); // #747, #751 + delete axis.eventArgs; + }); + } + if (isDirtyBox || hasStackedSeries) { + axis.redraw(); + } + }); + } + // the plot areas size has changed + if (isDirtyBox) { + chart.drawChartBox(); + } + // Fire an event before redrawing series, used by the boost module to + // clear previous series renderings. + fireEvent(chart, "predraw"); + // redraw affected series + series.forEach(function (serie) { + if ((isDirtyBox || serie.isDirty) && serie.visible) { + serie.redraw(); + } + // Set it here, otherwise we will have unlimited 'updatedData' calls + // for a hidden series after setData(). Fixes #6012 + serie.isDirtyData = false; + }); + // move tooltip or reset + if (pointer) { + pointer.reset(true); + } + // redraw if canvas + renderer.draw(); + // Fire the events + fireEvent(chart, "redraw"); + fireEvent(chart, "render"); + if (isHiddenChart) { + chart.temporaryDisplay(true); + } + // Fire callbacks that are put on hold until after the redraw + afterRedraw.forEach(function (callback) { + callback.call(); + }); + }; + /** + * Get an axis, series or point object by `id` as given in the configuration + * options. Returns `undefined` if no item is found. * - * @sample {highcharts} highcharts/demo/areaspline/ - * Area spline chart - * @sample {highstock} stock/demo/areaspline/ - * Area spline chart + * @sample highcharts/plotoptions/series-id/ + * Get series by id * - * @extends plotOptions.area - * @excluding step, boostThreshold, boostBlending - * @product highcharts highstock - * @apioption plotOptions.areaspline - */ - /** - * @see [fillColor](#plotOptions.areaspline.fillColor) - * @see [fillOpacity](#plotOptions.areaspline.fillOpacity) + * @function Highcharts.Chart#get * - * @apioption plotOptions.areaspline.color - */ - /** - * @see [color](#plotOptions.areaspline.color) - * @see [fillOpacity](#plotOptions.areaspline.fillOpacity) + * @param {string} id + * The id as given in the configuration options. * - * @apioption plotOptions.areaspline.fillColor + * @return {Highcharts.Axis|Highcharts.Series|Highcharts.Point|undefined} + * The retrieved item. */ + Chart.prototype.get = function (id) { + var ret, + series = this.series, + i; + /** + * @private + * @param {Highcharts.Axis|Highcharts.Series} item + * @return {boolean} + */ + function itemById(item) { + return item.id === id || (item.options && item.options.id === id); + } + ret = + // Search axes + find(this.axes, itemById) || + // Search series + find(this.series, itemById); + // Search points + for (i = 0; !ret && i < series.length; i++) { + ret = find(series[i].points || [], itemById); + } + return ret; + }; /** - * @see [color](#plotOptions.areaspline.color) - * @see [fillColor](#plotOptions.areaspline.fillColor) + * Create the Axis instances based on the config options. * - * @default {highcharts} 0.75 - * @default {highstock} 0.75 - * @apioption plotOptions.areaspline.fillOpacity - */ - defaultOptions.plotOptions.area, { - getStackPoints: areaProto.getStackPoints, - getGraphPath: areaProto.getGraphPath, - drawGraph: areaProto.drawGraph, - drawLegendSymbol: LegendSymbolMixin.drawRectangle - }); + * @private + * @function Highcharts.Chart#getAxes + * @fires Highcharts.Chart#event:afterGetAxes + * @fires Highcharts.Chart#event:getAxes + */ + Chart.prototype.getAxes = function () { + var chart = this, + options = this.options, + xAxisOptions = (options.xAxis = splat(options.xAxis || {})), + yAxisOptions = (options.yAxis = splat(options.yAxis || {})), + optionsArray; + fireEvent(this, "getAxes"); + // make sure the options are arrays and add some members + xAxisOptions.forEach(function (axis, i) { + axis.index = i; + axis.isX = true; + }); + yAxisOptions.forEach(function (axis, i) { + axis.index = i; + }); + // concatenate all axis options into one array + optionsArray = xAxisOptions.concat(yAxisOptions); + optionsArray.forEach(function (axisOptions) { + new Axis(chart, axisOptions); // eslint-disable-line no-new + }); + fireEvent(this, "afterGetAxes"); + }; /** - * A `areaspline` series. If the [type](#series.areaspline.type) option - * is not specified, it is inherited from [chart.type](#chart.type). + * Returns an array of all currently selected points in the chart. Points + * can be selected by clicking or programmatically by the + * {@link Highcharts.Point#select} + * function. * + * @sample highcharts/plotoptions/series-allowpointselect-line/ + * Get selected points * - * @extends series,plotOptions.areaspline - * @excluding dataParser, dataURL, step, boostThreshold, boostBlending - * @product highcharts highstock - * @apioption series.areaspline - */ - /** - * @see [fillColor](#series.areaspline.fillColor) - * @see [fillOpacity](#series.areaspline.fillOpacity) - * - * @apioption series.areaspline.color - */ - /** - * An array of data points for the series. For the `areaspline` series - * type, points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. The `x` values will be automatically - * calculated, either starting at 0 and incremented by 1, or from - * `pointStart` and `pointInterval` given in the series options. If the axis - * has categories, these will be used. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 2. An array of arrays with 2 values. In this case, the values correspond to - * `x,y`. If the first value is a string, it is applied as the name of the - * point, and the `x` value is inferred. - * ```js - * data: [ - * [0, 10], - * [1, 9], - * [2, 3] - * ] - * ``` - * - * 3. 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.areaspline.turboThreshold), this option is not - * available. - * ```js - * data: [{ - * x: 1, - * y: 4, - * name: "Point2", - * color: "#00FF00" - * }, { - * x: 1, - * y: 4, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects - * - * @type {Array<number|Array<(number|string),(number|null)>|null|*>} - * @extends series.line.data - * @product highcharts highstock - * @apioption series.areaspline.data - */ - /** - * @see [color](#series.areaspline.color) - * @see [fillOpacity](#series.areaspline.fillOpacity) + * @function Highcharts.Chart#getSelectedPoints * - * @apioption series.areaspline.fillColor + * @return {Array<Highcharts.Point>} + * The currently selected points. */ + Chart.prototype.getSelectedPoints = function () { + var points = []; + this.series.forEach(function (serie) { + // For one-to-one points inspect series.data in order to retrieve + // points outside the visible range (#6445). For grouped data, + // inspect the generated series.points. + points = points.concat( + serie.getPointsCollection().filter(function (point) { + return pick(point.selectedStaging, point.selected); + }) + ); + }); + return points; + }; /** - * @see [color](#series.areaspline.color) - * @see [fillColor](#series.areaspline.fillColor) + * Returns an array of all currently selected series in the chart. Series + * can be selected either programmatically by the + * {@link Highcharts.Series#select} + * function or by checking the checkbox next to the legend item if + * [series.showCheckBox](https://api.highcharts.com/highcharts/plotOptions.series.showCheckbox) + * is true. * - * @default {highcharts} 0.75 - * @default {highstock} 0.75 - * @apioption series.areaspline.fillOpacity - */ - ''; // adds doclets above into transpilat - - }); - _registerModule(_modules, 'Series/ColumnSeries.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Series/Series.js'], _modules['Core/Color/Color.js'], _modules['Core/Globals.js'], _modules['Mixins/LegendSymbol.js'], _modules['Series/LineSeries.js'], _modules['Core/Utilities.js']], function (A, BaseSeries, Color, H, LegendSymbolMixin, LineSeries, U) { - /* * + * @sample highcharts/members/chart-getselectedseries/ + * Get selected series * - * (c) 2010-2020 Torstein Honsi + * @function Highcharts.Chart#getSelectedSeries * - * License: www.highcharts.com/license + * @return {Array<Highcharts.Series>} + * The currently selected series. + */ + Chart.prototype.getSelectedSeries = function () { + return this.series.filter(function (serie) { + return serie.selected; + }); + }; + /** + * Set a new title or subtitle for the chart. * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @sample highcharts/members/chart-settitle/ + * Set title text and styles * - * */ - var animObject = A.animObject; - var color = Color.parse; - var noop = H.noop; - var clamp = U.clamp, - defined = U.defined, - extend = U.extend, - isArray = U.isArray, - isNumber = U.isNumber, - merge = U.merge, - pick = U.pick, - objectEach = U.objectEach; - /** - * Adjusted width and x offset of the columns for grouping. + * @function Highcharts.Chart#setTitle * - * @private - * @interface Highcharts.ColumnMetricsObject - */ /** - * Width of the columns. - * @name Highcharts.ColumnMetricsObject#width - * @type {number} - */ /** - * Offset of the columns. - * @name Highcharts.ColumnMetricsObject#offset - * @type {number} - */ - ''; // detach doclets above - /** - * The column series type. + * @param {Highcharts.TitleOptions} [titleOptions] + * New title options. The title text itself is set by the + * `titleOptions.text` property. * - * @private - * @class - * @name Highcharts.seriesTypes.column + * @param {Highcharts.SubtitleOptions} [subtitleOptions] + * New subtitle options. The subtitle text itself is set by the + * `subtitleOptions.text` property. * - * @augments Highcharts.Series + * @param {boolean} [redraw] + * Whether to redraw the chart or wait for a later call to + * `chart.redraw()`. */ - var ColumnSeries = BaseSeries.seriesType('column', 'line', - /** - * Column series display one column per value along an X axis. - * - * @sample {highcharts} highcharts/demo/column-basic/ - * Column chart - * @sample {highstock} stock/demo/column/ - * Column chart - * - * @extends plotOptions.line - * @excluding connectEnds, connectNulls, gapSize, gapUnit, linecap, - * lineWidth, marker, step, useOhlcData - * @product highcharts highstock - * @optionparent plotOptions.column - */ - { - /** - * The corner radius of the border surrounding each column or bar. - * - * @sample {highcharts} highcharts/plotoptions/column-borderradius/ - * Rounded columns - * - * @product highcharts highstock gantt - * - * @private - */ - borderRadius: 0, - /** - * When using automatic point colors pulled from the global - * [colors](colors) or series-specific - * [plotOptions.column.colors](series.colors) collections, this option - * determines whether the chart should receive one color per series or - * one color per point. - * - * In styled mode, the `colors` or `series.colors` arrays are not - * supported, and instead this option gives the points individual color - * class names on the form `highcharts-color-{n}`. - * - * @see [series colors](#plotOptions.column.colors) - * - * @sample {highcharts} highcharts/plotoptions/column-colorbypoint-false/ - * False by default - * @sample {highcharts} highcharts/plotoptions/column-colorbypoint-true/ - * True - * - * @type {boolean} - * @default false - * @since 2.0 - * @product highcharts highstock gantt - * @apioption plotOptions.column.colorByPoint - */ - /** - * A series specific or series type specific color set to apply instead - * of the global [colors](#colors) when [colorByPoint]( - * #plotOptions.column.colorByPoint) is true. - * - * @type {Array<Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject>} - * @since 3.0 - * @product highcharts highstock gantt - * @apioption plotOptions.column.colors - */ - /** - * When `true`, the columns will center in the category, ignoring null - * or missing points. When `false`, space will be reserved for null or - * missing points. - * - * @sample {highcharts} highcharts/series-column/centerincategory/ - * Center in category - * - * @since 8.0.1 - * @product highcharts highstock gantt - * - * @private - */ - centerInCategory: false, - /** - * Padding between each value groups, in x axis units. - * - * @sample {highcharts} highcharts/plotoptions/column-grouppadding-default/ - * 0.2 by default - * @sample {highcharts} highcharts/plotoptions/column-grouppadding-none/ - * No group padding - all columns are evenly spaced - * - * @product highcharts highstock gantt - * - * @private - */ - groupPadding: 0.2, - /** - * Whether to group non-stacked columns or to let them render - * independent of each other. Non-grouped columns will be laid out - * individually and overlap each other. - * - * @sample {highcharts} highcharts/plotoptions/column-grouping-false/ - * Grouping disabled - * @sample {highstock} highcharts/plotoptions/column-grouping-false/ - * Grouping disabled - * - * @type {boolean} - * @default true - * @since 2.3.0 - * @product highcharts highstock gantt - * @apioption plotOptions.column.grouping - */ - /** - * @ignore-option - * @private - */ - marker: null, - /** - * The maximum allowed pixel width for a column, translated to the - * height of a bar in a bar chart. This prevents the columns from - * becoming too wide when there is a small number of points in the - * chart. - * - * @see [pointWidth](#plotOptions.column.pointWidth) - * - * @sample {highcharts} highcharts/plotoptions/column-maxpointwidth-20/ - * Limited to 50 - * @sample {highstock} highcharts/plotoptions/column-maxpointwidth-20/ - * Limited to 50 - * - * @type {number} - * @since 4.1.8 - * @product highcharts highstock gantt - * @apioption plotOptions.column.maxPointWidth - */ - /** - * Padding between each column or bar, in x axis units. - * - * @sample {highcharts} highcharts/plotoptions/column-pointpadding-default/ - * 0.1 by default - * @sample {highcharts} highcharts/plotoptions/column-pointpadding-025/ - * 0.25 - * @sample {highcharts} highcharts/plotoptions/column-pointpadding-none/ - * 0 for tightly packed columns - * - * @product highcharts highstock gantt - * - * @private - */ - pointPadding: 0.1, - /** - * A pixel value specifying a fixed width for each column or bar point. - * When `null`, the width is calculated from the `pointPadding` and - * `groupPadding`. The width effects the dimension that is not based on - * the point value. For column series it is the hoizontal length and for - * bar series it is the vertical length. - * - * @see [maxPointWidth](#plotOptions.column.maxPointWidth) - * - * @sample {highcharts} highcharts/plotoptions/column-pointwidth-20/ - * 20px wide columns regardless of chart width or the amount of - * data points - * - * @type {number} - * @since 1.2.5 - * @product highcharts highstock gantt - * @apioption plotOptions.column.pointWidth - */ - /** - * A pixel value specifying a fixed width for the column or bar. - * Overrides pointWidth on the series. - * - * @see [series.pointWidth](#plotOptions.column.pointWidth) - * - * @type {number} - * @default undefined - * @since 7.0.0 - * @product highcharts highstock gantt - * @apioption series.column.data.pointWidth - */ - /** - * The minimal height for a column or width for a bar. By default, - * 0 values are not shown. To visualize a 0 (or close to zero) point, - * set the minimal point length to a pixel value like 3\. In stacked - * column charts, minPointLength might not be respected for tightly - * packed values. - * - * @sample {highcharts} highcharts/plotoptions/column-minpointlength/ - * Zero base value - * @sample {highcharts} highcharts/plotoptions/column-minpointlength-pos-and-neg/ - * Positive and negative close to zero values - * - * @product highcharts highstock gantt - * - * @private - */ - minPointLength: 0, - /** - * When the series contains less points than the crop threshold, all - * points are drawn, event if the points fall outside the visible plot - * area at the current zoom. The advantage of drawing all points - * (including markers and columns), is that animation is performed on - * updates. On the other hand, when the series contains more points than - * the crop threshold, the series data is cropped to only contain points - * that fall within the plot area. The advantage of cropping away - * invisible points is to increase performance on large series. - * - * @product highcharts highstock gantt - * - * @private - */ - cropThreshold: 50, - /** - * The X axis range that each point is valid for. This determines the - * width of the column. On a categorized axis, the range will be 1 - * by default (one category unit). On linear and datetime axes, the - * range will be computed as the distance between the two closest data - * points. - * - * The default `null` means it is computed automatically, but this - * option can be used to override the automatic value. - * - * This option is set by default to 1 if data sorting is enabled. - * - * @sample {highcharts} highcharts/plotoptions/column-pointrange/ - * Set the point range to one day on a data set with one week - * between the points - * - * @type {number|null} - * @since 2.3 - * @product highcharts highstock gantt - * - * @private - */ - pointRange: null, - states: { - /** - * Options for the hovered point. These settings override the normal - * state options when a point is moused over or touched. - * - * @extends plotOptions.series.states.hover - * @excluding halo, lineWidth, lineWidthPlus, marker - * @product highcharts highstock gantt - */ - hover: { - /** @ignore-option */ - halo: false, - /** - * A specific border color for the hovered point. Defaults to - * inherit the normal state border color. - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @product highcharts gantt - * @apioption plotOptions.column.states.hover.borderColor - */ - /** - * A specific color for the hovered point. - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @product highcharts gantt - * @apioption plotOptions.column.states.hover.color - */ - /** - * How much to brighten the point on interaction. Requires the - * main color to be defined in hex or rgb(a) format. - * - * In styled mode, the hover brightening is by default replaced - * with a fill-opacity set in the `.highcharts-point:hover` - * rule. - * - * @sample {highcharts} highcharts/plotoptions/column-states-hover-brightness/ - * Brighten by 0.5 - * - * @product highcharts highstock gantt - */ - brightness: 0.1 - }, - /** - * Options for the selected point. These settings override the - * normal state options when a point is selected. - * - * @extends plotOptions.series.states.select - * @excluding halo, lineWidth, lineWidthPlus, marker - * @product highcharts highstock gantt - */ - select: { - /** - * A specific color for the selected point. - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @default #cccccc - * @product highcharts highstock gantt - */ - color: '#cccccc', - /** - * A specific border color for the selected point. - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @default #000000 - * @product highcharts highstock gantt - */ - borderColor: '#000000' - } - }, - dataLabels: { - align: void 0, - verticalAlign: void 0, - /** - * The y position offset of the label relative to the point in - * pixels. - * - * @type {number} - */ - y: void 0 - }, - // false doesn't work well: https://jsfiddle.net/highcharts/hz8fopan/14/ - /** - * @ignore-option - * @private - */ - startFromThreshold: true, - stickyTracking: false, - tooltip: { - distance: 6 - }, - /** - * The Y axis value to serve as the base for the columns, for - * distinguishing between values above and below a threshold. If `null`, - * the columns extend from the padding Y axis minimum. - * - * @type {number|null} - * @since 2.0 - * @product highcharts - * - * @private - */ - threshold: 0, - /** - * The width of the border surrounding each column or bar. Defaults to - * `1` when there is room for a border, but to `0` when the columns are - * so dense that a border would cover the next column. - * - * In styled mode, the stroke width can be set with the - * `.highcharts-point` rule. - * - * @sample {highcharts} highcharts/plotoptions/column-borderwidth/ - * 2px black border - * - * @type {number} - * @default undefined - * @product highcharts highstock gantt - * @apioption plotOptions.column.borderWidth - */ - /** - * The color of the border surrounding each column or bar. - * - * In styled mode, the border stroke can be set with the - * `.highcharts-point` rule. - * - * @sample {highcharts} highcharts/plotoptions/column-bordercolor/ - * Dark gray border - * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @default #ffffff - * @product highcharts highstock gantt - * - * @private - */ - borderColor: '#ffffff' - }, - /** - * @lends seriesTypes.column.prototype - */ - { - cropShoulder: 0, - // When tooltip is not shared, this series (and derivatives) requires - // direct touch/hover. KD-tree does not apply. - directTouch: true, - trackerGroups: ['group', 'dataLabelsGroup'], - // use separate negative stacks, unlike area stacks where a negative - // point is substracted from previous (#1910) - negStacks: true, - /* eslint-disable valid-jsdoc */ - /** - * Initialize the series. Extends the basic Series.init method by - * marking other series of the same type as dirty. - * - * @private - * @function Highcharts.seriesTypes.column#init - * @return {void} - */ - init: function () { - LineSeries.prototype.init.apply(this, arguments); - var series = this, - chart = series.chart; - // if the series is added dynamically, force redraw of other - // series affected by a new column - if (chart.hasRendered) { - chart.series.forEach(function (otherSeries) { - if (otherSeries.type === series.type) { - otherSeries.isDirty = true; - } - }); - } - }, - /** - * Return the width and x offset of the columns adjusted for grouping, - * groupPadding, pointPadding, pointWidth etc. - * - * @private - * @function Highcharts.seriesTypes.column#getColumnMetrics - * @return {Highcharts.ColumnMetricsObject} - */ - getColumnMetrics: function () { - var series = this, - options = series.options, - xAxis = series.xAxis, - yAxis = series.yAxis, - reversedStacks = xAxis.options.reversedStacks, - // Keep backward compatibility: reversed xAxis had reversed - // stacks - reverseStacks = (xAxis.reversed && !reversedStacks) || - (!xAxis.reversed && reversedStacks), - stackKey, - stackGroups = {}, - columnCount = 0; - // Get the total number of column type series. This is called on - // every series. Consider moving this logic to a chart.orderStacks() - // function and call it on init, addSeries and removeSeries - if (options.grouping === false) { - columnCount = 1; - } - else { - series.chart.series.forEach(function (otherSeries) { - var otherYAxis = otherSeries.yAxis, - otherOptions = otherSeries.options, - columnIndex; - if (otherSeries.type === series.type && - (otherSeries.visible || - !series.chart.options.chart - .ignoreHiddenSeries) && - yAxis.len === otherYAxis.len && - yAxis.pos === otherYAxis.pos) { // #642, #2086 - if (otherOptions.stacking && otherOptions.stacking !== 'group') { - stackKey = otherSeries.stackKey; - if (typeof stackGroups[stackKey] === - 'undefined') { - stackGroups[stackKey] = columnCount++; - } - columnIndex = stackGroups[stackKey]; - } - else if (otherOptions.grouping !== false) { // #1162 - columnIndex = columnCount++; - } - otherSeries.columnIndex = columnIndex; - } - }); - } - var categoryWidth = Math.min(Math.abs(xAxis.transA) * ((xAxis.ordinal && xAxis.ordinal.slope) || - options.pointRange || - xAxis.closestPointRange || - xAxis.tickInterval || - 1), // #2610 - xAxis.len // #1535 - ), - groupPadding = categoryWidth * options.groupPadding, - groupWidth = categoryWidth - 2 * groupPadding, - pointOffsetWidth = groupWidth / (columnCount || 1), - pointWidth = Math.min(options.maxPointWidth || xAxis.len, - pick(options.pointWidth, - pointOffsetWidth * (1 - 2 * options.pointPadding))), - pointPadding = (pointOffsetWidth - pointWidth) / 2, - // #1251, #3737 - colIndex = (series.columnIndex || 0) + (reverseStacks ? 1 : 0), - pointXOffset = pointPadding + - (groupPadding + - colIndex * pointOffsetWidth - - (categoryWidth / 2)) * (reverseStacks ? -1 : 1); - // Save it for reading in linked series (Error bars particularly) - series.columnMetrics = { - width: pointWidth, - offset: pointXOffset, - paddedWidth: pointOffsetWidth, - columnCount: columnCount - }; - return series.columnMetrics; - }, - /** - * Make the columns crisp. The edges are rounded to the nearest full - * pixel. - * - * @private - * @function Highcharts.seriesTypes.column#crispCol - * @param {number} x - * @param {number} y - * @param {number} w - * @param {number} h - * @return {Highcharts.BBoxObject} - */ - crispCol: function (x, y, w, h) { - var chart = this.chart, - borderWidth = this.borderWidth, - xCrisp = -(borderWidth % 2 ? 0.5 : 0), - yCrisp = borderWidth % 2 ? 0.5 : 1, - right, - bottom, - fromTop; - if (chart.inverted && chart.renderer.isVML) { - yCrisp += 1; - } - // Horizontal. We need to first compute the exact right edge, then - // round it and compute the width from there. - if (this.options.crisp) { - right = Math.round(x + w) + xCrisp; - x = Math.round(x) + xCrisp; - w = right - x; - } - // Vertical - bottom = Math.round(y + h) + yCrisp; - fromTop = Math.abs(y) <= 0.5 && bottom > 0.5; // #4504, #4656 - y = Math.round(y) + yCrisp; - h = bottom - y; - // Top edges are exceptions - if (fromTop && h) { // #5146 - y -= 1; - h += 1; - } - return { - x: x, - y: y, - width: w, - height: h + Chart.prototype.setTitle = function ( + titleOptions, + subtitleOptions, + redraw + ) { + this.applyDescription("title", titleOptions); + this.applyDescription("subtitle", subtitleOptions); + // The initial call also adds the caption. On update, chart.update will + // relay to Chart.setCaption. + this.applyDescription("caption", void 0); + this.layOutTitles(redraw); + }; + /** + * Apply a title, subtitle or caption for the chart + * + * @private + * @function Highcharts.Chart#applyDescription + * @param name {string} + * Either title, subtitle or caption + * @param {Highcharts.TitleOptions|Highcharts.SubtitleOptions|Highcharts.CaptionOptions|undefined} explicitOptions + * The options to set, will be merged with default options. + */ + Chart.prototype.applyDescription = function (name, explicitOptions) { + var chart = this; + // Default style + var style = + name === "title" + ? { + color: "#333333", + fontSize: this.options.isStock ? "16px" : "18px", // #2944 + } + : { + color: "#666666", }; - }, - /** - * Adjust for missing columns, according to the `centerInCategory` - * option. Missing columns are either single points or stacks where the - * point or points are either missing or null. - * - * @private - * @function Highcharts.seriesTypes.column#adjustForMissingColumns - * @param {number} x - * The x coordinate of the column, left side - * @param {number} pointWidth - * The pointWidth, already computed upstream - * @param {Highcharts.ColumnPoint} point - * The point instance - * @param {Highcharts.ColumnMetricsObject} metrics - * The series-wide column metrics - * @return {number} - * The adjusted x position, or the original if not adjusted - */ - adjustForMissingColumns: function (x, pointWidth, point, metrics) { - var _this = this; - var stacking = this.options.stacking; - if (!point.isNull && metrics.columnCount > 1) { - var indexInCategory_1 = 0; - var totalInCategory_1 = 0; - // Loop over all the stacks on the Y axis. When stacking is - // enabled, these are real point stacks. When stacking is not - // enabled, but `centerInCategory` is true, there is one stack - // handling the grouping of points in each category. This is - // done in the `setGroupedPoints` function. - objectEach(this.yAxis.stacking && this.yAxis.stacking.stacks, function (stack) { - if (typeof point.x === 'number') { - var stackItem = stack[point.x.toString()]; - if (stackItem) { - var pointValues = stackItem.points[_this.index], - total = stackItem.total; - // If true `stacking` is enabled, count the - // total number of non-null stacks in the - // category, and note which index this point is - // within those stacks. - if (stacking) { - if (pointValues) { - indexInCategory_1 = totalInCategory_1; - } - if (stackItem.hasValidPoints) { - totalInCategory_1++; - } - // If `stacking` is not enabled, look for the - // index and total of the `group` stack. - } - else if (isArray(pointValues)) { - indexInCategory_1 = pointValues[1]; - totalInCategory_1 = total || 0; - } - } - } - }); - // Compute the adjusted x position - var boxWidth = (totalInCategory_1 - 1) * metrics.paddedWidth + - pointWidth; - x = (point.plotX || 0) + boxWidth / 2 - pointWidth - - indexInCategory_1 * metrics.paddedWidth; - } - return x; - }, - /** - * Translate each point to the plot area coordinate system and find - * shape positions - * - * @private - * @function Highcharts.seriesTypes.column#translate - */ - translate: function () { - var series = this, - chart = series.chart, - options = series.options, - dense = series.dense = - series.closestPointRange * series.xAxis.transA < 2, - borderWidth = series.borderWidth = pick(options.borderWidth, - dense ? 0 : 1 // #3635 - ), - xAxis = series.xAxis, - yAxis = series.yAxis, - threshold = options.threshold, - translatedThreshold = series.translatedThreshold = - yAxis.getThreshold(threshold), - minPointLength = pick(options.minPointLength, 5), - metrics = series.getColumnMetrics(), - seriesPointWidth = metrics.width, - // postprocessed for border width - seriesBarW = series.barW = - Math.max(seriesPointWidth, 1 + 2 * borderWidth), - seriesXOffset = series.pointXOffset = metrics.offset, - dataMin = series.dataMin, - dataMax = series.dataMax; - if (chart.inverted) { - translatedThreshold -= 0.5; // #3355 - } - // When the pointPadding is 0, we want the columns to be packed - // tightly, so we allow individual columns to have individual sizes. - // When pointPadding is greater, we strive for equal-width columns - // (#2694). - if (options.pointPadding) { - seriesBarW = Math.ceil(seriesBarW); - } - LineSeries.prototype.translate.apply(series); - // Record the new values - series.points.forEach(function (point) { - var yBottom = pick(point.yBottom, - translatedThreshold), - safeDistance = 999 + Math.abs(yBottom), - pointWidth = seriesPointWidth, - plotX = point.plotX || 0, - // Don't draw too far outside plot area (#1303, #2241, - // #4264) - plotY = clamp(point.plotY, -safeDistance, - yAxis.len + safeDistance), - barX = plotX + seriesXOffset, - barW = seriesBarW, - barY = Math.min(plotY, - yBottom), - up, - barH = Math.max(plotY, - yBottom) - barY; - // Handle options.minPointLength - if (minPointLength && Math.abs(barH) < minPointLength) { - barH = minPointLength; - up = (!yAxis.reversed && !point.negative) || - (yAxis.reversed && point.negative); - // Reverse zeros if there's no positive value in the series - // in visible range (#7046) - if (isNumber(threshold) && - isNumber(dataMax) && - point.y === threshold && - dataMax <= threshold && - // and if there's room for it (#7311) - (yAxis.min || 0) < threshold && - // if all points are the same value (i.e zero) not draw - // as negative points (#10646) - dataMin !== dataMax) { - up = !up; - } - // If stacked... - barY = (Math.abs(barY - translatedThreshold) > minPointLength ? - // ...keep position - yBottom - minPointLength : - // #1485, #4051 - translatedThreshold - - (up ? minPointLength : 0)); - } - // Handle point.options.pointWidth - // @todo Handle grouping/stacking too. Calculate offset properly - if (defined(point.options.pointWidth)) { - pointWidth = barW = - Math.ceil(point.options.pointWidth); - barX -= Math.round((pointWidth - seriesPointWidth) / 2); - } - // Adjust for null or missing points - if (options.centerInCategory) { - barX = series.adjustForMissingColumns(barX, pointWidth, point, metrics); - } - // Cache for access in polar - point.barX = barX; - point.pointWidth = pointWidth; - // Fix the tooltip on center of grouped columns (#1216, #424, - // #3648) - point.tooltipPos = chart.inverted ? - [ - yAxis.len + yAxis.pos - chart.plotLeft - plotY, - xAxis.len + xAxis.pos - chart.plotTop - (plotX || 0) - seriesXOffset - barW / 2, - barH - ] : - [barX + barW / 2, plotY + yAxis.pos - - chart.plotTop, barH]; - // Register shape type and arguments to be used in drawPoints - // Allow shapeType defined on pointClass level - point.shapeType = - series.pointClass.prototype.shapeType || 'rect'; - point.shapeArgs = series.crispCol.apply(series, point.isNull ? - // #3169, drilldown from null must have a position to work - // from #6585, dataLabel should be placed on xAxis, not - // floating in the middle of the chart - [barX, translatedThreshold, barW, 0] : - [barX, barY, barW, barH]); - }); - }, - getSymbol: noop, + // Merge default options with explicit options + var options = (this.options[name] = merge( + // Default styles + !this.styledMode && { style: style }, + this.options[name], + explicitOptions + )); + var elem = this[name]; + if (elem && explicitOptions) { + this[name] = elem = elem.destroy(); // remove old + } + if (options && !elem) { + elem = this.renderer + .text(options.text, 0, 0, options.useHTML) + .attr({ + align: options.align, + class: "highcharts-" + name, + zIndex: options.zIndex || 4, + }) + .add(); + // Update methods, shortcut to Chart.setTitle, Chart.setSubtitle and + // Chart.setCaption + elem.update = function (updateOptions) { + var fn = { + title: "setTitle", + subtitle: "setSubtitle", + caption: "setCaption", + }[name]; + chart[fn](updateOptions); + }; + // Presentational + if (!this.styledMode) { + elem.css(options.style); + } /** - * Use a solid rectangle like the area series types - * - * @private - * @function Highcharts.seriesTypes.column#drawLegendSymbol + * The chart title. The title has an `update` method that allows + * modifying the options directly or indirectly via + * `chart.update`. * - * @param {Highcharts.Legend} legend - * The legend object + * @sample highcharts/members/title-update/ + * Updating titles * - * @param {Highcharts.Series|Highcharts.Point} item - * The series (this) or point + * @name Highcharts.Chart#title + * @type {Highcharts.TitleObject} */ - drawLegendSymbol: LegendSymbolMixin.drawRectangle, /** - * Columns have no graph + * The chart subtitle. The subtitle has an `update` method that + * allows modifying the options directly or indirectly via + * `chart.update`. * - * @private - * @function Highcharts.seriesTypes.column#drawGraph - */ - drawGraph: function () { - this.group[this.dense ? 'addClass' : 'removeClass']('highcharts-dense-data'); - }, - /** - * Get presentational attributes - * - * @private - * @function Highcharts.seriesTypes.column#pointAttribs - * - * @param {Highcharts.ColumnPoint} point - * - * @param {string} state - * - * @return {Highcharts.SVGAttributes} + * @name Highcharts.Chart#subtitle + * @type {Highcharts.SubtitleObject} */ - pointAttribs: function (point, state) { - var options = this.options, stateOptions, ret, p2o = this.pointAttrToOptions || {}, strokeOption = p2o.stroke || 'borderColor', strokeWidthOption = p2o['stroke-width'] || 'borderWidth', fill = (point && point.color) || this.color, - // set to fill when borderColor null: - stroke = ((point && point[strokeOption]) || - options[strokeOption] || - this.color || - fill), strokeWidth = (point && point[strokeWidthOption]) || - options[strokeWidthOption] || - this[strokeWidthOption] || 0, dashstyle = (point && point.options.dashStyle) || options.dashStyle, opacity = pick(point && point.opacity, options.opacity, 1), zone, brightness; - // Handle zone colors - if (point && this.zones.length) { - zone = point.getZone(); - // When zones are present, don't use point.color (#4267). - // Changed order (#6527), added support for colorAxis (#10670) - fill = (point.options.color || - (zone && (zone.color || point.nonZonedColor)) || - this.color); - if (zone) { - stroke = zone.borderColor || stroke; - dashstyle = zone.dashStyle || dashstyle; - strokeWidth = zone.borderWidth || strokeWidth; - } - } - // Select or hover states - if (state && point) { - stateOptions = merge(options.states[state], - // #6401 - point.options.states && - point.options.states[state] || - {}); - brightness = stateOptions.brightness; - fill = - stateOptions.color || (typeof brightness !== 'undefined' && - color(fill) - .brighten(stateOptions.brightness) - .get()) || fill; - stroke = stateOptions[strokeOption] || stroke; - strokeWidth = - stateOptions[strokeWidthOption] || strokeWidth; - dashstyle = stateOptions.dashStyle || dashstyle; - opacity = pick(stateOptions.opacity, opacity); - } - ret = { - fill: fill, - stroke: stroke, - 'stroke-width': strokeWidth, - opacity: opacity + this[name] = elem; + } + }; + /** + * Internal function to lay out the chart title, subtitle and caption, and + * cache the full offset height for use in `getMargins`. The result is + * stored in `this.titleOffset`. + * + * @private + * @function Highcharts.Chart#layOutTitles + * + * @param {boolean} [redraw=true] + * @fires Highcharts.Chart#event:afterLayOutTitles + */ + Chart.prototype.layOutTitles = function (redraw) { + var titleOffset = [0, 0, 0], + requiresDirtyBox, + renderer = this.renderer, + spacingBox = this.spacingBox; + // Lay out the title and the subtitle respectively + ["title", "subtitle", "caption"].forEach(function (key) { + var title = this[key], + titleOptions = this.options[key], + verticalAlign = titleOptions.verticalAlign || "top", + offset = + key === "title" + ? -3 + : // Floating subtitle (#6574) + verticalAlign === "top" + ? titleOffset[0] + 2 + : 0, + titleSize, + height; + if (title) { + if (!this.styledMode) { + titleSize = titleOptions.style.fontSize; + } + titleSize = renderer.fontMetrics(titleSize, title).b; + title.css({ + width: + (titleOptions.width || + spacingBox.width + (titleOptions.widthAdjust || 0)) + "px", + }); + // Skip the cache for HTML (#3481, #11666) + height = Math.round(title.getBBox(titleOptions.useHTML).height); + title.align( + extend( + { + y: + verticalAlign === "bottom" + ? titleSize + : offset + titleSize, + height: height, + }, + titleOptions + ), + false, + "spacingBox" + ); + if (!titleOptions.floating) { + if (verticalAlign === "top") { + titleOffset[0] = Math.ceil(titleOffset[0] + height); + } else if (verticalAlign === "bottom") { + titleOffset[2] = Math.ceil(titleOffset[2] + height); + } + } + } + }, this); + // Handle title.margin and caption.margin + if ( + titleOffset[0] && + (this.options.title.verticalAlign || "top") === "top" + ) { + titleOffset[0] += this.options.title.margin; + } + if ( + titleOffset[2] && + this.options.caption.verticalAlign === "bottom" + ) { + titleOffset[2] += this.options.caption.margin; + } + requiresDirtyBox = + !this.titleOffset || + this.titleOffset.join(",") !== titleOffset.join(","); + // Used in getMargins + this.titleOffset = titleOffset; + fireEvent(this, "afterLayOutTitles"); + if (!this.isDirtyBox && requiresDirtyBox) { + this.isDirtyBox = this.isDirtyLegend = requiresDirtyBox; + // Redraw if necessary (#2719, #2744) + if (this.hasRendered && pick(redraw, true) && this.isDirtyBox) { + this.redraw(); + } + } + }; + /** + * Internal function to get the chart width and height according to options + * and container size. Sets {@link Chart.chartWidth} and + * {@link Chart.chartHeight}. + * + * @private + * @function Highcharts.Chart#getChartSize + */ + Chart.prototype.getChartSize = function () { + var chart = this, + optionsChart = chart.options.chart, + widthOption = optionsChart.width, + heightOption = optionsChart.height, + renderTo = chart.renderTo; + // Get inner width and height + if (!defined(widthOption)) { + chart.containerWidth = getStyle(renderTo, "width"); + } + if (!defined(heightOption)) { + chart.containerHeight = getStyle(renderTo, "height"); + } + /** + * The current pixel width of the chart. + * + * @name Highcharts.Chart#chartWidth + * @type {number} + */ + chart.chartWidth = Math.max( + // #1393 + 0, + widthOption || chart.containerWidth || 600 // #1460 + ); + /** + * The current pixel height of the chart. + * + * @name Highcharts.Chart#chartHeight + * @type {number} + */ + chart.chartHeight = Math.max( + 0, + relativeLength(heightOption, chart.chartWidth) || + (chart.containerHeight > 1 ? chart.containerHeight : 400) + ); + }; + /** + * If the renderTo element has no offsetWidth, most likely one or more of + * its parents are hidden. Loop up the DOM tree to temporarily display the + * parents, then save the original display properties, and when the true + * size is retrieved, reset them. Used on first render and on redraws. + * + * @private + * @function Highcharts.Chart#temporaryDisplay + * + * @param {boolean} [revert] + * Revert to the saved original styles. + */ + Chart.prototype.temporaryDisplay = function (revert) { + var node = this.renderTo, + tempStyle; + if (!revert) { + while (node && node.style) { + // When rendering to a detached node, it needs to be temporarily + // attached in order to read styling and bounding boxes (#5783, + // #7024). + if (!doc.body.contains(node) && !node.parentNode) { + node.hcOrigDetached = true; + doc.body.appendChild(node); + } + if ( + getStyle(node, "display", false) === "none" || + node.hcOricDetached + ) { + node.hcOrigStyle = { + display: node.style.display, + height: node.style.height, + overflow: node.style.overflow, }; - if (dashstyle) { - ret.dashstyle = dashstyle; - } - return ret; - }, - /** - * Draw the columns. For bars, the series.group is rotated, so the same - * coordinates apply for columns and bars. This method is inherited by - * scatter series. - * - * @private - * @function Highcharts.seriesTypes.column#drawPoints - */ - drawPoints: function () { - var series = this, - chart = this.chart, - options = series.options, - renderer = chart.renderer, - animationLimit = options.animationLimit || 250, - shapeArgs; - // draw the columns - series.points.forEach(function (point) { - var plotY = point.plotY, - graphic = point.graphic, - hasGraphic = !!graphic, - verb = graphic && chart.pointCount < animationLimit ? - 'animate' : 'attr'; - if (isNumber(plotY) && point.y !== null) { - shapeArgs = point.shapeArgs; - // When updating a series between 2d and 3d or cartesian and - // polar, the shape type changes. - if (graphic && point.hasNewShapeType()) { - graphic = graphic.destroy(); - } - // Set starting position for point sliding animation. - if (series.enabledDataSorting) { - point.startXPos = series.xAxis.reversed ? - -(shapeArgs ? shapeArgs.width : 0) : - series.xAxis.width; - } - if (!graphic) { - point.graphic = graphic = - renderer[point.shapeType](shapeArgs) - .add(point.group || series.group); - if (graphic && - series.enabledDataSorting && - chart.hasRendered && - chart.pointCount < animationLimit) { - graphic.attr({ - x: point.startXPos - }); - hasGraphic = true; - verb = 'animate'; - } - } - if (graphic && hasGraphic) { // update - graphic[verb](merge(shapeArgs)); - } - // Border radius is not stylable (#6900) - if (options.borderRadius) { - graphic[verb]({ - r: options.borderRadius - }); - } - // Presentational - if (!chart.styledMode) { - graphic[verb](series.pointAttribs(point, (point.selected && 'select'))) - .shadow(point.allowShadow !== false && options.shadow, null, options.stacking && !options.borderRadius); - } - graphic.addClass(point.getClassName(), true); - } - else if (graphic) { - point.graphic = graphic.destroy(); // #1269 - } - }); + tempStyle = { + display: "block", + overflow: "hidden", + }; + if (node !== this.renderTo) { + tempStyle.height = 0; + } + css(node, tempStyle); + // If it still doesn't have an offset width after setting + // display to block, it probably has an !important priority + // #2631, 6803 + if (!node.offsetWidth) { + node.style.setProperty("display", "block", "important"); + } + } + node = node.parentNode; + if (node === doc.body) { + break; + } + } + } else { + while (node && node.style) { + if (node.hcOrigStyle) { + css(node, node.hcOrigStyle); + delete node.hcOrigStyle; + } + if (node.hcOrigDetached) { + doc.body.removeChild(node); + node.hcOrigDetached = false; + } + node = node.parentNode; + } + } + }; + /** + * Set the {@link Chart.container|chart container's} class name, in + * addition to `highcharts-container`. + * + * @function Highcharts.Chart#setClassName + * + * @param {string} [className] + * The additional class name. + */ + Chart.prototype.setClassName = function (className) { + this.container.className = + "highcharts-container " + (className || ""); + }; + /** + * Get the containing element, determine the size and create the inner + * container div to hold the chart. + * + * @private + * @function Highcharts.Chart#afterGetContainer + * @fires Highcharts.Chart#event:afterGetContainer + */ + Chart.prototype.getContainer = function () { + var chart = this, + container, + options = chart.options, + optionsChart = options.chart, + chartWidth, + chartHeight, + renderTo = chart.renderTo, + indexAttrName = "data-highcharts-chart", + oldChartIndex, + Ren, + containerId = uniqueKey(), + containerStyle, + key; + if (!renderTo) { + chart.renderTo = renderTo = optionsChart.renderTo; + } + if (isString(renderTo)) { + chart.renderTo = renderTo = doc.getElementById(renderTo); + } + // Display an error if the renderTo is wrong + if (!renderTo) { + error(13, true, chart); + } + // If the container already holds a chart, destroy it. The check for + // hasRendered is there because web pages that are saved to disk from + // the browser, will preserve the data-highcharts-chart attribute and + // the SVG contents, but not an interactive chart. So in this case, + // charts[oldChartIndex] will point to the wrong chart if any (#2609). + oldChartIndex = pInt(attr(renderTo, indexAttrName)); + if ( + isNumber(oldChartIndex) && + charts[oldChartIndex] && + charts[oldChartIndex].hasRendered + ) { + charts[oldChartIndex].destroy(); + } + // Make a reference to the chart from the div + attr(renderTo, indexAttrName, chart.index); + // remove previous chart + renderTo.innerHTML = ""; + // If the container doesn't have an offsetWidth, it has or is a child of + // a node that has display:none. We need to temporarily move it out to a + // visible state to determine the size, else the legend and tooltips + // won't render properly. The skipClone option is used in sparklines as + // a micro optimization, saving about 1-2 ms each chart. + if (!optionsChart.skipClone && !renderTo.offsetWidth) { + chart.temporaryDisplay(); + } + // get the width and height + chart.getChartSize(); + chartWidth = chart.chartWidth; + chartHeight = chart.chartHeight; + // Allow table cells and flex-boxes to shrink without the chart blocking + // them out (#6427) + css(renderTo, { overflow: "hidden" }); + // Create the inner container + if (!chart.styledMode) { + containerStyle = extend( + { + position: "relative", + // needed for context menu (avoidscrollbars) and content + // overflow in IE + overflow: "hidden", + width: chartWidth + "px", + height: chartHeight + "px", + textAlign: "left", + lineHeight: "normal", + zIndex: 0, + "-webkit-tap-highlight-color": "rgba(0,0,0,0)", + userSelect: "none", // #13503 + }, + optionsChart.style + ); + } + /** + * The containing HTML element of the chart. The container is + * dynamically inserted into the element given as the `renderTo` + * parameter in the {@link Highcharts#chart} constructor. + * + * @name Highcharts.Chart#container + * @type {Highcharts.HTMLDOMElement} + */ + container = createElement( + "div", + { + id: containerId, }, - /** - * Animate the column heights one by one from zero. - * - * @private - * @function Highcharts.seriesTypes.column#animate - * - * @param {boolean} init - * Whether to initialize the animation or run it - */ - animate: function (init) { - var series = this, - yAxis = this.yAxis, - options = series.options, - inverted = this.chart.inverted, - attr = {}, - translateProp = inverted ? 'translateX' : 'translateY', - translateStart, - translatedThreshold; - if (init) { - attr.scaleY = 0.001; - translatedThreshold = clamp(yAxis.toPixels(options.threshold), yAxis.pos, yAxis.pos + yAxis.len); - if (inverted) { - attr.translateX = translatedThreshold - yAxis.len; - } - else { - attr.translateY = translatedThreshold; - } - // apply finnal clipping (used in Highstock) (#7083) - // animation is done by scaleY, so cliping is for panes - if (series.clipBox) { - series.setClip(); - } - series.group.attr(attr); - } - else { // run the animation - translateStart = series.group.attr(translateProp); - series.group.animate({ scaleY: 1 }, extend(animObject(series.options.animation), { - // Do the scale synchronously to ensure smooth - // updating (#5030, #7228) - step: function (val, fx) { - if (series.group) { - attr[translateProp] = translateStart + - fx.pos * (yAxis.pos - translateStart); - series.group.attr(attr); - } - } - })); + containerStyle, + renderTo + ); + chart.container = container; + // cache the cursor (#1650) + chart._cursor = container.style.cursor; + // Initialize the renderer + Ren = H[optionsChart.renderer] || H.Renderer; + /** + * The renderer instance of the chart. Each chart instance has only one + * associated renderer. + * + * @name Highcharts.Chart#renderer + * @type {Highcharts.SVGRenderer} + */ + chart.renderer = new Ren( + container, + chartWidth, + chartHeight, + null, + optionsChart.forExport, + options.exporting && options.exporting.allowHTML, + chart.styledMode + ); + // Set the initial animation from the options + setAnimation(void 0, chart); + chart.setClassName(optionsChart.className); + if (!chart.styledMode) { + chart.renderer.setStyle(optionsChart.style); + } else { + // Initialize definitions + for (key in options.defs) { + // eslint-disable-line guard-for-in + this.renderer.definition(options.defs[key]); + } + } + // Add a reference to the charts index + chart.renderer.chartIndex = chart.index; + fireEvent(this, "afterGetContainer"); + }; + /** + * Calculate margins by rendering axis labels in a preliminary position. + * Title, subtitle and legend have already been rendered at this stage, but + * will be moved into their final positions. + * + * @private + * @function Highcharts.Chart#getMargins + * @fires Highcharts.Chart#event:getMargins + */ + Chart.prototype.getMargins = function (skipAxes) { + var _a = this, + spacing = _a.spacing, + margin = _a.margin, + titleOffset = _a.titleOffset; + this.resetMargins(); + // Adjust for title and subtitle + if (titleOffset[0] && !defined(margin[0])) { + this.plotTop = Math.max(this.plotTop, titleOffset[0] + spacing[0]); + } + if (titleOffset[2] && !defined(margin[2])) { + this.marginBottom = Math.max( + this.marginBottom, + titleOffset[2] + spacing[2] + ); + } + // Adjust for legend + if (this.legend && this.legend.display) { + this.legend.adjustMargins(margin, spacing); + } + fireEvent(this, "getMargins"); + if (!skipAxes) { + this.getAxisMargins(); + } + }; + /** + * @private + * @function Highcharts.Chart#getAxisMargins + */ + Chart.prototype.getAxisMargins = function () { + var chart = this, + // [top, right, bottom, left] + axisOffset = (chart.axisOffset = [0, 0, 0, 0]), + colorAxis = chart.colorAxis, + margin = chart.margin, + getOffset = function (axes) { + axes.forEach(function (axis) { + if (axis.visible) { + axis.getOffset(); + } + }); + }; + // pre-render axes to get labels offset width + if (chart.hasCartesianSeries) { + getOffset(chart.axes); + } else if (colorAxis && colorAxis.length) { + getOffset(colorAxis); + } + // Add the axis offsets + marginNames.forEach(function (m, side) { + if (!defined(margin[side])) { + chart[m] += axisOffset[side]; + } + }); + chart.setChartSize(); + }; + /** + * Reflows the chart to its container. By default, the chart reflows + * automatically to its container following a `window.resize` event, as per + * the [chart.reflow](https://api.highcharts.com/highcharts/chart.reflow) + * option. However, there are no reliable events for div resize, so if the + * container is resized without a window resize event, this must be called + * explicitly. + * + * @sample highcharts/members/chart-reflow/ + * Resize div and reflow + * @sample highcharts/chart/events-container/ + * Pop up and reflow + * + * @function Highcharts.Chart#reflow + * + * @param {global.Event} [e] + * Event arguments. Used primarily when the function is called + * internally as a response to window resize. + */ + Chart.prototype.reflow = function (e) { + var chart = this, + optionsChart = chart.options.chart, + renderTo = chart.renderTo, + hasUserSize = + defined(optionsChart.width) && defined(optionsChart.height), + width = optionsChart.width || getStyle(renderTo, "width"), + height = optionsChart.height || getStyle(renderTo, "height"), + target = e ? e.target : win; + // Width and height checks for display:none. Target is doc in IE8 and + // Opera, win in Firefox, Chrome and IE9. + if ( + !hasUserSize && + !chart.isPrinting && + width && + height && + (target === win || target === doc) + ) { + if ( + width !== chart.containerWidth || + height !== chart.containerHeight + ) { + U.clearTimeout(chart.reflowTimeout); + // When called from window.resize, e is set, else it's called + // directly (#2224) + chart.reflowTimeout = syncTimeout( + function () { + // Set size, it may have been destroyed in the meantime + // (#1257) + if (chart.container) { + chart.setSize(void 0, void 0, false); + } + }, + e ? 100 : 0 + ); + } + chart.containerWidth = width; + chart.containerHeight = height; + } + }; + /** + * Toggle the event handlers necessary for auto resizing, depending on the + * `chart.reflow` option. + * + * @private + * @function Highcharts.Chart#setReflow + */ + Chart.prototype.setReflow = function (reflow) { + var chart = this; + if (reflow !== false && !this.unbindReflow) { + this.unbindReflow = addEvent(win, "resize", function (e) { + // a removed event listener still runs in Edge and IE if the + // listener was removed while the event runs, so check if the + // chart is not destroyed (#11609) + if (chart.options) { + chart.reflow(e); + } + }); + addEvent(this, "destroy", this.unbindReflow); + } else if (reflow === false && this.unbindReflow) { + // Unbind and unset + this.unbindReflow = this.unbindReflow(); + } + // The following will add listeners to re-fit the chart before and after + // printing (#2284). However it only works in WebKit. Should have worked + // in Firefox, but not supported in IE. + /* + if (win.matchMedia) { + win.matchMedia('print').addListener(function reflow() { + chart.reflow(); + }); } - }, + //*/ + }; + /** + * Resize the chart to a given width and height. In order to set the width + * only, the height argument may be skipped. To set the height only, pass + * `undefined` for the width. + * + * @sample highcharts/members/chart-setsize-button/ + * Test resizing from buttons + * @sample highcharts/members/chart-setsize-jquery-resizable/ + * Add a jQuery UI resizable + * @sample stock/members/chart-setsize/ + * Highstock with UI resizable + * + * @function Highcharts.Chart#setSize + * + * @param {number|null} [width] + * The new pixel width of the chart. Since v4.2.6, the argument can + * be `undefined` in order to preserve the current value (when + * setting height only), or `null` to adapt to the width of the + * containing element. + * + * @param {number|null} [height] + * The new pixel height of the chart. Since v4.2.6, the argument can + * be `undefined` in order to preserve the current value, or `null` + * in order to adapt to the height of the containing element. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] + * Whether and how to apply animation. + * + * @return {void} + * + * @fires Highcharts.Chart#event:endResize + * @fires Highcharts.Chart#event:resize + */ + Chart.prototype.setSize = function (width, height, animation) { + var chart = this, + renderer = chart.renderer, + globalAnimation; + // Handle the isResizing counter + chart.isResizing += 1; + // set the animation for the current process + setAnimation(animation, chart); + globalAnimation = renderer.globalAnimation; + chart.oldChartHeight = chart.chartHeight; + chart.oldChartWidth = chart.chartWidth; + if (typeof width !== "undefined") { + chart.options.chart.width = width; + } + if (typeof height !== "undefined") { + chart.options.chart.height = height; + } + chart.getChartSize(); + // Resize the container with the global animation applied if enabled + // (#2503) + if (!chart.styledMode) { + (globalAnimation ? animate : css)( + chart.container, + { + width: chart.chartWidth + "px", + height: chart.chartHeight + "px", + }, + globalAnimation + ); + } + chart.setChartSize(true); + renderer.setSize( + chart.chartWidth, + chart.chartHeight, + globalAnimation + ); + // handle axes + chart.axes.forEach(function (axis) { + axis.isDirty = true; + axis.setScale(); + }); + chart.isDirtyLegend = true; // force legend redraw + chart.isDirtyBox = true; // force redraw of plot and chart border + chart.layOutTitles(); // #2857 + chart.getMargins(); + chart.redraw(globalAnimation); + chart.oldChartHeight = null; + fireEvent(chart, "resize"); + // Fire endResize and set isResizing back. If animation is disabled, + // fire without delay + syncTimeout(function () { + if (chart) { + fireEvent(chart, "endResize", null, function () { + chart.isResizing -= 1; + }); + } + }, animObject(globalAnimation).duration); + }; + /** + * Set the public chart properties. This is done before and after the + * pre-render to determine margin sizes. + * + * @private + * @function Highcharts.Chart#setChartSize + * @fires Highcharts.Chart#event:afterSetChartSize + */ + Chart.prototype.setChartSize = function (skipAxes) { + var chart = this, + inverted = chart.inverted, + renderer = chart.renderer, + chartWidth = chart.chartWidth, + chartHeight = chart.chartHeight, + optionsChart = chart.options.chart, + spacing = chart.spacing, + clipOffset = chart.clipOffset, + clipX, + clipY, + plotLeft, + plotTop, + plotWidth, + plotHeight, + plotBorderWidth; + /** + * The current left position of the plot area in pixels. + * + * @name Highcharts.Chart#plotLeft + * @type {number} + */ + chart.plotLeft = plotLeft = Math.round(chart.plotLeft); + /** + * The current top position of the plot area in pixels. + * + * @name Highcharts.Chart#plotTop + * @type {number} + */ + chart.plotTop = plotTop = Math.round(chart.plotTop); + /** + * The current width of the plot area in pixels. + * + * @name Highcharts.Chart#plotWidth + * @type {number} + */ + chart.plotWidth = plotWidth = Math.max( + 0, + Math.round(chartWidth - plotLeft - chart.marginRight) + ); + /** + * The current height of the plot area in pixels. + * + * @name Highcharts.Chart#plotHeight + * @type {number} + */ + chart.plotHeight = plotHeight = Math.max( + 0, + Math.round(chartHeight - plotTop - chart.marginBottom) + ); + chart.plotSizeX = inverted ? plotHeight : plotWidth; + chart.plotSizeY = inverted ? plotWidth : plotHeight; + chart.plotBorderWidth = optionsChart.plotBorderWidth || 0; + // Set boxes used for alignment + chart.spacingBox = renderer.spacingBox = { + x: spacing[3], + y: spacing[0], + width: chartWidth - spacing[3] - spacing[1], + height: chartHeight - spacing[0] - spacing[2], + }; + chart.plotBox = renderer.plotBox = { + x: plotLeft, + y: plotTop, + width: plotWidth, + height: plotHeight, + }; + plotBorderWidth = 2 * Math.floor(chart.plotBorderWidth / 2); + clipX = Math.ceil(Math.max(plotBorderWidth, clipOffset[3]) / 2); + clipY = Math.ceil(Math.max(plotBorderWidth, clipOffset[0]) / 2); + chart.clipBox = { + x: clipX, + y: clipY, + width: Math.floor( + chart.plotSizeX - + Math.max(plotBorderWidth, clipOffset[1]) / 2 - + clipX + ), + height: Math.max( + 0, + Math.floor( + chart.plotSizeY - + Math.max(plotBorderWidth, clipOffset[2]) / 2 - + clipY + ) + ), + }; + if (!skipAxes) { + chart.axes.forEach(function (axis) { + axis.setAxisSize(); + axis.setAxisTranslation(); + }); + } + fireEvent(chart, "afterSetChartSize", { skipAxes: skipAxes }); + }; + /** + * Initial margins before auto size margins are applied. + * + * @private + * @function Highcharts.Chart#resetMargins + */ + Chart.prototype.resetMargins = function () { + fireEvent(this, "resetMargins"); + var chart = this, + chartOptions = chart.options.chart; + // Create margin and spacing array + ["margin", "spacing"].forEach(function splashArrays(target) { + var value = chartOptions[target], + values = isObject(value) ? value : [value, value, value, value]; + ["Top", "Right", "Bottom", "Left"].forEach(function ( + sideName, + side + ) { + chart[target][side] = pick( + chartOptions[target + sideName], + values[side] + ); + }); + }); + // Set margin names like chart.plotTop, chart.plotLeft, + // chart.marginRight, chart.marginBottom. + marginNames.forEach(function (m, side) { + chart[m] = pick(chart.margin[side], chart.spacing[side]); + }); + chart.axisOffset = [0, 0, 0, 0]; // top, right, bottom, left + chart.clipOffset = [0, 0, 0, 0]; + }; + /** + * Internal function to draw or redraw the borders and backgrounds for chart + * and plot area. + * + * @private + * @function Highcharts.Chart#drawChartBox + * @fires Highcharts.Chart#event:afterDrawChartBox + */ + Chart.prototype.drawChartBox = function () { + var chart = this, + optionsChart = chart.options.chart, + renderer = chart.renderer, + chartWidth = chart.chartWidth, + chartHeight = chart.chartHeight, + chartBackground = chart.chartBackground, + plotBackground = chart.plotBackground, + plotBorder = chart.plotBorder, + chartBorderWidth, + styledMode = chart.styledMode, + plotBGImage = chart.plotBGImage, + chartBackgroundColor = optionsChart.backgroundColor, + plotBackgroundColor = optionsChart.plotBackgroundColor, + plotBackgroundImage = optionsChart.plotBackgroundImage, + mgn, + bgAttr, + plotLeft = chart.plotLeft, + plotTop = chart.plotTop, + plotWidth = chart.plotWidth, + plotHeight = chart.plotHeight, + plotBox = chart.plotBox, + clipRect = chart.clipRect, + clipBox = chart.clipBox, + verb = "animate"; + // Chart area + if (!chartBackground) { + chart.chartBackground = chartBackground = renderer + .rect() + .addClass("highcharts-background") + .add(); + verb = "attr"; + } + if (!styledMode) { + // Presentational + chartBorderWidth = optionsChart.borderWidth || 0; + mgn = chartBorderWidth + (optionsChart.shadow ? 8 : 0); + bgAttr = { + fill: chartBackgroundColor || "none", + }; + if (chartBorderWidth || chartBackground["stroke-width"]) { + // #980 + bgAttr.stroke = optionsChart.borderColor; + bgAttr["stroke-width"] = chartBorderWidth; + } + chartBackground.attr(bgAttr).shadow(optionsChart.shadow); + } else { + chartBorderWidth = mgn = chartBackground.strokeWidth(); + } + chartBackground[verb]({ + x: mgn / 2, + y: mgn / 2, + width: chartWidth - mgn - (chartBorderWidth % 2), + height: chartHeight - mgn - (chartBorderWidth % 2), + r: optionsChart.borderRadius, + }); + // Plot background + verb = "animate"; + if (!plotBackground) { + verb = "attr"; + chart.plotBackground = plotBackground = renderer + .rect() + .addClass("highcharts-plot-background") + .add(); + } + plotBackground[verb](plotBox); + if (!styledMode) { + // Presentational attributes for the background + plotBackground + .attr({ + fill: plotBackgroundColor || "none", + }) + .shadow(optionsChart.plotShadow); + // Create the background image + if (plotBackgroundImage) { + if (!plotBGImage) { + chart.plotBGImage = renderer + .image( + plotBackgroundImage, + plotLeft, + plotTop, + plotWidth, + plotHeight + ) + .add(); + } else { + if (plotBackgroundImage !== plotBGImage.attr("href")) { + plotBGImage.attr("href", plotBackgroundImage); + } + plotBGImage.animate(plotBox); + } + } + } + // Plot clip + if (!clipRect) { + chart.clipRect = renderer.clipRect(clipBox); + } else { + clipRect.animate({ + width: clipBox.width, + height: clipBox.height, + }); + } + // Plot area border + verb = "animate"; + if (!plotBorder) { + verb = "attr"; + chart.plotBorder = plotBorder = renderer + .rect() + .addClass("highcharts-plot-border") + .attr({ + zIndex: 1, // Above the grid + }) + .add(); + } + if (!styledMode) { + // Presentational + plotBorder.attr({ + stroke: optionsChart.plotBorderColor, + "stroke-width": optionsChart.plotBorderWidth || 0, + fill: "none", + }); + } + plotBorder[verb]( + plotBorder.crisp( + { + x: plotLeft, + y: plotTop, + width: plotWidth, + height: plotHeight, + }, + -plotBorder.strokeWidth() + ) + ); // #3282 plotBorder should be negative; + // reset + chart.isDirtyBox = false; + fireEvent(this, "afterDrawChartBox"); + }; + /** + * Detect whether a certain chart property is needed based on inspecting its + * options and series. This mainly applies to the chart.inverted property, + * and in extensions to the chart.angular and chart.polar properties. + * + * @private + * @function Highcharts.Chart#propFromSeries + * @return {void} + */ + Chart.prototype.propFromSeries = function () { + var chart = this, + optionsChart = chart.options.chart, + klass, + seriesOptions = chart.options.series, + i, + value; + /** + * The flag is set to `true` if a series of the chart is inverted. + * + * @name Highcharts.Chart#inverted + * @type {boolean|undefined} + */ + ["inverted", "angular", "polar"].forEach(function (key) { + // The default series type's class + klass = + BaseSeries.seriesTypes[ + optionsChart.type || optionsChart.defaultSeriesType + ]; + // Get the value from available chart-wide properties + value = + // It is set in the options: + optionsChart[key] || + // The default series class: + (klass && klass.prototype[key]); + // requires it + // 4. Check if any the chart's series require it + i = seriesOptions && seriesOptions.length; + while (!value && i--) { + klass = BaseSeries.seriesTypes[seriesOptions[i].type]; + if (klass && klass.prototype[key]) { + value = true; + } + } + // Set the chart property + chart[key] = value; + }); + }; + /** + * Internal function to link two or more series together, based on the + * `linkedTo` option. This is done from `Chart.render`, and after + * `Chart.addSeries` and `Series.remove`. + * + * @private + * @function Highcharts.Chart#linkSeries + * @fires Highcharts.Chart#event:afterLinkSeries + */ + Chart.prototype.linkSeries = function () { + var chart = this, + chartSeries = chart.series; + // Reset links + chartSeries.forEach(function (series) { + series.linkedSeries.length = 0; + }); + // Apply new links + chartSeries.forEach(function (series) { + var linkedTo = series.options.linkedTo; + if (isString(linkedTo)) { + if (linkedTo === ":previous") { + linkedTo = chart.series[series.index - 1]; + } else { + linkedTo = chart.get(linkedTo); + } + // #3341 avoid mutual linking + if (linkedTo && linkedTo.linkedParent !== series) { + linkedTo.linkedSeries.push(series); + series.linkedParent = linkedTo; + if (linkedTo.enabledDataSorting) { + series.setDataSortingOptions(); + } + series.visible = pick( + series.options.visible, + linkedTo.options.visible, + series.visible + ); // #3879 + } + } + }); + fireEvent(this, "afterLinkSeries"); + }; + /** + * Render series for the chart. + * + * @private + * @function Highcharts.Chart#renderSeries + */ + Chart.prototype.renderSeries = function () { + this.series.forEach(function (serie) { + serie.translate(); + serie.render(); + }); + }; + /** + * Render labels for the chart. + * + * @private + * @function Highcharts.Chart#renderLabels + */ + Chart.prototype.renderLabels = function () { + var chart = this, + labels = chart.options.labels; + if (labels.items) { + labels.items.forEach(function (label) { + var style = extend(labels.style, label.style), + x = pInt(style.left) + chart.plotLeft, + y = pInt(style.top) + chart.plotTop + 12; + // delete to prevent rewriting in IE + delete style.left; + delete style.top; + chart.renderer + .text(label.html, x, y) + .attr({ zIndex: 2 }) + .css(style) + .add(); + }); + } + }; + /** + * Render all graphics for the chart. Runs internally on initialization. + * + * @private + * @function Highcharts.Chart#render + */ + Chart.prototype.render = function () { + var chart = this, + axes = chart.axes, + colorAxis = chart.colorAxis, + renderer = chart.renderer, + options = chart.options, + correction = 0, // correction for X axis labels + tempWidth, + tempHeight, + redoHorizontal, + redoVertical, + renderAxes = function (axes) { + axes.forEach(function (axis) { + if (axis.visible) { + axis.render(); + } + }); + }; + // Title + chart.setTitle(); + /** + * The overview of the chart's series. + * + * @name Highcharts.Chart#legend + * @type {Highcharts.Legend} + */ + chart.legend = new Legend(chart, options.legend); + // Get stacks + if (chart.getStacks) { + chart.getStacks(); + } + // Get chart margins + chart.getMargins(true); + chart.setChartSize(); + // Record preliminary dimensions for later comparison + tempWidth = chart.plotWidth; + axes.some(function (axis) { + if ( + axis.horiz && + axis.visible && + axis.options.labels.enabled && + axis.series.length + ) { + // 21 is the most common correction for X axis labels + correction = 21; + return true; + } + }); + // use Math.max to prevent negative plotHeight + chart.plotHeight = Math.max(chart.plotHeight - correction, 0); + tempHeight = chart.plotHeight; + // Get margins by pre-rendering axes + axes.forEach(function (axis) { + axis.setScale(); + }); + chart.getAxisMargins(); + // If the plot area size has changed significantly, calculate tick + // positions again + redoHorizontal = tempWidth / chart.plotWidth > 1.1; + // Height is more sensitive, use lower threshold + redoVertical = tempHeight / chart.plotHeight > 1.05; + if (redoHorizontal || redoVertical) { + axes.forEach(function (axis) { + if ( + (axis.horiz && redoHorizontal) || + (!axis.horiz && redoVertical) + ) { + // update to reflect the new margins + axis.setTickInterval(true); + } + }); + chart.getMargins(); // second pass to check for new labels + } + // Draw the borders and backgrounds + chart.drawChartBox(); + // Axes + if (chart.hasCartesianSeries) { + renderAxes(axes); + } else if (colorAxis && colorAxis.length) { + renderAxes(colorAxis); + } + // The series + if (!chart.seriesGroup) { + chart.seriesGroup = renderer + .g("series-group") + .attr({ zIndex: 3 }) + .add(); + } + chart.renderSeries(); + // Labels + chart.renderLabels(); + // Credits + chart.addCredits(); + // Handle responsiveness + if (chart.setResponsive) { + chart.setResponsive(); + } + // Handle scaling + chart.updateContainerScaling(); + // Set flag + chart.hasRendered = true; + }; + /** + * Set a new credits label for the chart. + * + * @sample highcharts/credits/credits-update/ + * Add and update credits + * + * @function Highcharts.Chart#addCredits + * + * @param {Highcharts.CreditsOptions} [credits] + * A configuration object for the new credits. + */ + Chart.prototype.addCredits = function (credits) { + var chart = this, + creds = merge(true, this.options.credits, credits); + if (creds.enabled && !this.credits) { /** - * Remove this series from the chart + * The chart's credits label. The label has an `update` method that + * allows setting new options as per the + * [credits options set](https://api.highcharts.com/highcharts/credits). * - * @private - * @function Highcharts.seriesTypes.column#remove + * @name Highcharts.Chart#credits + * @type {Highcharts.SVGElement} */ - remove: function () { - var series = this, - chart = series.chart; - // column and bar series affects other series of the same type - // as they are either stacked or grouped - if (chart.hasRendered) { - chart.series.forEach(function (otherSeries) { - if (otherSeries.type === series.type) { - otherSeries.isDirty = true; - } - }); + this.credits = this.renderer + .text(creds.text + (this.mapCredits || ""), 0, 0) + .addClass("highcharts-credits") + .on("click", function () { + if (creds.href) { + win.location.href = creds.href; } - LineSeries.prototype.remove.apply(series, arguments); + }) + .attr({ + align: creds.position.align, + zIndex: 8, + }); + if (!chart.styledMode) { + this.credits.css(creds.style); + } + this.credits.add().align(creds.position); + // Dynamically update + this.credits.update = function (options) { + chart.credits = chart.credits.destroy(); + chart.addCredits(options); + }; + } + }; + /** + * Handle scaling, #11329 - when there is scaling/transform on the container + * or on a parent element, we need to take this into account. We calculate + * the scaling once here and it is picked up where we need to use it + * (Pointer, Tooltip). + * + * @private + * @function Highcharts.Chart#updateContainerScaling + */ + Chart.prototype.updateContainerScaling = function () { + var container = this.container; + // #13342 - tooltip was not visible in Chrome, when chart + // updates height. + if ( + container.offsetWidth > 2 && // #13342 + container.offsetHeight > 2 && // #13342 + container.getBoundingClientRect + ) { + var bb = container.getBoundingClientRect(), + scaleX = bb.width / container.offsetWidth, + scaleY = bb.height / container.offsetHeight; + if (scaleX !== 1 || scaleY !== 1) { + this.containerScaling = { scaleX: scaleX, scaleY: scaleY }; + } else { + delete this.containerScaling; + } + } + }; + /** + * Remove the chart and purge memory. This method is called internally + * before adding a second chart into the same container, as well as on + * window unload to prevent leaks. + * + * @sample highcharts/members/chart-destroy/ + * Destroy the chart from a button + * @sample stock/members/chart-destroy/ + * Destroy with Highstock + * + * @function Highcharts.Chart#destroy + * + * @fires Highcharts.Chart#event:destroy + */ + Chart.prototype.destroy = function () { + var chart = this, + axes = chart.axes, + series = chart.series, + container = chart.container, + i, + parentNode = container && container.parentNode; + // fire the chart.destoy event + fireEvent(chart, "destroy"); + // Delete the chart from charts lookup array + if (chart.renderer.forExport) { + erase(charts, chart); // #6569 + } else { + charts[chart.index] = void 0; + } + H.chartCount--; + chart.renderTo.removeAttribute("data-highcharts-chart"); + // remove events + removeEvent(chart); + // ==== Destroy collections: + // Destroy axes + i = axes.length; + while (i--) { + axes[i] = axes[i].destroy(); + } + // Destroy scroller & scroller series before destroying base series + if (this.scroller && this.scroller.destroy) { + this.scroller.destroy(); + } + // Destroy each series + i = series.length; + while (i--) { + series[i] = series[i].destroy(); + } + // ==== Destroy chart properties: + [ + "title", + "subtitle", + "chartBackground", + "plotBackground", + "plotBGImage", + "plotBorder", + "seriesGroup", + "clipRect", + "credits", + "pointer", + "rangeSelector", + "legend", + "resetZoomButton", + "tooltip", + "renderer", + ].forEach(function (name) { + var prop = chart[name]; + if (prop && prop.destroy) { + chart[name] = prop.destroy(); + } + }); + // Remove container and all SVG, check container as it can break in IE + // when destroyed before finished loading + if (container) { + container.innerHTML = ""; + removeEvent(container); + if (parentNode) { + discardElement(container); + } + } + // clean it all up + objectEach(chart, function (val, key) { + delete chart[key]; + }); + }; + /** + * Prepare for first rendering after all data are loaded. + * + * @private + * @function Highcharts.Chart#firstRender + * @fires Highcharts.Chart#event:beforeRender + */ + Chart.prototype.firstRender = function () { + var chart = this, + options = chart.options; + // Hook for oldIE to check whether the chart is ready to render + if (chart.isReadyToRender && !chart.isReadyToRender()) { + return; + } + // Create the container + chart.getContainer(); + chart.resetMargins(); + chart.setChartSize(); + // Set the common chart properties (mainly invert) from the given series + chart.propFromSeries(); + // get axes + chart.getAxes(); + // Initialize the series + (isArray(options.series) ? options.series : []).forEach( + // #9680 + function (serieOptions) { + chart.initSeries(serieOptions); + } + ); + chart.linkSeries(); + chart.setSeriesData(); + // Run an event after axes and series are initialized, but before + // render. At this stage, the series data is indexed and cached in the + // xData and yData arrays, so we can access those before rendering. Used + // in Highstock. + fireEvent(chart, "beforeRender"); + // depends on inverted and on margins being set + if (Pointer) { + if (!H.hasTouch && (win.PointerEvent || win.MSPointerEvent)) { + chart.pointer = new MSPointer(chart, options); + } else { + /** + * The Pointer that keeps track of mouse and touch interaction. + * + * @memberof Highcharts.Chart + * @name pointer + * @type {Highcharts.Pointer} + * @instance + */ + chart.pointer = new Pointer(chart, options); + } + } + chart.render(); + // Fire the load event if there are no external images + if (!chart.renderer.imgCount && !chart.hasLoaded) { + chart.onload(); + } + // If the chart was rendered outside the top container, put it back in + // (#3679) + chart.temporaryDisplay(true); + }; + /** + * Internal function that runs on chart load, async if any images are loaded + * in the chart. Runs the callbacks and triggers the `load` and `render` + * events. + * + * @private + * @function Highcharts.Chart#onload + * @fires Highcharts.Chart#event:load + * @fires Highcharts.Chart#event:render + */ + Chart.prototype.onload = function () { + // Run callbacks, first the ones registered by modules, then user's one + this.callbacks.concat([this.callback]).forEach(function (fn) { + // Chart destroyed in its own callback (#3600) + if (fn && typeof this.index !== "undefined") { + fn.apply(this, [this]); + } + }, this); + fireEvent(this, "load"); + fireEvent(this, "render"); + // Set up auto resize, check for not destroyed (#6068) + if (defined(this.index)) { + this.setReflow(this.options.chart.reflow); + } + // Don't run again + this.hasLoaded = true; + }; + return Chart; + })(); + // Hook for adding callbacks in modules + Chart.prototype.callbacks = []; + /** + * Factory function for basic charts. + * + * @example + * // Render a chart in to div#container + * var chart = Highcharts.chart('container', { + * title: { + * text: 'My chart' + * }, + * series: [{ + * data: [1, 3, 2, 4] + * }] + * }); + * + * @function Highcharts.chart + * + * @param {string|Highcharts.HTMLDOMElement} [renderTo] + * The DOM element to render to, or its id. + * + * @param {Highcharts.Options} options + * The chart options structure. + * + * @param {Highcharts.ChartCallbackFunction} [callback] + * Function to run when the chart has loaded and and all external images + * are loaded. Defining a + * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load) + * handler is equivalent. + * + * @return {Highcharts.Chart} + * Returns the Chart object. + */ + function chart(a, b, c) { + return new Chart(a, b, c); + } + H.chart = chart; + H.Chart = Chart; + + return Chart; + } + ); + _registerModule( + _modules, + "Extensions/ScrollablePlotArea.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Chart/Chart.js"], + _modules["Core/Globals.js"], + _modules["Core/Utilities.js"], + ], + function (A, Chart, H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * Highcharts feature to make the Y axis stay fixed when scrolling the chart + * horizontally on mobile devices. Supports left and right side axes. + */ + /* + WIP on vertical scrollable plot area (#9378). To do: + - Bottom axis positioning + - Test with Gantt + - Look for size optimizing the code + - API and demos + */ + var stop = A.stop; + var addEvent = U.addEvent, + createElement = U.createElement, + pick = U.pick; + /** + * Options for a scrollable plot area. This feature provides a minimum size for + * the plot area of the chart. If the size gets smaller than this, typically + * on mobile devices, a native browser scrollbar is presented. This scrollbar + * provides smooth scrolling for the contents of the plot area, whereas the + * title, legend and unaffected axes are fixed. + * + * Since v7.1.2, a scrollable plot area can be defined for either horizontal or + * vertical scrolling, depending on whether the `minWidth` or `minHeight` + * option is set. + * + * @sample highcharts/chart/scrollable-plotarea + * Scrollable plot area + * @sample highcharts/chart/scrollable-plotarea-vertical + * Vertically scrollable plot area + * @sample {gantt} highcharts/chart/scrollable-plotarea-vertical + * Gantt chart with vertically scrollable plot area + * + * @since 6.1.0 + * @product highcharts gantt + * @apioption chart.scrollablePlotArea + */ + /** + * The minimum height for the plot area. If it gets smaller than this, the plot + * area will become scrollable. + * + * @type {number} + * @apioption chart.scrollablePlotArea.minHeight + */ + /** + * The minimum width for the plot area. If it gets smaller than this, the plot + * area will become scrollable. + * + * @type {number} + * @apioption chart.scrollablePlotArea.minWidth + */ + /** + * The initial scrolling position of the scrollable plot area. Ranges from 0 to + * 1, where 0 aligns the plot area to the left and 1 aligns it to the right. + * Typically we would use 1 if the chart has right aligned Y axes. + * + * @type {number} + * @apioption chart.scrollablePlotArea.scrollPositionX + */ + /** + * The initial scrolling position of the scrollable plot area. Ranges from 0 to + * 1, where 0 aligns the plot area to the top and 1 aligns it to the bottom. + * + * @type {number} + * @apioption chart.scrollablePlotArea.scrollPositionY + */ + /** + * The opacity of mask applied on one of the sides of the plot + * area. + * + * @sample {highcharts} highcharts/chart/scrollable-plotarea-opacity + * Disabled opacity for the mask + * + * @type {number} + * @default 0.85 + * @since 7.1.1 + * @apioption chart.scrollablePlotArea.opacity + */ + (""); // detach API doclets + /* eslint-disable no-invalid-this, valid-jsdoc */ + addEvent(Chart, "afterSetChartSize", function (e) { + var scrollablePlotArea = this.options.chart.scrollablePlotArea, + scrollableMinWidth = + scrollablePlotArea && scrollablePlotArea.minWidth, + scrollableMinHeight = + scrollablePlotArea && scrollablePlotArea.minHeight, + scrollablePixelsX, + scrollablePixelsY, + corrections; + if (!this.renderer.forExport) { + // The amount of pixels to scroll, the difference between chart + // width and scrollable width + if (scrollableMinWidth) { + this.scrollablePixelsX = scrollablePixelsX = Math.max( + 0, + scrollableMinWidth - this.chartWidth + ); + if (scrollablePixelsX) { + this.plotWidth += scrollablePixelsX; + if (this.inverted) { + this.clipBox.height += scrollablePixelsX; + this.plotBox.height += scrollablePixelsX; + } else { + this.clipBox.width += scrollablePixelsX; + this.plotBox.width += scrollablePixelsX; + } + corrections = { + // Corrections for right side + 1: { name: "right", value: scrollablePixelsX }, + }; + } + // Currently we can only do either X or Y + } else if (scrollableMinHeight) { + this.scrollablePixelsY = scrollablePixelsY = Math.max( + 0, + scrollableMinHeight - this.chartHeight + ); + if (scrollablePixelsY) { + this.plotHeight += scrollablePixelsY; + if (this.inverted) { + this.clipBox.width += scrollablePixelsY; + this.plotBox.width += scrollablePixelsY; + } else { + this.clipBox.height += scrollablePixelsY; + this.plotBox.height += scrollablePixelsY; + } + corrections = { + 2: { name: "bottom", value: scrollablePixelsY }, + }; + } + } + if (corrections && !e.skipAxes) { + this.axes.forEach(function (axis) { + // For right and bottom axes, only fix the plot line length + if (corrections[axis.side]) { + // Get the plot lines right in getPlotLinePath, + // temporarily set it to the adjusted plot width. + axis.getPlotLinePath = function () { + var marginName = corrections[axis.side].name, + correctionValue = corrections[axis.side].value, + // axis.right or axis.bottom + margin = this[marginName], + path; + // Temporarily adjust + this[marginName] = margin - correctionValue; + path = H.Axis.prototype.getPlotLinePath.apply( + this, + arguments + ); + // Reset + this[marginName] = margin; + return path; + }; + } else { + // Apply the corrected plotWidth + axis.setAxisSize(); + axis.setAxisTranslation(); + } + }); + } + } + }); + addEvent(Chart, "render", function () { + if (this.scrollablePixelsX || this.scrollablePixelsY) { + if (this.setUpScrolling) { + this.setUpScrolling(); + } + this.applyFixed(); + } else if (this.fixedDiv) { + // Has been in scrollable mode + this.applyFixed(); + } + }); + /** + * @private + * @function Highcharts.Chart#setUpScrolling + * @return {void} + */ + Chart.prototype.setUpScrolling = function () { + var _this = this; + var attribs = { + WebkitOverflowScrolling: "touch", + overflowX: "hidden", + overflowY: "hidden", + }; + if (this.scrollablePixelsX) { + attribs.overflowX = "auto"; + } + if (this.scrollablePixelsY) { + attribs.overflowY = "auto"; + } + // Insert a container with position relative + // that scrolling and fixed container renders to (#10555) + this.scrollingParent = createElement( + "div", + { + className: "highcharts-scrolling-parent", + }, + { + position: "relative", + }, + this.renderTo + ); + // Add the necessary divs to provide scrolling + this.scrollingContainer = createElement( + "div", + { + className: "highcharts-scrolling", + }, + attribs, + this.scrollingParent + ); + // On scroll, reset the chart position because it applies to the scrolled + // container + addEvent(this.scrollingContainer, "scroll", function () { + if (_this.pointer) { + delete _this.pointer.chartPosition; + } + }); + this.innerContainer = createElement( + "div", + { + className: "highcharts-inner-container", + }, + null, + this.scrollingContainer + ); + // Now move the container inside + this.innerContainer.appendChild(this.container); + // Don't run again + this.setUpScrolling = null; + }; + /** + * These elements are moved over to the fixed renderer and stay fixed when the + * user scrolls the chart + * @private + */ + Chart.prototype.moveFixedElements = function () { + var container = this.container, + fixedRenderer = this.fixedRenderer, + fixedSelectors = [ + ".highcharts-contextbutton", + ".highcharts-credits", + ".highcharts-legend", + ".highcharts-legend-checkbox", + ".highcharts-navigator-series", + ".highcharts-navigator-xaxis", + ".highcharts-navigator-yaxis", + ".highcharts-navigator", + ".highcharts-reset-zoom", + ".highcharts-scrollbar", + ".highcharts-subtitle", + ".highcharts-title", + ], + axisClass; + if (this.scrollablePixelsX && !this.inverted) { + axisClass = ".highcharts-yaxis"; + } else if (this.scrollablePixelsX && this.inverted) { + axisClass = ".highcharts-xaxis"; + } else if (this.scrollablePixelsY && !this.inverted) { + axisClass = ".highcharts-xaxis"; + } else if (this.scrollablePixelsY && this.inverted) { + axisClass = ".highcharts-yaxis"; + } + fixedSelectors.push(axisClass, axisClass + "-labels"); + fixedSelectors.forEach(function (className) { + [].forEach.call( + container.querySelectorAll(className), + function (elem) { + (elem.namespaceURI === fixedRenderer.SVG_NS + ? fixedRenderer.box + : fixedRenderer.box.parentNode + ).appendChild(elem); + elem.style.pointerEvents = "auto"; } + ); + }); + }; + /** + * @private + * @function Highcharts.Chart#applyFixed + * @return {void} + */ + Chart.prototype.applyFixed = function () { + var _a, _b; + var fixedRenderer, + scrollableWidth, + scrollableHeight, + firstTime = !this.fixedDiv, + scrollableOptions = this.options.chart.scrollablePlotArea; + // First render + if (firstTime) { + this.fixedDiv = createElement( + "div", + { + className: "highcharts-fixed", + }, + { + position: "absolute", + overflow: "hidden", + pointerEvents: "none", + zIndex: 2, + top: 0, + }, + null, + true + ); + (_a = this.scrollingContainer) === null || _a === void 0 + ? void 0 + : _a.parentNode.insertBefore( + this.fixedDiv, + this.scrollingContainer + ); + this.renderTo.style.overflow = "visible"; + this.fixedRenderer = fixedRenderer = new H.Renderer( + this.fixedDiv, + this.chartWidth, + this.chartHeight, + (_b = this.options.chart) === null || _b === void 0 + ? void 0 + : _b.style + ); + // Mask + this.scrollableMask = fixedRenderer + .path() + .attr({ + fill: this.options.chart.backgroundColor || "#fff", + "fill-opacity": pick(scrollableOptions.opacity, 0.85), + zIndex: -1, + }) + .addClass("highcharts-scrollable-mask") + .add(); + this.moveFixedElements(); + addEvent(this, "afterShowResetZoom", this.moveFixedElements); + addEvent(this, "afterLayOutTitles", this.moveFixedElements); + } else { + // Set the size of the fixed renderer to the visible width + this.fixedRenderer.setSize(this.chartWidth, this.chartHeight); + } + // Increase the size of the scrollable renderer and background + scrollableWidth = this.chartWidth + (this.scrollablePixelsX || 0); + scrollableHeight = this.chartHeight + (this.scrollablePixelsY || 0); + stop(this.container); + this.container.style.width = scrollableWidth + "px"; + this.container.style.height = scrollableHeight + "px"; + this.renderer.boxWrapper.attr({ + width: scrollableWidth, + height: scrollableHeight, + viewBox: [0, 0, scrollableWidth, scrollableHeight].join(" "), }); - /* eslint-enable valid-jsdoc */ - /** - * A `column` series. If the [type](#series.column.type) option is - * not specified, it is inherited from [chart.type](#chart.type). + this.chartBackground.attr({ + width: scrollableWidth, + height: scrollableHeight, + }); + this.scrollingContainer.style.height = this.chartHeight + "px"; + // Set scroll position + if (firstTime) { + if (scrollableOptions.scrollPositionX) { + this.scrollingContainer.scrollLeft = + this.scrollablePixelsX * scrollableOptions.scrollPositionX; + } + if (scrollableOptions.scrollPositionY) { + this.scrollingContainer.scrollTop = + this.scrollablePixelsY * scrollableOptions.scrollPositionY; + } + } + // Mask behind the left and right side + var axisOffset = this.axisOffset, + maskTop = this.plotTop - axisOffset[0] - 1, + maskLeft = this.plotLeft - axisOffset[3] - 1, + maskBottom = this.plotTop + this.plotHeight + axisOffset[2] + 1, + maskRight = this.plotLeft + this.plotWidth + axisOffset[1] + 1, + maskPlotRight = + this.plotLeft + this.plotWidth - (this.scrollablePixelsX || 0), + maskPlotBottom = + this.plotTop + this.plotHeight - (this.scrollablePixelsY || 0), + d; + if (this.scrollablePixelsX) { + d = [ + // Left side + ["M", 0, maskTop], + ["L", this.plotLeft - 1, maskTop], + ["L", this.plotLeft - 1, maskBottom], + ["L", 0, maskBottom], + ["Z"], + // Right side + ["M", maskPlotRight, maskTop], + ["L", this.chartWidth, maskTop], + ["L", this.chartWidth, maskBottom], + ["L", maskPlotRight, maskBottom], + ["Z"], + ]; + } else if (this.scrollablePixelsY) { + d = [ + // Top side + ["M", maskLeft, 0], + ["L", maskLeft, this.plotTop - 1], + ["L", maskRight, this.plotTop - 1], + ["L", maskRight, 0], + ["Z"], + // Bottom side + ["M", maskLeft, maskPlotBottom], + ["L", maskLeft, this.chartHeight], + ["L", maskRight, this.chartHeight], + ["L", maskRight, maskPlotBottom], + ["Z"], + ]; + } else { + d = [["M", 0, 0]]; + } + if (this.redrawTrigger !== "adjustHeight") { + this.scrollableMask.attr({ d: d }); + } + }; + } + ); + _registerModule( + _modules, + "Core/Axis/StackingAxis.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Utilities.js"], + ], + function (A, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var getDeferredAnimation = A.getDeferredAnimation; + var addEvent = U.addEvent, + destroyObjectProperties = U.destroyObjectProperties, + fireEvent = U.fireEvent, + objectEach = U.objectEach, + pick = U.pick; + /* eslint-disable valid-jsdoc */ + /** + * Adds stacking support to axes. + * @private + * @class + */ + var StackingAxisAdditions = /** @class */ (function () { + /* * * - * @extends series,plotOptions.column - * @excluding connectNulls, dataParser, dataURL, gapSize, gapUnit, linecap, - * lineWidth, marker, connectEnds, step - * @product highcharts highstock - * @apioption series.column - */ - /** - * An array of data points for the series. For the `column` series type, - * points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. The `x` values will be automatically - * calculated, either starting at 0 and incremented by 1, or from - * `pointStart` and `pointInterval` given in the series options. If the axis - * has categories, these will be used. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 2. An array of arrays with 2 values. In this case, the values correspond to - * `x,y`. If the first value is a string, it is applied as the name of the - * point, and the `x` value is inferred. - * ```js - * data: [ - * [0, 6], - * [1, 2], - * [2, 6] - * ] - * ``` - * - * 3. 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.column.turboThreshold), this option is not - * available. - * ```js - * data: [{ - * x: 1, - * y: 9, - * name: "Point2", - * color: "#00FF00" - * }, { - * x: 1, - * y: 6, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects - * - * @type {Array<number|Array<(number|string),(number|null)>|null|*>} - * @extends series.line.data - * @excluding marker - * @product highcharts highstock - * @apioption series.column.data - */ - /** - * The color of the border surrounding the column or bar. + * Constructors * - * In styled mode, the border stroke can be set with the `.highcharts-point` - * rule. + * */ + function StackingAxisAdditions(axis) { + this.oldStacks = {}; + this.stacks = {}; + this.stacksTouched = 0; + this.axis = axis; + } + /* * * - * @sample {highcharts} highcharts/plotoptions/column-bordercolor/ - * Dark gray border + * Functions * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @product highcharts highstock - * @apioption series.column.data.borderColor + * */ + /** + * Build the stacks from top down + * @private */ + StackingAxisAdditions.prototype.buildStacks = function () { + var stacking = this; + var axis = stacking.axis; + var axisSeries = axis.series; + var reversedStacks = pick(axis.options.reversedStacks, true); + var len = axisSeries.length; + var actualSeries, i; + if (!axis.isXAxis) { + stacking.usePercentage = false; + i = len; + while (i--) { + actualSeries = axisSeries[reversedStacks ? i : len - i - 1]; + actualSeries.setStackedPoints(); + actualSeries.setGroupedPoints(); + } + // Loop up again to compute percent and stream stack + for (i = 0; i < len; i++) { + axisSeries[i].modifyStacks(); + } + fireEvent(axis, "afterBuildStacks"); + } + }; /** - * The width of the border surrounding the column or bar. - * - * In styled mode, the stroke width can be set with the `.highcharts-point` - * rule. - * - * @sample {highcharts} highcharts/plotoptions/column-borderwidth/ - * 2px black border - * - * @type {number} - * @product highcharts highstock - * @apioption series.column.data.borderWidth + * @private */ + StackingAxisAdditions.prototype.cleanStacks = function () { + var stacking = this; + var axis = stacking.axis; + var stacks; + if (!axis.isXAxis) { + if (stacking.oldStacks) { + stacks = stacking.stacks = stacking.oldStacks; + } + // reset stacks + objectEach(stacks, function (type) { + objectEach(type, function (stack) { + stack.cumulative = stack.total; + }); + }); + } + }; /** - * A name for the dash style to use for the column or bar. Overrides - * dashStyle on the series. - * - * In styled mode, the stroke dash-array can be set with the same classes as - * listed under [data.color](#series.column.data.color). - * - * @see [series.pointWidth](#plotOptions.column.dashStyle) - * - * @type {Highcharts.DashStyleValue} - * @apioption series.column.data.dashStyle + * Set all the stacks to initial states and destroy unused ones. + * @private */ + StackingAxisAdditions.prototype.resetStacks = function () { + var stacking = this; + var axis = stacking.axis; + var stacks = stacking.stacks; + if (!axis.isXAxis) { + objectEach(stacks, function (type) { + objectEach(type, function (stack, key) { + // Clean up memory after point deletion (#1044, #4320) + if (stack.touched < stacking.stacksTouched) { + stack.destroy(); + delete type[key]; + // Reset stacks + } else { + stack.total = null; + stack.cumulative = null; + } + }); + }); + } + }; /** - * A pixel value specifying a fixed width for the column or bar. Overrides - * pointWidth on the series. The width effects the dimension that is not based - * on the point value. + * @private + */ + StackingAxisAdditions.prototype.renderStackTotals = function () { + var stacking = this; + var axis = stacking.axis; + var chart = axis.chart; + var renderer = chart.renderer; + var stacks = stacking.stacks; + var stackLabelsAnim = axis.options.stackLabels.animation; + var animationConfig = getDeferredAnimation(chart, stackLabelsAnim); + var stackTotalGroup = (stacking.stackTotalGroup = + stacking.stackTotalGroup || + renderer + .g("stack-labels") + .attr({ + visibility: "visible", + zIndex: 6, + opacity: 0, + }) + .add()); + // plotLeft/Top will change when y axis gets wider so we need to + // translate the stackTotalGroup at every render call. See bug #506 + // and #516 + stackTotalGroup.translate(chart.plotLeft, chart.plotTop); + // Render each stack total + objectEach(stacks, function (type) { + objectEach(type, function (stack) { + stack.render(stackTotalGroup); + }); + }); + stackTotalGroup.animate( + { + opacity: 1, + }, + animationConfig + ); + }; + return StackingAxisAdditions; + })(); + /** + * Axis with stacking support. + * @private + * @class + */ + var StackingAxis = /** @class */ (function () { + function StackingAxis() {} + /* * * - * @see [series.pointWidth](#plotOptions.column.pointWidth) + * Static Functions * - * @type {number} - * @apioption series.column.data.pointWidth + * */ + /** + * Extends axis with stacking support. + * @private */ + StackingAxis.compose = function (AxisClass) { + var axisProto = AxisClass.prototype; + addEvent(AxisClass, "init", StackingAxis.onInit); + addEvent(AxisClass, "destroy", StackingAxis.onDestroy); + }; /** - * @excluding halo, lineWidth, lineWidthPlus, marker - * @product highcharts highstock - * @apioption series.column.states.hover + * @private */ + StackingAxis.onDestroy = function () { + var stacking = this.stacking; + if (!stacking) { + return; + } + var stacks = stacking.stacks; + // Destroy each stack total + objectEach(stacks, function (stack, stackKey) { + destroyObjectProperties(stack); + stacks[stackKey] = null; + }); + if (stacking && stacking.stackTotalGroup) { + stacking.stackTotalGroup.destroy(); + } + }; /** - * @excluding halo, lineWidth, lineWidthPlus, marker - * @product highcharts highstock - * @apioption series.column.states.select + * @private */ - ''; // includes above doclets in transpilat + StackingAxis.onInit = function () { + var axis = this; + if (!axis.stacking) { + axis.stacking = new StackingAxisAdditions(axis); + } + }; + return StackingAxis; + })(); - return ColumnSeries; - }); - _registerModule(_modules, 'Series/BarSeries.js', [_modules['Core/Series/Series.js']], function (Series) { - /* * - * - * (c) 2010-2020 Torstein Honsi + return StackingAxis; + } + ); + _registerModule( + _modules, + "Mixins/LegendSymbol.js", + [_modules["Core/Globals.js"], _modules["Core/Utilities.js"]], + function (H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var merge = U.merge, + pick = U.pick; + /* eslint-disable valid-jsdoc */ + /** + * Legend symbol mixin. + * + * @private + * @mixin Highcharts.LegendSymbolMixin + */ + var LegendSymbolMixin = (H.LegendSymbolMixin = { + /** + * Get the series' symbol in the legend * - * License: www.highcharts.com/license + * @private + * @function Highcharts.LegendSymbolMixin.drawRectangle * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @param {Highcharts.Legend} legend + * The legend object * - * */ - /** - * Bar series type. + * @param {Highcharts.Point|Highcharts.Series} item + * The series (this) or point + */ + drawRectangle: function (legend, item) { + var options = legend.options, + symbolHeight = legend.symbolHeight, + square = options.squareSymbol, + symbolWidth = square ? symbolHeight : legend.symbolWidth; + item.legendSymbol = this.chart.renderer + .rect( + square ? (legend.symbolWidth - symbolHeight) / 2 : 0, + legend.baseline - symbolHeight + 1, // #3988 + symbolWidth, + symbolHeight, + pick(legend.options.symbolRadius, symbolHeight / 2) + ) + .addClass("highcharts-point") + .attr({ + zIndex: 3, + }) + .add(item.legendGroup); + }, + /** + * Get the series' symbol in the legend. This method should be overridable + * to create custom symbols through + * Highcharts.seriesTypes[type].prototype.drawLegendSymbols. * * @private - * @class - * @name Highcharts.seriesTypes.bar + * @function Highcharts.LegendSymbolMixin.drawLineMarker * - * @augments Highcharts.Series + * @param {Highcharts.Legend} legend + * The legend object. */ - Series.seriesType('bar', 'column', + drawLineMarker: function (legend) { + var options = this.options, + markerOptions = options.marker, + radius, + legendSymbol, + symbolWidth = legend.symbolWidth, + symbolHeight = legend.symbolHeight, + generalRadius = symbolHeight / 2, + renderer = this.chart.renderer, + legendItemGroup = this.legendGroup, + verticalCenter = + legend.baseline - Math.round(legend.fontMetrics.b * 0.3), + attr = {}; + // Draw the line + if (!this.chart.styledMode) { + attr = { + "stroke-width": options.lineWidth || 0, + }; + if (options.dashStyle) { + attr.dashstyle = options.dashStyle; + } + } + this.legendLine = renderer + .path([ + ["M", 0, verticalCenter], + ["L", symbolWidth, verticalCenter], + ]) + .addClass("highcharts-graph") + .attr(attr) + .add(legendItemGroup); + // Draw the marker + if (markerOptions && markerOptions.enabled !== false && symbolWidth) { + // Do not allow the marker to be larger than the symbolHeight + radius = Math.min( + pick(markerOptions.radius, generalRadius), + generalRadius + ); + // Restrict symbol markers size + if (this.symbol.indexOf("url") === 0) { + markerOptions = merge(markerOptions, { + width: symbolHeight, + height: symbolHeight, + }); + radius = 0; + } + this.legendSymbol = legendSymbol = renderer + .symbol( + this.symbol, + symbolWidth / 2 - radius, + verticalCenter - radius, + 2 * radius, + 2 * radius, + markerOptions + ) + .addClass("highcharts-point") + .add(legendItemGroup); + legendSymbol.isMarker = true; + } + }, + }); + + return LegendSymbolMixin; + } + ); + _registerModule( + _modules, + "Core/Series/CartesianSeries.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Series/Series.js"], + _modules["Core/Globals.js"], + _modules["Mixins/LegendSymbol.js"], + _modules["Core/Options.js"], + _modules["Core/Series/Point.js"], + _modules["Core/Renderer/SVG/SVGElement.js"], + _modules["Core/Utilities.js"], + ], + function (A, BaseSeries, H, LegendSymbolMixin, O, Point, SVGElement, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animObject = A.animObject; + var defaultOptions = O.defaultOptions; + var addEvent = U.addEvent, + arrayMax = U.arrayMax, + arrayMin = U.arrayMin, + clamp = U.clamp, + correctFloat = U.correctFloat, + defined = U.defined, + erase = U.erase, + error = U.error, + extend = U.extend, + find = U.find, + fireEvent = U.fireEvent, + getNestedProperty = U.getNestedProperty, + isArray = U.isArray, + isFunction = U.isFunction, + isNumber = U.isNumber, + isString = U.isString, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick, + removeEvent = U.removeEvent, + splat = U.splat, + syncTimeout = U.syncTimeout; + /** + * This is a placeholder type of the possible series options for + * [Highcharts](../highcharts/series), [Highstock](../highstock/series), + * [Highmaps](../highmaps/series), and [Gantt](../gantt/series). + * + * In TypeScript is this dynamically generated to reference all possible types + * of series options. + * + * @ignore-declaration + * @typedef {Highcharts.SeriesOptions|Highcharts.Dictionary<*>} Highcharts.SeriesOptionsType + */ + /** + * Options for `dataSorting`. + * + * @interface Highcharts.DataSortingOptionsObject + * @since 8.0.0 + */ /** + * Enable or disable data sorting for the series. + * @name Highcharts.DataSortingOptionsObject#enabled + * @type {boolean|undefined} + */ /** + * Whether to allow matching points by name in an update. + * @name Highcharts.DataSortingOptionsObject#matchByName + * @type {boolean|undefined} + */ /** + * Determines what data value should be used to sort by. + * @name Highcharts.DataSortingOptionsObject#sortKey + * @type {string|undefined} + */ + /** + * Function callback when a series has been animated. + * + * @callback Highcharts.SeriesAfterAnimateCallbackFunction + * + * @param {Highcharts.Series} this + * The series where the event occured. + * + * @param {Highcharts.SeriesAfterAnimateEventObject} event + * Event arguments. + */ + /** + * Event information regarding completed animation of a series. + * + * @interface Highcharts.SeriesAfterAnimateEventObject + */ /** + * Animated series. + * @name Highcharts.SeriesAfterAnimateEventObject#target + * @type {Highcharts.Series} + */ /** + * Event type. + * @name Highcharts.SeriesAfterAnimateEventObject#type + * @type {"afterAnimate"} + */ + /** + * Function callback when the checkbox next to the series' name in the legend is + * clicked. + * + * @callback Highcharts.SeriesCheckboxClickCallbackFunction + * + * @param {Highcharts.Series} this + * The series where the event occured. + * + * @param {Highcharts.SeriesCheckboxClickEventObject} event + * Event arguments. + */ + /** + * Event information regarding check of a series box. + * + * @interface Highcharts.SeriesCheckboxClickEventObject + */ /** + * Whether the box has been checked. + * @name Highcharts.SeriesCheckboxClickEventObject#checked + * @type {boolean} + */ /** + * Related series. + * @name Highcharts.SeriesCheckboxClickEventObject#item + * @type {Highcharts.Series} + */ /** + * Related series. + * @name Highcharts.SeriesCheckboxClickEventObject#target + * @type {Highcharts.Series} + */ /** + * Event type. + * @name Highcharts.SeriesCheckboxClickEventObject#type + * @type {"checkboxClick"} + */ + /** + * Function callback when a series is clicked. Return false to cancel toogle + * actions. + * + * @callback Highcharts.SeriesClickCallbackFunction + * + * @param {Highcharts.Series} this + * The series where the event occured. + * + * @param {Highcharts.SeriesClickEventObject} event + * Event arguments. + */ + /** + * Common information for a click event on a series. + * + * @interface Highcharts.SeriesClickEventObject + * @extends global.Event + */ /** + * Nearest point on the graph. + * @name Highcharts.SeriesClickEventObject#point + * @type {Highcharts.Point} + */ + /** + * Gets fired when the series is hidden after chart generation time, either by + * clicking the legend item or by calling `.hide()`. + * + * @callback Highcharts.SeriesHideCallbackFunction + * + * @param {Highcharts.Series} this + * The series where the event occured. + * + * @param {global.Event} event + * The event that occured. + */ + /** + * The SVG value used for the `stroke-linecap` and `stroke-linejoin` of a line + * graph. + * + * @typedef {"butt"|"round"|"square"|string} Highcharts.SeriesLinecapValue + */ + /** + * Gets fired when the legend item belonging to the series is clicked. The + * default action is to toggle the visibility of the series. This can be + * prevented by returning `false` or calling `event.preventDefault()`. + * + * @callback Highcharts.SeriesLegendItemClickCallbackFunction + * + * @param {Highcharts.Series} this + * The series where the event occured. + * + * @param {Highcharts.SeriesLegendItemClickEventObject} event + * The event that occured. + */ + /** + * Information about the event. + * + * @interface Highcharts.SeriesLegendItemClickEventObject + */ /** + * Related browser event. + * @name Highcharts.SeriesLegendItemClickEventObject#browserEvent + * @type {global.PointerEvent} + */ /** + * Prevent the default action of toggle the visibility of the series. + * @name Highcharts.SeriesLegendItemClickEventObject#preventDefault + * @type {Function} + */ /** + * Related series. + * @name Highcharts.SeriesCheckboxClickEventObject#target + * @type {Highcharts.Series} + */ /** + * Event type. + * @name Highcharts.SeriesCheckboxClickEventObject#type + * @type {"checkboxClick"} + */ + /** + * Gets fired when the mouse leaves the graph. + * + * @callback Highcharts.SeriesMouseOutCallbackFunction + * + * @param {Highcharts.Series} this + * Series where the event occured. + * + * @param {global.PointerEvent} event + * Event that occured. + */ + /** + * Gets fired when the mouse enters the graph. + * + * @callback Highcharts.SeriesMouseOverCallbackFunction + * + * @param {Highcharts.Series} this + * Series where the event occured. + * + * @param {global.PointerEvent} event + * Event that occured. + */ + /** + * Translation and scale for the plot area of a series. + * + * @interface Highcharts.SeriesPlotBoxObject + */ /** + * @name Highcharts.SeriesPlotBoxObject#scaleX + * @type {number} + */ /** + * @name Highcharts.SeriesPlotBoxObject#scaleY + * @type {number} + */ /** + * @name Highcharts.SeriesPlotBoxObject#translateX + * @type {number} + */ /** + * @name Highcharts.SeriesPlotBoxObject#translateY + * @type {number} + */ + /** + * Gets fired when the series is shown after chart generation time, either by + * clicking the legend item or by calling `.show()`. + * + * @callback Highcharts.SeriesShowCallbackFunction + * + * @param {Highcharts.Series} this + * Series where the event occured. + * + * @param {global.Event} event + * Event that occured. + */ + /** + * Possible key values for the series state options. + * + * @typedef {"hover"|"inactive"|"normal"|"select"} Highcharts.SeriesStateValue + */ + (""); // detach doclets above + var seriesTypes = BaseSeries.seriesTypes, + win = H.win; + /** + * This is the base series prototype that all other series types inherit from. + * A new series is initialized either through the + * [series](https://api.highcharts.com/highcharts/series) + * option structure, or after the chart is initialized, through + * {@link Highcharts.Chart#addSeries}. + * + * The object can be accessed in a number of ways. All series and point event + * handlers give a reference to the `series` object. The chart object has a + * {@link Highcharts.Chart#series|series} property that is a collection of all + * the chart's series. The point objects and axis objects also have the same + * reference. + * + * Another way to reference the series programmatically is by `id`. Add an id + * in the series configuration options, and get the series object by + * {@link Highcharts.Chart#get}. + * + * Configuration options for the series are given in three levels. Options for + * all series in a chart are given in the + * [plotOptions.series](https://api.highcharts.com/highcharts/plotOptions.series) + * object. Then options for all series of a specific type + * are given in the plotOptions of that type, for example `plotOptions.line`. + * Next, options for one single series are given in the series array, or as + * arguments to `chart.addSeries`. + * + * The data in the series is stored in various arrays. + * + * - First, `series.options.data` contains all the original config options for + * each point whether added by options or methods like `series.addPoint`. + * + * - Next, `series.data` contains those values converted to points, but in case + * the series data length exceeds the `cropThreshold`, or if the data is + * grouped, `series.data` doesn't contain all the points. It only contains the + * points that have been created on demand. + * + * - Then there's `series.points` that contains all currently visible point + * objects. In case of cropping, the cropped-away points are not part of this + * array. The `series.points` array starts at `series.cropStart` compared to + * `series.data` and `series.options.data`. If however the series data is + * grouped, these can't be correlated one to one. + * + * - `series.xData` and `series.processedXData` contain clean x values, + * equivalent to `series.data` and `series.points`. + * + * - `series.yData` and `series.processedYData` contain clean y values, + * equivalent to `series.data` and `series.points`. + * + * @class + * @name Highcharts.Series + * + * @param {Highcharts.Chart} chart + * The chart instance. + * + * @param {Highcharts.SeriesOptionsType|object} options + * The series options. + */ /** + * The line series is the base type and is therefor the series base prototype. + * + * @private + * @class + * @name Highcharts.seriesTypes.line + * + * @augments Highcharts.Series + */ + var CartesianSeries = BaseSeries.seriesType( + "line", /** - * A bar series is a special type of column series where the columns are - * horizontal. - * - * @sample highcharts/demo/bar-basic/ - * Bar chart + * Series options for specific data and the data itself. In TypeScript you + * have to cast the series options to specific series types, + to get all + * possible options for a series. + * + * @example + * // TypeScript example + * Highcharts.chart('container', { + * series: [{ + * color: '#06C', + * data: [[0, 1], + [2, 3]] + * } as Highcharts.SeriesLineOptions ] + * }); * - * @extends plotOptions.column - * @product highcharts - * @apioption plotOptions.bar - */ - /** - * @ignore + * @type {Array<*>} + * @apioption series */ - null, { - inverted: true - }); /** - * A `bar` series. If the [type](#series.bar.type) option is not specified, - * it is inherited from [chart.type](#chart.type). + * An id for the series. This can be used after render time to get a pointer + * to the series object through `chart.get()`. * - * @extends series,plotOptions.bar - * @excluding connectNulls, dashStyle, dataParser, dataURL, gapSize, gapUnit, - * linecap, lineWidth, marker, connectEnds, step - * @product highcharts - * @apioption series.bar - */ - /** - * An array of data points for the series. For the `bar` series type, - * points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. The `x` values will be automatically - * calculated, either starting at 0 and incremented by 1, or from - * `pointStart` and `pointInterval` given in the series options. If the axis - * has categories, these will be used. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 2. An array of arrays with 2 values. In this case, the values correspond to - * `x,y`. If the first value is a string, it is applied as the name of the - * point, and the `x` value is inferred. - * ```js - * data: [ - * [0, 5], - * [1, 10], - * [2, 3] - * ] - * ``` - * - * 3. 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.bar.turboThreshold), this option is not - * available. - * ```js - * data: [{ - * x: 1, - * y: 1, - * name: "Point2", - * color: "#00FF00" - * }, { - * x: 1, - * y: 10, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects - * - * @type {Array<number|Array<(number|string),(number|null)>|null|*>} - * @extends series.column.data - * @product highcharts - * @apioption series.bar.data + * @sample {highcharts} highcharts/plotoptions/series-id/ + * Get series by id + * + * @type {string} + * @since 1.2.0 + * @apioption series.id */ /** - * @excluding halo,lineWidth,lineWidthPlus,marker - * @product highcharts highstock - * @apioption series.bar.states.hover + * The index of the series in the chart, affecting the internal index in the + * `chart.series` array, the visible Z index as well as the order in the + * legend. + * + * @type {number} + * @since 2.3.0 + * @apioption series.index */ /** - * @excluding halo,lineWidth,lineWidthPlus,marker - * @product highcharts highstock - * @apioption series.bar.states.select - */ - ''; // gets doclets above into transpilat - - }); - _registerModule(_modules, 'Series/ScatterSeries.js', [_modules['Core/Series/Series.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (BaseSeries, H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * The sequential index of the series in the legend. * - * */ - var addEvent = U.addEvent; - var Series = H.Series; - /** - * Scatter series type. + * @see [legend.reversed](#legend.reversed), + * [yAxis.reversedStacks](#yAxis.reversedStacks) * - * @private - * @class - * @name Highcharts.seriesTypes.scatter + * @sample {highcharts|highstock} highcharts/series/legendindex/ + * Legend in opposite order * - * @augments Highcharts.Series + * @type {number} + * @apioption series.legendIndex */ - BaseSeries.seriesType('scatter', 'line', /** - * A scatter plot uses cartesian coordinates to display values for two - * variables for a set of data. + * The name of the series as shown in the legend, tooltip etc. * - * @sample {highcharts} highcharts/demo/scatter/ - * Scatter plot + * @sample {highcharts} highcharts/series/name/ + * Series name + * @sample {highmaps} maps/demo/category-map/ + * Series name * - * @extends plotOptions.line - * @excluding cropThreshold, pointPlacement, shadow, useOhlcData - * @product highcharts highstock - * @optionparent plotOptions.scatter + * @type {string} + * @apioption series.name */ - { - /** - * The width of the line connecting the data points. - * - * @sample {highcharts} highcharts/plotoptions/scatter-linewidth-none/ - * 0 by default - * @sample {highcharts} highcharts/plotoptions/scatter-linewidth-1/ - * 1px - * - * @product highcharts highstock - */ - lineWidth: 0, - findNearestPointBy: 'xy', - /** - * Apply a jitter effect for the rendered markers. When plotting - * discrete values, a little random noise may help telling the points - * apart. The jitter setting applies a random displacement of up to `n` - * axis units in either direction. So for example on a horizontal X - * axis, setting the `jitter.x` to 0.24 will render the point in a - * random position between 0.24 units to the left and 0.24 units to the - * right of the true axis position. On a category axis, setting it to - * 0.5 will fill up the bin and make the data appear continuous. - * - * When rendered on top of a box plot or a column series, a jitter value - * of 0.24 will correspond to the underlying series' default - * [groupPadding]( - * https://api.highcharts.com/highcharts/plotOptions.column.groupPadding) - * and [pointPadding]( - * https://api.highcharts.com/highcharts/plotOptions.column.pointPadding) - * settings. - * - * @sample {highcharts} highcharts/series-scatter/jitter - * Jitter on a scatter plot - * - * @sample {highcharts} highcharts/series-scatter/jitter-boxplot - * Jittered scatter plot on top of a box plot - * - * @product highcharts highstock - * @since 7.0.2 - */ - jitter: { - /** - * The maximal X offset for the random jitter effect. - */ - x: 0, - /** - * The maximal Y offset for the random jitter effect. - */ - y: 0 - }, - marker: { - enabled: true // Overrides auto-enabling in line series (#3647) - }, - /** - * Sticky tracking of mouse events. When true, the `mouseOut` event - * on a series isn't triggered until the mouse moves over another - * series, or out of the plot area. When false, the `mouseOut` event on - * a series is triggered when the mouse leaves the area around the - * series' graph or markers. This also implies the tooltip. When - * `stickyTracking` is false and `tooltip.shared` is false, the tooltip - * will be hidden when moving the mouse between series. - * - * @type {boolean} - * @default false - * @product highcharts highstock - * @apioption plotOptions.scatter.stickyTracking - */ - /** - * A configuration object for the tooltip rendering of each single - * series. Properties are inherited from [tooltip](#tooltip). - * Overridable properties are `headerFormat`, `pointFormat`, - * `yDecimals`, `xDateFormat`, `yPrefix` and `ySuffix`. Unlike other - * series, in a scatter plot the series.name by default shows in the - * headerFormat and point.x and point.y in the pointFormat. - * - * @product highcharts highstock - */ - tooltip: { - headerFormat: '<span style="color:{point.color}">\u25CF</span> ' + - '<span style="font-size: 10px"> {series.name}</span><br/>', - pointFormat: 'x: <b>{point.x}</b><br/>y: <b>{point.y}</b><br/>' - } - // Prototype members - }, { - sorted: false, - requireSorting: false, - noSharedTooltip: true, - trackerGroups: ['group', 'markerGroup', 'dataLabelsGroup'], - takeOrdinalPosition: false, - /* eslint-disable valid-jsdoc */ - /** - * @private - * @function Highcharts.seriesTypes.scatter#drawGraph - */ - drawGraph: function () { - if (this.options.lineWidth || - // In case we have a graph from before and we update the line - // width to 0 (#13816) - (this.options.lineWidth === 0 && - this.graph && - this.graph.strokeWidth())) { - Series.prototype.drawGraph.call(this); - } - }, - // Optionally add the jitter effect - applyJitter: function () { - var series = this, - jitter = this.options.jitter, - len = this.points.length; - /** - * Return a repeatable, pseudo-random number based on an integer - * seed. - * @private - */ - function unrandom(seed) { - var rand = Math.sin(seed) * 10000; - return rand - Math.floor(rand); - } - if (jitter) { - this.points.forEach(function (point, i) { - ['x', 'y'].forEach(function (dim, j) { - var axis, - plotProp = 'plot' + dim.toUpperCase(), - min, - max, - translatedJitter; - if (jitter[dim] && !point.isNull) { - axis = series[dim + 'Axis']; - translatedJitter = - jitter[dim] * axis.transA; - if (axis && !axis.isLog) { - // Identify the outer bounds of the jitter range - min = Math.max(0, point[plotProp] - translatedJitter); - max = Math.min(axis.len, point[plotProp] + translatedJitter); - // Find a random position within this range - point[plotProp] = min + - (max - min) * unrandom(i + j * len); - // Update clientX for the tooltip k-d-tree - if (dim === 'x') { - point.clientX = point.plotX; - } - } - } - }); - }); - } - } - /* eslint-enable valid-jsdoc */ - }); - /* eslint-disable no-invalid-this */ - addEvent(Series, 'afterTranslate', function () { - if (this.applyJitter) { - this.applyJitter(); - } - }); - /* eslint-enable no-invalid-this */ /** - * A `scatter` series. If the [type](#series.scatter.type) option is - * not specified, it is inherited from [chart.type](#chart.type). - * - * @extends series,plotOptions.scatter - * @excluding cropThreshold, dataParser, dataURL, useOhlcData - * @product highcharts highstock - * @apioption series.scatter - */ - /** - * An array of data points for the series. For the `scatter` series - * type, points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. The `x` values will be automatically - * calculated, either starting at 0 and incremented by 1, or from - * `pointStart` and `pointInterval` given in the series options. If the axis - * has categories, these will be used. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 2. An array of arrays with 2 values. In this case, the values correspond to - * `x,y`. If the first value is a string, it is applied as the name of the - * point, and the `x` value is inferred. - * ```js - * data: [ - * [0, 0], - * [1, 8], - * [2, 9] - * ] - * ``` - * - * 3. 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.scatter.turboThreshold), this option is not - * available. - * ```js - * data: [{ - * x: 1, - * y: 2, - * name: "Point2", - * color: "#00FF00" - * }, { - * x: 1, - * y: 4, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects - * - * @type {Array<number|Array<(number|string),(number|null)>|null|*>} - * @extends series.line.data - * @product highcharts highstock - * @apioption series.scatter.data - */ - ''; // adds doclets above to transpilat - - }); - _registerModule(_modules, 'Mixins/CenteredSeries.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * This option allows grouping series in a stacked chart. The stack option + * can be a string or anything else, as long as the grouped series' stack + * options match each other after conversion into a string. * - * */ - /** - * @private - * @interface Highcharts.RadianAngles - */ /** - * @name Highcharts.RadianAngles#end - * @type {number} - */ /** - * @name Highcharts.RadianAngles#start - * @type {number} - */ - var isNumber = U.isNumber, - pick = U.pick, - relativeLength = U.relativeLength; - var deg2rad = H.deg2rad; - /* eslint-disable valid-jsdoc */ - /** - * @private - * @mixin Highcharts.CenteredSeriesMixin - */ - var centeredSeriesMixin = H.CenteredSeriesMixin = { - /** - * Get the center of the pie based on the size and center options relative - * to the plot area. Borrowed by the polar and gauge series types. - * - * @private - * @function Highcharts.CenteredSeriesMixin.getCenter - * - * @return {Array<number>} - */ - getCenter: function () { - var options = this.options, - chart = this.chart, - slicingRoom = 2 * (options.slicedOffset || 0), - handleSlicingRoom, - plotWidth = chart.plotWidth - 2 * slicingRoom, - plotHeight = chart.plotHeight - 2 * slicingRoom, - centerOption = options.center, - smallestSize = Math.min(plotWidth, - plotHeight), - size = options.size, - innerSize = options.innerSize || 0, - positions, - i, - value; - if (typeof size === 'string') { - size = parseFloat(size); - } - if (typeof innerSize === 'string') { - innerSize = parseFloat(innerSize); - } - positions = [ - pick(centerOption[0], '50%'), - pick(centerOption[1], '50%'), - // Prevent from negative values - pick(size && size < 0 ? void 0 : options.size, '100%'), - pick(innerSize && innerSize < 0 ? void 0 : options.innerSize || 0, '0%') - ]; - // No need for inner size in angular (gauges) series but still required - // for pie series - if (chart.angular && !(this instanceof H.Series)) { - positions[3] = 0; - } - for (i = 0; i < 4; ++i) { - value = positions[i]; - handleSlicingRoom = i < 2 || (i === 2 && /%$/.test(value)); - // i == 0: centerX, relative to width - // i == 1: centerY, relative to height - // i == 2: size, relative to smallestSize - // i == 3: innerSize, relative to size - positions[i] = relativeLength(value, [plotWidth, plotHeight, smallestSize, positions[2]][i]) + (handleSlicingRoom ? slicingRoom : 0); - } - // innerSize cannot be larger than size (#3632) - if (positions[3] > positions[2]) { - positions[3] = positions[2]; - } - return positions; - }, - /** - * getStartAndEndRadians - Calculates start and end angles in radians. - * Used in series types such as pie and sunburst. - * - * @private - * @function Highcharts.CenteredSeriesMixin.getStartAndEndRadians - * - * @param {number} [start] - * Start angle in degrees. - * - * @param {number} [end] - * Start angle in degrees. - * - * @return {Highcharts.RadianAngles} - * Returns an object containing start and end angles as radians. - */ - getStartAndEndRadians: function (start, end) { - var startAngle = isNumber(start) ? start : 0, // must be a number - endAngle = ((isNumber(end) && // must be a number - end > startAngle && // must be larger than the start angle - // difference must be less than 360 degrees - (end - startAngle) < 360) ? - end : - startAngle + 360), - correction = -90; - return { - start: deg2rad * (startAngle + correction), - end: deg2rad * (endAngle + correction) - }; - } - }; - - return centeredSeriesMixin; - }); - _registerModule(_modules, 'Series/PieSeries.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Series/Series.js'], _modules['Mixins/CenteredSeries.js'], _modules['Core/Globals.js'], _modules['Mixins/LegendSymbol.js'], _modules['Series/LineSeries.js'], _modules['Core/Series/Point.js'], _modules['Core/Renderer/SVG/SVGRenderer.js'], _modules['Core/Utilities.js']], function (A, BaseSeries, CenteredSeriesMixin, H, LegendSymbolMixin, LineSeries, Point, SVGRenderer, U) { - /* * + * @sample {highcharts} highcharts/series/stack/ + * Stacked and grouped columns * - * (c) 2010-2020 Torstein Honsi + * @type {number|string} + * @since 2.1 + * @product highcharts highstock + * @apioption series.stack + */ + /** + * The type of series, for example `line` or `column`. By default, the + * series type is inherited from [chart.type](#chart.type), so unless the + * chart is a combination of series types, there is no need to set it on the + * series level. * - * License: www.highcharts.com/license + * @sample {highcharts} highcharts/series/type/ + * Line and column in the same chart + * @sample highcharts/series/type-dynamic/ + * Dynamic types with button selector + * @sample {highmaps} maps/demo/mapline-mappoint/ + * Multiple types in the same map * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @type {string} + * @apioption series.type + */ + /** + * When using dual or multiple x axes, this number defines which xAxis the + * particular series is connected to. It refers to either the + * {@link #xAxis.id|axis id} + * or the index of the axis in the xAxis array, with 0 being the first. * - * */ - var setAnimation = A.setAnimation; - var getStartAndEndRadians = CenteredSeriesMixin.getStartAndEndRadians; - var noop = H.noop; - var addEvent = U.addEvent, - clamp = U.clamp, - defined = U.defined, - fireEvent = U.fireEvent, - isNumber = U.isNumber, - merge = U.merge, - pick = U.pick, - relativeLength = U.relativeLength; - /** - * Pie series type. + * @type {number|string} + * @default 0 + * @product highcharts highstock + * @apioption series.xAxis + */ + /** + * When using dual or multiple y axes, this number defines which yAxis the + * particular series is connected to. It refers to either the + * {@link #yAxis.id|axis id} + * or the index of the axis in the yAxis array, with 0 being the first. * - * @private - * @class - * @name Highcharts.seriesTypes.pie + * @sample {highcharts} highcharts/series/yaxis/ + * Apply the column series to the secondary Y axis * - * @augments Highcharts.Series + * @type {number|string} + * @default 0 + * @product highcharts highstock + * @apioption series.yAxis */ - BaseSeries.seriesType('pie', 'line', /** - * A pie chart is a circular graphic which is divided into slices to - * illustrate numerical proportion. + * Define the visual z index of the series. * - * @sample highcharts/demo/pie-basic/ - * Pie chart + * @sample {highcharts} highcharts/plotoptions/series-zindex-default/ + * With no z index, the series defined last are on top + * @sample {highcharts} highcharts/plotoptions/series-zindex/ + * With a z index, the series with the highest z index is on top + * @sample {highstock} highcharts/plotoptions/series-zindex-default/ + * With no z index, the series defined last are on top + * @sample {highstock} highcharts/plotoptions/series-zindex/ + * With a z index, the series with the highest z index is on top * - * @extends plotOptions.line - * @excluding animationLimit, boostThreshold, connectEnds, connectNulls, - * cropThreshold, dashStyle, dataSorting, dragDrop, - * findNearestPointBy, getExtremesFromAll, label, lineWidth, - * marker, negativeColor, pointInterval, pointIntervalUnit, - * pointPlacement, pointStart, softThreshold, stacking, step, - * threshold, turboThreshold, zoneAxis, zones, dataSorting, - * boostBlending - * @product highcharts - * @optionparent plotOptions.pie + * @type {number} + * @product highcharts highstock + * @apioption series.zIndex + */ + void 0, + /** + * General options for all series types. + * + * @optionparent plotOptions.series */ { - /** - * @excluding legendItemClick - * @apioption plotOptions.pie.events + /** + * The SVG value used for the `stroke-linecap` and `stroke-linejoin` + * of a line graph. Round means that lines are rounded in the ends and + * bends. + * + * @type {Highcharts.SeriesLinecapValue} + * @default round + * @since 3.0.7 + * @apioption plotOptions.line.linecap + */ + /** + * Pixel width of the graph line. + * + * @see In styled mode, the line stroke-width can be set with the + * `.highcharts-graph` class name. + * + * @sample {highcharts} highcharts/plotoptions/series-linewidth-general/ + * On all series + * @sample {highcharts} highcharts/plotoptions/series-linewidth-specific/ + * On one single series + * + * @product highcharts highstock + * + * @private + */ + lineWidth: 2, + /** + * For some series, there is a limit that shuts down initial animation + * by default when the total number of points in the chart is too high. + * For example, for a column chart and its derivatives, animation does + * not run if there is more than 250 points totally. To disable this + * cap, set `animationLimit` to `Infinity`. + * + * @type {number} + * @apioption plotOptions.series.animationLimit + */ + /** + * Allow this series' points to be selected by clicking on the graphic + * (columns, point markers, pie slices, map areas etc). + * + * The selected points can be handled by point select and unselect + * events, or collectively by the [getSelectedPoints + * ](/class-reference/Highcharts.Chart#getSelectedPoints) function. + * + * And alternative way of selecting points is through dragging. + * + * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-line/ + * Line + * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-column/ + * Column + * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-pie/ + * Pie + * @sample {highcharts} highcharts/chart/events-selection-points/ + * Select a range of points through a drag selection + * @sample {highmaps} maps/plotoptions/series-allowpointselect/ + * Map area + * @sample {highmaps} maps/plotoptions/mapbubble-allowpointselect/ + * Map bubble + * + * @since 1.2.0 + * + * @private + */ + allowPointSelect: false, + /** + * When true, each point or column edge is rounded to its nearest pixel + * in order to render sharp on screen. In some cases, when there are a + * lot of densely packed columns, this leads to visible difference + * in column widths or distance between columns. In these cases, + * setting `crisp` to `false` may look better, even though each column + * is rendered blurry. + * + * @sample {highcharts} highcharts/plotoptions/column-crisp-false/ + * Crisp is false + * + * @since 5.0.10 + * @product highcharts highstock gantt + * + * @private + */ + crisp: true, + /** + * If true, a checkbox is displayed next to the legend item to allow + * selecting the series. The state of the checkbox is determined by + * the `selected` option. + * + * @productdesc {highmaps} + * Note that if a `colorAxis` is defined, the color axis is represented + * in the legend, not the series. + * + * @sample {highcharts} highcharts/plotoptions/series-showcheckbox-true/ + * Show select box + * + * @since 1.2.0 + * + * @private + */ + showCheckbox: false, + /** + * Enable or disable the initial animation when a series is displayed. + * The animation can also be set as a configuration object. Please + * note that this option only applies to the initial animation of the + * series itself. For other animations, see [chart.animation]( + * #chart.animation) and the animation parameter under the API methods. + * The following properties are supported: + * + * - `defer`: The animation delay time in milliseconds. + * + * - `duration`: The duration of the animation in milliseconds. + * + * - `easing`: Can be a string reference to an easing function set on + * the `Math` object or a function. See the _Custom easing function_ + * demo below. + * + * Due to poor performance, animation is disabled in old IE browsers + * for several chart types. + * + * @sample {highcharts} highcharts/plotoptions/series-animation-disabled/ + * Animation disabled + * @sample {highcharts} highcharts/plotoptions/series-animation-slower/ + * Slower animation + * @sample {highcharts} highcharts/plotoptions/series-animation-easing/ + * Custom easing function + * @sample {highstock} stock/plotoptions/animation-slower/ + * Slower animation + * @sample {highstock} stock/plotoptions/animation-easing/ + * Custom easing function + * @sample {highmaps} maps/plotoptions/series-animation-true/ + * Animation enabled on map series + * @sample {highmaps} maps/plotoptions/mapbubble-animation-false/ + * Disabled on mapbubble series + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + * @default {highcharts} true + * @default {highstock} true + * @default {highmaps} false + * + * @private + */ + animation: { + /** @internal */ + duration: 1000, + }, + /** + * @default 0 + * @type {number} + * @since 8.2.0 + * @apioption plotOptions.series.animation.defer + */ + /** + * An additional class name to apply to the series' graphical elements. + * This option does not replace default class names of the graphical + * element. + * + * @type {string} + * @since 5.0.0 + * @apioption plotOptions.series.className + */ + /** + * Disable this option to allow series rendering in the whole plotting + * area. + * + * **Note:** Clipping should be always enabled when + * [chart.zoomType](#chart.zoomType) is set + * + * @sample {highcharts} highcharts/plotoptions/series-clip/ + * Disabled clipping + * + * @default true + * @type {boolean} + * @since 3.0.0 + * @apioption plotOptions.series.clip + */ + /** + * The main color of the series. In line type series it applies to the + * line and the point markers unless otherwise specified. In bar type + * series it applies to the bars unless a color is specified per point. + * The default value is pulled from the `options.colors` array. + * + * In styled mode, the color can be defined by the + * [colorIndex](#plotOptions.series.colorIndex) option. Also, the series + * color can be set with the `.highcharts-series`, + * `.highcharts-color-{n}`, `.highcharts-{type}-series` or + * `.highcharts-series-{n}` class, or individual classes given by the + * `className` option. + * + * @productdesc {highmaps} + * In maps, the series color is rarely used, as most choropleth maps use + * the color to denote the value of each point. The series color can + * however be used in a map with multiple series holding categorized + * data. + * + * @sample {highcharts} highcharts/plotoptions/series-color-general/ + * General plot option + * @sample {highcharts} highcharts/plotoptions/series-color-specific/ + * One specific series + * @sample {highcharts} highcharts/plotoptions/series-color-area/ + * Area color + * @sample {highcharts} highcharts/series/infographic/ + * Pattern fill + * @sample {highmaps} maps/demo/category-map/ + * Category map by multiple series + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @apioption plotOptions.series.color + */ + /** + * Styled mode only. A specific color index to use for the series, so + * its graphic representations are given the class name + * `highcharts-color-{n}`. + * + * @type {number} + * @since 5.0.0 + * @apioption plotOptions.series.colorIndex + */ + /** + * Whether to connect a graph line across null points, or render a gap + * between the two points on either side of the null. + * + * @sample {highcharts} highcharts/plotoptions/series-connectnulls-false/ + * False by default + * @sample {highcharts} highcharts/plotoptions/series-connectnulls-true/ + * True + * + * @type {boolean} + * @default false + * @product highcharts highstock + * @apioption plotOptions.series.connectNulls + */ + /** + * You can set the cursor to "pointer" if you have click events attached + * to the series, to signal to the user that the points and lines can + * be clicked. + * + * In styled mode, the series cursor can be set with the same classes + * as listed under [series.color](#plotOptions.series.color). + * + * @sample {highcharts} highcharts/plotoptions/series-cursor-line/ + * On line graph + * @sample {highcharts} highcharts/plotoptions/series-cursor-column/ + * On columns + * @sample {highcharts} highcharts/plotoptions/series-cursor-scatter/ + * On scatter markers + * @sample {highstock} stock/plotoptions/cursor/ + * Pointer on a line graph + * @sample {highmaps} maps/plotoptions/series-allowpointselect/ + * Map area + * @sample {highmaps} maps/plotoptions/mapbubble-allowpointselect/ + * Map bubble + * + * @type {string|Highcharts.CursorValue} + * @apioption plotOptions.series.cursor + */ + /** + * A reserved subspace to store options and values for customized + * functionality. Here you can add additional data for your own event + * callbacks and formatter callbacks. + * + * @sample {highcharts} highcharts/point/custom/ + * Point and series with custom data + * + * @type {Highcharts.Dictionary<*>} + * @apioption plotOptions.series.custom + */ + /** + * Name of the dash style to use for the graph, or for some series types + * the outline of each shape. + * + * In styled mode, the + * [stroke dash-array](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/css/series-dashstyle/) + * can be set with the same classes as listed under + * [series.color](#plotOptions.series.color). + * + * @sample {highcharts} highcharts/plotoptions/series-dashstyle-all/ + * Possible values demonstrated + * @sample {highcharts} highcharts/plotoptions/series-dashstyle/ + * Chart suitable for printing in black and white + * @sample {highstock} highcharts/plotoptions/series-dashstyle-all/ + * Possible values demonstrated + * @sample {highmaps} highcharts/plotoptions/series-dashstyle-all/ + * Possible values demonstrated + * @sample {highmaps} maps/plotoptions/series-dashstyle/ + * Dotted borders on a map + * + * @type {Highcharts.DashStyleValue} + * @default Solid + * @since 2.1 + * @apioption plotOptions.series.dashStyle + */ + /** + * A description of the series to add to the screen reader information + * about the series. + * + * @type {string} + * @since 5.0.0 + * @requires modules/accessibility + * @apioption plotOptions.series.description + */ + /** + * Options for the series data sorting. + * + * @type {Highcharts.DataSortingOptionsObject} + * @since 8.0.0 + * @product highcharts highstock + * @apioption plotOptions.series.dataSorting + */ + /** + * Enable or disable data sorting for the series. Use [xAxis.reversed]( + * #xAxis.reversed) to change the sorting order. + * + * @sample {highcharts} highcharts/datasorting/animation/ + * Data sorting in scatter-3d + * @sample {highcharts} highcharts/datasorting/labels-animation/ + * Axis labels animation + * @sample {highcharts} highcharts/datasorting/dependent-sorting/ + * Dependent series sorting + * @sample {highcharts} highcharts/datasorting/independent-sorting/ + * Independent series sorting + * + * @type {boolean} + * @since 8.0.0 + * @apioption plotOptions.series.dataSorting.enabled + */ + /** + * Whether to allow matching points by name in an update. If this option + * is disabled, points will be matched by order. + * + * @sample {highcharts} highcharts/datasorting/match-by-name/ + * Enabled match by name + * + * @type {boolean} + * @since 8.0.0 + * @apioption plotOptions.series.dataSorting.matchByName + */ + /** + * Determines what data value should be used to sort by. + * + * @sample {highcharts} highcharts/datasorting/sort-key/ + * Sort key as `z` value + * + * @type {string} + * @since 8.0.0 + * @default y + * @apioption plotOptions.series.dataSorting.sortKey + */ + /** + * Enable or disable the mouse tracking for a specific series. This + * includes point tooltips and click events on graphs and points. For + * large datasets it improves performance. + * + * @sample {highcharts} highcharts/plotoptions/series-enablemousetracking-false/ + * No mouse tracking + * @sample {highmaps} maps/plotoptions/series-enablemousetracking-false/ + * No mouse tracking + * + * @type {boolean} + * @default true + * @apioption plotOptions.series.enableMouseTracking + */ + /** + * Whether to use the Y extremes of the total chart width or only the + * zoomed area when zooming in on parts of the X axis. By default, the + * Y axis adjusts to the min and max of the visible data. Cartesian + * series only. + * + * @type {boolean} + * @default false + * @since 4.1.6 + * @product highcharts highstock gantt + * @apioption plotOptions.series.getExtremesFromAll + */ + /** + * An array specifying which option maps to which key in the data point + * array. This makes it convenient to work with unstructured data arrays + * from different sources. + * + * @see [series.data](#series.line.data) + * + * @sample {highcharts|highstock} highcharts/series/data-keys/ + * An extended data array with keys + * @sample {highcharts|highstock} highcharts/series/data-nested-keys/ + * Nested keys used to access object properties + * + * @type {Array<string>} + * @since 4.1.6 + * @apioption plotOptions.series.keys + */ + /** + * The line cap used for line ends and line joins on the graph. + * + * @type {Highcharts.SeriesLinecapValue} + * @default round + * @product highcharts highstock + * @apioption plotOptions.series.linecap + */ + /** + * The [id](#series.id) of another series to link to. Additionally, + * the value can be ":previous" to link to the previous series. When + * two series are linked, only the first one appears in the legend. + * Toggling the visibility of this also toggles the linked series. + * + * If master series uses data sorting and linked series does not have + * its own sorting definition, the linked series will be sorted in the + * same order as the master one. + * + * @sample {highcharts|highstock} highcharts/demo/arearange-line/ + * Linked series + * + * @type {string} + * @since 3.0 + * @product highcharts highstock gantt + * @apioption plotOptions.series.linkedTo + */ + /** + * Options for the corresponding navigator series if `showInNavigator` + * is `true` for this series. Available options are the same as any + * series, documented at [plotOptions](#plotOptions.series) and + * [series](#series). + * + * These options are merged with options in [navigator.series]( + * #navigator.series), and will take precedence if the same option is + * defined both places. + * + * @see [navigator.series](#navigator.series) + * + * @type {Highcharts.PlotSeriesOptions} + * @since 5.0.0 + * @product highstock + * @apioption plotOptions.series.navigatorOptions + */ + /** + * The color for the parts of the graph or points that are below the + * [threshold](#plotOptions.series.threshold). Note that `zones` takes + * precedence over the negative color. Using `negativeColor` is + * equivalent to applying a zone with value of 0. + * + * @see In styled mode, a negative color is applied by setting this option + * to `true` combined with the `.highcharts-negative` class name. + * + * @sample {highcharts} highcharts/plotoptions/series-negative-color/ + * Spline, area and column + * @sample {highcharts} highcharts/plotoptions/arearange-negativecolor/ + * Arearange + * @sample {highcharts} highcharts/css/series-negative-color/ + * Styled mode + * @sample {highstock} highcharts/plotoptions/series-negative-color/ + * Spline, area and column + * @sample {highstock} highcharts/plotoptions/arearange-negativecolor/ + * Arearange + * @sample {highmaps} highcharts/plotoptions/series-negative-color/ + * Spline, area and column + * @sample {highmaps} highcharts/plotoptions/arearange-negativecolor/ + * Arearange + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 3.0 + * @apioption plotOptions.series.negativeColor + */ + /** + * Same as + * [accessibility.pointDescriptionFormatter](#accessibility.pointDescriptionFormatter), + * but for an individual series. Overrides the chart wide configuration. + * + * @type {Function} + * @since 5.0.12 + * @apioption plotOptions.series.pointDescriptionFormatter + */ + /** + * If no x values are given for the points in a series, `pointInterval` + * defines the interval of the x values. For example, if a series + * contains one value every decade starting from year 0, set + * `pointInterval` to `10`. In true `datetime` axes, the `pointInterval` + * is set in milliseconds. + * + * It can be also be combined with `pointIntervalUnit` to draw irregular + * time intervals. + * + * Please note that this options applies to the _series data_, not the + * interval of the axis ticks, which is independent. + * + * @sample {highcharts} highcharts/plotoptions/series-pointstart-datetime/ + * Datetime X axis + * @sample {highstock} stock/plotoptions/pointinterval-pointstart/ + * Using pointStart and pointInterval + * + * @type {number} + * @default 1 + * @product highcharts highstock gantt + * @apioption plotOptions.series.pointInterval + */ + /** + * On datetime series, this allows for setting the + * [pointInterval](#plotOptions.series.pointInterval) to irregular time + * units, `day`, `month` and `year`. A day is usually the same as 24 + * hours, but `pointIntervalUnit` also takes the DST crossover into + * consideration when dealing with local time. Combine this option with + * `pointInterval` to draw weeks, quarters, 6 months, 10 years etc. + * + * Please note that this options applies to the _series data_, not the + * interval of the axis ticks, which is independent. + * + * @sample {highcharts} highcharts/plotoptions/series-pointintervalunit/ + * One point a month + * @sample {highstock} highcharts/plotoptions/series-pointintervalunit/ + * One point a month + * + * @type {string} + * @since 4.1.0 + * @product highcharts highstock gantt + * @validvalue ["day", "month", "year"] + * @apioption plotOptions.series.pointIntervalUnit + */ + /** + * Possible values: `"on"`, `"between"`, `number`. + * + * In a column chart, when pointPlacement is `"on"`, the point will not + * create any padding of the X axis. In a polar column chart this means + * that the first column points directly north. If the pointPlacement is + * `"between"`, the columns will be laid out between ticks. This is + * useful for example for visualising an amount between two points in + * time or in a certain sector of a polar chart. + * + * Since Highcharts 3.0.2, the point placement can also be numeric, + * where 0 is on the axis value, -0.5 is between this value and the + * previous, and 0.5 is between this value and the next. Unlike the + * textual options, numeric point placement options won't affect axis + * padding. + * + * Note that pointPlacement needs a [pointRange]( + * #plotOptions.series.pointRange) to work. For column series this is + * computed, but for line-type series it needs to be set. + * + * For the `xrange` series type and gantt charts, if the Y axis is a + * category axis, the `pointPlacement` applies to the Y axis rather than + * the (typically datetime) X axis. + * + * Defaults to `undefined` in cartesian charts, `"between"` in polar + * charts. + * + * @see [xAxis.tickmarkPlacement](#xAxis.tickmarkPlacement) + * + * @sample {highcharts|highstock} highcharts/plotoptions/series-pointplacement-between/ + * Between in a column chart + * @sample {highcharts|highstock} highcharts/plotoptions/series-pointplacement-numeric/ + * Numeric placement for custom layout + * @sample {highcharts|highstock} maps/plotoptions/heatmap-pointplacement/ + * Placement in heatmap + * + * @type {string|number} + * @since 2.3.0 + * @product highcharts highstock gantt + * @apioption plotOptions.series.pointPlacement + */ + /** + * If no x values are given for the points in a series, pointStart + * defines on what value to start. For example, if a series contains one + * yearly value starting from 1945, set pointStart to 1945. + * + * @sample {highcharts} highcharts/plotoptions/series-pointstart-linear/ + * Linear + * @sample {highcharts} highcharts/plotoptions/series-pointstart-datetime/ + * Datetime + * @sample {highstock} stock/plotoptions/pointinterval-pointstart/ + * Using pointStart and pointInterval + * + * @type {number} + * @default 0 + * @product highcharts highstock gantt + * @apioption plotOptions.series.pointStart + */ + /** + * Whether to select the series initially. If `showCheckbox` is true, + * the checkbox next to the series name in the legend will be checked + * for a selected series. + * + * @sample {highcharts} highcharts/plotoptions/series-selected/ + * One out of two series selected + * + * @type {boolean} + * @default false + * @since 1.2.0 + * @apioption plotOptions.series.selected + */ + /** + * Whether to apply a drop shadow to the graph line. Since 2.3 the + * shadow can be an object configuration containing `color`, `offsetX`, + * `offsetY`, `opacity` and `width`. + * + * @sample {highcharts} highcharts/plotoptions/series-shadow/ + * Shadow enabled + * + * @type {boolean|Highcharts.ShadowOptionsObject} + * @default false + * @apioption plotOptions.series.shadow + */ + /** + * Whether to display this particular series or series type in the + * legend. Standalone series are shown in legend by default, and linked + * series are not. Since v7.2.0 it is possible to show series that use + * colorAxis by setting this option to `true`. + * + * @sample {highcharts} highcharts/plotoptions/series-showinlegend/ + * One series in the legend, one hidden + * + * @type {boolean} + * @apioption plotOptions.series.showInLegend + */ + /** + * Whether or not to show the series in the navigator. Takes precedence + * over [navigator.baseSeries](#navigator.baseSeries) if defined. + * + * @type {boolean} + * @since 5.0.0 + * @product highstock + * @apioption plotOptions.series.showInNavigator + */ + /** + * If set to `true`, the accessibility module will skip past the points + * in this series for keyboard navigation. + * + * @type {boolean} + * @since 5.0.12 + * @apioption plotOptions.series.skipKeyboardNavigation + */ + /** + * Whether to stack the values of each series on top of each other. + * Possible values are `undefined` to disable, `"normal"` to stack by + * value or `"percent"`. + * + * When stacking is enabled, data must be sorted + * in ascending X order. + * + * Some stacking options are related to specific series types. In the + * streamgraph series type, the stacking option is set to `"stream"`. + * The second one is `"overlap"`, which only applies to waterfall + * series. + * + * @see [yAxis.reversedStacks](#yAxis.reversedStacks) + * + * @sample {highcharts} highcharts/plotoptions/series-stacking-line/ + * Line + * @sample {highcharts} highcharts/plotoptions/series-stacking-column/ + * Column + * @sample {highcharts} highcharts/plotoptions/series-stacking-bar/ + * Bar + * @sample {highcharts} highcharts/plotoptions/series-stacking-area/ + * Area + * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-line/ + * Line + * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-column/ + * Column + * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-bar/ + * Bar + * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-area/ + * Area + * @sample {highcharts} highcharts/plotoptions/series-waterfall-with-normal-stacking + * Waterfall with normal stacking + * @sample {highcharts} highcharts/plotoptions/series-waterfall-with-overlap-stacking + * Waterfall with overlap stacking + * @sample {highstock} stock/plotoptions/stacking/ + * Area + * + * @type {string} + * @product highcharts highstock + * @validvalue ["normal", "overlap", "percent", "stream"] + * @apioption plotOptions.series.stacking + */ + /** + * Whether to apply steps to the line. Possible values are `left`, + * `center` and `right`. + * + * @sample {highcharts} highcharts/plotoptions/line-step/ + * Different step line options + * @sample {highcharts} highcharts/plotoptions/area-step/ + * Stepped, stacked area + * @sample {highstock} stock/plotoptions/line-step/ + * Step line + * + * @type {string} + * @since 1.2.5 + * @product highcharts highstock + * @validvalue ["left", "center", "right"] + * @apioption plotOptions.series.step + */ + /** + * The threshold, also called zero level or base level. For line type + * series this is only used in conjunction with + * [negativeColor](#plotOptions.series.negativeColor). + * + * @see [softThreshold](#plotOptions.series.softThreshold). + * + * @type {number} + * @default 0 + * @since 3.0 + * @product highcharts highstock + * @apioption plotOptions.series.threshold + */ + /** + * Set the initial visibility of the series. + * + * @sample {highcharts} highcharts/plotoptions/series-visible/ + * Two series, one hidden and one visible + * @sample {highstock} stock/plotoptions/series-visibility/ + * Hidden series + * + * @type {boolean} + * @default true + * @apioption plotOptions.series.visible + */ + /** + * Defines the Axis on which the zones are applied. + * + * @see [zones](#plotOptions.series.zones) + * + * @sample {highcharts} highcharts/series/color-zones-zoneaxis-x/ + * Zones on the X-Axis + * @sample {highstock} highcharts/series/color-zones-zoneaxis-x/ + * Zones on the X-Axis + * + * @type {string} + * @default y + * @since 4.1.0 + * @product highcharts highstock + * @apioption plotOptions.series.zoneAxis + */ + /** + * General event handlers for the series items. These event hooks can + * also be attached to the series at run time using the + * `Highcharts.addEvent` function. + * + * @declare Highcharts.SeriesEventsOptionsObject + * + * @private + */ + events: {}, + /** + * Fires after the series has finished its initial animation, or in case + * animation is disabled, immediately as the series is displayed. + * + * @sample {highcharts} highcharts/plotoptions/series-events-afteranimate/ + * Show label after animate + * @sample {highstock} highcharts/plotoptions/series-events-afteranimate/ + * Show label after animate + * + * @type {Highcharts.SeriesAfterAnimateCallbackFunction} + * @since 4.0 + * @product highcharts highstock gantt + * @context Highcharts.Series + * @apioption plotOptions.series.events.afterAnimate + */ + /** + * Fires when the checkbox next to the series' name in the legend is + * clicked. One parameter, `event`, is passed to the function. The state + * of the checkbox is found by `event.checked`. The checked item is + * found by `event.item`. Return `false` to prevent the default action + * which is to toggle the select state of the series. + * + * @sample {highcharts} highcharts/plotoptions/series-events-checkboxclick/ + * Alert checkbox status + * + * @type {Highcharts.SeriesCheckboxClickCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Series + * @apioption plotOptions.series.events.checkboxClick + */ + /** + * Fires when the series is clicked. One parameter, `event`, is passed + * to the function, containing common event information. Additionally, + * `event.point` holds a pointer to the nearest point on the graph. + * + * @sample {highcharts} highcharts/plotoptions/series-events-click/ + * Alert click info + * @sample {highstock} stock/plotoptions/series-events-click/ + * Alert click info + * @sample {highmaps} maps/plotoptions/series-events-click/ + * Display click info in subtitle + * + * @type {Highcharts.SeriesClickCallbackFunction} + * @context Highcharts.Series + * @apioption plotOptions.series.events.click + */ + /** + * Fires when the series is hidden after chart generation time, either + * by clicking the legend item or by calling `.hide()`. + * + * @sample {highcharts} highcharts/plotoptions/series-events-hide/ + * Alert when the series is hidden by clicking the legend item + * + * @type {Highcharts.SeriesHideCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Series + * @apioption plotOptions.series.events.hide + */ + /** + * Fires when the legend item belonging to the series is clicked. One + * parameter, `event`, is passed to the function. The default action + * is to toggle the visibility of the series. This can be prevented + * by returning `false` or calling `event.preventDefault()`. + * + * @sample {highcharts} highcharts/plotoptions/series-events-legenditemclick/ + * Confirm hiding and showing + * + * @type {Highcharts.SeriesLegendItemClickCallbackFunction} + * @context Highcharts.Series + * @apioption plotOptions.series.events.legendItemClick + */ + /** + * Fires when the mouse leaves the graph. One parameter, `event`, is + * passed to the function, containing common event information. If the + * [stickyTracking](#plotOptions.series) option is true, `mouseOut` + * doesn't happen before the mouse enters another graph or leaves the + * plot area. + * + * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-sticky/ + * With sticky tracking by default + * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-no-sticky/ + * Without sticky tracking + * + * @type {Highcharts.SeriesMouseOutCallbackFunction} + * @context Highcharts.Series + * @apioption plotOptions.series.events.mouseOut + */ + /** + * Fires when the mouse enters the graph. One parameter, `event`, is + * passed to the function, containing common event information. + * + * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-sticky/ + * With sticky tracking by default + * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-no-sticky/ + * Without sticky tracking + * + * @type {Highcharts.SeriesMouseOverCallbackFunction} + * @context Highcharts.Series + * @apioption plotOptions.series.events.mouseOver + */ + /** + * Fires when the series is shown after chart generation time, either + * by clicking the legend item or by calling `.show()`. + * + * @sample {highcharts} highcharts/plotoptions/series-events-show/ + * Alert when the series is shown by clicking the legend item. + * + * @type {Highcharts.SeriesShowCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Series + * @apioption plotOptions.series.events.show + */ + /** + * Options for the point markers of line-like series. Properties like + * `fillColor`, `lineColor` and `lineWidth` define the visual appearance + * of the markers. Other series types, like column series, don't have + * markers, but have visual options on the series level instead. + * + * In styled mode, the markers can be styled with the + * `.highcharts-point`, `.highcharts-point-hover` and + * `.highcharts-point-select` class names. + * + * @declare Highcharts.PointMarkerOptionsObject + * + * @private + */ + marker: { + /** + * Enable or disable the point marker. If `undefined`, the markers + * are hidden when the data is dense, and shown for more widespread + * data points. + * + * @sample {highcharts} highcharts/plotoptions/series-marker-enabled/ + * Disabled markers + * @sample {highcharts} highcharts/plotoptions/series-marker-enabled-false/ + * Disabled in normal state but enabled on hover + * @sample {highstock} stock/plotoptions/series-marker/ + * Enabled markers + * + * @type {boolean} + * @default {highcharts} undefined + * @default {highstock} false + * @apioption plotOptions.series.marker.enabled */ /** - * Fires when the checkbox next to the point name in the legend is - * clicked. One parameter, event, is passed to the function. The state - * of the checkbox is found by event.checked. The checked item is found - * by event.item. Return false to prevent the default action which is to - * toggle the select state of the series. + * The threshold for how dense the point markers should be before + * they are hidden, given that `enabled` is not defined. The number + * indicates the horizontal distance between the two closest points + * in the series, as multiples of the `marker.radius`. In other + * words, the default value of 2 means points are hidden if + * overlapping horizontally. * - * @sample {highcharts} highcharts/plotoptions/series-events-checkboxclick/ - * Alert checkbox status + * @sample highcharts/plotoptions/series-marker-enabledthreshold + * A higher threshold * - * @type {Function} - * @since 1.2.0 - * @product highcharts - * @context Highcharts.Point - * @apioption plotOptions.pie.events.checkboxClick + * @since 6.0.5 */ + enabledThreshold: 2, /** - * Fires when the legend item belonging to the pie point (slice) is - * clicked. The `this` keyword refers to the point itself. One - * parameter, `event`, is passed to the function, containing common - * event information. The default action is to toggle the visibility of - * the point. This can be prevented by calling `event.preventDefault()`. + * The fill color of the point marker. When `undefined`, the series' + * or point's color is used. * - * @sample {highcharts} highcharts/plotoptions/pie-point-events-legenditemclick/ - * Confirm toggle visibility + * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/ + * White fill * - * @type {Highcharts.PointLegendItemClickCallbackFunction} - * @since 1.2.0 - * @product highcharts - * @apioption plotOptions.pie.point.events.legendItemClick + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @apioption plotOptions.series.marker.fillColor */ /** - * The center of the pie chart relative to the plot area. Can be - * percentages or pixel values. The default behaviour (as of 3.0) is to - * center the pie so that all slices and data labels are within the plot - * area. As a consequence, the pie may actually jump around in a chart - * with dynamic values, as the data labels move. In that case, the - * center should be explicitly set, for example to `["50%", "50%"]`. - * - * @sample {highcharts} highcharts/plotoptions/pie-center/ - * Centered at 100, 100 - * - * @type {Array<(number|string|null),(number|string|null)>} - * @default [null, null] - * @product highcharts + * Image markers only. Set the image width explicitly. When using + * this option, a `width` must also be set. * - * @private + * @sample {highcharts} highcharts/plotoptions/series-marker-width-height/ + * Fixed width and height + * @sample {highstock} highcharts/plotoptions/series-marker-width-height/ + * Fixed width and height + * + * @type {number} + * @since 4.0.4 + * @apioption plotOptions.series.marker.height */ - center: [null, null], /** - * The color of the pie series. A pie series is represented as an empty - * circle if the total sum of its values is 0. Use this property to - * define the color of its border. + * The color of the point marker's outline. When `undefined`, the + * series' or point's color is used. * - * In styled mode, the color can be defined by the - * [colorIndex](#plotOptions.series.colorIndex) option. Also, the series - * color can be set with the `.highcharts-series`, - * `.highcharts-color-{n}`, `.highcharts-{type}-series` or - * `.highcharts-series-{n}` class, or individual classes given by the - * `className` option. + * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/ + * Inherit from series color (undefined) * - * @sample {highcharts} highcharts/plotoptions/pie-emptyseries/ - * Empty pie series + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + */ + lineColor: "#ffffff", + /** + * The width of the point marker's outline. * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @default #cccccc - * @apioption plotOptions.pie.color + * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/ + * 2px blue marker */ + lineWidth: 0, /** - * @product highcharts + * The radius of the point marker. * - * @private + * @sample {highcharts} highcharts/plotoptions/series-marker-radius/ + * Bigger markers + * + * @default {highstock} 2 */ - clip: false, + radius: 4, /** - * @ignore-option + * A predefined shape or symbol for the marker. When undefined, the + * symbol is pulled from options.symbols. Other possible values are + * `'circle'`, `'square'`,`'diamond'`, `'triangle'` and + * `'triangle-down'`. * - * @private + * Additionally, the URL to a graphic can be given on this form: + * `'url(graphic.png)'`. Note that for the image to be applied to + * exported charts, its URL needs to be accessible by the export + * server. + * + * Custom callbacks for symbol path generation can also be added to + * `Highcharts.SVGRenderer.prototype.symbols`. The callback is then + * used by its method name, as shown in the demo. + * + * @sample {highcharts} highcharts/plotoptions/series-marker-symbol/ + * Predefined, graphic and custom markers + * @sample {highstock} highcharts/plotoptions/series-marker-symbol/ + * Predefined, graphic and custom markers + * + * @type {string} + * @apioption plotOptions.series.marker.symbol */ - colorByPoint: true, /** - * A series specific or series type specific color set to use instead - * of the global [colors](#colors). + * Image markers only. Set the image width explicitly. When using + * this option, a `height` must also be set. * - * @sample {highcharts} highcharts/demo/pie-monochrome/ - * Set default colors for all pies + * @sample {highcharts} highcharts/plotoptions/series-marker-width-height/ + * Fixed width and height + * @sample {highstock} highcharts/plotoptions/series-marker-width-height/ + * Fixed width and height * - * @type {Array<Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject>} - * @since 3.0 - * @product highcharts - * @apioption plotOptions.pie.colors + * @type {number} + * @since 4.0.4 + * @apioption plotOptions.series.marker.width */ /** - * @declare Highcharts.SeriesPieDataLabelsOptionsObject - * @extends plotOptions.series.dataLabels - * @excluding align, allowOverlap, inside, staggerLines, step - * @private + * States for a single point marker. + * + * @declare Highcharts.PointStatesOptionsObject */ - dataLabels: { + states: { + /** + * The normal state of a single point marker. Currently only + * used for setting animation when returning to normal state + * from hover. + * + * @declare Highcharts.PointStatesNormalOptionsObject + */ + normal: { + /** + * Animation when returning to normal state after hovering. + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + */ + animation: true, + }, + /** + * The hover state for a single point marker. + * + * @declare Highcharts.PointStatesHoverOptionsObject + */ + hover: { + /** + * Animation when hovering over the marker. + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + */ + animation: { + /** @internal */ + duration: 50, + }, /** - * Alignment method for data labels. Possible values are: - * - * - `toPlotEdges`: Each label touches the nearest vertical edge of - * the plot area. - * - * - `connectors`: Connectors have the same x position and the - * widest label of each half (left & right) touches the nearest - * vertical edge of the plot area. - * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-alignto-connectors/ - * alignTo: connectors - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-alignto-plotedges/ - * alignTo: plotEdges + * Enable or disable the point marker. * - * @type {string} - * @since 7.0.0 - * @product highcharts - * @apioption plotOptions.pie.dataLabels.alignTo + * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-enabled/ + * Disabled hover state */ - allowOverlap: true, + enabled: true, /** - * The color of the line connecting the data label to the pie slice. - * The default color is the same as the point's color. - * - * In styled mode, the connector stroke is given in the - * `.highcharts-data-label-connector` class. - * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorcolor/ - * Blue connectors - * @sample {highcharts} highcharts/css/pie-point/ - * Styled connectors + * The fill color of the marker in hover state. When + * `undefined`, the series' or point's fillColor for normal + * state is used. * * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @since 2.1 - * @product highcharts - * @apioption plotOptions.pie.dataLabels.connectorColor + * @apioption plotOptions.series.marker.states.hover.fillColor */ /** - * The distance from the data label to the connector. Note that - * data labels also have a default `padding`, so in order for the - * connector to touch the text, the `padding` must also be 0. + * The color of the point marker's outline. When + * `undefined`, the series' or point's lineColor for normal + * state is used. * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorpadding/ - * No padding + * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-linecolor/ + * White fill color, black line color * - * @since 2.1 - * @product highcharts + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @apioption plotOptions.series.marker.states.hover.lineColor */ - connectorPadding: 5, /** - * Specifies the method that is used to generate the connector path. - * Highcharts provides 3 built-in connector shapes: `'fixedOffset'` - * (default), `'straight'` and `'crookedLine'`. Using - * `'crookedLine'` has the most sense (in most of the cases) when - * `'alignTo'` is set. + * The width of the point marker's outline. When + * `undefined`, the series' or point's lineWidth for normal + * state is used. * - * Users can provide their own method by passing a function instead - * of a String. 3 arguments are passed to the callback: + * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-linewidth/ + * 3px line width * - * - Object that holds the information about the coordinates of the - * label (`x` & `y` properties) and how the label is located in - * relation to the pie (`alignment` property). `alignment` can by - * one of the following: - * `'left'` (pie on the left side of the data label), - * `'right'` (pie on the right side of the data label) or - * `'center'` (data label overlaps the pie). - * - * - Object that holds the information about the position of the - * connector. Its `touchingSliceAt` porperty tells the position - * of the place where the connector touches the slice. - * - * - Data label options - * - * The function has to return an SVG path definition in array form - * (see the example). + * @type {number} + * @apioption plotOptions.series.marker.states.hover.lineWidth + */ + /** + * The radius of the point marker. In hover state, it + * defaults to the normal state's radius + 2 as per the + * [radiusPlus](#plotOptions.series.marker.states.hover.radiusPlus) + * option. * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorshape-string/ - * connectorShape is a String - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorshape-function/ - * connectorShape is a function + * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-radius/ + * 10px radius * - * @type {string|Function} - * @since 7.0.0 - * @product highcharts + * @type {number} + * @apioption plotOptions.series.marker.states.hover.radius */ - connectorShape: 'fixedOffset', /** - * The width of the line connecting the data label to the pie slice. - * - * In styled mode, the connector stroke width is given in the - * `.highcharts-data-label-connector` class. + * The number of pixels to increase the radius of the + * hovered point. * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorwidth-disabled/ - * Disable the connector - * @sample {highcharts} highcharts/css/pie-point/ - * Styled connectors + * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/ + * 5 pixels greater radius on hover + * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/ + * 5 pixels greater radius on hover * - * @type {number} - * @default 1 - * @since 2.1 - * @product highcharts - * @apioption plotOptions.pie.dataLabels.connectorWidth + * @since 4.0.3 */ + radiusPlus: 2, /** - * Works only if `connectorShape` is `'crookedLine'`. It defines how - * far from the vertical plot edge the coonnector path should be - * crooked. + * The additional line width for a hovered point. * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-crookdistance/ - * crookDistance set to 90% + * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/ + * 2 pixels wider on hover + * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/ + * 2 pixels wider on hover * - * @since 7.0.0 - * @product highcharts + * @since 4.0.3 */ - crookDistance: '70%', + lineWidthPlus: 1, + }, + /** + * The appearance of the point marker when selected. In order to + * allow a point to be selected, set the + * `series.allowPointSelect` option to true. + * + * @declare Highcharts.PointStatesSelectOptionsObject + */ + select: { /** - * The distance of the data label from the pie's edge. Negative - * numbers put the data label on top of the pie slices. Can also be - * defined as a percentage of pie's radius. Connectors are only - * shown for data labels outside the pie. + * Enable or disable visible feedback for selection. * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-distance/ - * Data labels on top of the pie + * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-enabled/ + * Disabled select state * - * @type {number|string} - * @since 2.1 - * @product highcharts + * @type {boolean} + * @default true + * @apioption plotOptions.series.marker.states.select.enabled */ - distance: 30, - enabled: true, /** - * A - * [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) - * for the data label. Available variables are the same as for - * `formatter`. + * The radius of the point marker. In hover state, it + * defaults to the normal state's radius + 2. * - * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ - * Add a unit + * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-radius/ + * 10px radius for selected points * - * @type {string} - * @default undefined - * @since 3.0 - * @apioption plotOptions.pie.dataLabels.format + * @type {number} + * @apioption plotOptions.series.marker.states.select.radius */ - // eslint-disable-next-line valid-jsdoc /** - * Callback JavaScript function to format the data label. Note that - * if a `format` is defined, the format takes precedence and the - * formatter is ignored. + * The fill color of the point marker. * - * @type {Highcharts.DataLabelsFormatterCallbackFunction} - * @default function () { return this.point.isNull ? void 0 : this.point.name; } + * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-fillcolor/ + * Solid red discs for selected points + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} */ - formatter: function () { - return this.point.isNull ? void 0 : this.point.name; - }, + fillColor: "#cccccc", /** - * Whether to render the connector as a soft arc or a line with - * sharp break. Works only if `connectorShape` equals to - * `fixedOffset`. + * The color of the point marker's outline. When + * `undefined`, the series' or point's color is used. * - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-softconnector-true/ - * Soft - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-softconnector-false/ - * Non soft + * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-linecolor/ + * Red line color for selected points * - * @since 2.1.7 - * @product highcharts + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} */ - softConnector: true, + lineColor: "#000000", /** - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow - * Long labels truncated with an ellipsis - * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow-wrap - * Long labels are wrapped + * The width of the point marker's outline. * - * @type {Highcharts.CSSObject} - * @apioption plotOptions.pie.dataLabels.style + * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-linewidth/ + * 3px line width for selected points */ - x: 0 + lineWidth: 2, + }, }, + }, + /** + * Properties for each single point. + * + * @declare Highcharts.PlotSeriesPointOptions + * + * @private + */ + point: { + /** + * Fires when a point is clicked. One parameter, `event`, is passed + * to the function, containing common event information. + * + * If the `series.allowPointSelect` option is true, the default + * action for the point's click event is to toggle the point's + * select state. Returning `false` cancels this action. + * + * @sample {highcharts} highcharts/plotoptions/series-point-events-click/ + * Click marker to alert values + * @sample {highcharts} highcharts/plotoptions/series-point-events-click-column/ + * Click column + * @sample {highcharts} highcharts/plotoptions/series-point-events-click-url/ + * Go to URL + * @sample {highmaps} maps/plotoptions/series-point-events-click/ + * Click marker to display values + * @sample {highmaps} maps/plotoptions/series-point-events-click-url/ + * Go to URL + * + * @type {Highcharts.PointClickCallbackFunction} + * @context Highcharts.Point + * @apioption plotOptions.series.point.events.click + */ /** - * If the total sum of the pie's values is 0, the series is represented - * as an empty circle . The `fillColor` option defines the color of that - * circle. Use [pie.borderWidth](#plotOptions.pie.borderWidth) to set - * the border thickness. + * Fires when the mouse leaves the area close to the point. One + * parameter, `event`, is passed to the function, containing common + * event information. * - * @sample {highcharts} highcharts/plotoptions/pie-emptyseries/ - * Empty pie series + * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/ + * Show values in the chart's corner on mouse over * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @private + * @type {Highcharts.PointMouseOutCallbackFunction} + * @context Highcharts.Point + * @apioption plotOptions.series.point.events.mouseOut + */ + /** + * Fires when the mouse enters the area close to the point. One + * parameter, `event`, is passed to the function, containing common + * event information. + * + * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/ + * Show values in the chart's corner on mouse over + * + * @type {Highcharts.PointMouseOverCallbackFunction} + * @context Highcharts.Point + * @apioption plotOptions.series.point.events.mouseOver + */ + /** + * Fires when the point is removed using the `.remove()` method. One + * parameter, `event`, is passed to the function. Returning `false` + * cancels the operation. + * + * @sample {highcharts} highcharts/plotoptions/series-point-events-remove/ + * Remove point and confirm + * + * @type {Highcharts.PointRemoveCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Point + * @apioption plotOptions.series.point.events.remove + */ + /** + * Fires when the point is selected either programmatically or + * following a click on the point. One parameter, `event`, is passed + * to the function. Returning `false` cancels the operation. + * + * @sample {highcharts} highcharts/plotoptions/series-point-events-select/ + * Report the last selected point + * @sample {highmaps} maps/plotoptions/series-allowpointselect/ + * Report select and unselect + * + * @type {Highcharts.PointSelectCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Point + * @apioption plotOptions.series.point.events.select + */ + /** + * Fires when the point is unselected either programmatically or + * following a click on the point. One parameter, `event`, is passed + * to the function. + * Returning `false` cancels the operation. + * + * @sample {highcharts} highcharts/plotoptions/series-point-events-unselect/ + * Report the last unselected point + * @sample {highmaps} maps/plotoptions/series-allowpointselect/ + * Report select and unselect + * + * @type {Highcharts.PointUnselectCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Point + * @apioption plotOptions.series.point.events.unselect + */ + /** + * Fires when the point is updated programmatically through the + * `.update()` method. One parameter, `event`, is passed to the + * function. The new point options can be accessed through + * `event.options`. Returning `false` cancels the operation. + * + * @sample {highcharts} highcharts/plotoptions/series-point-events-update/ + * Confirm point updating + * + * @type {Highcharts.PointUpdateCallbackFunction} + * @since 1.2.0 + * @context Highcharts.Point + * @apioption plotOptions.series.point.events.update */ - fillColor: void 0, /** - * The end angle of the pie in degrees where 0 is top and 90 is right. - * Defaults to `startAngle` plus 360. + * Events for each single point. * - * @sample {highcharts} highcharts/demo/pie-semi-circle/ - * Semi-circle donut + * @declare Highcharts.PointEventsOptionsObject + */ + events: {}, + }, + /** + * Options for the series data labels, appearing next to each data + * point. + * + * Since v6.2.0, multiple data labels can be applied to each single + * point by defining them as an array of configs. + * + * In styled mode, the data labels can be styled with the + * `.highcharts-data-label-box` and `.highcharts-data-label` class names + * ([see example](https://www.highcharts.com/samples/highcharts/css/series-datalabels)). + * + * @sample {highcharts} highcharts/plotoptions/series-datalabels-enabled + * Data labels enabled + * @sample {highcharts} highcharts/plotoptions/series-datalabels-multiple + * Multiple data labels on a bar series + * @sample {highcharts} highcharts/css/series-datalabels + * Style mode example + * + * @type {*|Array<*>} + * @product highcharts highstock highmaps gantt + * + * @private + */ + dataLabels: { + /** + * Enable or disable the initial animation when a series is + * displayed for the `dataLabels`. The animation can also be set as + * a configuration object. Please note that this option only + * applies to the initial animation. + * For other animations, see [chart.animation](#chart.animation) + * and the animation parameter under the API methods. + * The following properties are supported: + * + * - `defer`: The animation delay time in milliseconds. + * + * @sample {highcharts} highcharts/plotoptions/animation-defer/ + * Animation defer settings + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + * @since 8.2.0 + * @apioption plotOptions.series.dataLabels.animation + */ + animation: {}, + /** + * The animation delay time in milliseconds. + * Set to `0` renders dataLabel immediately. + * As `undefined` inherits defer time from the [series.animation.defer](#plotOptions.series.animation.defer). * * @type {number} - * @since 1.3.6 - * @product highcharts - * @apioption plotOptions.pie.endAngle + * @since 8.2.0 + * @apioption plotOptions.series.dataLabels.animation.defer */ /** - * Equivalent to [chart.ignoreHiddenSeries](#chart.ignoreHiddenSeries), - * this option tells whether the series shall be redrawn as if the - * hidden point were `null`. + * The alignment of the data label compared to the point. If + * `right`, the right side of the label should be touching the + * point. For points with an extent, like columns, the alignments + * also dictates how to align it inside the box, as given with the + * [inside](#plotOptions.column.dataLabels.inside) + * option. Can be one of `left`, `center` or `right`. * - * The default value changed from `false` to `true` with Highcharts - * 3.0. + * @sample {highcharts} highcharts/plotoptions/series-datalabels-align-left/ + * Left aligned + * @sample {highcharts} highcharts/plotoptions/bar-datalabels-align-inside-bar/ + * Data labels inside the bar * - * @sample {highcharts} highcharts/plotoptions/pie-ignorehiddenpoint/ - * True, the hiddden point is ignored + * @type {Highcharts.AlignValue|null} + */ + align: "center", + /** + * Whether to allow data labels to overlap. To make the labels less + * sensitive for overlapping, the + * [dataLabels.padding](#plotOptions.series.dataLabels.padding) + * can be set to 0. * - * @since 2.3.0 - * @product highcharts + * @sample {highcharts} highcharts/plotoptions/series-datalabels-allowoverlap-false/ + * Don't allow overlap * - * @private + * @type {boolean} + * @default false + * @since 4.1.0 + * @apioption plotOptions.series.dataLabels.allowOverlap */ - ignoreHiddenPoint: true, /** - * @ignore-option + * The background color or gradient for the data label. * - * @private + * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ + * Data labels box options + * @sample {highmaps} maps/plotoptions/series-datalabels-box/ + * Data labels box options + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 2.2.1 + * @apioption plotOptions.series.dataLabels.backgroundColor */ - inactiveOtherPoints: true, /** - * The size of the inner diameter for the pie. A size greater than 0 - * renders a donut chart. Can be a percentage or pixel value. - * Percentages are relative to the pie size. Pixel values are given as - * integers. + * The border color for the data label. Defaults to `undefined`. * + * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ + * Data labels box options * - * Note: in Highcharts < 4.1.2, the percentage was relative to the plot - * area, not the pie size. + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 2.2.1 + * @apioption plotOptions.series.dataLabels.borderColor + */ + /** + * The border radius in pixels for the data label. * - * @sample {highcharts} highcharts/plotoptions/pie-innersize-80px/ - * 80px inner size - * @sample {highcharts} highcharts/plotoptions/pie-innersize-50percent/ - * 50% of the plot area - * @sample {highcharts} highcharts/demo/3d-pie-donut/ - * 3D donut + * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ + * Data labels box options + * @sample {highmaps} maps/plotoptions/series-datalabels-box/ + * Data labels box options * - * @type {number|string} + * @type {number} * @default 0 - * @since 2.0 - * @product highcharts - * @apioption plotOptions.pie.innerSize + * @since 2.2.1 + * @apioption plotOptions.series.dataLabels.borderRadius */ /** - * @ignore-option + * The border width in pixels for the data label. * - * @private + * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ + * Data labels box options + * + * @type {number} + * @default 0 + * @since 2.2.1 + * @apioption plotOptions.series.dataLabels.borderWidth */ - legendType: 'point', /** - * @ignore-option + * A class name for the data label. Particularly in styled mode, + * this can be used to give each series' or point's data label + * unique styling. In addition to this option, a default color class + * name is added so that we can give the labels a contrast text + * shadow. * - * @private + * @sample {highcharts} highcharts/css/data-label-contrast/ + * Contrast text shadow + * @sample {highcharts} highcharts/css/series-datalabels/ + * Styling by CSS + * + * @type {string} + * @since 5.0.0 + * @apioption plotOptions.series.dataLabels.className */ - marker: null, /** - * The minimum size for a pie in response to auto margins. The pie will - * try to shrink to make room for data labels in side the plot area, - * but only to this size. + * The text color for the data labels. Defaults to `undefined`. For + * certain series types, like column or map, the data labels can be + * drawn inside the points. In this case the data label will be + * drawn with maximum contrast by default. Additionally, it will be + * given a `text-outline` style with the opposite color, to further + * increase the contrast. This can be overridden by setting the + * `text-outline` style to `none` in the `dataLabels.style` option. * - * @type {number|string} - * @default 80 - * @since 3.0 - * @product highcharts - * @apioption plotOptions.pie.minSize + * @sample {highcharts} highcharts/plotoptions/series-datalabels-color/ + * Red data labels + * @sample {highmaps} maps/demo/color-axis/ + * White data labels + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @apioption plotOptions.series.dataLabels.color */ /** - * The diameter of the pie relative to the plot area. Can be a - * percentage or pixel value. Pixel values are given as integers. The - * default behaviour (as of 3.0) is to scale to the plot area and give - * room for data labels within the plot area. - * [slicedOffset](#plotOptions.pie.slicedOffset) is also included in the - * default size calculation. As a consequence, the size of the pie may - * vary when points are updated and data labels more around. In that - * case it is best to set a fixed value, for example `"75%"`. - * - * @sample {highcharts} highcharts/plotoptions/pie-size/ - * Smaller pie - * - * @type {number|string|null} - * @product highcharts + * Whether to hide data labels that are outside the plot area. By + * default, the data label is moved inside the plot area according + * to the + * [overflow](#plotOptions.series.dataLabels.overflow) + * option. * - * @private + * @type {boolean} + * @default true + * @since 2.3.3 + * @apioption plotOptions.series.dataLabels.crop */ - size: null, /** - * Whether to display this particular series or series type in the - * legend. Since 2.1, pies are not shown in the legend by default. + * Whether to defer displaying the data labels until the initial + * series animation has finished. Setting to `false` renders the + * data label immediately. If set to `true` inherits the defer + * time set in [plotOptions.series.animation](#plotOptions.series.animation). * - * @sample {highcharts} highcharts/plotoptions/series-showinlegend/ - * One series in the legend, one hidden + * @sample highcharts/plotoptions/animation-defer + * Set defer time * - * @product highcharts + * @since 4.0.0 + * @product highcharts highstock gantt + */ + defer: true, + /** + * Enable or disable the data labels. * - * @private + * @sample {highcharts} highcharts/plotoptions/series-datalabels-enabled/ + * Data labels enabled + * @sample {highmaps} maps/demo/color-axis/ + * Data labels enabled + * + * @type {boolean} + * @default false + * @apioption plotOptions.series.dataLabels.enabled */ - showInLegend: false, /** - * If a point is sliced, moved out from the center, how many pixels - * should it be moved?. + * A declarative filter to control of which data labels to display. + * The declarative filter is designed for use when callback + * functions are not available, like when the chart options require + * a pure JSON structure or for use with graphical editors. For + * programmatic control, use the `formatter` instead, and return + * `undefined` to disable a single data label. * - * @sample {highcharts} highcharts/plotoptions/pie-slicedoffset-20/ - * 20px offset + * @example + * filter: { + * property: 'percentage', + * operator: '>', + * value: 4 + * } * - * @product highcharts + * @sample {highcharts} highcharts/demo/pie-monochrome + * Data labels filtered by percentage * - * @private + * @declare Highcharts.DataLabelsFilterOptionsObject + * @since 6.0.3 + * @apioption plotOptions.series.dataLabels.filter + */ + /** + * The operator to compare by. Can be one of `>`, `<`, `>=`, `<=`, + * `==`, and `===`. + * + * @type {string} + * @validvalue [">", "<", ">=", "<=", "==", "==="] + * @apioption plotOptions.series.dataLabels.filter.operator */ - slicedOffset: 10, /** - * The start angle of the pie slices in degrees where 0 is top and 90 - * right. + * The point property to filter by. Point options are passed + * directly to properties, additionally there are `y` value, + * `percentage` and others listed under {@link Highcharts.Point} + * members. * - * @sample {highcharts} highcharts/plotoptions/pie-startangle-90/ - * Start from right + * @type {string} + * @apioption plotOptions.series.dataLabels.filter.property + */ + /** + * The value to compare against. * * @type {number} - * @default 0 - * @since 2.3.4 - * @product highcharts - * @apioption plotOptions.pie.startAngle + * @apioption plotOptions.series.dataLabels.filter.value */ /** - * Sticky tracking of mouse events. When true, the `mouseOut` event - * on a series isn't triggered until the mouse moves over another - * series, or out of the plot area. When false, the `mouseOut` event on - * a series is triggered when the mouse leaves the area around the - * series' graph or markers. This also implies the tooltip. When - * `stickyTracking` is false and `tooltip.shared` is false, the tooltip - * will be hidden when moving the mouse between series. + * A + * [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) + * for the data label. Available variables are the same as for + * `formatter`. + * + * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ + * Add a unit + * @sample {highmaps} maps/plotoptions/series-datalabels-format/ + * Formatted value in the data label + * + * @type {string} + * @default y + * @default point.value + * @since 3.0 + * @apioption plotOptions.series.dataLabels.format + */ + // eslint-disable-next-line valid-jsdoc + /** + * Callback JavaScript function to format the data label. Note that + * if a `format` is defined, the format takes precedence and the + * formatter is ignored. * - * @product highcharts + * @sample {highmaps} maps/plotoptions/series-datalabels-format/ + * Formatted value * - * @private + * @type {Highcharts.DataLabelsFormatterCallbackFunction} */ - stickyTracking: false, - tooltip: { - followPointer: true + formatter: function () { + var numberFormatter = this.series.chart.numberFormatter; + return typeof this.y !== "number" + ? "" + : numberFormatter(this.y, -1); }, /** - * The color of the border surrounding each slice. When `null`, the - * border takes the same color as the slice fill. This can be used - * together with a `borderWidth` to fill drawing gaps created by - * antialiazing artefacts in borderless pies. - * - * In styled mode, the border stroke is given in the `.highcharts-point` - * class. - * - * @sample {highcharts} highcharts/plotoptions/pie-bordercolor-black/ - * Black border + * For points with an extent, like columns or map areas, whether to + * align the data label inside the box or to the actual value point. + * Defaults to `false` in most cases, `true` in stacked columns. * - * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} - * @default #ffffff - * @product highcharts - * - * @private + * @type {boolean} + * @since 3.0 + * @apioption plotOptions.series.dataLabels.inside */ - borderColor: '#ffffff', /** - * The width of the border surrounding each slice. - * - * When setting the border width to 0, there may be small gaps between - * the slices due to SVG antialiasing artefacts. To work around this, - * keep the border width at 0.5 or 1, but set the `borderColor` to - * `null` instead. - * - * In styled mode, the border stroke width is given in the - * `.highcharts-point` class. - * - * @sample {highcharts} highcharts/plotoptions/pie-borderwidth/ - * 3px border + * Format for points with the value of null. Works analogously to + * [format](#plotOptions.series.dataLabels.format). `nullFormat` can + * be applied only to series which support displaying null points. * - * @product highcharts + * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ + * Format data label and tooltip for null point. * - * @private - */ - borderWidth: 1, - /** - * @ignore-options - * @private + * @type {boolean|string} + * @since 7.1.0 + * @apioption plotOptions.series.dataLabels.nullFormat */ - lineWidth: void 0, - states: { - /** - * @extends plotOptions.series.states.hover - * @excluding marker, lineWidth, lineWidthPlus - * @product highcharts - */ - hover: { - /** - * How much to brighten the point on interaction. Requires the - * main color to be defined in hex or rgb(a) format. - * - * In styled mode, the hover brightness is by default replaced - * by a fill-opacity given in the `.highcharts-point-hover` - * class. - * - * @sample {highcharts} highcharts/plotoptions/pie-states-hover-brightness/ - * Brightened by 0.5 - * - * @product highcharts - */ - brightness: 0.1 - } - } - }, - /* eslint-disable valid-jsdoc */ - /** - * @lends seriesTypes.pie.prototype - */ - { - isCartesian: false, - requireSorting: false, - directTouch: true, - noSharedTooltip: true, - trackerGroups: ['group', 'dataLabelsGroup'], - axisTypes: [], - pointAttribs: BaseSeries.seriesTypes.column.prototype.pointAttribs, /** - * Animate the pies in + * Callback JavaScript function that defines formatting for points + * with the value of null. Works analogously to + * [formatter](#plotOptions.series.dataLabels.formatter). + * `nullPointFormatter` can be applied only to series which support + * displaying null points. * - * @private - * @function Highcharts.seriesTypes.pie#animate + * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ + * Format data label and tooltip for null point. * - * @param {boolean} [init=false] + * @type {Highcharts.DataLabelsFormatterCallbackFunction} + * @since 7.1.0 + * @apioption plotOptions.series.dataLabels.nullFormatter */ - animate: function (init) { - var series = this, - points = series.points, - startAngleRad = series.startAngleRad; - if (!init) { - points.forEach(function (point) { - var graphic = point.graphic, - args = point.shapeArgs; - if (graphic && args) { - // start values - graphic.attr({ - // animate from inner radius (#779) - r: pick(point.startR, (series.center && series.center[3] / 2)), - start: startAngleRad, - end: startAngleRad - }); - // animate - graphic.animate({ - r: args.r, - start: args.start, - end: args.end - }, series.options.animation); - } - }); - } - }, - // Define hasData function for non-cartesian series. - // Returns true if the series has points at all. - hasData: function () { - return !!this.processedXData.length; // != 0 - }, /** - * Recompute total chart sum and update percentages of points. + * How to handle data labels that flow outside the plot area. The + * default is `"justify"`, which aligns them inside the plot area. + * For columns and bars, this means it will be moved inside the bar. + * To display data labels outside the plot area, set `crop` to + * `false` and `overflow` to `"allow"`. * - * @private - * @function Highcharts.seriesTypes.pie#updateTotals - * @return {void} + * @type {Highcharts.DataLabelsOverflowValue} + * @default justify + * @since 3.0.6 + * @apioption plotOptions.series.dataLabels.overflow */ - updateTotals: function () { - var i, - total = 0, - points = this.points, - len = points.length, - point, - ignoreHiddenPoint = this.options.ignoreHiddenPoint; - // Get the total sum - for (i = 0; i < len; i++) { - point = points[i]; - total += (ignoreHiddenPoint && !point.visible) ? - 0 : - point.isNull ? - 0 : - point.y; - } - this.total = total; - // Set each point's properties - for (i = 0; i < len; i++) { - point = points[i]; - point.percentage = - (total > 0 && (point.visible || !ignoreHiddenPoint)) ? - point.y / total * 100 : - 0; - point.total = total; - } - }, /** - * Extend the generatePoints method by adding total and percentage - * properties to each point + * When either the `borderWidth` or the `backgroundColor` is set, + * this is the padding within the box. * - * @private - * @function Highcharts.seriesTypes.pie#generatePoints - * @return {void} + * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ + * Data labels box options + * @sample {highmaps} maps/plotoptions/series-datalabels-box/ + * Data labels box options + * + * @since 2.2.1 */ - generatePoints: function () { - LineSeries.prototype.generatePoints.call(this); - this.updateTotals(); - }, + padding: 5, /** - * Utility for getting the x value from a given y, used for - * anticollision logic in data labels. Added point for using specific - * points' label distance. - * @private + * Aligns data labels relative to points. If `center` alignment is + * not possible, it defaults to `right`. + * + * @type {Highcharts.AlignValue} + * @default center + * @apioption plotOptions.series.dataLabels.position */ - getX: function (y, left, point) { - var center = this.center, - // Variable pie has individual radius - radius = this.radii ? - this.radii[point.index] : - center[2] / 2, - angle, - x; - angle = Math.asin(clamp((y - center[1]) / (radius + point.labelDistance), -1, 1)); - x = center[0] + - (left ? -1 : 1) * - (Math.cos(angle) * (radius + point.labelDistance)) + - (point.labelDistance > 0 ? - (left ? -1 : 1) * this.options.dataLabels.padding : - 0); - return x; - }, /** - * Do translation for pie slices + * Text rotation in degrees. Note that due to a more complex + * structure, backgrounds, borders and padding will be lost on a + * rotated data label. * - * @private - * @function Highcharts.seriesTypes.pie#translate - * @param {Array<number>} [positions] - * @return {void} + * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/ + * Vertical labels + * + * @type {number} + * @default 0 + * @apioption plotOptions.series.dataLabels.rotation */ - translate: function (positions) { - this.generatePoints(); - var series = this, - cumulative = 0, - precision = 1000, // issue #172 - options = series.options, - slicedOffset = options.slicedOffset, - connectorOffset = slicedOffset + (options.borderWidth || 0), - finalConnectorOffset, - start, - end, - angle, - radians = getStartAndEndRadians(options.startAngle, - options.endAngle), - startAngleRad = series.startAngleRad = radians.start, - endAngleRad = series.endAngleRad = radians.end, - circ = endAngleRad - startAngleRad, // 2 * Math.PI, - points = series.points, - // the x component of the radius vector for a given point - radiusX, - radiusY, - labelDistance = options.dataLabels.distance, - ignoreHiddenPoint = options.ignoreHiddenPoint, - i, - len = points.length, - point; - // Get positions - either an integer or a percentage string must be - // given. If positions are passed as a parameter, we're in a - // recursive loop for adjusting space for data labels. - if (!positions) { - series.center = positions = series.getCenter(); - } - // Calculate the geometry for each point - for (i = 0; i < len; i++) { - point = points[i]; - // set start and end angle - start = startAngleRad + (cumulative * circ); - if (!ignoreHiddenPoint || point.visible) { - cumulative += point.percentage / 100; - } - end = startAngleRad + (cumulative * circ); - // set the shape - point.shapeType = 'arc'; - point.shapeArgs = { - x: positions[0], - y: positions[1], - r: positions[2] / 2, - innerR: positions[3] / 2, - start: Math.round(start * precision) / precision, - end: Math.round(end * precision) / precision - }; - // Used for distance calculation for specific point. - point.labelDistance = pick((point.options.dataLabels && - point.options.dataLabels.distance), labelDistance); - // Compute point.labelDistance if it's defined as percentage - // of slice radius (#8854) - point.labelDistance = relativeLength(point.labelDistance, point.shapeArgs.r); - // Saved for later dataLabels distance calculation. - series.maxLabelDistance = Math.max(series.maxLabelDistance || 0, point.labelDistance); - // The angle must stay within -90 and 270 (#2645) - angle = (end + start) / 2; - if (angle > 1.5 * Math.PI) { - angle -= 2 * Math.PI; - } - else if (angle < -Math.PI / 2) { - angle += 2 * Math.PI; - } - // Center for the sliced out slice - point.slicedTranslation = { - translateX: Math.round(Math.cos(angle) * slicedOffset), - translateY: Math.round(Math.sin(angle) * slicedOffset) - }; - // set the anchor point for tooltips - radiusX = Math.cos(angle) * positions[2] / 2; - radiusY = Math.sin(angle) * positions[2] / 2; - point.tooltipPos = [ - positions[0] + radiusX * 0.7, - positions[1] + radiusY * 0.7 - ]; - point.half = angle < -Math.PI / 2 || angle > Math.PI / 2 ? - 1 : - 0; - point.angle = angle; - // Set the anchor point for data labels. Use point.labelDistance - // instead of labelDistance // #1174 - // finalConnectorOffset - not override connectorOffset value. - finalConnectorOffset = Math.min(connectorOffset, point.labelDistance / 5); // #1678 - point.labelPosition = { - natural: { - // initial position of the data label - it's utilized for - // finding the final position for the label - x: positions[0] + radiusX + Math.cos(angle) * - point.labelDistance, - y: positions[1] + radiusY + Math.sin(angle) * - point.labelDistance - }, - 'final': { - // used for generating connector path - - // initialized later in drawDataLabels function - // x: undefined, - // y: undefined - }, - // left - pie on the left side of the data label - // right - pie on the right side of the data label - // center - data label overlaps the pie - alignment: point.labelDistance < 0 ? - 'center' : point.half ? 'right' : 'left', - connectorPosition: { - breakAt: { - x: positions[0] + radiusX + Math.cos(angle) * - finalConnectorOffset, - y: positions[1] + radiusY + Math.sin(angle) * - finalConnectorOffset - }, - touchingSliceAt: { - x: positions[0] + radiusX, - y: positions[1] + radiusY - } - } - }; - } - fireEvent(series, 'afterTranslate'); - }, /** - * Called internally to draw auxiliary graph in pie-like series in - * situtation when the default graph is not sufficient enough to present - * the data well. Auxiliary graph is saved in the same object as - * regular graph. + * The shadow of the box. Works best with `borderWidth` or + * `backgroundColor`. Since 2.3 the shadow can be an object + * configuration containing `color`, `offsetX`, `offsetY`, `opacity` + * and `width`. * - * @private - * @function Highcharts.seriesTypes.pie#drawEmpty + * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/ + * Data labels box options + * + * @type {boolean|Highcharts.ShadowOptionsObject} + * @default false + * @since 2.2.1 + * @apioption plotOptions.series.dataLabels.shadow */ - drawEmpty: function () { - var centerX, - centerY, - start = this.startAngleRad, - end = this.endAngleRad, - options = this.options; - // Draw auxiliary graph if there're no visible points. - if (this.total === 0 && this.center) { - centerX = this.center[0]; - centerY = this.center[1]; - if (!this.graph) { - this.graph = this.chart.renderer - .arc(centerX, centerY, this.center[1] / 2, 0, start, end) - .addClass('highcharts-empty-series') - .add(this.group); - } - this.graph.attr({ - d: SVGRenderer.prototype.symbols.arc(centerX, centerY, this.center[2] / 2, 0, { - start: start, - end: end, - innerR: this.center[3] / 2 - }) - }); - if (!this.chart.styledMode) { - this.graph.attr({ - 'stroke-width': options.borderWidth, - fill: options.fillColor || 'none', - stroke: options.color || - '#cccccc' - }); - } - } - else if (this.graph) { // Destroy the graph object. - this.graph = this.graph.destroy(); - } - }, /** - * Draw the data points + * The name of a symbol to use for the border around the label. + * Symbols are predefined functions on the Renderer object. * - * @private - * @function Highcharts.seriesTypes.pie#drawPoints - * @return {void} - */ - redrawPoints: function () { - var series = this, - chart = series.chart, - renderer = chart.renderer, - groupTranslation, - graphic, - pointAttr, - shapeArgs, - shadow = series.options.shadow; - this.drawEmpty(); - if (shadow && !series.shadowGroup && !chart.styledMode) { - series.shadowGroup = renderer.g('shadow') - .attr({ zIndex: -1 }) - .add(series.group); - } - // draw the slices - series.points.forEach(function (point) { - var animateTo = {}; - graphic = point.graphic; - if (!point.isNull && graphic) { - shapeArgs = point.shapeArgs; - // If the point is sliced, use special translation, else use - // plot area translation - groupTranslation = point.getTranslate(); - if (!chart.styledMode) { - // Put the shadow behind all points - var shadowGroup = point.shadowGroup; - if (shadow && !shadowGroup) { - shadowGroup = point.shadowGroup = renderer - .g('shadow') - .add(series.shadowGroup); - } - if (shadowGroup) { - shadowGroup.attr(groupTranslation); - } - pointAttr = series.pointAttribs(point, (point.selected && 'select')); - } - // Draw the slice - if (!point.delayedRendering) { - graphic - .setRadialReference(series.center); - if (!chart.styledMode) { - merge(true, animateTo, pointAttr); - } - merge(true, animateTo, shapeArgs, groupTranslation); - graphic.animate(animateTo); - } - else { - graphic - .setRadialReference(series.center) - .attr(shapeArgs) - .attr(groupTranslation); - if (!chart.styledMode) { - graphic - .attr(pointAttr) - .attr({ 'stroke-linejoin': 'round' }) - .shadow(shadow, shadowGroup); - } - point.delayedRendering = false; - } - graphic.attr({ - visibility: point.visible ? 'inherit' : 'hidden' - }); - graphic.addClass(point.getClassName()); - } - else if (graphic) { - point.graphic = graphic.destroy(); - } - }); + * @sample {highcharts} highcharts/plotoptions/series-datalabels-shape/ + * A callout for annotations + * + * @type {string} + * @default square + * @since 4.1.2 + * @apioption plotOptions.series.dataLabels.shape + */ + /** + * Styles for the label. The default `color` setting is + * `"contrast"`, which is a pseudo color that Highcharts picks up + * and applies the maximum contrast to the underlying point item, + * for example the bar in a bar chart. + * + * The `textOutline` is a pseudo property that applies an outline of + * the given width with the given color, which by default is the + * maximum contrast to the text. So a bright text color will result + * in a black text outline for maximum readability on a mixed + * background. In some cases, especially with grayscale text, the + * text outline doesn't work well, in which cases it can be disabled + * by setting it to `"none"`. When `useHTML` is true, the + * `textOutline` will not be picked up. In this, case, the same + * effect can be acheived through the `text-shadow` CSS property. + * + * For some series types, where each point has an extent, like for + * example tree maps, the data label may overflow the point. There + * are two strategies for handling overflow. By default, the text + * will wrap to multiple lines. The other strategy is to set + * `style.textOverflow` to `ellipsis`, which will keep the text on + * one line plus it will break inside long words. + * + * @sample {highcharts} highcharts/plotoptions/series-datalabels-style/ + * Bold labels + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow/ + * Long labels truncated with an ellipsis in a pie + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow-wrap/ + * Long labels are wrapped in a pie + * @sample {highmaps} maps/demo/color-axis/ + * Bold labels + * + * @type {Highcharts.CSSObject} + * @since 4.1.0 + * @apioption plotOptions.series.dataLabels.style + */ + style: { + /** @internal */ + fontSize: "11px", + /** @internal */ + fontWeight: "bold", + /** @internal */ + color: "contrast", + /** @internal */ + textOutline: "1px contrast", }, /** - * Slices in pie chart are initialized in DOM, but it's shapes and - * animations are normally run in `drawPoints()`. - * @private + * Options for a label text which should follow marker's shape. + * Border and background are disabled for a label that follows a + * path. + * + * **Note:** Only SVG-based renderer supports this option. Setting + * `useHTML` to true will disable this option. + * + * @declare Highcharts.DataLabelsTextPathOptionsObject + * @since 7.1.0 + * @apioption plotOptions.series.dataLabels.textPath */ - drawPoints: function () { - var renderer = this.chart.renderer; - this.points.forEach(function (point) { - // When updating a series between 2d and 3d or cartesian and - // polar, the shape type changes. - if (point.graphic && point.hasNewShapeType()) { - point.graphic = point.graphic.destroy(); - } - if (!point.graphic) { - point.graphic = renderer[point.shapeType](point.shapeArgs) - .add(point.series.group); - point.delayedRendering = true; - } - }); - }, /** - * @private - * @deprecated - * @function Highcharts.seriesTypes.pie#searchPoint + * Presentation attributes for the text path. + * + * @type {Highcharts.SVGAttributes} + * @since 7.1.0 + * @apioption plotOptions.series.dataLabels.textPath.attributes */ - searchPoint: noop, /** - * Utility for sorting data labels + * Enable or disable `textPath` option for link's or marker's data + * labels. * - * @private - * @function Highcharts.seriesTypes.pie#sortByAngle - * @param {Array<Highcharts.Point>} points - * @param {number} sign - * @return {void} + * @type {boolean} + * @since 7.1.0 + * @apioption plotOptions.series.dataLabels.textPath.enabled */ - sortByAngle: function (points, sign) { - points.sort(function (a, b) { - return ((typeof a.angle !== 'undefined') && - (b.angle - a.angle) * sign); - }); - }, /** - * Use a simple symbol from LegendSymbolMixin. + * Whether to + * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html) + * to render the labels. * - * @private - * @borrows Highcharts.LegendSymbolMixin.drawRectangle as Highcharts.seriesTypes.pie#drawLegendSymbol + * @type {boolean} + * @default false + * @apioption plotOptions.series.dataLabels.useHTML */ - drawLegendSymbol: LegendSymbolMixin.drawRectangle, /** - * Use the getCenter method from drawLegendSymbol. + * The vertical alignment of a data label. Can be one of `top`, + * `middle` or `bottom`. The default value depends on the data, for + * instance in a column chart, the label is above positive values + * and below negative values. * - * @private - * @borrows Highcharts.CenteredSeriesMixin.getCenter as Highcharts.seriesTypes.pie#getCenter + * @type {Highcharts.VerticalAlignValue|null} + * @since 2.3.3 */ - getCenter: CenteredSeriesMixin.getCenter, + verticalAlign: "bottom", /** - * Pies don't have point marker symbols. + * The x position offset of the label relative to the point in + * pixels. * - * @deprecated - * @private - * @function Highcharts.seriesTypes.pie#getSymbol + * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/ + * Vertical and positioned + * @sample {highcharts} highcharts/plotoptions/bar-datalabels-align-inside-bar/ + * Data labels inside the bar */ - getSymbol: noop, + x: 0, /** - * @private - * @type {null} + * The Z index of the data labels. The default Z index puts it above + * the series. Use a Z index of 2 to display it behind the series. + * + * @type {number} + * @default 6 + * @since 2.3.5 + * @apioption plotOptions.series.dataLabels.z */ - drawGraph: null - }, - /** - * @lends seriesTypes.pie.prototype.pointClass.prototype - */ - { /** - * Initialize the pie slice + * The y position offset of the label relative to the point in + * pixels. * - * @private - * @function Highcharts.seriesTypes.pie#pointClass#init - * @return {Highcharts.Point} - */ - init: function () { - Point.prototype.init.apply(this, arguments); - var point = this, - toggleSlice; - point.name = pick(point.name, 'Slice'); - // add event listener for select - toggleSlice = function (e) { - point.slice(e.type === 'select'); - }; - addEvent(point, 'select', toggleSlice); - addEvent(point, 'unselect', toggleSlice); - return point; + * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/ + * Vertical and positioned + */ + y: 0, + }, + /** + * When the series contains less points than the crop threshold, all + * points are drawn, even if the points fall outside the visible plot + * area at the current zoom. The advantage of drawing all points + * (including markers and columns), is that animation is performed on + * updates. On the other hand, when the series contains more points than + * the crop threshold, the series data is cropped to only contain points + * that fall within the plot area. The advantage of cropping away + * invisible points is to increase performance on large series. + * + * @since 2.2 + * @product highcharts highstock + * + * @private + */ + cropThreshold: 300, + /** + * Opacity of a series parts: line, fill (e.g. area) and dataLabels. + * + * @see [states.inactive.opacity](#plotOptions.series.states.inactive.opacity) + * + * @since 7.1.0 + * + * @private + */ + opacity: 1, + /** + * The width of each point on the x axis. For example in a column chart + * with one value each day, the pointRange would be 1 day (= 24 * 3600 + * * 1000 milliseconds). This is normally computed automatically, but + * this option can be used to override the automatic value. + * + * @product highstock + * + * @private + */ + pointRange: 0, + /** + * When this is true, the series will not cause the Y axis to cross + * the zero plane (or [threshold](#plotOptions.series.threshold) option) + * unless the data actually crosses the plane. + * + * For example, if `softThreshold` is `false`, a series of 0, 1, 2, + * 3 will make the Y axis show negative values according to the + * `minPadding` option. If `softThreshold` is `true`, the Y axis starts + * at 0. + * + * @since 4.1.9 + * @product highcharts highstock + * + * @private + */ + softThreshold: true, + /** + * @declare Highcharts.SeriesStatesOptionsObject + * + * @private + */ + states: { + /** + * The normal state of a series, or for point items in column, pie + * and similar series. Currently only used for setting animation + * when returning to normal state from hover. + * + * @declare Highcharts.SeriesStatesNormalOptionsObject + */ + normal: { + /** + * Animation when returning to normal state after hovering. + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + */ + animation: true, }, /** - * Negative points are not valid (#1530, #3623, #5322) - * - * @private - * @function Highcharts.seriesTypes.pie#pointClass#isValid - * @return {boolean} - */ - isValid: function () { - return isNumber(this.y) && this.y >= 0; + * Options for the hovered series. These settings override the + * normal state options when a series is moused over or touched. + * + * @declare Highcharts.SeriesStatesHoverOptionsObject + */ + hover: { + /** + * Enable separate styles for the hovered series to visualize + * that the user hovers either the series itself or the legend. + * + * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled/ + * Line + * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled-column/ + * Column + * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled-pie/ + * Pie + * + * @type {boolean} + * @default true + * @since 1.2 + * @apioption plotOptions.series.states.hover.enabled + */ + /** + * Animation setting for hovering the graph in line-type series. + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + * @since 5.0.8 + * @product highcharts highstock + */ + animation: { + /** + * The duration of the hover animation in milliseconds. By + * default the hover state animates quickly in, and slowly + * back to normal. + * + * @internal + */ + duration: 50, + }, + /** + * Pixel width of the graph line. By default this property is + * undefined, and the `lineWidthPlus` property dictates how much + * to increase the linewidth from normal state. + * + * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidth/ + * 5px line on hover + * + * @type {number} + * @product highcharts highstock + * @apioption plotOptions.series.states.hover.lineWidth + */ + /** + * The additional line width for the graph of a hovered series. + * + * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/ + * 5 pixels wider + * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/ + * 5 pixels wider + * + * @since 4.0.3 + * @product highcharts highstock + */ + lineWidthPlus: 1, + /** + * In Highcharts 1.0, the appearance of all markers belonging + * to the hovered series. For settings on the hover state of the + * individual point, see + * [marker.states.hover](#plotOptions.series.marker.states.hover). + * + * @deprecated + * + * @extends plotOptions.series.marker + * @excluding states + * @product highcharts highstock + */ + marker: { + // lineWidth: base + 1, + // radius: base + 1 + }, + /** + * Options for the halo appearing around the hovered point in + * line-type series as well as outside the hovered slice in pie + * charts. By default the halo is filled by the current point or + * series color with an opacity of 0.25\. The halo can be + * disabled by setting the `halo` option to `null`. + * + * In styled mode, the halo is styled with the + * `.highcharts-halo` class, with colors inherited from + * `.highcharts-color-{n}`. + * + * @sample {highcharts} highcharts/plotoptions/halo/ + * Halo options + * @sample {highstock} highcharts/plotoptions/halo/ + * Halo options + * + * @declare Highcharts.SeriesStatesHoverHaloOptionsObject + * @type {null|*} + * @since 4.0 + * @product highcharts highstock + */ + halo: { + /** + * A collection of SVG attributes to override the appearance + * of the halo, for example `fill`, `stroke` and + * `stroke-width`. + * + * @type {Highcharts.SVGAttributes} + * @since 4.0 + * @product highcharts highstock + * @apioption plotOptions.series.states.hover.halo.attributes + */ + /** + * The pixel size of the halo. For point markers this is the + * radius of the halo. For pie slices it is the width of the + * halo outside the slice. For bubbles it defaults to 5 and + * is the width of the halo outside the bubble. + * + * @since 4.0 + * @product highcharts highstock + */ + size: 10, + /** + * Opacity for the halo unless a specific fill is overridden + * using the `attributes` setting. Note that Highcharts is + * only able to apply opacity to colors of hex or rgb(a) + * formats. + * + * @since 4.0 + * @product highcharts highstock + */ + opacity: 0.25, + }, }, /** - * Toggle the visibility of the pie slice + * Specific options for point in selected states, after being + * selected by + * [allowPointSelect](#plotOptions.series.allowPointSelect) + * or programmatically. * - * @private - * @function Highcharts.seriesTypes.pie#pointClass#setVisible - * @param {boolean} vis - * Whether to show the slice or not. If undefined, the visibility - * is toggled. - * @param {boolean} [redraw=false] - * @return {void} + * @sample maps/plotoptions/series-allowpointselect/ + * Allow point select demo + * + * @declare Highcharts.SeriesStatesSelectOptionsObject + * @extends plotOptions.series.states.hover + * @excluding brightness */ - setVisible: function (vis, redraw) { - var point = this, - series = point.series, - chart = series.chart, - ignoreHiddenPoint = series.options.ignoreHiddenPoint; - redraw = pick(redraw, ignoreHiddenPoint); - if (vis !== point.visible) { - // If called without an argument, toggle visibility - point.visible = point.options.visible = vis = - typeof vis === 'undefined' ? !point.visible : vis; - // update userOptions.data - series.options.data[series.data.indexOf(point)] = - point.options; - // Show and hide associated elements. This is performed - // regardless of redraw or not, because chart.redraw only - // handles full series. - ['graphic', 'dataLabel', 'connector', 'shadowGroup'].forEach(function (key) { - if (point[key]) { - point[key][vis ? 'show' : 'hide'](true); - } - }); - if (point.legendItem) { - chart.legend.colorizeItem(point, vis); - } - // #4170, hide halo after hiding point - if (!vis && point.state === 'hover') { - point.setState(''); + select: { + animation: { + /** @internal */ + duration: 0, + }, + }, + /** + * The opposite state of a hover for series. + * + * @sample highcharts/plotoptions/series-states-inactive-disabled + * Disabled inactive state + * + * @declare Highcharts.SeriesStatesInactiveOptionsObject + */ + inactive: { + /** + * Enable or disable the inactive state for a series + * + * @sample highcharts/plotoptions/series-states-inactive-disabled + * Disabled inactive state + * + * @type {boolean} + * @default true + * @apioption plotOptions.series.states.inactive.enabled + */ + /** + * The animation for entering the inactive state. + * + * @type {boolean|Partial<Highcharts.AnimationOptionsObject>} + */ + animation: { + /** @internal */ + duration: 50, + }, + /** + * Opacity of series elements (dataLabels, line, area). + * + * @type {number} + */ + opacity: 0.2, + }, + }, + /** + * Sticky tracking of mouse events. When true, the `mouseOut` event on a + * series isn't triggered until the mouse moves over another series, or + * out of the plot area. When false, the `mouseOut` event on a series is + * triggered when the mouse leaves the area around the series' graph or + * markers. This also implies the tooltip when not shared. When + * `stickyTracking` is false and `tooltip.shared` is false, the tooltip + * will be hidden when moving the mouse between series. Defaults to true + * for line and area type series, but to false for columns, pies etc. + * + * **Note:** The boost module will force this option because of + * technical limitations. + * + * @sample {highcharts} highcharts/plotoptions/series-stickytracking-true/ + * True by default + * @sample {highcharts} highcharts/plotoptions/series-stickytracking-false/ + * False + * + * @default {highcharts} true + * @default {highstock} true + * @default {highmaps} false + * @since 2.0 + * + * @private + */ + stickyTracking: true, + /** + * A configuration object for the tooltip rendering of each single + * series. Properties are inherited from [tooltip](#tooltip), but only + * the following properties can be defined on a series level. + * + * @declare Highcharts.SeriesTooltipOptionsObject + * @since 2.3 + * @extends tooltip + * @excluding animation, backgroundColor, borderColor, borderRadius, + * borderWidth, className, crosshairs, enabled, formatter, + * headerShape, hideDelay, outside, padding, positioner, + * shadow, shape, shared, snap, split, stickOnContact, + * style, useHTML + * @apioption plotOptions.series.tooltip + */ + /** + * When a series contains a data array that is longer than this, only + * one dimensional arrays of numbers, or two dimensional arrays with + * x and y values are allowed. Also, only the first point is tested, + * and the rest are assumed to be the same format. This saves expensive + * data checking and indexing in long series. Set it to `0` disable. + * + * Note: + * In boost mode turbo threshold is forced. Only array of numbers or + * two dimensional arrays are allowed. + * + * @since 2.2 + * @product highcharts highstock gantt + * + * @private + */ + turboThreshold: 1000, + /** + * An array defining zones within a series. Zones can be applied to the + * X axis, Y axis or Z axis for bubbles, according to the `zoneAxis` + * option. The zone definitions have to be in ascending order regarding + * to the value. + * + * In styled mode, the color zones are styled with the + * `.highcharts-zone-{n}` class, or custom classed from the `className` + * option + * ([view live demo](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/css/color-zones/)). + * + * @see [zoneAxis](#plotOptions.series.zoneAxis) + * + * @sample {highcharts} highcharts/series/color-zones-simple/ + * Color zones + * @sample {highstock} highcharts/series/color-zones-simple/ + * Color zones + * + * @declare Highcharts.SeriesZonesOptionsObject + * @type {Array<*>} + * @since 4.1.0 + * @product highcharts highstock + * @apioption plotOptions.series.zones + */ + /** + * Styled mode only. A custom class name for the zone. + * + * @sample highcharts/css/color-zones/ + * Zones styled by class name + * + * @type {string} + * @since 5.0.0 + * @apioption plotOptions.series.zones.className + */ + /** + * Defines the color of the series. + * + * @see [series color](#plotOptions.series.color) + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 4.1.0 + * @product highcharts highstock + * @apioption plotOptions.series.zones.color + */ + /** + * A name for the dash style to use for the graph. + * + * @see [plotOptions.series.dashStyle](#plotOptions.series.dashStyle) + * + * @sample {highcharts|highstock} highcharts/series/color-zones-dashstyle-dot/ + * Dashed line indicates prognosis + * + * @type {Highcharts.DashStyleValue} + * @since 4.1.0 + * @product highcharts highstock + * @apioption plotOptions.series.zones.dashStyle + */ + /** + * Defines the fill color for the series (in area type series) + * + * @see [fillColor](#plotOptions.area.fillColor) + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 4.1.0 + * @product highcharts highstock + * @apioption plotOptions.series.zones.fillColor + */ + /** + * The value up to where the zone extends, if undefined the zones + * stretches to the last value in the series. + * + * @type {number} + * @since 4.1.0 + * @product highcharts highstock + * @apioption plotOptions.series.zones.value + */ + /** + * When using dual or multiple color axes, this number defines which + * colorAxis the particular series is connected to. It refers to + * either the + * {@link #colorAxis.id|axis id} + * or the index of the axis in the colorAxis array, with 0 being the + * first. Set this option to false to prevent a series from connecting + * to the default color axis. + * + * Since v7.2.0 the option can also be an axis id or an axis index + * instead of a boolean flag. + * + * @sample highcharts/coloraxis/coloraxis-with-pie/ + * Color axis with pie series + * @sample highcharts/coloraxis/multiple-coloraxis/ + * Multiple color axis + * + * @type {number|string|boolean} + * @default 0 + * @product highcharts highstock highmaps + * @apioption plotOptions.series.colorAxis + */ + /** + * Determines what data value should be used to calculate point color + * if `colorAxis` is used. Requires to set `min` and `max` if some + * custom point property is used or if approximation for data grouping + * is set to `'sum'`. + * + * @sample highcharts/coloraxis/custom-color-key/ + * Custom color key + * @sample highcharts/coloraxis/changed-default-color-key/ + * Changed default color key + * + * @type {string} + * @default y + * @since 7.2.0 + * @product highcharts highstock highmaps + * @apioption plotOptions.series.colorKey + */ + /** + * Determines whether the series should look for the nearest point + * in both dimensions or just the x-dimension when hovering the series. + * Defaults to `'xy'` for scatter series and `'x'` for most other + * series. If the data has duplicate x-values, it is recommended to + * set this to `'xy'` to allow hovering over all points. + * + * Applies only to series types using nearest neighbor search (not + * direct hover) for tooltip. + * + * @sample {highcharts} highcharts/series/findnearestpointby/ + * Different hover behaviors + * @sample {highstock} highcharts/series/findnearestpointby/ + * Different hover behaviors + * @sample {highmaps} highcharts/series/findnearestpointby/ + * Different hover behaviors + * + * @since 5.0.10 + * @validvalue ["x", "xy"] + * + * @private + */ + findNearestPointBy: "x", + }, + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** @lends Highcharts.Series.prototype */ + { + axisTypes: ["xAxis", "yAxis"], + coll: "series", + colorCounter: 0, + cropShoulder: 1, + directTouch: false, + isCartesian: true, + // each point's x and y values are stored in this.xData and this.yData + parallelArrays: ["x", "y"], + pointClass: Point, + requireSorting: true, + sorted: true, + init: function (chart, options) { + fireEvent(this, "init", { options: options }); + var series = this, + events, + chartSeries = chart.series, + lastSeries; + // A lookup over those events that are added by _options_ (not + // programmatically). These are updated through Series.update() + // (#10861). + this.eventOptions = this.eventOptions || {}; + // The 'eventsToUnbind' property moved from prototype into the + // Series init to avoid reference to the same array between + // the different series and charts. #12959, #13937 + this.eventsToUnbind = []; + /** + * Read only. The chart that the series belongs to. + * + * @name Highcharts.Series#chart + * @type {Highcharts.Chart} + */ + series.chart = chart; + /** + * Read only. The series' type, like "line", "area", "column" etc. + * The type in the series options anc can be altered using + * {@link Series#update}. + * + * @name Highcharts.Series#type + * @type {string} + */ + /** + * Read only. The series' current options. To update, use + * {@link Series#update}. + * + * @name Highcharts.Series#options + * @type {Highcharts.SeriesOptionsType} + */ + series.options = options = series.setOptions(options); + series.linkedSeries = []; + // bind the axes + series.bindAxes(); + // set some variables + extend(series, { + /** + * The series name as given in the options. Defaults to + * "Series {n}". + * + * @name Highcharts.Series#name + * @type {string} + */ + name: options.name, + state: "", + /** + * Read only. The series' visibility state as set by {@link + * Series#show}, {@link Series#hide}, or in the initial + * configuration. + * + * @name Highcharts.Series#visible + * @type {boolean} + */ + visible: options.visible !== false, + /** + * Read only. The series' selected state as set by {@link + * Highcharts.Series#select}. + * + * @name Highcharts.Series#selected + * @type {boolean} + */ + selected: options.selected === true, // false by default + }); + // Register event listeners + events = options.events; + objectEach(events, function (event, eventType) { + if (isFunction(event)) { + // If event does not exist, or is changed by Series.update + if (series.eventOptions[eventType] !== event) { + // Remove existing if set by option + if (isFunction(series.eventOptions[eventType])) { + removeEvent( + series, + eventType, + series.eventOptions[eventType] + ); + } + series.eventOptions[eventType] = event; + addEvent(series, eventType, event); + } + } + }); + if ( + (events && events.click) || + (options.point && + options.point.events && + options.point.events.click) || + options.allowPointSelect + ) { + chart.runTrackerClick = true; + } + series.getColor(); + series.getSymbol(); + // Initialize the parallel data arrays + series.parallelArrays.forEach(function (key) { + if (!series[key + "Data"]) { + series[key + "Data"] = []; + } + }); + // Mark cartesian + if (series.isCartesian) { + chart.hasCartesianSeries = true; + } + // Get the index and register the series in the chart. The index is + // one more than the current latest series index (#5960). + if (chartSeries.length) { + lastSeries = chartSeries[chartSeries.length - 1]; + } + series._i = pick(lastSeries && lastSeries._i, -1) + 1; + series.opacity = series.options.opacity; + // Insert the series and re-order all series above the insertion + // point. + chart.orderSeries(this.insert(chartSeries)); + // Set options for series with sorting and set data later. + if (options.dataSorting && options.dataSorting.enabled) { + series.setDataSortingOptions(); + } else if (!series.points && !series.data) { + series.setData(options.data, false); + } + fireEvent(this, "afterInit"); + }, + /** + * Check whether the series item is itself or inherits from a certain + * series type. + * + * @function Highcharts.Series#is + * @param {string} type The type of series to check for, can be either + * featured or custom series types. For example `column`, `pie`, + * `ohlc` etc. + * + * @return {boolean} + * True if this item is or inherits from the given type. + */ + is: function (type) { + return seriesTypes[type] && this instanceof seriesTypes[type]; + }, + /** + * Insert the series in a collection with other series, either the chart + * series or yAxis series, in the correct order according to the index + * option. Used internally when adding series. + * + * @private + * @function Highcharts.Series#insert + * @param {Array<Highcharts.Series>} collection + * A collection of series, like `chart.series` or `xAxis.series`. + * @return {number} + * The index of the series in the collection. + */ + insert: function (collection) { + var indexOption = this.options.index, + i; + // Insert by index option + if (isNumber(indexOption)) { + i = collection.length; + while (i--) { + // Loop down until the interted element has higher index + if ( + indexOption >= + pick(collection[i].options.index, collection[i]._i) + ) { + collection.splice(i + 1, 0, this); + break; + } + } + if (i === -1) { + collection.unshift(this); + } + i = i + 1; + // Or just push it to the end + } else { + collection.push(this); + } + return pick(i, collection.length - 1); + }, + /** + * Set the xAxis and yAxis properties of cartesian series, and register + * the series in the `axis.series` array. + * + * @private + * @function Highcharts.Series#bindAxes + * @return {void} + * @exception 18 + */ + bindAxes: function () { + var series = this, + seriesOptions = series.options, + chart = series.chart, + axisOptions; + fireEvent(this, "bindAxes", null, function () { + // repeat for xAxis and yAxis + (series.axisTypes || []).forEach(function (AXIS) { + // loop through the chart's axis objects + chart[AXIS].forEach(function (axis) { + axisOptions = axis.options; + // apply if the series xAxis or yAxis option mathches + // the number of the axis, or if undefined, use the + // first axis + if ( + seriesOptions[AXIS] === axisOptions.index || + (typeof seriesOptions[AXIS] !== "undefined" && + seriesOptions[AXIS] === axisOptions.id) || + (typeof seriesOptions[AXIS] === "undefined" && + axisOptions.index === 0) + ) { + // register this series in the axis.series lookup + series.insert(axis.series); + // set this series.xAxis or series.yAxis reference + /** + * Read only. The unique xAxis object associated + * with the series. + * + * @name Highcharts.Series#xAxis + * @type {Highcharts.Axis} + */ + /** + * Read only. The unique yAxis object associated + * with the series. + * + * @name Highcharts.Series#yAxis + * @type {Highcharts.Axis} + */ + series[AXIS] = axis; + // mark dirty for redraw + axis.isDirty = true; + } + }); + // The series needs an X and an Y axis + if (!series[AXIS] && series.optionalAxis !== AXIS) { + error(18, true, chart); + } + }); + }); + fireEvent(this, "afterBindAxes"); + }, + /** + * For simple series types like line and column, the data values are + * held in arrays like xData and yData for quick lookup to find extremes + * and more. For multidimensional series like bubble and map, this can + * be extended with arrays like zData and valueData by adding to the + * `series.parallelArrays` array. + * + * @private + * @function Highcharts.Series#updateParallelArrays + * @param {Highcharts.Point} point + * @param {number|string} i + * @return {void} + */ + updateParallelArrays: function (point, i) { + var series = point.series, + args = arguments, + fn = isNumber(i) + ? // Insert the value in the given position + function (key) { + var val = + key === "y" && series.toYData + ? series.toYData(point) + : point[key]; + series[key + "Data"][i] = val; + } + : // Apply the method specified in i with the following + // arguments as arguments + function (key) { + Array.prototype[i].apply( + series[key + "Data"], + Array.prototype.slice.call(args, 2) + ); + }; + series.parallelArrays.forEach(fn); + }, + /** + * Define hasData functions for series. These return true if there + * are data points on this series within the plot area. + * + * @private + * @function Highcharts.Series#hasData + * @return {boolean} + */ + hasData: function () { + return ( + (this.visible && + typeof this.dataMax !== "undefined" && + typeof this.dataMin !== "undefined") || // #3703 + (this.visible && this.yData && this.yData.length > 0) // #9758 + ); + }, + /** + * Return an auto incremented x value based on the pointStart and + * pointInterval options. This is only used if an x value is not given + * for the point that calls autoIncrement. + * + * @private + * @function Highcharts.Series#autoIncrement + * @return {number} + */ + autoIncrement: function () { + var options = this.options, + xIncrement = this.xIncrement, + date, + pointInterval, + pointIntervalUnit = options.pointIntervalUnit, + time = this.chart.time; + xIncrement = pick(xIncrement, options.pointStart, 0); + this.pointInterval = pointInterval = pick( + this.pointInterval, + options.pointInterval, + 1 + ); + // Added code for pointInterval strings + if (pointIntervalUnit) { + date = new time.Date(xIncrement); + if (pointIntervalUnit === "day") { + time.set("Date", date, time.get("Date", date) + pointInterval); + } else if (pointIntervalUnit === "month") { + time.set( + "Month", + date, + time.get("Month", date) + pointInterval + ); + } else if (pointIntervalUnit === "year") { + time.set( + "FullYear", + date, + time.get("FullYear", date) + pointInterval + ); + } + pointInterval = date.getTime() - xIncrement; + } + this.xIncrement = xIncrement + pointInterval; + return xIncrement; + }, + /** + * Internal function to set properties for series if data sorting is + * enabled. + * + * @private + * @function Highcharts.Series#setDataSortingOptions + * @return {void} + */ + setDataSortingOptions: function () { + var options = this.options; + extend(this, { + requireSorting: false, + sorted: false, + enabledDataSorting: true, + allowDG: false, + }); + // To allow unsorted data for column series. + if (!defined(options.pointRange)) { + options.pointRange = 1; + } + }, + /** + * Set the series options by merging from the options tree. Called + * internally on initializing and updating series. This function will + * not redraw the series. For API usage, use {@link Series#update}. + * @private + * @function Highcharts.Series#setOptions + * @param {Highcharts.SeriesOptionsType} itemOptions + * The series options. + * @return {Highcharts.SeriesOptionsType} + * @fires Highcharts.Series#event:afterSetOptions + */ + setOptions: function (itemOptions) { + var chart = this.chart, + chartOptions = chart.options, + plotOptions = chartOptions.plotOptions, + userOptions = chart.userOptions || {}, + seriesUserOptions = merge(itemOptions), + options, + zones, + zone, + styledMode = chart.styledMode, + e = { + plotOptions: plotOptions, + userOptions: seriesUserOptions, + }; + fireEvent(this, "setOptions", e); + // These may be modified by the event + var typeOptions = e.plotOptions[this.type], + userPlotOptions = userOptions.plotOptions || {}; + // use copy to prevent undetected changes (#9762) + /** + * Contains series options by the user without defaults. + * @name Highcharts.Series#userOptions + * @type {Highcharts.SeriesOptionsType} + */ + this.userOptions = e.userOptions; + options = merge( + typeOptions, + plotOptions.series, + // #3881, chart instance plotOptions[type] should trump + // plotOptions.series + userOptions.plotOptions && userOptions.plotOptions[this.type], + seriesUserOptions + ); + // The tooltip options are merged between global and series specific + // options. Importance order asscendingly: + // globals: (1)tooltip, (2)plotOptions.series, + // (3)plotOptions[this.type] + // init userOptions with possible later updates: 4-6 like 1-3 and + // (7)this series options + this.tooltipOptions = merge( + defaultOptions.tooltip, // 1 + defaultOptions.plotOptions.series && + defaultOptions.plotOptions.series.tooltip, // 2 + defaultOptions.plotOptions[this.type].tooltip, // 3 + chartOptions.tooltip.userOptions, // 4 + plotOptions.series && plotOptions.series.tooltip, // 5 + plotOptions[this.type].tooltip, // 6 + seriesUserOptions.tooltip // 7 + ); + // When shared tooltip, stickyTracking is true by default, + // unless user says otherwise. + this.stickyTracking = pick( + seriesUserOptions.stickyTracking, + userPlotOptions[this.type] && + userPlotOptions[this.type].stickyTracking, + userPlotOptions.series && userPlotOptions.series.stickyTracking, + this.tooltipOptions.shared && !this.noSharedTooltip + ? true + : options.stickyTracking + ); + // Delete marker object if not allowed (#1125) + if (typeOptions.marker === null) { + delete options.marker; + } + // Handle color zones + this.zoneAxis = options.zoneAxis; + zones = this.zones = (options.zones || []).slice(); + if ( + (options.negativeColor || options.negativeFillColor) && + !options.zones + ) { + zone = { + value: + options[this.zoneAxis + "Threshold"] || + options.threshold || + 0, + className: "highcharts-negative", + }; + if (!styledMode) { + zone.color = options.negativeColor; + zone.fillColor = options.negativeFillColor; + } + zones.push(zone); + } + if (zones.length) { + // Push one extra zone for the rest + if (defined(zones[zones.length - 1].value)) { + zones.push( + styledMode + ? {} + : { + color: this.color, + fillColor: this.fillColor, + } + ); + } + } + fireEvent(this, "afterSetOptions", { options: options }); + return options; + }, + /** + * Return series name in "Series {Number}" format or the one defined by + * a user. This method can be simply overridden as series name format + * can vary (e.g. technical indicators). + * + * @function Highcharts.Series#getName + * @return {string} + * The series name. + */ + getName: function () { + // #4119 + return pick(this.options.name, "Series " + (this.index + 1)); + }, + /** + * @private + * @function Highcharts.Series#getCyclic + * @param {string} prop + * @param {*} [value] + * @param {Highcharts.Dictionary<any>} [defaults] + * @return {void} + */ + getCyclic: function (prop, value, defaults) { + var i, + chart = this.chart, + userOptions = this.userOptions, + indexName = prop + "Index", + counterName = prop + "Counter", + len = defaults + ? defaults.length + : pick( + chart.options.chart[prop + "Count"], + chart[prop + "Count"] + ), + setting; + if (!value) { + // Pick up either the colorIndex option, or the _colorIndex + // after Series.update() + setting = pick( + userOptions[indexName], + userOptions["_" + indexName] + ); + if (defined(setting)) { + // after Series.update() + i = setting; + } else { + // #6138 + if (!chart.series.length) { + chart[counterName] = 0; + } + userOptions["_" + indexName] = i = chart[counterName] % len; + chart[counterName] += 1; + } + if (defaults) { + value = defaults[i]; + } + } + // Set the colorIndex + if (typeof i !== "undefined") { + this[indexName] = i; + } + this[prop] = value; + }, + /** + * Get the series' color based on either the options or pulled from + * global options. + * + * @private + * @function Highcharts.Series#getColor + * @return {void} + */ + getColor: function () { + if (this.chart.styledMode) { + this.getCyclic("color"); + } else if (this.options.colorByPoint) { + // #4359, selected slice got series.color even when colorByPoint + // was set. + this.options.color = null; + } else { + this.getCyclic( + "color", + this.options.color || + defaultOptions.plotOptions[this.type].color, + this.chart.options.colors + ); + } + }, + /** + * Get all points' instances created for this series. + * + * @private + * @function Highcharts.Series#getPointsCollection + * @return {Array<Highcharts.Point>} + */ + getPointsCollection: function () { + return (this.hasGroupedData ? this.points : this.data) || []; + }, + /** + * Get the series' symbol based on either the options or pulled from + * global options. + * + * @private + * @function Highcharts.Series#getSymbol + * @return {void} + */ + getSymbol: function () { + var seriesMarkerOption = this.options.marker; + this.getCyclic( + "symbol", + seriesMarkerOption.symbol, + this.chart.options.symbols + ); + }, + /** + * Finds the index of an existing point that matches the given point + * options. + * + * @private + * @function Highcharts.Series#findPointIndex + * @param {Highcharts.PointOptionsObject} optionsObject + * The options of the point. + * @param {number} fromIndex + * The index to start searching from, used for optimizing + * series with required sorting. + * @returns {number|undefined} + * Returns the index of a matching point, or undefined if no + * match is found. + */ + findPointIndex: function (optionsObject, fromIndex) { + var id = optionsObject.id, + x = optionsObject.x, + oldData = this.points, + matchingPoint, + matchedById, + pointIndex, + matchKey, + dataSorting = this.options.dataSorting; + if (id) { + matchingPoint = this.chart.get(id); + } else if (this.linkedParent || this.enabledDataSorting) { + matchKey = + dataSorting && dataSorting.matchByName ? "name" : "index"; + matchingPoint = find(oldData, function (oldPoint) { + return ( + !oldPoint.touched && + oldPoint[matchKey] === optionsObject[matchKey] + ); + }); + // Add unmatched point as a new point + if (!matchingPoint) { + return void 0; + } + } + if (matchingPoint) { + pointIndex = matchingPoint && matchingPoint.index; + if (typeof pointIndex !== "undefined") { + matchedById = true; + } + } + // Search for the same X in the existing data set + if (typeof pointIndex === "undefined" && isNumber(x)) { + pointIndex = this.xData.indexOf(x, fromIndex); + } + // Reduce pointIndex if data is cropped + if ( + pointIndex !== -1 && + typeof pointIndex !== "undefined" && + this.cropped + ) { + pointIndex = + pointIndex >= this.cropStart + ? pointIndex - this.cropStart + : pointIndex; + } + if ( + !matchedById && + oldData[pointIndex] && + oldData[pointIndex].touched + ) { + pointIndex = void 0; + } + return pointIndex; + }, + /** + * @private + * @borrows LegendSymbolMixin.drawLineMarker as Highcharts.Series#drawLegendSymbol + */ + drawLegendSymbol: LegendSymbolMixin.drawLineMarker, + /** + * Internal function called from setData. If the point count is the same + * as is was, or if there are overlapping X values, just run + * Point.update which is cheaper, allows animation, and keeps references + * to points. This also allows adding or removing points if the X-es + * don't match. + * + * @private + * @function Highcharts.Series#updateData + * + * @param {Array<Highcharts.PointOptionsType>} data + * + * @return {boolean} + */ + updateData: function (data, animation) { + var options = this.options, + dataSorting = options.dataSorting, + oldData = this.points, + pointsToAdd = [], + hasUpdatedByKey, + i, + point, + lastIndex, + requireSorting = this.requireSorting, + equalLength = data.length === oldData.length, + succeeded = true; + this.xIncrement = null; + // Iterate the new data + data.forEach(function (pointOptions, i) { + var id, + x, + pointIndex, + optionsObject = + (defined(pointOptions) && + this.pointClass.prototype.optionsToObject.call( + { series: this }, + pointOptions + )) || + {}; + // Get the x of the new data point + x = optionsObject.x; + id = optionsObject.id; + if (id || isNumber(x)) { + pointIndex = this.findPointIndex(optionsObject, lastIndex); + // Matching X not found + // or used already due to ununique x values (#8995), + // add point (but later) + if (pointIndex === -1 || typeof pointIndex === "undefined") { + pointsToAdd.push(pointOptions); + // Matching X found, update + } else if ( + oldData[pointIndex] && + pointOptions !== options.data[pointIndex] + ) { + oldData[pointIndex].update(pointOptions, false, null, false); + // Mark it touched, below we will remove all points that + // are not touched. + oldData[pointIndex].touched = true; + // Speed optimize by only searching after last known + // index. Performs ~20% bettor on large data sets. + if (requireSorting) { + lastIndex = pointIndex + 1; + } + // Point exists, no changes, don't remove it + } else if (oldData[pointIndex]) { + oldData[pointIndex].touched = true; + } + // If the length is equal and some of the nodes had a + // match in the same position, we don't want to remove + // non-matches. + if ( + !equalLength || + i !== pointIndex || + (dataSorting && dataSorting.enabled) || + this.hasDerivedData + ) { + hasUpdatedByKey = true; + } + } else { + // Gather all points that are not matched + pointsToAdd.push(pointOptions); + } + }, this); + // Remove points that don't exist in the updated data set + if (hasUpdatedByKey) { + i = oldData.length; + while (i--) { + point = oldData[i]; + if (point && !point.touched && point.remove) { + point.remove(false, animation); + } + } + // If we did not find keys (ids or x-values), and the length is the + // same, update one-to-one + } else if (equalLength && (!dataSorting || !dataSorting.enabled)) { + data.forEach(function (point, i) { + // .update doesn't exist on a linked, hidden series (#3709) + // (#10187) + if (oldData[i].update && point !== oldData[i].y) { + oldData[i].update(point, false, null, false); + } + }); + // Don't add new points since those configs are used above + pointsToAdd.length = 0; + // Did not succeed in updating data + } else { + succeeded = false; + } + oldData.forEach(function (point) { + if (point) { + point.touched = false; + } + }); + if (!succeeded) { + return false; + } + // Add new points + pointsToAdd.forEach(function (point) { + this.addPoint(point, false, null, null, false); + }, this); + if (this.xIncrement === null && this.xData && this.xData.length) { + this.xIncrement = arrayMax(this.xData); + this.autoIncrement(); + } + return true; + }, + /** + * Apply a new set of data to the series and optionally redraw it. The + * new data array is passed by reference (except in case of + * `updatePoints`), and may later be mutated when updating the chart + * data. + * + * Note the difference in behaviour when setting the same amount of + * points, or a different amount of points, as handled by the + * `updatePoints` parameter. + * + * @sample highcharts/members/series-setdata/ + * Set new data from a button + * @sample highcharts/members/series-setdata-pie/ + * Set data in a pie + * @sample stock/members/series-setdata/ + * Set new data in Highstock + * @sample maps/members/series-setdata/ + * Set new data in Highmaps + * + * @function Highcharts.Series#setData + * + * @param {Array<Highcharts.PointOptionsType>} data + * Takes an array of data in the same format as described under + * `series.{type}.data` for the given series type, for example a + * line series would take data in the form described under + * [series.line.data](https://api.highcharts.com/highcharts/series.line.data). + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the series is altered. If + * doing more operations on the chart, it is a good idea to set + * redraw to false and call {@link Chart#redraw} after. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * When the updated data is the same length as the existing data, + * points will be updated by default, and animation visualizes + * how the points are changed. Set false to disable animation, or + * a configuration object to set duration or easing. + * + * @param {boolean} [updatePoints=true] + * When this is true, points will be updated instead of replaced + * whenever possible. This occurs a) when the updated data is the + * same length as the existing data, b) when points are matched + * by their id's, or c) when points can be matched by X values. + * This allows updating with animation and performs better. In + * this case, the original array is not passed by reference. Set + * `false` to prevent. + * + * @return {void} + */ + setData: function (data, redraw, animation, updatePoints) { + var series = this, + oldData = series.points, + oldDataLength = (oldData && oldData.length) || 0, + dataLength, + options = series.options, + chart = series.chart, + dataSorting = options.dataSorting, + firstPoint = null, + xAxis = series.xAxis, + i, + turboThreshold = options.turboThreshold, + pt, + xData = this.xData, + yData = this.yData, + pointArrayMap = series.pointArrayMap, + valueCount = pointArrayMap && pointArrayMap.length, + keys = options.keys, + indexOfX = 0, + indexOfY = 1, + updatedData; + data = data || []; + dataLength = data.length; + redraw = pick(redraw, true); + if (dataSorting && dataSorting.enabled) { + data = this.sortData(data); + } + // First try to run Point.update which is cheaper, allows animation, + // and keeps references to points. + if ( + updatePoints !== false && + dataLength && + oldDataLength && + !series.cropped && + !series.hasGroupedData && + series.visible && + // Soft updating has no benefit in boost, and causes JS error + // (#8355) + !series.isSeriesBoosting + ) { + updatedData = this.updateData(data, animation); + } + if (!updatedData) { + // Reset properties + series.xIncrement = null; + series.colorCounter = 0; // for series with colorByPoint (#1547) + // Update parallel arrays + this.parallelArrays.forEach(function (key) { + series[key + "Data"].length = 0; + }); + // In turbo mode, only one- or twodimensional arrays of numbers + // are allowed. The first value is tested, and we assume that + // all the rest are defined the same way. Although the 'for' + // loops are similar, they are repeated inside each if-else + // conditional for max performance. + if (turboThreshold && dataLength > turboThreshold) { + firstPoint = series.getFirstValidPoint(data); + if (isNumber(firstPoint)) { + // assume all points are numbers + for (i = 0; i < dataLength; i++) { + xData[i] = this.autoIncrement(); + yData[i] = data[i]; + } + // Assume all points are arrays when first point is + } else if (isArray(firstPoint)) { + if (valueCount) { + // [x, low, high] or [x, o, h, l, c] + for (i = 0; i < dataLength; i++) { + pt = data[i]; + xData[i] = pt[0]; + yData[i] = pt.slice(1, valueCount + 1); + } + } else { + // [x, y] + if (keys) { + indexOfX = keys.indexOf("x"); + indexOfY = keys.indexOf("y"); + indexOfX = indexOfX >= 0 ? indexOfX : 0; + indexOfY = indexOfY >= 0 ? indexOfY : 1; } - // Handle ignore hidden slices - if (ignoreHiddenPoint) { - series.isDirty = true; + for (i = 0; i < dataLength; i++) { + pt = data[i]; + xData[i] = pt[indexOfX]; + yData[i] = pt[indexOfY]; } - if (redraw) { - chart.redraw(); + } + } else { + // Highcharts expects configs to be numbers or arrays in + // turbo mode + error(12, false, chart); + } + } else { + for (i = 0; i < dataLength; i++) { + // stray commas in oldIE: + if (typeof data[i] !== "undefined") { + pt = { series: series }; + series.pointClass.prototype.applyOptions.apply(pt, [ + data[i], + ]); + series.updateParallelArrays(pt, i); + } + } + } + // Forgetting to cast strings to numbers is a common caveat when + // handling CSV or JSON + if (yData && isString(yData[0])) { + error(14, true, chart); + } + series.data = []; + series.options.data = series.userOptions.data = data; + // destroy old points + i = oldDataLength; + while (i--) { + if (oldData[i] && oldData[i].destroy) { + oldData[i].destroy(); + } + } + // reset minRange (#878) + if (xAxis) { + xAxis.minRange = xAxis.userMinRange; + } + // redraw + series.isDirty = chart.isDirtyBox = true; + series.isDirtyData = !!oldData; + animation = false; + } + // Typically for pie series, points need to be processed and + // generated prior to rendering the legend + if (options.legendType === "point") { + this.processData(); + this.generatePoints(); + } + if (redraw) { + chart.redraw(animation); + } + }, + /** + * Internal function to sort series data + * + * @private + * @function Highcharts.Series#sortData + * @param {Array<Highcharts.PointOptionsType>} data + * Force data grouping. + * @return {Array<Highcharts.PointOptionsObject>} + */ + sortData: function (data) { + var series = this, + options = series.options, + dataSorting = options.dataSorting, + sortKey = dataSorting.sortKey || "y", + sortedData, + getPointOptionsObject = function (series, pointOptions) { + return ( + (defined(pointOptions) && + series.pointClass.prototype.optionsToObject.call( + { + series: series, + }, + pointOptions + )) || + {} + ); + }; + data.forEach(function (pointOptions, i) { + data[i] = getPointOptionsObject(series, pointOptions); + data[i].index = i; + }, this); + // Sorting + sortedData = data.concat().sort(function (a, b) { + var aValue = getNestedProperty(sortKey, a); + var bValue = getNestedProperty(sortKey, b); + return bValue < aValue ? -1 : bValue > aValue ? 1 : 0; + }); + // Set x value depending on the position in the array + sortedData.forEach(function (point, i) { + point.x = i; + }, this); + // Set the same x for linked series points if they don't have their + // own sorting + if (series.linkedSeries) { + series.linkedSeries.forEach(function (linkedSeries) { + var options = linkedSeries.options, + seriesData = options.data; + if ( + (!options.dataSorting || !options.dataSorting.enabled) && + seriesData + ) { + seriesData.forEach(function (pointOptions, i) { + seriesData[i] = getPointOptionsObject( + linkedSeries, + pointOptions + ); + if (data[i]) { + seriesData[i].x = data[i].x; + seriesData[i].index = i; } + }); + linkedSeries.setData(seriesData, false); } - }, - /** - * Set or toggle whether the slice is cut out from the pie - * - * @private - * @function Highcharts.seriesTypes.pie#pointClass#slice - * @param {boolean} sliced - * When undefined, the slice state is toggled. - * @param {boolean} redraw - * Whether to redraw the chart. True by default. - * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} - * Animation options. - * @return {void} - */ - slice: function (sliced, redraw, animation) { - var point = this, - series = point.series, - chart = series.chart; - setAnimation(animation, chart); - // redraw is true by default - redraw = pick(redraw, true); + }); + } + return data; + }, + /** + * Internal function to process the data by cropping away unused data + * points if the series is longer than the crop threshold. This saves + * computing time for large series. + * + * @private + * @function Highcharts.Series#getProcessedData + * @param {boolean} [forceExtremesFromAll] + * Force getting extremes of a total series data range. + * @return {Highcharts.SeriesProcessedDataObject} + */ + getProcessedData: function (forceExtremesFromAll) { + var series = this, + // copied during slice operation: + processedXData = series.xData, + processedYData = series.yData, + dataLength = processedXData.length, + croppedData, + cropStart = 0, + cropped, + distance, + closestPointRange, + xAxis = series.xAxis, + i, // loop variable + options = series.options, + cropThreshold = options.cropThreshold, + getExtremesFromAll = + forceExtremesFromAll || + series.getExtremesFromAll || + options.getExtremesFromAll, // #4599 + isCartesian = series.isCartesian, + xExtremes, + val2lin = xAxis && xAxis.val2lin, + isLog = !!(xAxis && xAxis.logarithmic), + throwOnUnsorted = series.requireSorting, + min, + max; + if (xAxis) { + // corrected for log axis (#3053) + xExtremes = xAxis.getExtremes(); + min = xExtremes.min; + max = xExtremes.max; + } + // optionally filter out points outside the plot area + if ( + isCartesian && + series.sorted && + !getExtremesFromAll && + (!cropThreshold || dataLength > cropThreshold || series.forceCrop) + ) { + // it's outside current extremes + if ( + processedXData[dataLength - 1] < min || + processedXData[0] > max + ) { + processedXData = []; + processedYData = []; + // only crop if it's actually spilling out + } else if ( + series.yData && + (processedXData[0] < min || + processedXData[dataLength - 1] > max) + ) { + croppedData = this.cropData( + series.xData, + series.yData, + min, + max + ); + processedXData = croppedData.xData; + processedYData = croppedData.yData; + cropStart = croppedData.start; + cropped = true; + } + } + // Find the closest distance between processed points + i = processedXData.length || 1; + while (--i) { + distance = isLog + ? val2lin(processedXData[i]) - val2lin(processedXData[i - 1]) + : processedXData[i] - processedXData[i - 1]; + if ( + distance > 0 && + (typeof closestPointRange === "undefined" || + distance < closestPointRange) + ) { + closestPointRange = distance; + // Unsorted data is not supported by the line tooltip, as well + // as data grouping and navigation in Stock charts (#725) and + // width calculation of columns (#1900) + } else if (distance < 0 && throwOnUnsorted) { + error(15, false, series.chart); + throwOnUnsorted = false; // Only once + } + } + return { + xData: processedXData, + yData: processedYData, + cropped: cropped, + cropStart: cropStart, + closestPointRange: closestPointRange, + }; + }, + /** + * Internal function to apply processed data. + * In Highstock, this function is extended to provide data grouping. + * + * @private + * @function Highcharts.Series#processData + * @param {boolean} [force] + * Force data grouping. + * @return {boolean|undefined} + */ + processData: function (force) { + var series = this, + xAxis = series.xAxis, + processedData; + // If the series data or axes haven't changed, don't go through + // this. Return false to pass the message on to override methods + // like in data grouping. + if ( + series.isCartesian && + !series.isDirty && + !xAxis.isDirty && + !series.yAxis.isDirty && + !force + ) { + return false; + } + processedData = series.getProcessedData(); + // Record the properties + series.cropped = processedData.cropped; // undefined or true + series.cropStart = processedData.cropStart; + series.processedXData = processedData.xData; + series.processedYData = processedData.yData; + series.closestPointRange = series.basePointRange = + processedData.closestPointRange; + }, + /** + * Iterate over xData and crop values between min and max. Returns + * object containing crop start/end cropped xData with corresponding + * part of yData, dataMin and dataMax within the cropped range. + * + * @private + * @function Highcharts.Series#cropData + * @param {Array<number>} xData + * @param {Array<number>} yData + * @param {number} min + * @param {number} max + * @param {number} [cropShoulder] + * @return {Highcharts.SeriesCropDataObject} + */ + cropData: function (xData, yData, min, max, cropShoulder) { + var dataLength = xData.length, + cropStart = 0, + cropEnd = dataLength, + i, + j; + // line-type series need one point outside + cropShoulder = pick(cropShoulder, this.cropShoulder); + // iterate up to find slice start + for (i = 0; i < dataLength; i++) { + if (xData[i] >= min) { + cropStart = Math.max(0, i - cropShoulder); + break; + } + } + // proceed to find slice end + for (j = i; j < dataLength; j++) { + if (xData[j] > max) { + cropEnd = j + cropShoulder; + break; + } + } + return { + xData: xData.slice(cropStart, cropEnd), + yData: yData.slice(cropStart, cropEnd), + start: cropStart, + end: cropEnd, + }; + }, + /** + * Generate the data point after the data has been processed by cropping + * away unused points and optionally grouped in Highcharts Stock. + * + * @private + * @function Highcharts.Series#generatePoints + */ + generatePoints: function () { + var series = this, + options = series.options, + dataOptions = options.data, + data = series.data, + dataLength, + processedXData = series.processedXData, + processedYData = series.processedYData, + PointClass = series.pointClass, + processedDataLength = processedXData.length, + cropStart = series.cropStart || 0, + cursor, + hasGroupedData = series.hasGroupedData, + keys = options.keys, + point, + points = [], + i; + if (!data && !hasGroupedData) { + var arr = []; + arr.length = dataOptions.length; + data = series.data = arr; + } + if (keys && hasGroupedData) { + // grouped data has already applied keys (#6590) + series.options.keys = false; + } + for (i = 0; i < processedDataLength; i++) { + cursor = cropStart + i; + if (!hasGroupedData) { + point = data[cursor]; + // #970: + if (!point && typeof dataOptions[cursor] !== "undefined") { + data[cursor] = point = new PointClass().init( + series, + dataOptions[cursor], + processedXData[i] + ); + } + } else { + // splat the y data in case of ohlc data array + point = new PointClass().init( + series, + [processedXData[i]].concat(splat(processedYData[i])) + ); /** - * Pie series only. Whether to display a slice offset from the - * center. - * @name Highcharts.Point#sliced - * @type {boolean|undefined} + * Highstock only. If a point object is created by data + * grouping, it doesn't reflect actual points in the raw + * data. In this case, the `dataGroup` property holds + * information that points back to the raw data. + * + * - `dataGroup.start` is the index of the first raw data + * point in the group. + * + * - `dataGroup.length` is the amount of points in the + * group. + * + * @product highstock + * + * @name Highcharts.Point#dataGroup + * @type {Highcharts.DataGroupingInfoObject|undefined} */ - // if called without an argument, toggle - point.sliced = point.options.sliced = sliced = - defined(sliced) ? sliced : !point.sliced; - // update userOptions.data - series.options.data[series.data.indexOf(point)] = - point.options; - if (point.graphic) { - point.graphic.animate(this.getTranslate()); - } - if (point.shadowGroup) { - point.shadowGroup.animate(this.getTranslate()); + point.dataGroup = series.groupMap[i]; + if (point.dataGroup.options) { + point.options = point.dataGroup.options; + extend(point, point.dataGroup.options); + // Collision of props and options (#9770) + delete point.dataLabels; } - }, + } + if (point) { + // #6279 + /** + * Contains the point's index in the `Series.points` array. + * + * @name Highcharts.Point#index + * @type {number} + * @readonly + */ + point.index = cursor; // For faster access in Point.update + points[i] = point; + } + } + // restore keys options (#6590) + series.options.keys = keys; + // Hide cropped-away points - this only runs when the number of + // points is above cropThreshold, or when swithching view from + // non-grouped data to grouped data (#637) + if ( + data && + (processedDataLength !== (dataLength = data.length) || + hasGroupedData) + ) { + for (i = 0; i < dataLength; i++) { + // when has grouped data, clear all points + if (i === cropStart && !hasGroupedData) { + i += processedDataLength; + } + if (data[i]) { + data[i].destroyElements(); + data[i].plotX = void 0; // #1003 + } + } + } /** - * @private - * @function Highcharts.seriesTypes.pie#pointClass#getTranslate - * @return {Highcharts.TranslationAttributes} - */ - getTranslate: function () { - return this.sliced ? this.slicedTranslation : { - translateX: 0, - translateY: 0 + * Read only. An array containing those values converted to points. + * In case the series data length exceeds the `cropThreshold`, or if + * the data is grouped, `series.data` doesn't contain all the + * points. Also, in case a series is hidden, the `data` array may be + * empty. To access raw values, `series.options.data` will always be + * up to date. `Series.data` only contains the points that have been + * created on demand. To modify the data, use + * {@link Highcharts.Series#setData} or + * {@link Highcharts.Point#update}. + * + * @see Series.points + * + * @name Highcharts.Series#data + * @type {Array<Highcharts.Point>} + */ + series.data = data; + /** + * An array containing all currently visible point objects. In case + * of cropping, the cropped-away points are not part of this array. + * The `series.points` array starts at `series.cropStart` compared + * to `series.data` and `series.options.data`. If however the series + * data is grouped, these can't be correlated one to one. To modify + * the data, use {@link Highcharts.Series#setData} or + * {@link Highcharts.Point#update}. + * + * @name Highcharts.Series#points + * @type {Array<Highcharts.Point>} + */ + series.points = points; + fireEvent(this, "afterGeneratePoints"); + }, + /** + * Get current X extremes for the visible data. + * + * @private + * @function Highcharts.Series#getXExtremes + * + * @param {Array<number>} xData + * The data to inspect. Defaults to the current data within the + * visible range. + * @return {Highcharts.RangeObject} + */ + getXExtremes: function (xData) { + return { + min: arrayMin(xData), + max: arrayMax(xData), + }; + }, + /** + * Calculate Y extremes for the visible data. The result is returned + * as an object with `dataMin` and `dataMax` properties. + * + * @private + * @function Highcharts.Series#getExtremes + * @param {Array<number>} [yData] + * The data to inspect. Defaults to the current data within the + * visible range. + * @param {boolean} [forceExtremesFromAll] + * Force getting extremes of a total series data range. + * @return {Highcharts.DataExtremesObject} + */ + getExtremes: function (yData, forceExtremesFromAll) { + var xAxis = this.xAxis, + yAxis = this.yAxis, + xData = this.processedXData || this.xData, + yDataLength, + activeYData = [], + activeCounter = 0, + // #2117, need to compensate for log X axis + xExtremes, + xMin = 0, + xMax = 0, + validValue, + withinRange, + // Handle X outside the viewed area. This does not work with + // non-sorted data like scatter (#7639). + shoulder = this.requireSorting ? this.cropShoulder : 0, + positiveValuesOnly = yAxis ? yAxis.positiveValuesOnly : false, + x, + y, + i, + j; + yData = yData || this.stackedYData || this.processedYData || []; + yDataLength = yData.length; + if (xAxis) { + xExtremes = xAxis.getExtremes(); + xMin = xExtremes.min; + xMax = xExtremes.max; + } + for (i = 0; i < yDataLength; i++) { + x = xData[i]; + y = yData[i]; + // For points within the visible range, including the first + // point outside the visible range (#7061), consider y extremes. + validValue = + (isNumber(y) || isArray(y)) && + (y.length || y > 0 || !positiveValuesOnly); + withinRange = + forceExtremesFromAll || + this.getExtremesFromAll || + this.options.getExtremesFromAll || + this.cropped || + !xAxis || // for colorAxis support + ((xData[i + shoulder] || x) >= xMin && + (xData[i - shoulder] || x) <= xMax); + if (validValue && withinRange) { + j = y.length; + if (j) { + // array, like ohlc or range data + while (j--) { + if (isNumber(y[j])) { + // #7380, #11513 + activeYData[activeCounter++] = y[j]; + } + } + } else { + activeYData[activeCounter++] = y; + } + } + } + var dataExtremes = { + dataMin: arrayMin(activeYData), + dataMax: arrayMax(activeYData), + }; + fireEvent(this, "afterGetExtremes", { dataExtremes: dataExtremes }); + return dataExtremes; + }, + /** + * Set the current data extremes as `dataMin` and `dataMax` on the + * Series item. Use this only when the series properties should be + * updated. + * + * @private + * @function Highcharts.Series#applyExtremes + * @return {void} + */ + applyExtremes: function () { + var dataExtremes = this.getExtremes(); + /** + * Contains the minimum value of the series' data point. Some series + * types like `networkgraph` do not support this property as they + * lack a `y`-value. + * @name Highcharts.Series#dataMin + * @type {number|undefined} + * @readonly + */ + this.dataMin = dataExtremes.dataMin; + /** + * Contains the maximum value of the series' data point. Some series + * types like `networkgraph` do not support this property as they + * lack a `y`-value. + * @name Highcharts.Series#dataMax + * @type {number|undefined} + * @readonly + */ + this.dataMax = dataExtremes.dataMax; + return dataExtremes; + }, + /** + * Find and return the first non null point in the data + * + * @private + * @function Highcharts.Series.getFirstValidPoint + * @param {Array<Highcharts.PointOptionsType>} data + * Array of options for points + * + * @return {Highcharts.PointOptionsType} + */ + getFirstValidPoint: function (data) { + var firstPoint = null, + dataLength = data.length, + i = 0; + while (firstPoint === null && i < dataLength) { + firstPoint = data[i]; + i++; + } + return firstPoint; + }, + /** + * Translate data points from raw data values to chart specific + * positioning data needed later in the `drawPoints` and `drawGraph` + * functions. This function can be overridden in plugins and custom + * series type implementations. + * + * @function Highcharts.Series#translate + * @return {void} + * @fires Highcharts.Series#events:translate + */ + translate: function () { + if (!this.processedXData) { + // hidden series + this.processData(); + } + this.generatePoints(); + var series = this, + options = series.options, + stacking = options.stacking, + xAxis = series.xAxis, + categories = xAxis.categories, + enabledDataSorting = series.enabledDataSorting, + yAxis = series.yAxis, + points = series.points, + dataLength = points.length, + hasModifyValue = !!series.modifyValue, + i, + pointPlacement = series.pointPlacementToXValue(), // #7860 + dynamicallyPlaced = Boolean(pointPlacement), + threshold = options.threshold, + stackThreshold = options.startFromThreshold ? threshold : 0, + plotX, + lastPlotX, + stackIndicator, + zoneAxis = this.zoneAxis || "y", + closestPointRangePx = Number.MAX_VALUE; + /** + * Plotted coordinates need to be within a limited range. Drawing + * too far outside the viewport causes various rendering issues + * (#3201, #3923, #7555). + * @private + */ + function limitedRange(val) { + return clamp(val, -1e5, 1e5); + } + // Translate each point + for (i = 0; i < dataLength; i++) { + var point = points[i], + xValue = point.x, + yValue = point.y, + yBottom = point.low, + stack = + stacking && + yAxis.stacking && + yAxis.stacking.stacks[ + (series.negStacks && + yValue < (stackThreshold ? 0 : threshold) + ? "-" + : "") + series.stackKey + ], + pointStack, + stackValues; + if ( + (yAxis.positiveValuesOnly && + !yAxis.validatePositiveValue(yValue)) || + (xAxis.positiveValuesOnly && + !xAxis.validatePositiveValue(xValue)) + ) { + point.isNull = true; + } + // Get the plotX translation + point.plotX = plotX = correctFloat( + // #5236 + limitedRange( + xAxis.translate( + // #3923 + xValue, + 0, + 0, + 0, + 1, + pointPlacement, + this.type === "flags" + ) + ) // #3923 + ); + // Calculate the bottom y value for stacked series + if (stacking && series.visible && stack && stack[xValue]) { + stackIndicator = series.getStackIndicator( + stackIndicator, + xValue, + series.index + ); + if (!point.isNull) { + pointStack = stack[xValue]; + stackValues = pointStack.points[stackIndicator.key]; + } + } + if (isArray(stackValues)) { + yBottom = stackValues[0]; + yValue = stackValues[1]; + if ( + yBottom === stackThreshold && + stackIndicator.key === stack[xValue].base + ) { + yBottom = pick(isNumber(threshold) && threshold, yAxis.min); + } + // #1200, #1232 + if (yAxis.positiveValuesOnly && yBottom <= 0) { + yBottom = null; + } + point.total = point.stackTotal = pointStack.total; + point.percentage = + pointStack.total && (point.y / pointStack.total) * 100; + point.stackY = yValue; + // Place the stack label + // in case of variwide series (where widths of points are + // different in most cases), stack labels are positioned + // wrongly, so the call of the setOffset is omited here and + // labels are correctly positioned later, at the end of the + // variwide's translate function (#10962) + if (!series.irregularWidths) { + pointStack.setOffset( + series.pointXOffset || 0, + series.barW || 0 + ); + } + } + // Set translated yBottom or remove it + point.yBottom = defined(yBottom) + ? limitedRange(yAxis.translate(yBottom, 0, 1, 0, 1)) + : null; + // general hook, used for Highstock compare mode + if (hasModifyValue) { + yValue = series.modifyValue(yValue, point); + } + // Set the the plotY value, reset it for redraws + // #3201 + point.plotY = + typeof yValue === "number" && yValue !== Infinity + ? limitedRange(yAxis.translate(yValue, 0, 1, 0, 1)) + : void 0; + point.isInside = this.isPointInside(point); + // Set client related positions for mouse tracking + point.clientX = dynamicallyPlaced + ? correctFloat( + xAxis.translate(xValue, 0, 0, 0, 1, pointPlacement) + ) + : plotX; // #1514, #5383, #5518 + // Negative points. For bubble charts, this means negative z + // values (#9728) + point.negative = + point[zoneAxis] < + (options[zoneAxis + "Threshold"] || threshold || 0); + // some API data + point.category = + categories && typeof categories[point.x] !== "undefined" + ? categories[point.x] + : point.x; + // Determine auto enabling of markers (#3635, #5099) + if (!point.isNull && point.visible !== false) { + if (typeof lastPlotX !== "undefined") { + closestPointRangePx = Math.min( + closestPointRangePx, + Math.abs(plotX - lastPlotX) + ); + } + lastPlotX = plotX; + } + // Find point zone + point.zone = this.zones.length && point.getZone(); + // Animate new points with data sorting + if (!point.graphic && series.group && enabledDataSorting) { + point.isNew = true; + } + } + series.closestPointRangePx = closestPointRangePx; + fireEvent(this, "afterTranslate"); + }, + /** + * Return the series points with null points filtered out. + * + * @function Highcharts.Series#getValidPoints + * + * @param {Array<Highcharts.Point>} [points] + * The points to inspect, defaults to {@link Series.points}. + * + * @param {boolean} [insideOnly=false] + * Whether to inspect only the points that are inside the visible + * view. + * + * @param {boolean} [allowNull=false] + * Whether to allow null points to pass as valid points. + * + * @return {Array<Highcharts.Point>} + * The valid points. + */ + getValidPoints: function (points, insideOnly, allowNull) { + var chart = this.chart; + // #3916, #5029, #5085 + return (points || this.points || []).filter(function isValidPoint( + point + ) { + if ( + insideOnly && + !chart.isInsidePlot(point.plotX, point.plotY, chart.inverted) + ) { + return false; + } + return point.visible !== false && (allowNull || !point.isNull); + }); + }, + /** + * Get the clipping for the series. Could be called for a series to + * initiate animating the clip or to set the final clip (only width + * and x). + * + * @private + * @function Highcharts.Series#getClip + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * Initialize the animation. + * @param {boolean} [finalBox] + * Final size for the clip - end state for the animation. + * @return {Highcharts.Dictionary<number>} + */ + getClipBox: function (animation, finalBox) { + var series = this, + options = series.options, + chart = series.chart, + inverted = chart.inverted, + xAxis = series.xAxis, + yAxis = xAxis && series.yAxis, + clipBox, + scrollablePlotAreaOptions = + chart.options.chart.scrollablePlotArea || {}; + if (animation && options.clip === false && yAxis) { + // support for not clipped series animation (#10450) + clipBox = inverted + ? { + y: -chart.chartWidth + yAxis.len + yAxis.pos, + height: chart.chartWidth, + width: chart.chartHeight, + x: -chart.chartHeight + xAxis.len + xAxis.pos, + } + : { + y: -yAxis.pos, + height: chart.chartHeight, + width: chart.chartWidth, + x: -xAxis.pos, + }; + // x and width will be changed later when setting for animation + // initial state in Series.setClip + } else { + clipBox = series.clipBox || chart.clipBox; + if (finalBox) { + clipBox.width = chart.plotSizeX; + clipBox.x = + (chart.scrollablePixelsX || 0) * + (scrollablePlotAreaOptions.scrollPositionX || 0); + } + } + return !finalBox + ? clipBox + : { + width: clipBox.width, + x: clipBox.x, }; - }, - /** - * @private - * @function Highcharts.seriesTypes.pie#pointClass#haloPath - * @param {number} size - * @return {Highcharts.SVGPathArray} - */ - haloPath: function (size) { - var shapeArgs = this.shapeArgs; - return this.sliced || !this.visible ? - [] : - this.series.chart.renderer.symbols.arc(shapeArgs.x, shapeArgs.y, shapeArgs.r + size, shapeArgs.r + size, { - // Substract 1px to ensure the background is not bleeding - // through between the halo and the slice (#7495). - innerR: shapeArgs.r - 1, - start: shapeArgs.start, - end: shapeArgs.end - }); - }, - connectorShapes: { - // only one available before v7.0.0 - fixedOffset: function (labelPosition, connectorPosition, options) { - var breakAt = connectorPosition.breakAt, - touchingSliceAt = connectorPosition.touchingSliceAt, - lineSegment = options.softConnector ? [ - 'C', - // 1st control point (of the curve) - labelPosition.x + - // 5 gives the connector a little horizontal bend - (labelPosition.alignment === 'left' ? -5 : 5), - labelPosition.y, - 2 * breakAt.x - touchingSliceAt.x, - 2 * breakAt.y - touchingSliceAt.y, - breakAt.x, - breakAt.y // - ] : [ - 'L', - breakAt.x, - breakAt.y - ]; - // assemble the path - return ([ - ['M', labelPosition.x, labelPosition.y], - lineSegment, - ['L', touchingSliceAt.x, touchingSliceAt.y] - ]); - }, - straight: function (labelPosition, connectorPosition) { - var touchingSliceAt = connectorPosition.touchingSliceAt; - // direct line to the slice - return [ - ['M', labelPosition.x, labelPosition.y], - ['L', touchingSliceAt.x, touchingSliceAt.y] - ]; - }, - crookedLine: function (labelPosition, connectorPosition, options) { - var touchingSliceAt = connectorPosition.touchingSliceAt, - series = this.series, - pieCenterX = series.center[0], - plotWidth = series.chart.plotWidth, - plotLeft = series.chart.plotLeft, - alignment = labelPosition.alignment, - radius = this.shapeArgs.r, - crookDistance = relativeLength(// % to fraction - options.crookDistance, 1), - crookX = alignment === 'left' ? - pieCenterX + radius + (plotWidth + plotLeft - - pieCenterX - radius) * (1 - crookDistance) : - plotLeft + (pieCenterX - radius) * crookDistance, - segmentWithCrook = [ - 'L', - crookX, - labelPosition.y - ], - useCrook = true; - // crookedLine formula doesn't make sense if the path overlaps - // the label - use straight line instead in that case - if (alignment === 'left' ? - (crookX > labelPosition.x || crookX < touchingSliceAt.x) : - (crookX < labelPosition.x || crookX > touchingSliceAt.x)) { - useCrook = false; - } - // assemble the path - var path = [ - ['M', - labelPosition.x, - labelPosition.y] - ]; - if (useCrook) { - path.push(segmentWithCrook); - } - path.push(['L', touchingSliceAt.x, touchingSliceAt.y]); - return path; - } - }, - /** - * Extendable method for getting the path of the connector between the - * data label and the pie slice. - */ - getConnectorPath: function () { - var labelPosition = this.labelPosition, - options = this.series.options.dataLabels, - connectorShape = options.connectorShape, - predefinedShapes = this.connectorShapes; - // find out whether to use the predefined shape - if (predefinedShapes[connectorShape]) { - connectorShape = predefinedShapes[connectorShape]; + }, + /** + * Set the clipping for the series. For animated series it is called + * twice, first to initiate animating the clip then the second time + * without the animation to set the final clip. + * + * @private + * @function Highcharts.Series#setClip + * @param {boolean|Highcharts.AnimationOptionsObject} [animation] + */ + setClip: function (animation) { + var chart = this.chart, + options = this.options, + renderer = chart.renderer, + inverted = chart.inverted, + seriesClipBox = this.clipBox, + clipBox = this.getClipBox(animation), + sharedClipKey = + this.sharedClipKey || + [ + "_sharedClip", + animation && animation.duration, + animation && animation.easing, + clipBox.height, + options.xAxis, + options.yAxis, + ].join(","), // #4526 + clipRect = chart[sharedClipKey], + markerClipRect = chart[sharedClipKey + "m"]; + if (animation) { + clipBox.width = 0; + if (inverted) { + clipBox.x = + chart.plotHeight + + (options.clip !== false ? 0 : chart.plotTop); + } + } + // If a clipping rectangle with the same properties is currently + // present in the chart, use that. + if (!clipRect) { + // When animation is set, prepare the initial positions + if (animation) { + chart[sharedClipKey + "m"] = markerClipRect = renderer.clipRect( + // include the width of the first marker + inverted ? chart.plotSizeX + 99 : -99, + inverted ? -chart.plotLeft : -chart.plotTop, + 99, + inverted ? chart.chartWidth : chart.chartHeight + ); + } + chart[sharedClipKey] = clipRect = renderer.clipRect(clipBox); + // Create hashmap for series indexes + clipRect.count = { length: 0 }; + // When the series is rendered again before starting animating, in + // compliance to a responsive rule + } else if (!chart.hasLoaded) { + clipRect.attr(clipBox); + } + if (animation) { + if (!clipRect.count[this.index]) { + clipRect.count[this.index] = true; + clipRect.count.length += 1; + } + } + if (options.clip !== false || animation) { + this.group.clip( + animation || seriesClipBox ? clipRect : chart.clipRect + ); + this.markerGroup.clip(markerClipRect); + this.sharedClipKey = sharedClipKey; + } + // Remove the shared clipping rectangle when all series are shown + if (!animation) { + if (clipRect.count[this.index]) { + delete clipRect.count[this.index]; + clipRect.count.length -= 1; + } + if ( + clipRect.count.length === 0 && + sharedClipKey && + chart[sharedClipKey] + ) { + if (!seriesClipBox) { + chart[sharedClipKey] = chart[sharedClipKey].destroy(); + } + if (chart[sharedClipKey + "m"]) { + chart[sharedClipKey + "m"] = + chart[sharedClipKey + "m"].destroy(); + } + } + } + }, + /** + * Animate in the series. Called internally twice. First with the `init` + * parameter set to true, which sets up the initial state of the + * animation. Then when ready, it is called with the `init` parameter + * undefined, in order to perform the actual animation. After the + * second run, the function is removed. + * + * @function Highcharts.Series#animate + * + * @param {boolean} [init] + * Initialize the animation. + */ + animate: function (init) { + var series = this, + chart = series.chart, + animation = animObject(series.options.animation), + clipRect, + sharedClipKey, + finalBox; + // Initialize the animation. Set up the clipping rectangle. + if (!chart.hasRendered) { + if (init) { + series.setClip(animation); + // Run the animation + } else { + sharedClipKey = this.sharedClipKey; + clipRect = chart[sharedClipKey]; + finalBox = series.getClipBox(animation, true); + if (clipRect) { + clipRect.animate(finalBox, animation); + } + if (chart[sharedClipKey + "m"]) { + chart[sharedClipKey + "m"].animate( + { + width: finalBox.width + 99, + x: finalBox.x - (chart.inverted ? 0 : 99), + }, + animation + ); } - return connectorShape.call(this, { - // pass simplified label position object for user's convenience - x: labelPosition.final.x, - y: labelPosition.final.y, - alignment: labelPosition.alignment - }, labelPosition.connectorPosition, options); + } } - } - /* eslint-enable valid-jsdoc */ - ); - /** - * A `pie` series. If the [type](#series.pie.type) option is not specified, - * it is inherited from [chart.type](#chart.type). - * - * @extends series,plotOptions.pie - * @excluding cropThreshold, dataParser, dataURL, stack, xAxis, yAxis, - * dataSorting, step, boostThreshold, boostBlending - * @product highcharts - * @apioption series.pie - */ - /** - * An array of data points for the series. For the `pie` series type, - * points can be given in the following ways: - * - * 1. An array of numerical values. In this case, the numerical values will be - * interpreted as `y` options. Example: - * ```js - * data: [0, 5, 3, 5] - * ``` - * - * 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.pie.turboThreshold), - * this option is not available. - * ```js - * data: [{ - * y: 1, - * name: "Point2", - * color: "#00FF00" - * }, { - * y: 7, - * name: "Point1", - * color: "#FF00FF" - * }] - * ``` - * - * @sample {highcharts} highcharts/chart/reflow-true/ - * Numerical values - * @sample {highcharts} highcharts/series/data-array-of-arrays/ - * Arrays of numeric x and y - * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ - * Arrays of datetime x and y - * @sample {highcharts} highcharts/series/data-array-of-name-value/ - * Arrays of point.name and y - * @sample {highcharts} highcharts/series/data-array-of-objects/ - * Config objects - * - * @type {Array<number|Array<string,(number|null)>|null|*>} - * @extends series.line.data - * @excluding marker, x - * @product highcharts - * @apioption series.pie.data - */ - /** - * @type {Highcharts.SeriesPieDataLabelsOptionsObject} - * @product highcharts - * @apioption series.pie.data.dataLabels - */ - /** - * The sequential index of the data point in the legend. - * - * @type {number} - * @product highcharts - * @apioption series.pie.data.legendIndex - */ - /** - * Whether to display a slice offset from the center. - * - * @sample {highcharts} highcharts/point/sliced/ - * One sliced point - * - * @type {boolean} - * @product highcharts - * @apioption series.pie.data.sliced - */ - /** - * @excluding legendItemClick - * @product highcharts - * @apioption series.pie.events - */ - ''; // placeholder for transpiled doclets above - - }); - _registerModule(_modules, 'Core/Series/DataLabels.js', [_modules['Core/Animation/AnimationUtilities.js'], _modules['Core/Globals.js'], _modules['Core/Series/CartesianSeries.js'], _modules['Core/Utilities.js']], function (A, H, CartesianSeries, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var getDeferredAnimation = A.getDeferredAnimation; - var noop = H.noop, - seriesTypes = H.seriesTypes; - var arrayMax = U.arrayMax, - clamp = U.clamp, - defined = U.defined, - extend = U.extend, - fireEvent = U.fireEvent, - format = U.format, - isArray = U.isArray, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick, - relativeLength = U.relativeLength, - splat = U.splat, - stableSort = U.stableSort; - /** - * Callback JavaScript function to format the data label as a string. Note that - * if a `format` is defined, the format takes precedence and the formatter is - * ignored. - * - * @callback Highcharts.DataLabelsFormatterCallbackFunction - * - * @param {Highcharts.PointLabelObject} this - * Data label context to format - * - * @param {Highcharts.DataLabelsOptions} options - * [API options](/highcharts/plotOptions.series.dataLabels) of the data label - * - * @return {number|string|null|undefined} - * Formatted data label text - */ - /** - * Values for handling data labels that flow outside the plot area. - * - * @typedef {"allow"|"justify"} Highcharts.DataLabelsOverflowValue - */ - ''; // detach doclets above - /* eslint-disable valid-jsdoc */ - /** - * General distribution algorithm for distributing labels of differing size - * along a confined length in two dimensions. The algorithm takes an array of - * objects containing a size, a target and a rank. It will place the labels as - * close as possible to their targets, skipping the lowest ranked labels if - * necessary. - * - * @private - * @function Highcharts.distribute - * @param {Highcharts.DataLabelsBoxArray} boxes - * @param {number} len - * @param {number} [maxDistance] - * @return {void} - */ - H.distribute = function (boxes, len, maxDistance) { - var i, - overlapping = true, - origBoxes = boxes, // Original array will be altered with added .pos - restBoxes = [], // The outranked overshoot - box, - target, - total = 0, - reducedLen = origBoxes.reducedLen || len; - /** - * @private - */ - function sortByTarget(a, b) { - return a.target - b.target; + }, + /** + * This runs after animation to land on the final plot clipping. + * + * @private + * @function Highcharts.Series#afterAnimate + * @fires Highcharts.Series#event:afterAnimate + */ + afterAnimate: function () { + this.setClip(); + fireEvent(this, "afterAnimate"); + this.finishedAnimating = true; + }, + /** + * Draw the markers for line-like series types, and columns or other + * graphical representation for {@link Point} objects for other series + * types. The resulting element is typically stored as + * {@link Point.graphic}, and is created on the first call and updated + * and moved on subsequent calls. + * + * @function Highcharts.Series#drawPoints + */ + drawPoints: function () { + var series = this, + points = series.points, + chart = series.chart, + i, + point, + graphic, + verb, + options = series.options, + seriesMarkerOptions = options.marker, + pointMarkerOptions, + hasPointMarker, + markerGroup = series[series.specialGroup] || series.markerGroup, + xAxis = series.xAxis, + markerAttribs, + globallyEnabled = pick( + seriesMarkerOptions.enabled, + !xAxis || xAxis.isRadial ? true : null, + // Use larger or equal as radius is null in bubbles (#6321) + series.closestPointRangePx >= + seriesMarkerOptions.enabledThreshold * + seriesMarkerOptions.radius + ); + if ( + seriesMarkerOptions.enabled !== false || + series._hasPointMarkers + ) { + for (i = 0; i < points.length; i++) { + point = points[i]; + graphic = point.graphic; + verb = graphic ? "animate" : "attr"; + pointMarkerOptions = point.marker || {}; + hasPointMarker = !!point.marker; + var shouldDrawMarker = + ((globallyEnabled && + typeof pointMarkerOptions.enabled === "undefined") || + pointMarkerOptions.enabled) && + !point.isNull && + point.visible !== false; + // only draw the point if y is defined + if (shouldDrawMarker) { + // Shortcuts + var symbol = pick(pointMarkerOptions.symbol, series.symbol); + markerAttribs = series.markerAttribs( + point, + point.selected && "select" + ); + // Set starting position for point sliding animation. + if (series.enabledDataSorting) { + point.startXPos = xAxis.reversed + ? -markerAttribs.width + : xAxis.width; + } + var isInside = point.isInside !== false; + if (graphic) { + // update + // Since the marker group isn't clipped, each + // individual marker must be toggled + graphic[isInside ? "show" : "hide"](isInside).animate( + markerAttribs + ); + } else if ( + isInside && + (markerAttribs.width > 0 || point.hasImage) + ) { + /** + * The graphic representation of the point. + * Typically this is a simple shape, like a `rect` + * for column charts or `path` for line markers, but + * for some complex series types like boxplot or 3D + * charts, the graphic may be a `g` element + * containing other shapes. The graphic is generated + * the first time {@link Series#drawPoints} runs, + * and updated and moved on subsequent runs. + * + * @name Point#graphic + * @type {SVGElement} + */ + point.graphic = graphic = chart.renderer + .symbol( + symbol, + markerAttribs.x, + markerAttribs.y, + markerAttribs.width, + markerAttribs.height, + hasPointMarker + ? pointMarkerOptions + : seriesMarkerOptions + ) + .add(markerGroup); + // Sliding animation for new points + if (series.enabledDataSorting && chart.hasRendered) { + graphic.attr({ + x: point.startXPos, + }); + verb = "animate"; + } + } + if (graphic && verb === "animate") { + // update + // Since the marker group isn't clipped, each + // individual marker must be toggled + graphic[isInside ? "show" : "hide"](isInside).animate( + markerAttribs + ); + } + // Presentational attributes + if (graphic && !chart.styledMode) { + graphic[verb]( + series.pointAttribs(point, point.selected && "select") + ); + } + if (graphic) { + graphic.addClass(point.getClassName(), true); + } + } else if (graphic) { + point.graphic = graphic.destroy(); // #1269 + } + } + } + }, + /** + * Get non-presentational attributes for a point. Used internally for + * both styled mode and classic. Can be overridden for different series + * types. + * + * @see Series#pointAttribs + * + * @function Highcharts.Series#markerAttribs + * + * @param {Highcharts.Point} point + * The Point to inspect. + * + * @param {string} [state] + * The state, can be either `hover`, `select` or undefined. + * + * @return {Highcharts.SVGAttributes} + * A hash containing those attributes that are not settable from + * CSS. + */ + markerAttribs: function (point, state) { + var seriesOptions = this.options, + seriesMarkerOptions = seriesOptions.marker, + seriesStateOptions, + pointMarkerOptions = point.marker || {}, + symbol = pointMarkerOptions.symbol || seriesMarkerOptions.symbol, + pointStateOptions, + radius = pick( + pointMarkerOptions.radius, + seriesMarkerOptions.radius + ), + attribs; + // Handle hover and select states + if (state) { + seriesStateOptions = seriesMarkerOptions.states[state]; + pointStateOptions = + pointMarkerOptions.states && pointMarkerOptions.states[state]; + radius = pick( + pointStateOptions && pointStateOptions.radius, + seriesStateOptions && seriesStateOptions.radius, + radius + + ((seriesStateOptions && seriesStateOptions.radiusPlus) || 0) + ); + } + point.hasImage = symbol && symbol.indexOf("url") === 0; + if (point.hasImage) { + radius = 0; // and subsequently width and height is not set + } + attribs = { + // Math.floor for #1843: + x: seriesOptions.crisp + ? Math.floor(point.plotX) - radius + : point.plotX - radius, + y: point.plotY - radius, + }; + if (radius) { + attribs.width = attribs.height = 2 * radius; + } + return attribs; + }, + /** + * Internal function to get presentational attributes for each point. + * Unlike {@link Series#markerAttribs}, this function should return + * those attributes that can also be set in CSS. In styled mode, + * `pointAttribs` won't be called. + * + * @private + * @function Highcharts.Series#pointAttribs + * + * @param {Highcharts.Point} [point] + * The point instance to inspect. + * + * @param {string} [state] + * The point state, can be either `hover`, `select` or 'normal'. + * If undefined, normal state is assumed. + * + * @return {Highcharts.SVGAttributes} + * The presentational attributes to be set on the point. + */ + pointAttribs: function (point, state) { + var seriesMarkerOptions = this.options.marker, + seriesStateOptions, + pointOptions = point && point.options, + pointMarkerOptions = (pointOptions && pointOptions.marker) || {}, + pointStateOptions, + color = this.color, + pointColorOption = pointOptions && pointOptions.color, + pointColor = point && point.color, + strokeWidth = pick( + pointMarkerOptions.lineWidth, + seriesMarkerOptions.lineWidth + ), + zoneColor = point && point.zone && point.zone.color, + fill, + stroke, + opacity = 1; + color = pointColorOption || zoneColor || pointColor || color; + fill = + pointMarkerOptions.fillColor || + seriesMarkerOptions.fillColor || + color; + stroke = + pointMarkerOptions.lineColor || + seriesMarkerOptions.lineColor || + color; + // Handle hover and select states + state = state || "normal"; + if (state) { + seriesStateOptions = seriesMarkerOptions.states[state]; + pointStateOptions = + (pointMarkerOptions.states && + pointMarkerOptions.states[state]) || + {}; + strokeWidth = pick( + pointStateOptions.lineWidth, + seriesStateOptions.lineWidth, + strokeWidth + + pick( + pointStateOptions.lineWidthPlus, + seriesStateOptions.lineWidthPlus, + 0 + ) + ); + fill = + pointStateOptions.fillColor || + seriesStateOptions.fillColor || + fill; + stroke = + pointStateOptions.lineColor || + seriesStateOptions.lineColor || + stroke; + opacity = pick( + pointStateOptions.opacity, + seriesStateOptions.opacity, + opacity + ); + } + return { + stroke: stroke, + "stroke-width": strokeWidth, + fill: fill, + opacity: opacity, + }; + }, + /** + * Clear DOM objects and free up memory. + * + * @private + * @function Highcharts.Series#destroy + * @param {boolean} [keepEventsForUpdate] + * @return {void} + * @fires Highcharts.Series#event:destroy + */ + destroy: function (keepEventsForUpdate) { + var series = this, + chart = series.chart, + issue134 = /AppleWebKit\/533/.test(win.navigator.userAgent), + destroy, + i, + data = series.data || [], + point, + axis; + // add event hook + fireEvent(series, "destroy"); + // remove events + this.removeEvents(keepEventsForUpdate); + // erase from axes + (series.axisTypes || []).forEach(function (AXIS) { + axis = series[AXIS]; + if (axis && axis.series) { + erase(axis.series, series); + axis.isDirty = axis.forceRedraw = true; + } + }); + // remove legend items + if (series.legendItem) { + series.chart.legend.destroyItem(series); } - // If the total size exceeds the len, remove those boxes with the lowest - // rank - i = boxes.length; + // destroy all points with their elements + i = data.length; while (i--) { - total += boxes[i].size; + point = data[i]; + if (point && point.destroy) { + point.destroy(); + } } - // Sort by rank, then slice away overshoot - if (total > reducedLen) { - stableSort(boxes, function (a, b) { - return (b.rank || 0) - (a.rank || 0); - }); - i = 0; - total = 0; - while (total <= reducedLen) { - total += boxes[i].size; - i++; - } - restBoxes = boxes.splice(i - 1, boxes.length); + series.points = null; + // Clear the animation timeout if we are destroying the series + // during initial animation + U.clearTimeout(series.animationTimeout); + // Destroy all SVGElements associated to the series + objectEach(series, function (val, prop) { + // Survive provides a hook for not destroying + if (val instanceof SVGElement && !val.survive) { + // issue 134 workaround + destroy = issue134 && prop === "group" ? "hide" : "destroy"; + val[destroy](); + } + }); + // remove from hoverSeries + if (chart.hoverSeries === series) { + chart.hoverSeries = null; } - // Order by target - stableSort(boxes, sortByTarget); - // So far we have been mutating the original array. Now - // create a copy with target arrays - boxes = boxes.map(function (box) { - return { - size: box.size, - targets: [box.target], - align: pick(box.align, 0.5) - }; + erase(chart.series, series); + chart.orderSeries(); + // clear all members + objectEach(series, function (val, prop) { + if (!keepEventsForUpdate || prop !== "hcEvents") { + delete series[prop]; + } }); - while (overlapping) { - // Initial positions: target centered in box - i = boxes.length; - while (i--) { - box = boxes[i]; - // Composite box, average of targets - target = (Math.min.apply(0, box.targets) + - Math.max.apply(0, box.targets)) / 2; - box.pos = clamp(target - box.size * box.align, 0, len - box.size); - } - // Detect overlap and join boxes - i = boxes.length; - overlapping = false; - while (i--) { - // Overlap - if (i > 0 && - boxes[i - 1].pos + boxes[i - 1].size > - boxes[i].pos) { - // Add this size to the previous box - boxes[i - 1].size += boxes[i].size; - boxes[i - 1].targets = boxes[i - 1] - .targets - .concat(boxes[i].targets); - boxes[i - 1].align = 0.5; - // Overlapping right, push left - if (boxes[i - 1].pos + boxes[i - 1].size > len) { - boxes[i - 1].pos = len - boxes[i - 1].size; - } - boxes.splice(i, 1); // Remove this item - overlapping = true; - } - } + }, + /** + * Get the graph path. + * + * @private + * @function Highcharts.Series#getGraphPath + * @param {Array<Highcharts.Point>} points + * @param {boolean} [nullsAsZeroes] + * @param {boolean} [connectCliffs] + * @return {Highcharts.SVGPathArray} + */ + getGraphPath: function (points, nullsAsZeroes, connectCliffs) { + var series = this, + options = series.options, + step = options.step, + reversed, + graphPath = [], + xMap = [], + gap; + points = points || series.points; + // Bottom of a stack is reversed + reversed = points.reversed; + if (reversed) { + points.reverse(); } - // Add the rest (hidden boxes) - origBoxes.push.apply(origBoxes, restBoxes); - // Now the composite boxes are placed, we need to put the original boxes - // within them - i = 0; - boxes.some(function (box) { - var posInCompositeBox = 0; - if (box.targets.some(function () { - origBoxes[i].pos = box.pos + posInCompositeBox; - // If the distance between the position and the target exceeds - // maxDistance, abort the loop and decrease the length in increments - // of 10% to recursively reduce the number of visible boxes by - // rank. Once all boxes are within the maxDistance, we're good. - if (typeof maxDistance !== 'undefined' && - Math.abs(origBoxes[i].pos - origBoxes[i].target) > maxDistance) { - // Reset the positions that are already set - origBoxes.slice(0, i + 1).forEach(function (box) { - delete box.pos; - }); - // Try with a smaller length - origBoxes.reducedLen = - (origBoxes.reducedLen || len) - (len * 0.1); - // Recurse - if (origBoxes.reducedLen > len * 0.1) { - H.distribute(origBoxes, len, maxDistance); - } - // Exceeded maxDistance => abort - return true; - } - posInCompositeBox += origBoxes[i].size; - i++; - })) { - // Exceeded maxDistance => abort - return true; - } + // Reverse the steps (#5004) + step = + { + right: 1, + center: 2, + }[step] || + (step && 3); + if (step && reversed) { + step = 4 - step; + } + // Remove invalid points, especially in spline (#5015) + points = this.getValidPoints( + points, + false, + !(options.connectNulls && !nullsAsZeroes && !connectCliffs) + ); + // Build the line + points.forEach(function (point, i) { + var plotX = point.plotX, + plotY = point.plotY, + lastPoint = points[i - 1], + // the path to this point from the previous + pathToPoint; + if ( + (point.leftCliff || (lastPoint && lastPoint.rightCliff)) && + !connectCliffs + ) { + gap = true; // ... and continue + } + // Line series, nullsAsZeroes is not handled + if (point.isNull && !defined(nullsAsZeroes) && i > 0) { + gap = !options.connectNulls; + // Area series, nullsAsZeroes is set + } else if (point.isNull && !nullsAsZeroes) { + gap = true; + } else { + if (i === 0 || gap) { + pathToPoint = [["M", point.plotX, point.plotY]]; + // Generate the spline as defined in the SplineSeries object + } else if (series.getPointSpline) { + pathToPoint = [series.getPointSpline(points, point, i)]; + } else if (step) { + if (step === 1) { + // right + pathToPoint = [["L", lastPoint.plotX, plotY]]; + } else if (step === 2) { + // center + pathToPoint = [ + ["L", (lastPoint.plotX + plotX) / 2, lastPoint.plotY], + ["L", (lastPoint.plotX + plotX) / 2, plotY], + ]; + } else { + pathToPoint = [["L", plotX, lastPoint.plotY]]; + } + pathToPoint.push(["L", plotX, plotY]); + } else { + // normal line to next point + pathToPoint = [["L", plotX, plotY]]; + } + // Prepare for animation. When step is enabled, there are + // two path nodes for each x value. + xMap.push(point.x); + if (step) { + xMap.push(point.x); + if (step === 2) { + // step = center (#8073) + xMap.push(point.x); + } + } + graphPath.push.apply(graphPath, pathToPoint); + gap = false; + } }); - // Add the rest (hidden) boxes and sort by target - stableSort(origBoxes, sortByTarget); - }; - /** - * Draw the data labels - * - * @private - * @function Highcharts.Series#drawDataLabels - * @return {void} - * @fires Highcharts.Series#event:afterDrawDataLabels - */ - CartesianSeries.prototype.drawDataLabels = function () { + graphPath.xMap = xMap; + series.graphPath = graphPath; + return graphPath; + }, + /** + * Draw the graph. Called internally when rendering line-like series + * types. The first time it generates the `series.graph` item and + * optionally other series-wide items like `series.area` for area + * charts. On subsequent calls these items are updated with new + * positions and attributes. + * + * @function Highcharts.Series#drawGraph + */ + drawGraph: function () { + var series = this, + options = this.options, + graphPath = (this.gappedPath || this.getGraphPath).call(this), + styledMode = this.chart.styledMode, + props = [["graph", "highcharts-graph"]]; + // Presentational properties + if (!styledMode) { + props[0].push( + options.lineColor || this.color || "#cccccc", // when colorByPoint = true + options.dashStyle + ); + } + props = series.getZonesGraphs(props); + // Draw the graph + props.forEach(function (prop, i) { + var graphKey = prop[0], + graph = series[graphKey], + verb = graph ? "animate" : "attr", + attribs; + if (graph) { + graph.endX = series.preventGraphAnimation + ? null + : graphPath.xMap; + graph.animate({ d: graphPath }); + } else if (graphPath.length) { + // #1487 + /** + * SVG element of area-based charts. Can be used for styling + * purposes. If zones are configured, this element will be + * hidden and replaced by multiple zone areas, accessible + * via `series['zone-area-x']` (where x is a number, + * starting with 0). + * + * @name Highcharts.Series#area + * @type {Highcharts.SVGElement|undefined} + */ + /** + * SVG element of line-based charts. Can be used for styling + * purposes. If zones are configured, this element will be + * hidden and replaced by multiple zone lines, accessible + * via `series['zone-graph-x']` (where x is a number, + * starting with 0). + * + * @name Highcharts.Series#graph + * @type {Highcharts.SVGElement|undefined} + */ + series[graphKey] = graph = series.chart.renderer + .path(graphPath) + .addClass(prop[1]) + .attr({ zIndex: 1 }) // #1069 + .add(series.group); + } + if (graph && !styledMode) { + attribs = { + stroke: prop[2], + "stroke-width": options.lineWidth, + // Polygon series use filled graph + fill: (series.fillGraph && series.color) || "none", + }; + if (prop[3]) { + attribs.dashstyle = prop[3]; + } else if (options.linecap !== "square") { + attribs["stroke-linecap"] = attribs["stroke-linejoin"] = + "round"; + } + graph[verb](attribs) + // Add shadow to normal series (0) or to first + // zone (1) #3932 + .shadow(i < 2 && options.shadow); + } + // Helpers for animation + if (graph) { + graph.startX = graphPath.xMap; + graph.isArea = graphPath.isArea; // For arearange animation + } + }); + }, + /** + * Get zones properties for building graphs. Extendable by series with + * multiple lines within one series. + * + * @private + * @function Highcharts.Series#getZonesGraphs + * + * @param {Array<Array<string>>} props + * + * @return {Array<Array<string>>} + */ + getZonesGraphs: function (props) { + // Add the zone properties if any + this.zones.forEach(function (zone, i) { + var propset = [ + "zone-graph-" + i, + "highcharts-graph highcharts-zone-graph-" + + i + + " " + + (zone.className || ""), + ]; + if (!this.chart.styledMode) { + propset.push( + zone.color || this.color, + zone.dashStyle || this.options.dashStyle + ); + } + props.push(propset); + }, this); + return props; + }, + /** + * Clip the graphs into zones for colors and styling. + * + * @private + * @function Highcharts.Series#applyZones + * @return {void} + */ + applyZones: function () { + var series = this, + chart = this.chart, + renderer = chart.renderer, + zones = this.zones, + translatedFrom, + translatedTo, + clips = this.clips || [], + clipAttr, + graph = this.graph, + area = this.area, + chartSizeMax = Math.max(chart.chartWidth, chart.chartHeight), + axis = this[(this.zoneAxis || "y") + "Axis"], + extremes, + reversed, + inverted = chart.inverted, + horiz, + pxRange, + pxPosMin, + pxPosMax, + ignoreZones = false, + zoneArea, + zoneGraph; + if ( + zones.length && + (graph || area) && + axis && + typeof axis.min !== "undefined" + ) { + reversed = axis.reversed; + horiz = axis.horiz; + // The use of the Color Threshold assumes there are no gaps + // so it is safe to hide the original graph and area + // unless it is not waterfall series, then use showLine property + // to set lines between columns to be visible (#7862) + if (graph && !this.showLine) { + graph.hide(); + } + if (area) { + area.hide(); + } + // Create the clips + extremes = axis.getExtremes(); + zones.forEach(function (threshold, i) { + translatedFrom = reversed + ? horiz + ? chart.plotWidth + : 0 + : horiz + ? 0 + : axis.toPixels(extremes.min) || 0; + translatedFrom = clamp( + pick(translatedTo, translatedFrom), + 0, + chartSizeMax + ); + translatedTo = clamp( + Math.round( + axis.toPixels(pick(threshold.value, extremes.max), true) || + 0 + ), + 0, + chartSizeMax + ); + if (ignoreZones) { + translatedFrom = translatedTo = axis.toPixels(extremes.max); + } + pxRange = Math.abs(translatedFrom - translatedTo); + pxPosMin = Math.min(translatedFrom, translatedTo); + pxPosMax = Math.max(translatedFrom, translatedTo); + if (axis.isXAxis) { + clipAttr = { + x: inverted ? pxPosMax : pxPosMin, + y: 0, + width: pxRange, + height: chartSizeMax, + }; + if (!horiz) { + clipAttr.x = chart.plotHeight - clipAttr.x; + } + } else { + clipAttr = { + x: 0, + y: inverted ? pxPosMax : pxPosMin, + width: chartSizeMax, + height: pxRange, + }; + if (horiz) { + clipAttr.y = chart.plotWidth - clipAttr.y; + } + } + // VML SUPPPORT + if (inverted && renderer.isVML) { + if (axis.isXAxis) { + clipAttr = { + x: 0, + y: reversed ? pxPosMin : pxPosMax, + height: clipAttr.width, + width: chart.chartWidth, + }; + } else { + clipAttr = { + x: clipAttr.y - chart.plotLeft - chart.spacingBox.x, + y: 0, + width: clipAttr.height, + height: chart.chartHeight, + }; + } + } + // END OF VML SUPPORT + if (clips[i]) { + clips[i].animate(clipAttr); + } else { + clips[i] = renderer.clipRect(clipAttr); + } + // when no data, graph zone is not applied and after setData + // clip was ignored. As a result, it should be applied each + // time. + zoneArea = series["zone-area-" + i]; + zoneGraph = series["zone-graph-" + i]; + if (graph && zoneGraph) { + zoneGraph.clip(clips[i]); + } + if (area && zoneArea) { + zoneArea.clip(clips[i]); + } + // if this zone extends out of the axis, ignore the others + ignoreZones = threshold.value > extremes.max; + // Clear translatedTo for indicators + if (series.resetZones && translatedTo === 0) { + translatedTo = void 0; + } + }); + this.clips = clips; + } else if (series.visible) { + // If zones were removed, restore graph and area + if (graph) { + graph.show(true); + } + if (area) { + area.show(true); + } + } + }, + /** + * Initialize and perform group inversion on series.group and + * series.markerGroup. + * + * @private + * @function Highcharts.Series#invertGroups + * @param {boolean} [inverted] + * @return {void} + */ + invertGroups: function (inverted) { var series = this, - chart = series.chart, - seriesOptions = series.options, - seriesDlOptions = seriesOptions.dataLabels, - points = series.points, - pointOptions, - hasRendered = series.hasRendered || 0, - dataLabelsGroup, - dataLabelAnim = seriesDlOptions.animation, - animationConfig = seriesDlOptions.defer ? - getDeferredAnimation(chart, - dataLabelAnim, - series) : - { defer: 0, - duration: 0 }, - renderer = chart.renderer; + chart = series.chart; /** - * Handle the dataLabels.filter option. * @private */ - function applyFilter(point, options) { - var filter = options.filter, - op, - prop, - val; - if (filter) { - op = filter.operator; - prop = point[filter.property]; - val = filter.value; - if ((op === '>' && prop > val) || - (op === '<' && prop < val) || - (op === '>=' && prop >= val) || - (op === '<=' && prop <= val) || - (op === '==' && prop == val) || // eslint-disable-line eqeqeq - (op === '===' && prop === val)) { - return true; + function setInvert() { + ["group", "markerGroup"].forEach(function (groupName) { + if (series[groupName]) { + // VML/HTML needs explicit attributes for flipping + if (chart.renderer.isVML) { + series[groupName].attr({ + width: series.yAxis.len, + height: series.xAxis.len, + }); + } + series[groupName].width = series.yAxis.len; + series[groupName].height = series.xAxis.len; + // If inverted polar, don't invert series group + series[groupName].invert( + series.isRadialSeries ? false : inverted + ); + } + }); + } + // Pie, go away (#1736) + if (!series.xAxis) { + return; + } + // A fixed size is needed for inversion to work + series.eventsToUnbind.push(addEvent(chart, "resize", setInvert)); + // Do it now + setInvert(); + // On subsequent render and redraw, just do setInvert without + // setting up events again + series.invertGroups = setInvert; + }, + /** + * General abstraction for creating plot groups like series.group, + * series.dataLabelsGroup and series.markerGroup. On subsequent calls, + * the group will only be adjusted to the updated plot size. + * + * @private + * @function Highcharts.Series#plotGroup + * @param {string} prop + * @param {string} name + * @param {string} visibility + * @param {number} [zIndex] + * @param {Highcharts.SVGElement} [parent] + * @return {Highcharts.SVGElement} + */ + plotGroup: function (prop, name, visibility, zIndex, parent) { + var group = this[prop], + isNew = !group, + attrs = { + visibility: visibility, + zIndex: zIndex || 0.1, // IE8 and pointer logic use this + }; + // Avoid setting undefined opacity, or in styled mode + if ( + typeof this.opacity !== "undefined" && + !this.chart.styledMode && + this.state !== "inactive" // #13719 + ) { + attrs.opacity = this.opacity; + } + // Generate it on first call + if (isNew) { + this[prop] = group = this.chart.renderer.g().add(parent); + } + // Add the class names, and replace existing ones as response to + // Series.update (#6660) + group.addClass( + "highcharts-" + + name + + " highcharts-series-" + + this.index + + " highcharts-" + + this.type + + "-series " + + (defined(this.colorIndex) + ? "highcharts-color-" + this.colorIndex + " " + : "") + + (this.options.className || "") + + (group.hasClass("highcharts-tracker") + ? " highcharts-tracker" + : ""), + true + ); + // Place it on first and subsequent (redraw) calls + group.attr(attrs)[isNew ? "attr" : "animate"](this.getPlotBox()); + return group; + }, + /** + * Get the translation and scale for the plot area of this series. + * + * @function Highcharts.Series#getPlotBox + * + * @return {Highcharts.SeriesPlotBoxObject} + */ + getPlotBox: function () { + var chart = this.chart, + xAxis = this.xAxis, + yAxis = this.yAxis; + // Swap axes for inverted (#2339) + if (chart.inverted) { + xAxis = yAxis; + yAxis = this.xAxis; + } + return { + translateX: xAxis ? xAxis.left : chart.plotLeft, + translateY: yAxis ? yAxis.top : chart.plotTop, + scaleX: 1, + scaleY: 1, + }; + }, + /** + * Removes the event handlers attached previously with addEvents. + * + * @private + * @function Highcharts.Series#removeEvents + * @param {boolean} [keepEventsForUpdate] + * @return {void} + */ + removeEvents: function (keepEventsForUpdate) { + var series = this; + if (!keepEventsForUpdate) { + // remove all events + removeEvent(series); + } else if (series.eventsToUnbind.length) { + // remove only internal events for proper update + // #12355 - solves problem with multiple destroy events + series.eventsToUnbind.forEach(function (unbind) { + unbind(); + }); + series.eventsToUnbind.length = 0; + } + }, + /** + * Render the graph and markers. Called internally when first rendering + * and later when redrawing the chart. This function can be extended in + * plugins, but normally shouldn't be called directly. + * + * @function Highcharts.Series#render + * + * @return {void} + * + * @fires Highcharts.Series#event:afterRender + */ + render: function () { + var series = this, + chart = series.chart, + group, + options = series.options, + animOptions = animObject(options.animation), + // Animation doesn't work in IE8 quirks when the group div is + // hidden, and looks bad in other oldIE + animDuration = + !series.finishedAnimating && + chart.renderer.isSVG && + animOptions.duration, + visibility = series.visible ? "inherit" : "hidden", // #2597 + zIndex = options.zIndex, + hasRendered = series.hasRendered, + chartSeriesGroup = chart.seriesGroup, + inverted = chart.inverted; + fireEvent(this, "render"); + // the group + group = series.plotGroup( + "group", + "series", + visibility, + zIndex, + chartSeriesGroup + ); + series.markerGroup = series.plotGroup( + "markerGroup", + "markers", + visibility, + zIndex, + chartSeriesGroup + ); + // initiate the animation + if (animDuration && series.animate) { + series.animate(true); + } + // SVGRenderer needs to know this before drawing elements (#1089, + // #1795) + group.inverted = + series.isCartesian || series.invertable ? inverted : false; + // Draw the graph if any + if (series.drawGraph) { + series.drawGraph(); + series.applyZones(); + } + // Draw the points + if (series.visible) { + series.drawPoints(); + } + /* series.points.forEach(function (point) { + if (point.redraw) { + point.redraw(); } - return false; - } - return true; + }); */ + // Draw the data labels + if (series.drawDataLabels) { + series.drawDataLabels(); + } + // In pie charts, slices are added to the DOM, but actual rendering + // is postponed until labels reserved their space + if (series.redrawPoints) { + series.redrawPoints(); } + // draw the mouse tracking area + if ( + series.drawTracker && + series.options.enableMouseTracking !== false + ) { + series.drawTracker(); + } + // Handle inverted series and tracker groups + series.invertGroups(inverted); + // Initial clipping, must be defined after inverting groups for VML. + // Applies to columns etc. (#3839). + if ( + options.clip !== false && + !series.sharedClipKey && + !hasRendered + ) { + group.clip(chart.clipRect); + } + // Run the animation + if (animDuration && series.animate) { + series.animate(); + } + // Call the afterAnimate function on animation complete (but don't + // overwrite the animation.complete option which should be available + // to the user). + if (!hasRendered) { + // Additional time if defer is defined before afterAnimate + // will be triggered + if (animDuration && animOptions.defer) { + animDuration += animOptions.defer; + } + series.animationTimeout = syncTimeout(function () { + series.afterAnimate(); + }, animDuration || 0); + } + // Means data is in accordance with what you see + series.isDirty = false; + // (See #322) series.isDirty = series.isDirtyData = false; // means + // data is in accordance with what you see + series.hasRendered = true; + fireEvent(series, "afterRender"); + }, + /** + * Redraw the series. This function is called internally from + * `chart.redraw` and normally shouldn't be called directly. + * + * @private + * @function Highcharts.Series#redraw + * @return {void} + */ + redraw: function () { + var series = this, + chart = series.chart, + // cache it here as it is set to false in render, but used after + wasDirty = series.isDirty || series.isDirtyData, + group = series.group, + xAxis = series.xAxis, + yAxis = series.yAxis; + // reposition on resize + if (group) { + if (chart.inverted) { + group.attr({ + width: chart.plotWidth, + height: chart.plotHeight, + }); + } + group.animate({ + translateX: pick(xAxis && xAxis.left, chart.plotLeft), + translateY: pick(yAxis && yAxis.top, chart.plotTop), + }); + } + series.translate(); + series.render(); + if (wasDirty) { + // #3868, #3945 + delete this.kdTree; + } + }, + kdAxisArray: ["clientX", "plotY"], + /** + * @private + * @function Highcharts.Series#searchPoint + * @param {Highcharts.PointerEventObject} e + * @param {boolean} [compareX] + * @return {Highcharts.Point} + */ + searchPoint: function (e, compareX) { + var series = this, + xAxis = series.xAxis, + yAxis = series.yAxis, + inverted = series.chart.inverted; + return this.searchKDTree( + { + clientX: inverted + ? xAxis.len - e.chartY + xAxis.pos + : e.chartX - xAxis.pos, + plotY: inverted + ? yAxis.len - e.chartX + yAxis.pos + : e.chartY - yAxis.pos, + }, + compareX, + e + ); + }, + /** + * Build the k-d-tree that is used by mouse and touch interaction to get + * the closest point. Line-like series typically have a one-dimensional + * tree where points are searched along the X axis, while scatter-like + * series typically search in two dimensions, X and Y. + * + * @private + * @function Highcharts.Series#buildKDTree + * @param {Highcharts.PointerEventObject} [e] + * @return {void} + */ + buildKDTree: function (e) { + // Prevent multiple k-d-trees from being built simultaneously + // (#6235) + this.buildingKdTree = true; + var series = this, + dimensions = + series.options.findNearestPointBy.indexOf("y") > -1 ? 2 : 1; /** - * Merge two objects that can be arrays. If one of them is an array, the - * other is merged into each element. If both are arrays, each element is - * merged by index. If neither are arrays, we use normal merge. + * Internal function * @private */ - function mergeArrays(one, two) { - var res = [], - i; - if (isArray(one) && !isArray(two)) { - res = one.map(function (el) { - return merge(el, two); - }); - } - else if (isArray(two) && !isArray(one)) { - res = two.map(function (el) { - return merge(one, el); - }); - } - else if (!isArray(one) && !isArray(two)) { - res = merge(one, two); - } - else { - i = Math.max(one.length, two.length); - while (i--) { - res[i] = merge(one[i], two[i]); - } - } - return res; - } - // Merge in plotOptions.dataLabels for series - seriesDlOptions = mergeArrays(mergeArrays(chart.options.plotOptions && - chart.options.plotOptions.series && - chart.options.plotOptions.series.dataLabels, chart.options.plotOptions && - chart.options.plotOptions[series.type] && - chart.options.plotOptions[series.type].dataLabels), seriesDlOptions); - fireEvent(this, 'drawDataLabels'); - if (isArray(seriesDlOptions) || - seriesDlOptions.enabled || - series._hasPointLabels) { - // Create a separate group for the data labels to avoid rotation - dataLabelsGroup = series.plotGroup('dataLabelsGroup', 'data-labels', !hasRendered ? 'hidden' : 'inherit', // #5133, #10220 - seriesDlOptions.zIndex || 6); - dataLabelsGroup.attr({ opacity: +hasRendered }); // #3300 - if (!hasRendered) { - var group = series.dataLabelsGroup; - if (group) { - if (series.visible) { // #2597, #3023, #3024 - dataLabelsGroup.show(true); - } - group[seriesOptions.animation ? 'animate' : 'attr']({ opacity: 1 }, animationConfig); - } - } - // Make the labels for each point - points.forEach(function (point) { - // Merge in series options for the point. - // @note dataLabelAttribs (like pointAttribs) would eradicate - // the need for dlOptions, and simplify the section below. - pointOptions = splat(mergeArrays(seriesDlOptions, point.dlOptions || // dlOptions is used in treemaps - (point.options && point.options.dataLabels))); - // Handle each individual data label for this point - pointOptions.forEach(function (labelOptions, i) { - // Options for one datalabel - var labelEnabled = (labelOptions.enabled && - // #2282, #4641, #7112, #10049 - (!point.isNull || point.dataLabelOnNull) && - applyFilter(point, - labelOptions)), - labelConfig, - formatString, - labelText, - style, - rotation, - attr, - dataLabel = point.dataLabels ? point.dataLabels[i] : - point.dataLabel, - connector = point.connectors ? point.connectors[i] : - point.connector, - labelDistance = pick(labelOptions.distance, - point.labelDistance), - isNew = !dataLabel; - if (labelEnabled) { - // Create individual options structure that can be extended - // without affecting others - labelConfig = point.getLabelConfig(); - formatString = pick(labelOptions[point.formatPrefix + 'Format'], labelOptions.format); - labelText = defined(formatString) ? - format(formatString, labelConfig, chart) : - (labelOptions[point.formatPrefix + 'Formatter'] || - labelOptions.formatter).call(labelConfig, labelOptions); - style = labelOptions.style; - rotation = labelOptions.rotation; - if (!chart.styledMode) { - // Determine the color - style.color = pick(labelOptions.color, style.color, series.color, '#000000'); - // Get automated contrast color - if (style.color === 'contrast') { - point.contrastColor = renderer.getContrast((point.color || series.color)); - style.color = (!defined(labelDistance) && - labelOptions.inside) || - labelDistance < 0 || - !!seriesOptions.stacking ? - point.contrastColor : - '#000000'; - } - else { - delete point.contrastColor; - } - if (seriesOptions.cursor) { - style.cursor = seriesOptions.cursor; - } - } - attr = { - r: labelOptions.borderRadius || 0, - rotation: rotation, - padding: labelOptions.padding, - zIndex: 1 - }; - if (!chart.styledMode) { - attr.fill = labelOptions.backgroundColor; - attr.stroke = labelOptions.borderColor; - attr['stroke-width'] = labelOptions.borderWidth; - } - // Remove unused attributes (#947) - objectEach(attr, function (val, name) { - if (typeof val === 'undefined') { - delete attr[name]; - } - }); - } - // If the point is outside the plot area, destroy it. #678, #820 - if (dataLabel && (!labelEnabled || !defined(labelText))) { - point.dataLabel = - point.dataLabel && point.dataLabel.destroy(); - if (point.dataLabels) { - // Remove point.dataLabels if this was the last one - if (point.dataLabels.length === 1) { - delete point.dataLabels; - } - else { - delete point.dataLabels[i]; - } - } - if (!i) { - delete point.dataLabel; - } - if (connector) { - point.connector = point.connector.destroy(); - if (point.connectors) { - // Remove point.connectors if this was the last one - if (point.connectors.length === 1) { - delete point.connectors; - } - else { - delete point.connectors[i]; - } - } - } - // Individual labels are disabled if the are explicitly disabled - // in the point options, or if they fall outside the plot area. - } - else if (labelEnabled && defined(labelText)) { - if (!dataLabel) { - // Create new label element - point.dataLabels = point.dataLabels || []; - dataLabel = point.dataLabels[i] = rotation ? - // Labels don't rotate, use text element - renderer.text(labelText, 0, -9999, labelOptions.useHTML) - .addClass('highcharts-data-label') : - // We can use label - renderer.label(labelText, 0, -9999, labelOptions.shape, null, null, labelOptions.useHTML, null, 'data-label'); - // Store for backwards compatibility - if (!i) { - point.dataLabel = dataLabel; - } - dataLabel.addClass(' highcharts-data-label-color-' + point.colorIndex + - ' ' + (labelOptions.className || '') + - ( // #3398 - labelOptions.useHTML ? - ' highcharts-tracker' : - '')); - } - else { - // Use old element and just update text - attr.text = labelText; - } - // Store data label options for later access - dataLabel.options = labelOptions; - dataLabel.attr(attr); - if (!chart.styledMode) { - // Styles must be applied before add in order to read - // text bounding box - dataLabel.css(style).shadow(labelOptions.shadow); - } - if (!dataLabel.added) { - dataLabel.add(dataLabelsGroup); - } - if (labelOptions.textPath && !labelOptions.useHTML) { - dataLabel.setTextPath((point.getDataLabelPath && - point.getDataLabelPath(dataLabel)) || point.graphic, labelOptions.textPath); - if (point.dataLabelPath && - !labelOptions.textPath.enabled) { - // clean the DOM - point.dataLabelPath = point.dataLabelPath.destroy(); - } - } - // Now the data label is created and placed at 0,0, so we - // need to align it - series.alignDataLabel(point, dataLabel, labelOptions, null, isNew); - } - }); + function _kdtree(points, depth, dimensions) { + var axis, + median, + length = points && points.length; + if (length) { + // alternate between the axis + axis = series.kdAxisArray[depth % dimensions]; + // sort point array + points.sort(function (a, b) { + return a[axis] - b[axis]; }); + median = Math.floor(length / 2); + // build and return nod + return { + point: points[median], + left: _kdtree(points.slice(0, median), depth + 1, dimensions), + right: _kdtree( + points.slice(median + 1), + depth + 1, + dimensions + ), + }; + } } - fireEvent(this, 'afterDrawDataLabels'); - }; - /** - * Align each individual data label. - * - * @private - * @function Highcharts.Series#alignDataLabel - * @param {Highcharts.Point} point - * @param {Highcharts.SVGElement} dataLabel - * @param {Highcharts.DataLabelsOptions} options - * @param {Highcharts.BBoxObject} alignTo - * @param {boolean} [isNew] - * @return {void} - */ - CartesianSeries.prototype.alignDataLabel = function (point, dataLabel, options, alignTo, isNew) { + /** + * Start the recursive build process with a clone of the points + * array and null points filtered out. (#3873) + * @private + */ + function startRecursive() { + series.kdTree = _kdtree( + series.getValidPoints( + null, + // For line-type series restrict to plot area, but + // column-type series not (#3916, #4511) + !series.directTouch + ), + dimensions, + dimensions + ); + series.buildingKdTree = false; + } + delete series.kdTree; + // For testing tooltips, don't build async. Also if touchstart, we + // may be dealing with click events on mobile, so don't delay + // (#6817). + syncTimeout( + startRecursive, + series.options.kdNow || (e && e.type === "touchstart") ? 0 : 1 + ); + }, + /** + * @private + * @function Highcharts.Series#searchKDTree + * @param {Highcharts.KDPointSearchObject} point + * @param {boolean} [compareX] + * @param {Highcharts.PointerEventObject} [e] + * @return {Highcharts.Point|undefined} + */ + searchKDTree: function (point, compareX, e) { var series = this, - chart = this.chart, - inverted = this.isCartesian && chart.inverted, - enabledDataSorting = this.enabledDataSorting, - plotX = pick(point.dlBox && point.dlBox.centerX, - point.plotX, -9999), - plotY = pick(point.plotY, -9999), - bBox = dataLabel.getBBox(), - baseline, - rotation = options.rotation, - normRotation, - negRotation, - align = options.align, - rotCorr, // rotation correction - isInsidePlot = chart.isInsidePlot(plotX, - Math.round(plotY), - inverted), - // Math.round for rounding errors (#2683), alignTo to allow column - // labels (#2700) - alignAttr, // the final position; - justify = pick(options.overflow, (enabledDataSorting ? 'none' : 'justify')) === 'justify', visible = this.visible && - point.visible !== false && - (point.series.forceDL || - (enabledDataSorting && !justify) || - isInsidePlot || - ( - // If the data label is inside the align box, it is enough - // that parts of the align box is inside the plot area - // (#12370) - options.inside && alignTo && chart.isInsidePlot(plotX, inverted ? - alignTo.x + 1 : - alignTo.y + alignTo.height - 1, inverted))), setStartPos = function (alignOptions) { - if (enabledDataSorting && series.xAxis && !justify) { - series.setDataLabelStartPos(point, dataLabel, isNew, isInsidePlot, alignOptions); - } - }; - if (visible) { - baseline = chart.renderer.fontMetrics(chart.styledMode ? void 0 : options.style.fontSize, dataLabel).b; - // The alignment box is a singular point - alignTo = extend({ - x: inverted ? this.yAxis.len - plotY : plotX, - y: Math.round(inverted ? this.xAxis.len - plotX : plotY), - width: 0, - height: 0 - }, alignTo); - // Add the text size for alignment calculation - extend(options, { - width: bBox.width, - height: bBox.height - }); - // Allow a hook for changing alignment in the last moment, then do the - // alignment - if (rotation) { - justify = false; // Not supported for rotated text - rotCorr = chart.renderer.rotCorr(baseline, rotation); // #3723 - alignAttr = { - x: (alignTo.x + - (options.x || 0) + - alignTo.width / 2 + - rotCorr.x), - y: (alignTo.y + - (options.y || 0) + - { top: 0, middle: 0.5, bottom: 1 }[options.verticalAlign] * - alignTo.height) - }; - setStartPos(alignAttr); // data sorting - dataLabel[isNew ? 'attr' : 'animate'](alignAttr) - .attr({ - align: align - }); - // Compensate for the rotated label sticking out on the sides - normRotation = (rotation + 720) % 360; - negRotation = normRotation > 180 && normRotation < 360; - if (align === 'left') { - alignAttr.y -= negRotation ? bBox.height : 0; - } - else if (align === 'center') { - alignAttr.x -= bBox.width / 2; - alignAttr.y -= bBox.height / 2; - } - else if (align === 'right') { - alignAttr.x -= bBox.width; - alignAttr.y -= negRotation ? 0 : bBox.height; - } - dataLabel.placed = true; - dataLabel.alignAttr = alignAttr; - } - else { - setStartPos(alignTo); // data sorting - dataLabel.align(options, null, alignTo); - alignAttr = dataLabel.alignAttr; - } - // Handle justify or crop - if (justify && alignTo.height >= 0) { // #8830 - this.justifyDataLabel(dataLabel, options, alignAttr, bBox, alignTo, isNew); - // Now check that the data label is within the plot area - } - else if (pick(options.crop, true)) { - visible = - chart.isInsidePlot(alignAttr.x, alignAttr.y) && - chart.isInsidePlot(alignAttr.x + bBox.width, alignAttr.y + bBox.height); - } - // When we're using a shape, make it possible with a connector or an - // arrow pointing to thie point - if (options.shape && !rotation) { - dataLabel[isNew ? 'attr' : 'animate']({ - anchorX: inverted ? - chart.plotWidth - point.plotY : - point.plotX, - anchorY: inverted ? - chart.plotHeight - point.plotX : - point.plotY - }); - } + kdX = this.kdAxisArray[0], + kdY = this.kdAxisArray[1], + kdComparer = compareX ? "distX" : "dist", + kdDimensions = + series.options.findNearestPointBy.indexOf("y") > -1 ? 2 : 1; + /** + * Set the one and two dimensional distance on the point object. + * @private + */ + function setDistance(p1, p2) { + var x = + defined(p1[kdX]) && defined(p2[kdX]) + ? Math.pow(p1[kdX] - p2[kdX], 2) + : null, + y = + defined(p1[kdY]) && defined(p2[kdY]) + ? Math.pow(p1[kdY] - p2[kdY], 2) + : null, + r = (x || 0) + (y || 0); + p2.dist = defined(r) ? Math.sqrt(r) : Number.MAX_VALUE; + p2.distX = defined(x) ? Math.sqrt(x) : Number.MAX_VALUE; } - // To use alignAttr property in hideOverlappingLabels - if (isNew && enabledDataSorting) { - dataLabel.placed = false; + /** + * @private + */ + function _search(search, tree, depth, dimensions) { + var point = tree.point, + axis = series.kdAxisArray[depth % dimensions], + tdist, + sideA, + sideB, + ret = point, + nPoint1, + nPoint2; + setDistance(search, point); + // Pick side based on distance to splitting point + tdist = search[axis] - point[axis]; + sideA = tdist < 0 ? "left" : "right"; + sideB = tdist < 0 ? "right" : "left"; + // End of tree + if (tree[sideA]) { + nPoint1 = _search(search, tree[sideA], depth + 1, dimensions); + ret = nPoint1[kdComparer] < ret[kdComparer] ? nPoint1 : point; + } + if (tree[sideB]) { + // compare distance to current best to splitting point to + // decide wether to check side B or not + if (Math.sqrt(tdist * tdist) < ret[kdComparer]) { + nPoint2 = _search(search, tree[sideB], depth + 1, dimensions); + ret = nPoint2[kdComparer] < ret[kdComparer] ? nPoint2 : ret; + } + } + return ret; + } + if (!this.kdTree && !this.buildingKdTree) { + this.buildKDTree(e); } - // Show or hide based on the final aligned position - if (!visible && (!enabledDataSorting || justify)) { - dataLabel.hide(true); - dataLabel.placed = false; // don't animate back in + if (this.kdTree) { + return _search(point, this.kdTree, kdDimensions, kdDimensions); } + }, + /** + * @private + * @function Highcharts.Series#pointPlacementToXValue + * @return {number} + */ + pointPlacementToXValue: function () { + var _a = this, + _b = _a.options, + pointPlacement = _b.pointPlacement, + pointRange = _b.pointRange, + axis = _a.xAxis; + var factor = pointPlacement; + // Point placement is relative to each series pointRange (#5889) + if (factor === "between") { + factor = axis.reversed ? -0.5 : 0.5; // #11955 + } + return isNumber(factor) + ? factor * pick(pointRange, axis.pointRange) + : 0; + }, + /** + * @private + * @function Highcharts.Series#isPointInside + * @param {Highcharts.Point} point + * @return {boolean} + */ + isPointInside: function (point) { + var isInside = + typeof point.plotY !== "undefined" && + typeof point.plotX !== "undefined" && + point.plotY >= 0 && + point.plotY <= this.yAxis.len && // #3519 + point.plotX >= 0 && + point.plotX <= this.xAxis.len; + return isInside; + }, + } + ); // end Series prototype + /** + * A line series displays information as a series of data points connected by + * straight line segments. + * + * @sample {highcharts} highcharts/demo/line-basic/ + * Line chart + * @sample {highstock} stock/demo/basic-line/ + * Line chart + * + * @extends plotOptions.series + * @product highcharts highstock + * @apioption plotOptions.line + */ + /** + * The SVG value used for the `stroke-linecap` and `stroke-linejoin` + * of a line graph. Round means that lines are rounded in the ends and + * bends. + * + * @type {Highcharts.SeriesLinecapValue} + * @default round + * @since 3.0.7 + * @apioption plotOptions.line.linecap + */ + /** + * A `line` series. If the [type](#series.line.type) option is not + * specified, it is inherited from [chart.type](#chart.type). + * + * @extends series,plotOptions.line + * @excluding dataParser,dataURL + * @product highcharts highstock + * @apioption series.line + */ + /** + * An array of data points for the series. For the `line` series type, + * points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. The `x` values will be automatically + * calculated, either starting at 0 and incremented by 1, or from + * `pointStart` and `pointInterval` given in the series options. If the axis + * has categories, these will be used. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 2. An array of arrays with 2 values. In this case, the values correspond to + * `x,y`. If the first value is a string, it is applied as the name of the + * point, and the `x` value is inferred. + * ```js + * data: [ + * [0, 1], + * [1, 2], + * [2, 8] + * ] + * ``` + * + * 3. 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.line.turboThreshold), + * this option is not available. + * ```js + * data: [{ + * x: 1, + * y: 9, + * name: "Point2", + * color: "#00FF00" + * }, { + * x: 1, + * y: 6, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * **Note:** In TypeScript you have to extend `PointOptionsObject` with an + * additional declaration to allow custom data types: + * ```ts + * declare module `highcharts` { + * interface PointOptionsObject { + * custom: Record<string, (boolean|number|string)>; + * } + * } + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @declare Highcharts.PointOptionsObject + * @type {Array<number|Array<(number|string),(number|null)>|null|*>} + * @apioption series.line.data + */ + /** + * An additional, individual class name for the data point's graphic + * representation. + * + * @type {string} + * @since 5.0.0 + * @product highcharts gantt + * @apioption series.line.data.className + */ + /** + * Individual color for the point. By default the color is pulled from + * the global `colors` array. + * + * In styled mode, the `color` option doesn't take effect. Instead, use + * `colorIndex`. + * + * @sample {highcharts} highcharts/point/color/ + * Mark the highest point + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @product highcharts highstock gantt + * @apioption series.line.data.color + */ + /** + * A specific color index to use for the point, so its graphic representations + * are given the class name `highcharts-color-{n}`. In styled mode this will + * change the color of the graphic. In non-styled mode, the color by is set by + * the `fill` attribute, so the change in class name won't have a visual effect + * by default. + * + * @type {number} + * @since 5.0.0 + * @product highcharts gantt + * @apioption series.line.data.colorIndex + */ + /** + * A reserved subspace to store options and values for customized functionality. + * Here you can add additional data for your own event callbacks and formatter + * callbacks. + * + * @sample {highcharts} highcharts/point/custom/ + * Point and series with custom data + * + * @type {Highcharts.Dictionary<*>} + * @apioption series.line.data.custom + */ + /** + * Individual data label for each point. The options are the same as + * the ones for [plotOptions.series.dataLabels]( + * #plotOptions.series.dataLabels). + * + * @sample highcharts/point/datalabels/ + * Show a label for the last value + * + * @declare Highcharts.DataLabelsOptions + * @extends plotOptions.line.dataLabels + * @product highcharts highstock gantt + * @apioption series.line.data.dataLabels + */ + /** + * A description of the point to add to the screen reader information + * about the point. + * + * @type {string} + * @since 5.0.0 + * @requires modules/accessibility + * @apioption series.line.data.description + */ + /** + * An id for the point. This can be used after render time to get a + * pointer to the point object through `chart.get()`. + * + * @sample {highcharts} highcharts/point/id/ + * Remove an id'd point + * + * @type {string} + * @since 1.2.0 + * @product highcharts highstock gantt + * @apioption series.line.data.id + */ + /** + * The rank for this point's data label in case of collision. If two + * data labels are about to overlap, only the one with the highest `labelrank` + * will be drawn. + * + * @type {number} + * @apioption series.line.data.labelrank + */ + /** + * The name of the point as shown in the legend, tooltip, dataLabels, etc. + * + * @see [xAxis.uniqueNames](#xAxis.uniqueNames) + * + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Point names + * + * @type {string} + * @apioption series.line.data.name + */ + /** + * Whether the data point is selected initially. + * + * @type {boolean} + * @default false + * @product highcharts highstock gantt + * @apioption series.line.data.selected + */ + /** + * The x value of the point. For datetime axes, the X value is the timestamp + * in milliseconds since 1970. + * + * @type {number} + * @product highcharts highstock + * @apioption series.line.data.x + */ + /** + * The y value of the point. + * + * @type {number|null} + * @product highcharts highstock + * @apioption series.line.data.y + */ + /** + * The individual point events. + * + * @extends plotOptions.series.point.events + * @product highcharts highstock gantt + * @apioption series.line.data.events + */ + /** + * Options for the point markers of line-like series. + * + * @declare Highcharts.PointMarkerOptionsObject + * @extends plotOptions.series.marker + * @product highcharts highstock + * @apioption series.line.data.marker + */ + (""); // include precedent doclets in transpilat + + return CartesianSeries; + } + ); + _registerModule( + _modules, + "Series/LineSeries.js", + [_modules["Core/Series/CartesianSeries.js"], _modules["Core/Globals.js"]], + function (CartesianSeries, H) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + H.Series = CartesianSeries; // backwards compatibility + + return H.Series; + } + ); + _registerModule( + _modules, + "Extensions/Stacking.js", + [ + _modules["Core/Axis/Axis.js"], + _modules["Core/Chart/Chart.js"], + _modules["Core/Globals.js"], + _modules["Core/Axis/StackingAxis.js"], + _modules["Core/Utilities.js"], + ], + function (Axis, Chart, H, StackingAxis, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var correctFloat = U.correctFloat, + defined = U.defined, + destroyObjectProperties = U.destroyObjectProperties, + format = U.format, + isNumber = U.isNumber, + pick = U.pick; + /** + * Stack of data points + * + * @product highcharts + * + * @interface Highcharts.StackItemObject + */ /** + * Alignment settings + * @name Highcharts.StackItemObject#alignOptions + * @type {Highcharts.AlignObject} + */ /** + * Related axis + * @name Highcharts.StackItemObject#axis + * @type {Highcharts.Axis} + */ /** + * Cumulative value of the stacked data points + * @name Highcharts.StackItemObject#cumulative + * @type {number} + */ /** + * True if on the negative side + * @name Highcharts.StackItemObject#isNegative + * @type {boolean} + */ /** + * Related SVG element + * @name Highcharts.StackItemObject#label + * @type {Highcharts.SVGElement} + */ /** + * Related stack options + * @name Highcharts.StackItemObject#options + * @type {Highcharts.YAxisStackLabelsOptions} + */ /** + * Total value of the stacked data points + * @name Highcharts.StackItemObject#total + * @type {number} + */ /** + * Shared x value of the stack + * @name Highcharts.StackItemObject#x + * @type {number} + */ + (""); // detached doclets above + var Series = H.Series; + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * The class for stacks. Each stack, on a specific X value and either negative + * or positive, has its own stack item. + * + * @private + * @class + * @name Highcharts.StackItem + * @param {Highcharts.Axis} axis + * @param {Highcharts.YAxisStackLabelsOptions} options + * @param {boolean} isNegative + * @param {number} x + * @param {Highcharts.OptionsStackingValue} [stackOption] + */ + var StackItem = /** @class */ (function () { + function StackItem(axis, options, isNegative, x, stackOption) { + var inverted = axis.chart.inverted; + this.axis = axis; + // Tells if the stack is negative + this.isNegative = isNegative; + // Save the options to be able to style the label + this.options = options = options || {}; + // Save the x value to be able to position the label later + this.x = x; + // Initialize total value + this.total = null; + // This will keep each points' extremes stored by series.index and point + // index + this.points = {}; + this.hasValidPoints = false; + // Save the stack option on the series configuration object, + // and whether to treat it as percent + this.stack = stackOption; + this.leftCliff = 0; + this.rightCliff = 0; + // The align options and text align varies on whether the stack is + // negative and if the chart is inverted or not. + // First test the user supplied value, then use the dynamic. + this.alignOptions = { + align: + options.align || + (inverted ? (isNegative ? "left" : "right") : "center"), + verticalAlign: + options.verticalAlign || + (inverted ? "middle" : isNegative ? "bottom" : "top"), + y: options.y, + x: options.x, + }; + this.textAlign = + options.textAlign || + (inverted ? (isNegative ? "right" : "left") : "center"); + } + /** + * @private + * @function Highcharts.StackItem#destroy + */ + StackItem.prototype.destroy = function () { + destroyObjectProperties(this, this.axis); }; /** - * Set starting position for data label sorting animation. + * Renders the stack total label and adds it to the stack label group. * * @private - * @function Highcharts.Series#setDataLabelStartPos - * @param {Highcharts.SVGElement} dataLabel - * @param {Highcharts.ColumnPoint} point - * @param {boolean | undefined} [isNew] - * @param {boolean} [isInside] - * @param {Highcharts.AlignObject} [alignOptions] - * - * @return {void} - */ - CartesianSeries.prototype.setDataLabelStartPos = function (point, dataLabel, isNew, isInside, alignOptions) { - var chart = this.chart, - inverted = chart.inverted, - xAxis = this.xAxis, - reversed = xAxis.reversed, - labelCenter = inverted ? dataLabel.height / 2 : dataLabel.width / 2, - pointWidth = point.pointWidth, - halfWidth = pointWidth ? pointWidth / 2 : 0, - startXPos, - startYPos; - startXPos = inverted ? - alignOptions.x : - (reversed ? - -labelCenter - halfWidth : - xAxis.width - labelCenter + halfWidth); - startYPos = inverted ? - (reversed ? - this.yAxis.height - labelCenter + halfWidth : - -labelCenter - halfWidth) : alignOptions.y; - dataLabel.startXPos = startXPos; - dataLabel.startYPos = startYPos; - // We need to handle visibility in case of sorting point outside plot area - if (!isInside) { - dataLabel - .attr({ opacity: 1 }) - .animate({ opacity: 0 }, void 0, dataLabel.hide); - } - else if (dataLabel.visibility === 'hidden') { - dataLabel.show(); - dataLabel - .attr({ opacity: 0 }) - .animate({ opacity: 1 }); - } - // Save start position on first render, but do not change position - if (!chart.hasRendered) { - return; + * @function Highcharts.StackItem#render + * @param {Highcharts.SVGElement} group + */ + StackItem.prototype.render = function (group) { + var chart = this.axis.chart, + options = this.options, + formatOption = options.format, + attr = {}, + str = formatOption // format the text in the label + ? format(formatOption, this, chart) + : options.formatter.call(this); + // Change the text to reflect the new total and set visibility to hidden + // in case the serie is hidden + if (this.label) { + this.label.attr({ text: str, visibility: "hidden" }); + } else { + // Create new label + this.label = chart.renderer.label( + str, + null, + null, + options.shape, + null, + null, + options.useHTML, + false, + "stack-labels" + ); + attr = { + r: options.borderRadius || 0, + text: str, + rotation: options.rotation, + padding: pick(options.padding, 5), + visibility: "hidden", // hidden until setOffset is called + }; + if (!chart.styledMode) { + attr.fill = options.backgroundColor; + attr.stroke = options.borderColor; + attr["stroke-width"] = options.borderWidth; + this.label.css(options.style); } - // Set start position - if (isNew) { - dataLabel.attr({ x: dataLabel.startXPos, y: dataLabel.startYPos }); + this.label.attr(attr); + if (!this.label.added) { + this.label.add(group); // add to the labels-group } - dataLabel.placed = true; + } + // Rank it higher than data labels (#8742) + this.label.labelrank = chart.plotHeight; }; /** - * If data labels fall partly outside the plot area, align them back in, in a - * way that doesn't hide the point. + * Sets the offset that the stack has from the x value and repositions the + * label. * * @private - * @function Highcharts.Series#justifyDataLabel - * @param {Highcharts.SVGElement} dataLabel - * @param {Highcharts.DataLabelsOptions} options - * @param {Highcharts.SVGAttributes} alignAttr - * @param {Highcharts.BBoxObject} bBox - * @param {Highcharts.BBoxObject} [alignTo] - * @param {boolean} [isNew] - * @return {boolean|undefined} - */ - CartesianSeries.prototype.justifyDataLabel = function (dataLabel, options, alignAttr, bBox, alignTo, isNew) { - var chart = this.chart, - align = options.align, - verticalAlign = options.verticalAlign, - off, - justified, - padding = dataLabel.box ? 0 : (dataLabel.padding || 0); - var _a = options.x, - x = _a === void 0 ? 0 : _a, - _b = options.y, - y = _b === void 0 ? 0 : _b; - // Off left - off = alignAttr.x + padding; - if (off < 0) { - if (align === 'right' && x >= 0) { - options.align = 'left'; - options.inside = true; - } - else { - x -= off; - } - justified = true; - } - // Off right - off = alignAttr.x + bBox.width - padding; - if (off > chart.plotWidth) { - if (align === 'left' && x <= 0) { - options.align = 'right'; - options.inside = true; - } - else { - x += chart.plotWidth - off; - } - justified = true; - } - // Off top - off = alignAttr.y + padding; - if (off < 0) { - if (verticalAlign === 'bottom' && y >= 0) { - options.verticalAlign = 'top'; - options.inside = true; - } - else { - y -= off; - } - justified = true; - } - // Off bottom - off = alignAttr.y + bBox.height - padding; - if (off > chart.plotHeight) { - if (verticalAlign === 'top' && y <= 0) { - options.verticalAlign = 'bottom'; - options.inside = true; - } - else { - y += chart.plotHeight - off; - } - justified = true; + * @function Highcarts.StackItem#setOffset + * @param {number} xOffset + * @param {number} xWidth + * @param {number} [boxBottom] + * @param {number} [boxTop] + * @param {number} [defaultX] + */ + StackItem.prototype.setOffset = function ( + xOffset, + xWidth, + boxBottom, + boxTop, + defaultX + ) { + var stackItem = this, + axis = stackItem.axis, + chart = axis.chart, + // stack value translated mapped to chart coordinates + y = axis.translate( + axis.stacking.usePercentage + ? 100 + : boxTop + ? boxTop + : stackItem.total, + 0, + 0, + 0, + 1 + ), + yZero = axis.translate(boxBottom ? boxBottom : 0), // stack origin + // stack height: + h = defined(y) && Math.abs(y - yZero), + // x position: + x = pick(defaultX, chart.xAxis[0].translate(stackItem.x)) + xOffset, + stackBox = + defined(y) && + stackItem.getStackBox(chart, stackItem, x, y, xWidth, h, axis), + label = stackItem.label, + isNegative = stackItem.isNegative, + isJustify = + pick(stackItem.options.overflow, "justify") === "justify", + textAlign = stackItem.textAlign, + visible; + if (label && stackBox) { + var bBox = label.getBBox(), + padding = label.padding, + boxOffsetX, + boxOffsetY; + if (textAlign === "left") { + boxOffsetX = chart.inverted ? -padding : padding; + } else if (textAlign === "right") { + boxOffsetX = bBox.width; + } else { + if (chart.inverted && textAlign === "center") { + boxOffsetX = bBox.width / 2; + } else { + boxOffsetX = chart.inverted + ? isNegative + ? bBox.width + padding + : -padding + : bBox.width / 2; + } + } + boxOffsetY = chart.inverted + ? bBox.height / 2 + : isNegative + ? -padding + : bBox.height; + // Reset alignOptions property after justify #12337 + stackItem.alignOptions.x = pick(stackItem.options.x, 0); + stackItem.alignOptions.y = pick(stackItem.options.y, 0); + // Set the stackBox position + stackBox.x -= boxOffsetX; + stackBox.y -= boxOffsetY; + // Align the label to the box + label.align(stackItem.alignOptions, null, stackBox); + // Check if label is inside the plotArea #12294 + if ( + chart.isInsidePlot( + label.alignAttr.x + boxOffsetX - stackItem.alignOptions.x, + label.alignAttr.y + boxOffsetY - stackItem.alignOptions.y + ) + ) { + label.show(); + } else { + // Move label away to avoid the overlapping issues + label.alignAttr.y = -9999; + isJustify = false; } - if (justified) { - options.x = x; - options.y = y; - dataLabel.placed = !isNew; - dataLabel.align(options, void 0, alignTo); + if (isJustify) { + // Justify stackLabel into the stackBox + Series.prototype.justifyDataLabel.call( + this.axis, + label, + stackItem.alignOptions, + label.alignAttr, + bBox, + stackBox + ); + } + label.attr({ + x: label.alignAttr.x, + y: label.alignAttr.y, + }); + if (pick(!isJustify && stackItem.options.crop, true)) { + visible = + isNumber(label.x) && + isNumber(label.y) && + chart.isInsidePlot(label.x - padding + label.width, label.y) && + chart.isInsidePlot(label.x + padding, label.y); + if (!visible) { + label.hide(); + } } - return justified; + } }; - if (seriesTypes.pie) { - seriesTypes.pie.prototype.dataLabelPositioners = { - // Based on the value computed in Highcharts' distribute algorithm. - radialDistributionY: function (point) { - return point.top + point.distributeBox.pos; - }, - // get the x - use the natural x position for labels near the - // top and bottom, to prevent the top and botton slice - // connectors from touching each other on either side - // Based on the value computed in Highcharts' distribute algorithm. - radialDistributionX: function (series, point, y, naturalY) { - return series.getX(y < point.top + 2 || y > point.bottom - 2 ? - naturalY : - y, point.half, point); - }, - // dataLabels.distance determines the x position of the label - justify: function (point, radius, seriesCenter) { - return seriesCenter[0] + (point.half ? -1 : 1) * - (radius + point.labelDistance); - }, - // Left edges of the left-half labels touch the left edge of the plot - // area. Right edges of the right-half labels touch the right edge of - // the plot area. - alignToPlotEdges: function (dataLabel, half, plotWidth, plotLeft) { - var dataLabelWidth = dataLabel.getBBox().width; - return half ? dataLabelWidth + plotLeft : - plotWidth - dataLabelWidth - plotLeft; - }, - // Connectors of each side end in the same x position. Labels are - // aligned to them. Left edge of the widest left-half label touches the - // left edge of the plot area. Right edge of the widest right-half label - // touches the right edge of the plot area. - alignToConnectors: function (points, half, plotWidth, plotLeft) { - var maxDataLabelWidth = 0, - dataLabelWidth; - // find widest data label - points.forEach(function (point) { - dataLabelWidth = point.dataLabel.getBBox().width; - if (dataLabelWidth > maxDataLabelWidth) { - maxDataLabelWidth = dataLabelWidth; - } - }); - return half ? maxDataLabelWidth + plotLeft : - plotWidth - maxDataLabelWidth - plotLeft; - } - }; - /** - * Override the base drawDataLabels method by pie specific functionality - * - * @private - * @function Highcharts.seriesTypes.pie#drawDataLabels - * @return {void} - */ - seriesTypes.pie.prototype.drawDataLabels = function () { - var series = this, - data = series.data, - point, - chart = series.chart, - options = series.options.dataLabels || {}, - connectorPadding = options.connectorPadding, - connectorWidth, - plotWidth = chart.plotWidth, - plotHeight = chart.plotHeight, - plotLeft = chart.plotLeft, - maxWidth = Math.round(chart.chartWidth / 3), - connector, - seriesCenter = series.center, - radius = seriesCenter[2] / 2, - centerY = seriesCenter[1], - dataLabel, - dataLabelWidth, - // labelPos, - labelPosition, - labelHeight, - // divide the points into right and left halves for anti collision - halves = [ - [], - [] // left - ], - x, - y, - visibility, - j, - overflow = [0, 0, 0, 0], // top, right, bottom, left - dataLabelPositioners = series.dataLabelPositioners, - pointDataLabelsOptions; - // get out if not enabled - if (!series.visible || - (!options.enabled && - !series._hasPointLabels)) { - return; - } - // Reset all labels that have been shortened - data.forEach(function (point) { - if (point.dataLabel && point.visible && point.dataLabel.shortened) { - point.dataLabel - .attr({ - width: 'auto' - }).css({ - width: 'auto', - textOverflow: 'clip' - }); - point.dataLabel.shortened = false; - } - }); - // run parent method - CartesianSeries.prototype.drawDataLabels.apply(series); - data.forEach(function (point) { - if (point.dataLabel) { - if (point.visible) { // #407, #2510 - // Arrange points for detection collision - halves[point.half].push(point); - // Reset positions (#4905) - point.dataLabel._pos = null; - // Avoid long labels squeezing the pie size too far down - if (!defined(options.style.width) && - !defined(point.options.dataLabels && - point.options.dataLabels.style && - point.options.dataLabels.style.width)) { - if (point.dataLabel.getBBox().width > maxWidth) { - point.dataLabel.css({ - // Use a fraction of the maxWidth to avoid - // wrapping close to the end of the string. - width: Math.round(maxWidth * 0.7) + 'px' - }); - point.dataLabel.shortened = true; - } - } - } - else { - point.dataLabel = point.dataLabel.destroy(); - // Workaround to make pies destroy multiple datalabels - // correctly. This logic needs rewriting to support multiple - // datalabels fully. - if (point.dataLabels && point.dataLabels.length === 1) { - delete point.dataLabels; - } - } - } - }); - /* Loop over the points in each half, starting from the top and bottom - * of the pie to detect overlapping labels. - */ - halves.forEach(function (points, i) { - var top, - bottom, - length = points.length, - positions = [], - naturalY, - sideOverflow, - size, - distributionLength; - if (!length) { - return; - } - // Sort by angle - series.sortByAngle(points, i - 0.5); - // Only do anti-collision when we have dataLabels outside the pie - // and have connectors. (#856) - if (series.maxLabelDistance > 0) { - top = Math.max(0, centerY - radius - series.maxLabelDistance); - bottom = Math.min(centerY + radius + series.maxLabelDistance, chart.plotHeight); - points.forEach(function (point) { - // check if specific points' label is outside the pie - if (point.labelDistance > 0 && point.dataLabel) { - // point.top depends on point.labelDistance value - // Used for calculation of y value in getX method - point.top = Math.max(0, centerY - radius - point.labelDistance); - point.bottom = Math.min(centerY + radius + point.labelDistance, chart.plotHeight); - size = point.dataLabel.getBBox().height || 21; - // point.positionsIndex is needed for getting index of - // parameter related to specific point inside positions - // array - not every point is in positions array. - point.distributeBox = { - target: point.labelPosition.natural.y - - point.top + size / 2, - size: size, - rank: point.y - }; - positions.push(point.distributeBox); - } - }); - distributionLength = bottom + size - top; - H.distribute(positions, distributionLength, distributionLength / 5); - } - // Now the used slots are sorted, fill them up sequentially - for (j = 0; j < length; j++) { - point = points[j]; - // labelPos = point.labelPos; - labelPosition = point.labelPosition; - dataLabel = point.dataLabel; - visibility = point.visible === false ? 'hidden' : 'inherit'; - naturalY = labelPosition.natural.y; - y = naturalY; - if (positions && defined(point.distributeBox)) { - if (typeof point.distributeBox.pos === 'undefined') { - visibility = 'hidden'; - } - else { - labelHeight = point.distributeBox.size; - // Find label's y position - y = dataLabelPositioners - .radialDistributionY(point); - } - } - // It is needed to delete point.positionIndex for - // dynamically added points etc. - delete point.positionIndex; // @todo unused - // Find label's x position - // justify is undocumented in the API - preserve support for it - if (options.justify) { - x = dataLabelPositioners.justify(point, radius, seriesCenter); - } - else { - switch (options.alignTo) { - case 'connectors': - x = dataLabelPositioners.alignToConnectors(points, i, plotWidth, plotLeft); - break; - case 'plotEdges': - x = dataLabelPositioners.alignToPlotEdges(dataLabel, i, plotWidth, plotLeft); - break; - default: - x = dataLabelPositioners.radialDistributionX(series, point, y, naturalY); - } - } - // Record the placement and visibility - dataLabel._attr = { - visibility: visibility, - align: labelPosition.alignment - }; - pointDataLabelsOptions = point.options.dataLabels || {}; - dataLabel._pos = { - x: (x + - pick(pointDataLabelsOptions.x, options.x) + // (#12985) - ({ - left: connectorPadding, - right: -connectorPadding - }[labelPosition.alignment] || 0)), - // 10 is for the baseline (label vs text) - y: (y + - pick(pointDataLabelsOptions.y, options.y) - // (#12985) - 10) - }; - // labelPos.x = x; - // labelPos.y = y; - labelPosition.final.x = x; - labelPosition.final.y = y; - // Detect overflowing data labels - if (pick(options.crop, true)) { - dataLabelWidth = dataLabel.getBBox().width; - sideOverflow = null; - // Overflow left - if (x - dataLabelWidth < connectorPadding && - i === 1 // left half - ) { - sideOverflow = Math.round(dataLabelWidth - x + connectorPadding); - overflow[3] = Math.max(sideOverflow, overflow[3]); - // Overflow right - } - else if (x + dataLabelWidth > plotWidth - connectorPadding && - i === 0 // right half - ) { - sideOverflow = Math.round(x + dataLabelWidth - plotWidth + connectorPadding); - overflow[1] = Math.max(sideOverflow, overflow[1]); - } - // Overflow top - if (y - labelHeight / 2 < 0) { - overflow[0] = Math.max(Math.round(-y + labelHeight / 2), overflow[0]); - // Overflow left - } - else if (y + labelHeight / 2 > plotHeight) { - overflow[2] = Math.max(Math.round(y + labelHeight / 2 - plotHeight), overflow[2]); - } - dataLabel.sideOverflow = sideOverflow; - } - } // for each point - }); // for each half - // Do not apply the final placement and draw the connectors until we - // have verified that labels are not spilling over. - if (arrayMax(overflow) === 0 || - this.verifyDataLabelOverflow(overflow)) { - // Place the labels in the final position - this.placeDataLabels(); - this.points.forEach(function (point) { - // #8864: every connector can have individual options - pointDataLabelsOptions = - merge(options, point.options.dataLabels); - connectorWidth = - pick(pointDataLabelsOptions.connectorWidth, 1); - // Draw the connector - if (connectorWidth) { - var isNew; - connector = point.connector; - dataLabel = point.dataLabel; - if (dataLabel && - dataLabel._pos && - point.visible && - point.labelDistance > 0) { - visibility = dataLabel._attr.visibility; - isNew = !connector; - if (isNew) { - point.connector = connector = chart.renderer - .path() - .addClass('highcharts-data-label-connector ' + - ' highcharts-color-' + point.colorIndex + - (point.className ? - ' ' + point.className : - '')) - .add(series.dataLabelsGroup); - if (!chart.styledMode) { - connector.attr({ - 'stroke-width': connectorWidth, - 'stroke': (pointDataLabelsOptions.connectorColor || - point.color || - '#666666') - }); - } - } - connector[isNew ? 'attr' : 'animate']({ - d: point.getConnectorPath() - }); - connector.attr('visibility', visibility); - } - else if (connector) { - point.connector = connector.destroy(); - } - } - }); - } - }; - /** - * Extendable method for getting the path of the connector between the data - * label and the pie slice. - * - * @private - * @function Highcharts.seriesTypes.pie#connectorPath - * - * @param {*} labelPos - * - * @return {Highcharts.SVGPathArray} - */ - // TODO: depracated - remove it - /* - seriesTypes.pie.prototype.connectorPath = function (labelPos) { - var x = labelPos.x, - y = labelPos.y; - return pick(this.options.dataLabels.softConnector, true) ? [ - 'M', - // end of the string at the label - x + (labelPos[6] === 'left' ? 5 : -5), y, - 'C', - x, y, // first break, next to the label - 2 * labelPos[2] - labelPos[4], 2 * labelPos[3] - labelPos[5], - labelPos[2], labelPos[3], // second break - 'L', - labelPos[4], labelPos[5] // base - ] : [ - 'M', - // end of the string at the label - x + (labelPos[6] === 'left' ? 5 : -5), y, - 'L', - labelPos[2], labelPos[3], // second break - 'L', - labelPos[4], labelPos[5] // base - ]; - }; - */ - /** - * Perform the final placement of the data labels after we have verified - * that they fall within the plot area. - * - * @private - * @function Highcharts.seriesTypes.pie#placeDataLabels - * @return {void} - */ - seriesTypes.pie.prototype.placeDataLabels = function () { - this.points.forEach(function (point) { - var dataLabel = point.dataLabel, - _pos; - if (dataLabel && point.visible) { - _pos = dataLabel._pos; - if (_pos) { - // Shorten data labels with ellipsis if they still overflow - // after the pie has reached minSize (#223). - if (dataLabel.sideOverflow) { - dataLabel._attr.width = - Math.max(dataLabel.getBBox().width - - dataLabel.sideOverflow, 0); - dataLabel.css({ - width: dataLabel._attr.width + 'px', - textOverflow: ((this.options.dataLabels.style || {}) - .textOverflow || - 'ellipsis') - }); - dataLabel.shortened = true; - } - dataLabel.attr(dataLabel._attr); - dataLabel[dataLabel.moved ? 'animate' : 'attr'](_pos); - dataLabel.moved = true; - } - else if (dataLabel) { - dataLabel.attr({ y: -9999 }); - } - } - // Clear for update - delete point.distributeBox; - }, this); - }; - seriesTypes.pie.prototype.alignDataLabel = noop; - /** - * Verify whether the data labels are allowed to draw, or we should run more - * translation and data label positioning to keep them inside the plot area. - * Returns true when data labels are ready to draw. - * - * @private - * @function Highcharts.seriesTypes.pie#verifyDataLabelOverflow - * @param {Array<number>} overflow - * @return {boolean} - */ - seriesTypes.pie.prototype.verifyDataLabelOverflow = function (overflow) { - var center = this.center, - options = this.options, - centerOption = options.center, - minSize = options.minSize || 80, - newSize = minSize, - // If a size is set, return true and don't try to shrink the pie - // to fit the labels. - ret = options.size !== null; - if (!ret) { - // Handle horizontal size and center - if (centerOption[0] !== null) { // Fixed center - newSize = Math.max(center[2] - - Math.max(overflow[1], overflow[3]), minSize); - } - else { // Auto center - newSize = Math.max( - // horizontal overflow - center[2] - overflow[1] - overflow[3], minSize); - // horizontal center - center[0] += (overflow[3] - overflow[1]) / 2; - } - // Handle vertical size and center - if (centerOption[1] !== null) { // Fixed center - newSize = clamp(newSize, minSize, center[2] - Math.max(overflow[0], overflow[2])); - } - else { // Auto center - newSize = clamp(newSize, minSize, - // vertical overflow - center[2] - overflow[0] - overflow[2]); - // vertical center - center[1] += (overflow[0] - overflow[2]) / 2; - } - // If the size must be decreased, we need to run translate and - // drawDataLabels again - if (newSize < center[2]) { - center[2] = newSize; - center[3] = Math.min(// #3632 - relativeLength(options.innerSize || 0, newSize), newSize); - this.translate(center); - if (this.drawDataLabels) { - this.drawDataLabels(); - } - // Else, return true to indicate that the pie and its labels is - // within the plot area - } - else { - ret = true; - } - } - return ret; - }; - } - if (seriesTypes.column) { - /** - * Override the basic data label alignment by adjusting for the position of - * the column. - * - * @private - * @function Highcharts.seriesTypes.column#alignDataLabel - * @param {Highcharts.Point} point - * @param {Highcharts.SVGElement} dataLabel - * @param {Highcharts.DataLabelsOptions} options - * @param {Highcharts.BBoxObject} alignTo - * @param {boolean} [isNew] - * @return {void} - */ - seriesTypes.column.prototype.alignDataLabel = function (point, dataLabel, options, alignTo, isNew) { - var inverted = this.chart.inverted, - series = point.series, - // data label box for alignment - dlBox = point.dlBox || point.shapeArgs, - below = pick(point.below, // range series - point.plotY > - pick(this.translatedThreshold, - series.yAxis.len)), - // draw it inside the box? - inside = pick(options.inside, !!this.options.stacking), - overshoot; - // Align to the column itself, or the top of it - if (dlBox) { // Area range uses this method but not alignTo - alignTo = merge(dlBox); - if (alignTo.y < 0) { - alignTo.height += alignTo.y; - alignTo.y = 0; - } - // If parts of the box overshoots outside the plot area, modify the - // box to center the label inside - overshoot = alignTo.y + alignTo.height - series.yAxis.len; - if (overshoot > 0 && overshoot < alignTo.height) { - alignTo.height -= overshoot; - } - if (inverted) { - alignTo = { - x: series.yAxis.len - alignTo.y - alignTo.height, - y: series.xAxis.len - alignTo.x - alignTo.width, - width: alignTo.height, - height: alignTo.width - }; - } - // Compute the alignment box - if (!inside) { - if (inverted) { - alignTo.x += below ? 0 : alignTo.width; - alignTo.width = 0; - } - else { - alignTo.y += below ? alignTo.height : 0; - alignTo.height = 0; - } - } - } - // When alignment is undefined (typically columns and bars), display the - // individual point below or above the point depending on the threshold - options.align = pick(options.align, !inverted || inside ? 'center' : below ? 'right' : 'left'); - options.verticalAlign = pick(options.verticalAlign, inverted || inside ? 'middle' : below ? 'top' : 'bottom'); - // Call the parent method - CartesianSeries.prototype.alignDataLabel.call(this, point, dataLabel, options, alignTo, isNew); - // If label was justified and we have contrast, set it: - if (options.inside && point.contrastColor) { - dataLabel.css({ - color: point.contrastColor - }); - } - }; - } - - }); - _registerModule(_modules, 'Extensions/OverlappingDataLabels.js', [_modules['Core/Chart/Chart.js'], _modules['Core/Utilities.js']], function (Chart, U) { - /* * + /** + * @private + * @function Highcharts.StackItem#getStackBox + * + * @param {Highcharts.Chart} chart * - * Highcharts module to hide overlapping data labels. - * This module is included in Highcharts. + * @param {Highcharts.StackItem} stackItem * - * (c) 2009-2020 Torstein Honsi + * @param {number} x * - * License: www.highcharts.com/license + * @param {number} y * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * @param {number} xWidth * - * */ - var addEvent = U.addEvent, - fireEvent = U.fireEvent, - isArray = U.isArray, - isNumber = U.isNumber, - objectEach = U.objectEach, - pick = U.pick; - /** - * Internal type - * @private - */ - /* eslint-disable no-invalid-this */ - // Collect potensial overlapping data labels. Stack labels probably don't need - // to be considered because they are usually accompanied by data labels that lie - // inside the columns. - addEvent(Chart, 'render', function collectAndHide() { - var labels = []; - // Consider external label collectors - (this.labelCollectors || []).forEach(function (collector) { - labels = labels.concat(collector()); - }); - (this.yAxis || []).forEach(function (yAxis) { - if (yAxis.stacking && - yAxis.options.stackLabels && - !yAxis.options.stackLabels.allowOverlap) { - objectEach(yAxis.stacking.stacks, function (stack) { - objectEach(stack, function (stackItem) { - labels.push(stackItem.label); - }); - }); + * @param {number} h + * + * @param {Highcharts.Axis} axis + * + * @return {Highcharts.BBoxObject} + */ + StackItem.prototype.getStackBox = function ( + chart, + stackItem, + x, + y, + xWidth, + h, + axis + ) { + var reversed = stackItem.axis.reversed, + inverted = chart.inverted, + axisPos = + axis.height + + axis.pos - + (inverted ? chart.plotLeft : chart.plotTop), + neg = + (stackItem.isNegative && !reversed) || + (!stackItem.isNegative && reversed); // #4056 + return { + x: inverted + ? neg + ? y - axis.right + : y - h + axis.pos - chart.plotLeft + : x + chart.xAxis[0].transB - chart.plotLeft, + y: inverted + ? axis.height - x - xWidth + : neg + ? axisPos - y - h + : axisPos - y, + width: inverted ? h : xWidth, + height: inverted ? xWidth : h, + }; + }; + return StackItem; + })(); + /** + * Generate stacks for each series and calculate stacks total values + * + * @private + * @function Highcharts.Chart#getStacks + */ + Chart.prototype.getStacks = function () { + var chart = this, + inverted = chart.inverted; + // reset stacks for each yAxis + chart.yAxis.forEach(function (axis) { + if (axis.stacking && axis.stacking.stacks && axis.hasVisibleSeries) { + axis.stacking.oldStacks = axis.stacking.stacks; + } + }); + chart.series.forEach(function (series) { + var xAxisOptions = (series.xAxis && series.xAxis.options) || {}; + if ( + series.options.stacking && + (series.visible === true || + chart.options.chart.ignoreHiddenSeries === false) + ) { + series.stackKey = [ + series.type, + pick(series.options.stack, ""), + inverted ? xAxisOptions.top : xAxisOptions.left, + inverted ? xAxisOptions.height : xAxisOptions.width, + ].join(","); + } + }); + }; + // Stacking methods defined on the Axis prototype + StackingAxis.compose(Axis); + // Stacking methods defined for Series prototype + /** + * Set grouped points in a stack-like object. When `centerInCategory` is true, + * and `stacking` is not enabled, we need a pseudo (horizontal) stack in order + * to handle grouping of points within the same category. + * + * @private + * @function Highcharts.Series#setStackedPoints + * @return {void} + */ + Series.prototype.setGroupedPoints = function () { + if ( + this.options.centerInCategory && + (this.is("column") || this.is("columnrange")) && + // With stacking enabled, we already have stacks that we can compute + // from + !this.options.stacking && + // With only one series, we don't need to consider centerInCategory + this.chart.series.length > 1 + ) { + Series.prototype.setStackedPoints.call(this, "group"); + } + }; + /** + * Adds series' points value to corresponding stack + * + * @private + * @function Highcharts.Series#setStackedPoints + */ + Series.prototype.setStackedPoints = function (stackingParam) { + var stacking = stackingParam || this.options.stacking; + if ( + !stacking || + (this.visible !== true && + this.chart.options.chart.ignoreHiddenSeries !== false) + ) { + return; + } + var series = this, + xData = series.processedXData, + yData = series.processedYData, + stackedYData = [], + yDataLength = yData.length, + seriesOptions = series.options, + threshold = seriesOptions.threshold, + stackThreshold = pick( + seriesOptions.startFromThreshold && threshold, + 0 + ), + stackOption = seriesOptions.stack, + stackKey = stackingParam + ? series.type + "," + stacking + : series.stackKey, + negKey = "-" + stackKey, + negStacks = series.negStacks, + yAxis = series.yAxis, + stacks = yAxis.stacking.stacks, + oldStacks = yAxis.stacking.oldStacks, + stackIndicator, + isNegative, + stack, + other, + key, + pointKey, + i, + x, + y; + yAxis.stacking.stacksTouched += 1; + // loop over the non-null y values and read them into a local array + for (i = 0; i < yDataLength; i++) { + x = xData[i]; + y = yData[i]; + stackIndicator = series.getStackIndicator( + stackIndicator, + x, + series.index + ); + pointKey = stackIndicator.key; + // Read stacked values into a stack based on the x value, + // the sign of y and the stack key. Stacking is also handled for null + // values (#739) + isNegative = negStacks && y < (stackThreshold ? 0 : threshold); + key = isNegative ? negKey : stackKey; + // Create empty object for this stack if it doesn't exist yet + if (!stacks[key]) { + stacks[key] = {}; + } + // Initialize StackItem for this x + if (!stacks[key][x]) { + if (oldStacks[key] && oldStacks[key][x]) { + stacks[key][x] = oldStacks[key][x]; + stacks[key][x].total = null; + } else { + stacks[key][x] = new StackItem( + yAxis, + yAxis.options.stackLabels, + isNegative, + x, + stackOption + ); + } + } + // If the StackItem doesn't exist, create it first + stack = stacks[key][x]; + if (y !== null) { + stack.points[pointKey] = stack.points[series.index] = [ + pick(stack.cumulative, stackThreshold), + ]; + // Record the base of the stack + if (!defined(stack.cumulative)) { + stack.base = pointKey; + } + stack.touched = yAxis.stacking.stacksTouched; + // In area charts, if there are multiple points on the same X value, + // let the area fill the full span of those points + if (stackIndicator.index > 0 && series.singleStacks === false) { + stack.points[pointKey][0] = + stack.points[series.index + "," + x + ",0"][0]; + } + // When updating to null, reset the point stack (#7493) + } else { + stack.points[pointKey] = stack.points[series.index] = null; + } + // Add value to the stack total + if (stacking === "percent") { + // Percent stacked column, totals are the same for the positive and + // negative stacks + other = isNegative ? stackKey : negKey; + if (negStacks && stacks[other] && stacks[other][x]) { + other = stacks[other][x]; + stack.total = other.total = + Math.max(other.total, stack.total) + Math.abs(y) || 0; + // Percent stacked areas + } else { + stack.total = correctFloat(stack.total + (Math.abs(y) || 0)); + } + } else if (stacking === "group") { + // In this stack, the total is the number of valid points + if (y !== null) { + stack.total = (stack.total || 0) + 1; + } + } else { + stack.total = correctFloat(stack.total + (y || 0)); + } + if (stacking === "group") { + // This point's index within the stack, pushed to stack.points[1] + stack.cumulative = (stack.total || 1) - 1; + } else { + stack.cumulative = + pick(stack.cumulative, stackThreshold) + (y || 0); + } + if (y !== null) { + stack.points[pointKey].push(stack.cumulative); + stackedYData[i] = stack.cumulative; + stack.hasValidPoints = true; + } + } + if (stacking === "percent") { + yAxis.stacking.usePercentage = true; + } + if (stacking !== "group") { + this.stackedYData = stackedYData; // To be used in getExtremes + } + // Reset old stacks + yAxis.stacking.oldStacks = {}; + }; + /** + * Iterate over all stacks and compute the absolute values to percent + * + * @private + * @function Highcharts.Series#modifyStacks + */ + Series.prototype.modifyStacks = function () { + var series = this, + yAxis = series.yAxis, + stackKey = series.stackKey, + stacks = yAxis.stacking.stacks, + processedXData = series.processedXData, + stackIndicator, + stacking = series.options.stacking; + if (series[stacking + "Stacker"]) { + // Modifier function exists + [stackKey, "-" + stackKey].forEach(function (key) { + var i = processedXData.length, + x, + stack, + pointExtremes; + while (i--) { + x = processedXData[i]; + stackIndicator = series.getStackIndicator( + stackIndicator, + x, + series.index, + key + ); + stack = stacks[key] && stacks[key][x]; + pointExtremes = stack && stack.points[stackIndicator.key]; + if (pointExtremes) { + series[stacking + "Stacker"](pointExtremes, stack, i); + } + } + }); + } + }; + /** + * Modifier function for percent stacks. Blows up the stack to 100%. + * + * @private + * @function Highcharts.Series#percentStacker + * @param {Array<number>} pointExtremes + * @param {Highcharts.StackItem} stack + * @param {number} i + */ + Series.prototype.percentStacker = function (pointExtremes, stack, i) { + var totalFactor = stack.total ? 100 / stack.total : 0; + // Y bottom value + pointExtremes[0] = correctFloat(pointExtremes[0] * totalFactor); + // Y value + pointExtremes[1] = correctFloat(pointExtremes[1] * totalFactor); + this.stackedYData[i] = pointExtremes[1]; + }; + /** + * Get stack indicator, according to it's x-value, to determine points with the + * same x-value + * + * @private + * @function Highcharts.Series#getStackIndicator + * @param {Highcharts.StackItemIndicatorObject|undefined} stackIndicator + * @param {number} x + * @param {number} index + * @param {string} [key] + * @return {Highcharts.StackItemIndicatorObject} + */ + Series.prototype.getStackIndicator = function ( + stackIndicator, + x, + index, + key + ) { + // Update stack indicator, when: + // first point in a stack || x changed || stack type (negative vs positive) + // changed: + if ( + !defined(stackIndicator) || + stackIndicator.x !== x || + (key && stackIndicator.key !== key) + ) { + stackIndicator = { + x: x, + index: 0, + key: key, + }; + } else { + stackIndicator.index++; + } + stackIndicator.key = [index, x, stackIndicator.index].join(","); + return stackIndicator; + }; + H.StackItem = StackItem; + + return H.StackItem; + } + ); + _registerModule( + _modules, + "Core/Dynamics.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Axis/Axis.js"], + _modules["Core/Series/Series.js"], + _modules["Core/Chart/Chart.js"], + _modules["Core/Globals.js"], + _modules["Series/LineSeries.js"], + _modules["Core/Options.js"], + _modules["Core/Series/Point.js"], + _modules["Core/Time.js"], + _modules["Core/Utilities.js"], + ], + function (A, Axis, BaseSeries, Chart, H, LineSeries, O, Point, Time, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animate = A.animate, + setAnimation = A.setAnimation; + var seriesTypes = BaseSeries.seriesTypes; + var time = O.time; + var addEvent = U.addEvent, + createElement = U.createElement, + css = U.css, + defined = U.defined, + erase = U.erase, + error = U.error, + extend = U.extend, + fireEvent = U.fireEvent, + isArray = U.isArray, + isNumber = U.isNumber, + isObject = U.isObject, + isString = U.isString, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick, + relativeLength = U.relativeLength, + splat = U.splat; + /* eslint-disable valid-jsdoc */ + /** + * Remove settings that have not changed, to avoid unnecessary rendering or + * computing (#9197). + * @private + */ + H.cleanRecursively = function (newer, older) { + var result = {}; + objectEach(newer, function (val, key) { + var ob; + // Dive into objects (except DOM nodes) + if ( + isObject(newer[key], true) && + !newer.nodeType && // #10044 + older[key] + ) { + ob = H.cleanRecursively(newer[key], older[key]); + if (Object.keys(ob).length) { + result[key] = ob; + } + // Arrays, primitives and DOM nodes are copied directly + } else if (isObject(newer[key]) || newer[key] !== older[key]) { + result[key] = newer[key]; + } + }); + return result; + }; + // Extend the Chart prototype for dynamic methods + extend( + Chart.prototype, + /** @lends Highcharts.Chart.prototype */ { + /** + * Add a series to the chart after render time. Note that this method should + * never be used when adding data synchronously at chart render time, as it + * adds expense to the calculations and rendering. When adding data at the + * same time as the chart is initialized, add the series as a configuration + * option instead. With multiple axes, the `offset` is dynamically adjusted. + * + * @sample highcharts/members/chart-addseries/ + * Add a series from a button + * @sample stock/members/chart-addseries/ + * Add a series in Highstock + * + * @function Highcharts.Chart#addSeries + * + * @param {Highcharts.SeriesOptionsType} options + * The config options for the series. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after adding. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * Whether to apply animation, and optionally animation + * configuration. + * + * @return {Highcharts.Series} + * The newly created series object. + * + * @fires Highcharts.Chart#event:addSeries + * @fires Highcharts.Chart#event:afterAddSeries + */ + addSeries: function (options, redraw, animation) { + var series, + chart = this; + if (options) { + // <- not necessary + redraw = pick(redraw, true); // defaults to true + fireEvent(chart, "addSeries", { options: options }, function () { + series = chart.initSeries(options); + chart.isDirtyLegend = true; + chart.linkSeries(); + if (series.enabledDataSorting) { + // We need to call `setData` after `linkSeries` + series.setData(options.data, false); } - }); - (this.series || []).forEach(function (series) { - var dlOptions = series.options.dataLabels; - if (series.visible && - !(dlOptions.enabled === false && !series._hasPointLabels)) { // #3866 - (series.nodes || series.points).forEach(function (point) { - if (point.visible) { - var dataLabels = (isArray(point.dataLabels) ? - point.dataLabels : - (point.dataLabel ? [point.dataLabel] : [])); - dataLabels.forEach(function (label) { - var options = label.options; - label.labelrank = pick(options.labelrank, point.labelrank, point.shapeArgs && point.shapeArgs.height); // #4118 - if (!options.allowOverlap) { - labels.push(label); - } - }); - } - }); + fireEvent(chart, "afterAddSeries", { series: series }); + if (redraw) { + chart.redraw(animation); } + }); + } + return series; + }, + /** + * Add an axis to the chart after render time. Note that this method should + * never be used when adding data synchronously at chart render time, as it + * adds expense to the calculations and rendering. When adding data at the + * same time as the chart is initialized, add the axis as a configuration + * option instead. + * + * @sample highcharts/members/chart-addaxis/ + * Add and remove axes + * + * @function Highcharts.Chart#addAxis + * + * @param {Highcharts.AxisOptions} options + * The axis options. + * + * @param {boolean} [isX=false] + * Whether it is an X axis or a value axis. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after adding. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] + * Whether and how to apply animation in the redraw. + * + * @return {Highcharts.Axis} + * The newly generated Axis object. + */ + addAxis: function (options, isX, redraw, animation) { + return this.createAxis(isX ? "xAxis" : "yAxis", { + axis: options, + redraw: redraw, + animation: animation, }); - this.hideOverlappingLabels(labels); - }); - /** - * Hide overlapping labels. Labels are moved and faded in and out on zoom to - * provide a smooth visual imression. - * - * @private - * @function Highcharts.Chart#hideOverlappingLabels - * @param {Array<Highcharts.SVGElement>} labels - * Rendered data labels - * @requires modules/overlapping-datalabels - */ - Chart.prototype.hideOverlappingLabels = function (labels) { + }, + /** + * Add a color axis to the chart after render time. Note that this method + * should never be used when adding data synchronously at chart render time, + * as it adds expense to the calculations and rendering. When adding data at + * the same time as the chart is initialized, add the axis as a + * configuration option instead. + * + * @sample highcharts/members/chart-addaxis/ + * Add and remove axes + * + * @function Highcharts.Chart#addColorAxis + * + * @param {Highcharts.ColorAxisOptions} options + * The axis options. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after adding. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] + * Whether and how to apply animation in the redraw. + * + * @return {Highcharts.ColorAxis} + * The newly generated Axis object. + */ + addColorAxis: function (options, redraw, animation) { + return this.createAxis("colorAxis", { + axis: options, + redraw: redraw, + animation: animation, + }); + }, + /** + * Factory for creating different axis types. + * + * @private + * @function Highcharts.Chart#createAxis + * + * @param {string} type + * An axis type. + * + * @param {...Array<*>} arguments + * All arguments for the constructor. + * + * @return {Highcharts.Axis | Highcharts.ColorAxis} + * The newly generated Axis object. + */ + createAxis: function (type, options) { + var chartOptions = this.options, + isColorAxis = type === "colorAxis", + axisOptions = options.axis, + redraw = options.redraw, + animation = options.animation, + userOptions = merge(axisOptions, { + index: this[type].length, + isX: type === "xAxis", + }), + axis; + if (isColorAxis) { + axis = new H.ColorAxis(this, userOptions); + } else { + axis = new Axis(this, userOptions); + } + // Push the new axis options to the chart options + chartOptions[type] = splat(chartOptions[type] || {}); + chartOptions[type].push(userOptions); + if (isColorAxis) { + this.isDirtyLegend = true; + // Clear before 'bindAxes' (#11924) + this.axes.forEach(function (axis) { + axis.series = []; + }); + this.series.forEach(function (series) { + series.bindAxes(); + series.isDirtyData = true; + }); + } + if (pick(redraw, true)) { + this.redraw(animation); + } + return axis; + }, + /** + * Dim the chart and show a loading text or symbol. Options for the loading + * screen are defined in {@link + * https://api.highcharts.com/highcharts/loading|the loading options}. + * + * @sample highcharts/members/chart-hideloading/ + * Show and hide loading from a button + * @sample highcharts/members/chart-showloading/ + * Apply different text labels + * @sample stock/members/chart-show-hide-loading/ + * Toggle loading in Highstock + * + * @function Highcharts.Chart#showLoading + * + * @param {string} [str] + * An optional text to show in the loading label instead of the + * default one. The default text is set in + * [lang.loading](https://api.highcharts.com/highcharts/lang.loading). + */ + showLoading: function (str) { var chart = this, - len = labels.length, - ren = chart.renderer, - label, - i, - j, - label1, - label2, - box1, - box2, - isLabelAffected = false, - isIntersectRect = function (box1, - box2) { - return !(box2.x >= box1.x + box1.width || - box2.x + box2.width <= box1.x || - box2.y >= box1.y + box1.height || - box2.y + box2.height <= box1.y); - }, - // Get the box with its position inside the chart, as opposed to getBBox - // that only reports the position relative to the parent. - getAbsoluteBox = function (label) { - var pos, - parent, - bBox, - // Substract the padding if no background or border (#4333) - padding = label.box ? 0 : (label.padding || 0), - lineHeightCorrection = 0, - xOffset = 0, - boxWidth, - alignValue; - if (label && - (!label.alignAttr || label.placed)) { - pos = label.alignAttr || { - x: label.attr('x'), - y: label.attr('y') - }; - parent = label.parentGroup; - // Get width and height if pure text nodes (stack labels) - if (!label.width) { - bBox = label.getBBox(); - label.width = bBox.width; - label.height = bBox.height; - // Labels positions are computed from top left corner, so - // we need to substract the text height from text nodes too. - lineHeightCorrection = ren - .fontMetrics(null, label.element).h; - } - boxWidth = label.width - 2 * padding; - alignValue = { - left: '0', - center: '0.5', - right: '1' - }[label.alignValue]; - if (alignValue) { - xOffset = +alignValue * boxWidth; - } - else if (isNumber(label.x) && Math.round(label.x) !== label.translateX) { - xOffset = label.x - label.translateX; - } - return { - x: pos.x + (parent.translateX || 0) + padding - - (xOffset || 0), - y: pos.y + (parent.translateY || 0) + padding - - lineHeightCorrection, - width: label.width - 2 * padding, - height: label.height - 2 * padding - }; + options = chart.options, + loadingDiv = chart.loadingDiv, + loadingOptions = options.loading, + setLoadingSize = function () { + if (loadingDiv) { + css(loadingDiv, { + left: chart.plotLeft + "px", + top: chart.plotTop + "px", + width: chart.plotWidth + "px", + height: chart.plotHeight + "px", + }); + } + }; + // create the layer at the first call + if (!loadingDiv) { + chart.loadingDiv = loadingDiv = createElement( + "div", + { + className: "highcharts-loading highcharts-loading-hidden", + }, + null, + chart.container + ); + chart.loadingSpan = createElement( + "span", + { className: "highcharts-loading-inner" }, + null, + loadingDiv + ); + addEvent(chart, "redraw", setLoadingSize); // #1080 + } + loadingDiv.className = "highcharts-loading"; + // Update text + chart.loadingSpan.innerHTML = pick(str, options.lang.loading, ""); + if (!chart.styledMode) { + // Update visuals + css( + loadingDiv, + extend(loadingOptions.style, { + zIndex: 10, + }) + ); + css(chart.loadingSpan, loadingOptions.labelStyle); + // Show it + if (!chart.loadingShown) { + css(loadingDiv, { + opacity: 0, + display: "", + }); + animate( + loadingDiv, + { + opacity: loadingOptions.style.opacity || 0.5, + }, + { + duration: loadingOptions.showDuration || 0, + } + ); + } + } + chart.loadingShown = true; + setLoadingSize(); + }, + /** + * Hide the loading layer. + * + * @see Highcharts.Chart#showLoading + * + * @sample highcharts/members/chart-hideloading/ + * Show and hide loading from a button + * @sample stock/members/chart-show-hide-loading/ + * Toggle loading in Highstock + * + * @function Highcharts.Chart#hideLoading + */ + hideLoading: function () { + var options = this.options, + loadingDiv = this.loadingDiv; + if (loadingDiv) { + loadingDiv.className = + "highcharts-loading highcharts-loading-hidden"; + if (!this.styledMode) { + animate( + loadingDiv, + { + opacity: 0, + }, + { + duration: options.loading.hideDuration || 100, + complete: function () { + css(loadingDiv, { display: "none" }); + }, + } + ); + } + } + this.loadingShown = false; + }, + /** + * These properties cause isDirtyBox to be set to true when updating. Can be + * extended from plugins. + */ + propsRequireDirtyBox: [ + "backgroundColor", + "borderColor", + "borderWidth", + "borderRadius", + "plotBackgroundColor", + "plotBackgroundImage", + "plotBorderColor", + "plotBorderWidth", + "plotShadow", + "shadow", + ], + /** + * These properties require a full reflow of chart elements, best + * implemented through running `Chart.setSize` internally (#8190). + * @type {Array} + */ + propsRequireReflow: [ + "margin", + "marginTop", + "marginRight", + "marginBottom", + "marginLeft", + "spacing", + "spacingTop", + "spacingRight", + "spacingBottom", + "spacingLeft", + ], + /** + * These properties cause all series to be updated when updating. Can be + * extended from plugins. + */ + propsRequireUpdateSeries: [ + "chart.inverted", + "chart.polar", + "chart.ignoreHiddenSeries", + "chart.type", + "colors", + "plotOptions", + "time", + "tooltip", + ], + /** + * These collections (arrays) implement update() methods with support for + * one-to-one option. + */ + collectionsWithUpdate: ["xAxis", "yAxis", "zAxis", "series"], + /** + * A generic function to update any element of the chart. Elements can be + * enabled and disabled, moved, re-styled, re-formatted etc. + * + * A special case is configuration objects that take arrays, for example + * [xAxis](https://api.highcharts.com/highcharts/xAxis), + * [yAxis](https://api.highcharts.com/highcharts/yAxis) or + * [series](https://api.highcharts.com/highcharts/series). For these + * collections, an `id` option is used to map the new option set to an + * existing object. If an existing object of the same id is not found, the + * corresponding item is updated. So for example, running `chart.update` + * with a series item without an id, will cause the existing chart's series + * with the same index in the series array to be updated. When the + * `oneToOne` parameter is true, `chart.update` will also take care of + * adding and removing items from the collection. Read more under the + * parameter description below. + * + * Note that when changing series data, `chart.update` may mutate the passed + * data options. + * + * See also the + * [responsive option set](https://api.highcharts.com/highcharts/responsive). + * Switching between `responsive.rules` basically runs `chart.update` under + * the hood. + * + * @sample highcharts/members/chart-update/ + * Update chart geometry + * + * @function Highcharts.Chart#update + * + * @param {Highcharts.Options} options + * A configuration object for the new chart options. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart. + * + * @param {boolean} [oneToOne=false] + * When `true`, the `series`, `xAxis`, `yAxis` and `annotations` + * collections will be updated one to one, and items will be either + * added or removed to match the new updated options. For example, + * if the chart has two series and we call `chart.update` with a + * configuration containing three series, one will be added. If we + * call `chart.update` with one series, one will be removed. Setting + * an empty `series` array will remove all series, but leaving out + * the`series` property will leave all series untouched. If the + * series have id's, the new series options will be matched by id, + * and the remaining ones removed. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] + * Whether to apply animation, and optionally animation + * configuration. + * + * @fires Highcharts.Chart#event:update + * @fires Highcharts.Chart#event:afterUpdate + */ + update: function (options, redraw, oneToOne, animation) { + var chart = this, + adders = { + credits: "addCredits", + title: "setTitle", + subtitle: "setSubtitle", + caption: "setCaption", + }, + optionsChart, + updateAllAxes, + updateAllSeries, + newWidth, + newHeight, + runSetSize, + isResponsiveOptions = options.isResponsiveOptions, + itemsForRemoval = []; + fireEvent(chart, "update", { options: options }); + // If there are responsive rules in action, undo the responsive rules + // before we apply the updated options and replay the responsive rules + // on top from the chart.redraw function (#9617). + if (!isResponsiveOptions) { + chart.setResponsive(false, true); + } + options = H.cleanRecursively(options, chart.options); + merge(true, chart.userOptions, options); + // If the top-level chart option is present, some special updates are + // required + optionsChart = options.chart; + if (optionsChart) { + merge(true, chart.options.chart, optionsChart); + // Setter function + if ("className" in optionsChart) { + chart.setClassName(optionsChart.className); + } + if ("reflow" in optionsChart) { + chart.setReflow(optionsChart.reflow); + } + if ( + "inverted" in optionsChart || + "polar" in optionsChart || + "type" in optionsChart + ) { + // Parse options.chart.inverted and options.chart.polar together + // with the available series. + chart.propFromSeries(); + updateAllAxes = true; + } + if ("alignTicks" in optionsChart) { + // #6452 + updateAllAxes = true; + } + objectEach(optionsChart, function (val, key) { + if ( + chart.propsRequireUpdateSeries.indexOf("chart." + key) !== -1 + ) { + updateAllSeries = true; } - }; - for (i = 0; i < len; i++) { - label = labels[i]; - if (label) { - // Mark with initial opacity - label.oldOpacity = label.opacity; - label.newOpacity = 1; - label.absoluteBox = getAbsoluteBox(label); + // Only dirty box + if (chart.propsRequireDirtyBox.indexOf(key) !== -1) { + chart.isDirtyBox = true; } + // Chart setSize + if (chart.propsRequireReflow.indexOf(key) !== -1) { + if (isResponsiveOptions) { + chart.isDirtyBox = true; + } else { + runSetSize = true; + } + } + }); + if (!chart.styledMode && "style" in optionsChart) { + chart.renderer.setStyle(optionsChart.style); + } + } + // Moved up, because tooltip needs updated plotOptions (#6218) + if (!chart.styledMode && options.colors) { + this.options.colors = options.colors; } - // Prevent a situation in a gradually rising slope, that each label will - // hide the previous one because the previous one always has lower rank. - labels.sort(function (a, b) { - return (b.labelrank || 0) - (a.labelrank || 0); + if (options.time) { + // Maintaining legacy global time. If the chart is instanciated + // first with global time, then updated with time options, we need + // to create a new Time instance to avoid mutating the global time + // (#10536). + if (this.time === time) { + this.time = new Time(options.time); + } + // If we're updating, the time class is different from other chart + // classes (chart.legend, chart.tooltip etc) in that it doesn't know + // about the chart. The other chart[something].update functions also + // set the chart.options[something]. For the time class however we + // need to update the chart options separately. #14230. + merge(true, chart.options.time, options.time); + } + // Some option stuctures correspond one-to-one to chart objects that + // have update methods, for example + // options.credits => chart.credits + // options.legend => chart.legend + // options.title => chart.title + // options.tooltip => chart.tooltip + // options.subtitle => chart.subtitle + // options.mapNavigation => chart.mapNavigation + // options.navigator => chart.navigator + // options.scrollbar => chart.scrollbar + objectEach(options, function (val, key) { + if (chart[key] && typeof chart[key].update === "function") { + chart[key].update(val, false); + // If a one-to-one object does not exist, look for an adder function + } else if (typeof chart[adders[key]] === "function") { + chart[adders[key]](val); + // Else, just merge the options. For nodes like loading, noData, + // plotOptions + } else if ( + key !== "color" && + chart.collectionsWithUpdate.indexOf(key) === -1 + ) { + merge(true, chart.options[key], options[key]); + } + if ( + key !== "chart" && + chart.propsRequireUpdateSeries.indexOf(key) !== -1 + ) { + updateAllSeries = true; + } }); - // Detect overlapping labels - for (i = 0; i < len; i++) { - label1 = labels[i]; - box1 = label1 && label1.absoluteBox; - for (j = i + 1; j < len; ++j) { - label2 = labels[j]; - box2 = label2 && label2.absoluteBox; - if (box1 && - box2 && - label1 !== label2 && // #6465, polar chart with connectEnds - label1.newOpacity !== 0 && - label2.newOpacity !== 0) { - if (isIntersectRect(box1, box2)) { - (label1.labelrank < label2.labelrank ? label1 : label2) - .newOpacity = 0; - } + // Setters for collections. For axes and series, each item is referred + // by an id. If the id is not found, it defaults to the corresponding + // item in the collection, so setting one series without an id, will + // update the first series in the chart. Setting two series without + // an id will update the first and the second respectively (#6019) + // chart.update and responsive. + this.collectionsWithUpdate.forEach(function (coll) { + var indexMap; + if (options[coll]) { + // In stock charts, the navigator series are also part of the + // chart.series array, but those series should not be handled + // here (#8196). + if (coll === "series") { + indexMap = []; + chart[coll].forEach(function (s, i) { + if (!s.options.isInternal) { + indexMap.push(pick(s.options.index, i)); + } + }); + } + splat(options[coll]).forEach(function (newOptions, i) { + var hasId = defined(newOptions.id); + var item; + // Match by id + if (hasId) { + item = chart.get(newOptions.id); + } + // No match by id found, match by index instead + if (!item) { + item = chart[coll][indexMap ? indexMap[i] : i]; + // Check if we grabbed an item with an exising but + // different id (#13541) + if (item && hasId && defined(item.options.id)) { + item = void 0; + } + } + if (item && item.coll === coll) { + item.update(newOptions, false); + if (oneToOne) { + item.touched = true; + } + } + // If oneToOne and no matching item is found, add one + if (!item && oneToOne && chart.collectionsWithInit[coll]) { + chart.collectionsWithInit[coll][0].apply( + chart, + // [newOptions, ...extraArguments, redraw=false] + [newOptions] + .concat( + // Not all initializers require extra args + chart.collectionsWithInit[coll][1] || [] + ) + .concat([false]) + ).touched = true; + } + }); + // Add items for removal + if (oneToOne) { + chart[coll].forEach(function (item) { + if (!item.touched && !item.options.isInternal) { + itemsForRemoval.push(item); + } else { + delete item.touched; } + }); } + } + }); + itemsForRemoval.forEach(function (item) { + if (item.remove) { + item.remove(false); + } + }); + if (updateAllAxes) { + chart.axes.forEach(function (axis) { + axis.update({}, false); + }); } - // Hide or show - labels.forEach(function (label) { - var complete, - newOpacity; - if (label) { - newOpacity = label.newOpacity; - if (label.oldOpacity !== newOpacity) { - // Make sure the label is completely hidden to avoid catching - // clicks (#4362) - if (label.alignAttr && label.placed) { // data labels - label[newOpacity ? 'removeClass' : 'addClass']('highcharts-data-label-hidden'); - complete = function () { - if (!chart.styledMode) { - label.css({ pointerEvents: newOpacity ? 'auto' : 'none' }); - } - label.visibility = newOpacity ? 'inherit' : 'hidden'; - }; - isLabelAffected = true; - // Animate or set the opacity - label.alignAttr.opacity = newOpacity; - label[label.isOld ? 'animate' : 'attr'](label.alignAttr, null, complete); - fireEvent(chart, 'afterHideOverlappingLabel'); - } - else { // other labels, tick labels - label.attr({ - opacity: newOpacity - }); - } - } - label.isOld = true; + // Certain options require the whole series structure to be thrown away + // and rebuilt + if (updateAllSeries) { + chart.getSeriesOrderByLinks().forEach(function (series) { + // Avoid removed navigator series + if (series.chart) { + series.update({}, false); + } + }, this); + } + // Update size. Redraw is forced. + newWidth = optionsChart && optionsChart.width; + newHeight = optionsChart && optionsChart.height; + if (isString(newHeight)) { + newHeight = relativeLength( + newHeight, + newWidth || chart.chartWidth + ); + } + if ( + // In this case, run chart.setSize with newWidth and newHeight which + // are undefined, only for reflowing chart elements because margin + // or spacing has been set (#8190) + runSetSize || + // In this case, the size is actually set + (isNumber(newWidth) && newWidth !== chart.chartWidth) || + (isNumber(newHeight) && newHeight !== chart.chartHeight) + ) { + chart.setSize(newWidth, newHeight, animation); + } else if (pick(redraw, true)) { + chart.redraw(animation); + } + fireEvent(chart, "afterUpdate", { + options: options, + redraw: redraw, + animation: animation, + }); + }, + /** + * Shortcut to set the subtitle options. This can also be done from {@link + * Chart#update} or {@link Chart#setTitle}. + * + * @function Highcharts.Chart#setSubtitle + * + * @param {Highcharts.SubtitleOptions} options + * New subtitle options. The subtitle text itself is set by the + * `options.text` property. + */ + setSubtitle: function (options, redraw) { + this.applyDescription("subtitle", options); + this.layOutTitles(redraw); + }, + /** + * Set the caption options. This can also be done from {@link + * Chart#update}. + * + * @function Highcharts.Chart#setCaption + * + * @param {Highcharts.CaptionOptions} options + * New caption options. The caption text itself is set by the + * `options.text` property. + */ + setCaption: function (options, redraw) { + this.applyDescription("caption", options); + this.layOutTitles(redraw); + }, + } + ); + /** + * These collections (arrays) implement `Chart.addSomethig` method used in + * chart.update() to create new object in the collection. Equivalent for + * deleting is resolved by simple `Somethig.remove()`. + * + * Note: We need to define these references after initializers are bound to + * chart's prototype. + */ + Chart.prototype.collectionsWithInit = { + // collectionName: [ initializingMethod, [extraArguments] ] + xAxis: [Chart.prototype.addAxis, [true]], + yAxis: [Chart.prototype.addAxis, [false]], + series: [Chart.prototype.addSeries], + }; + // extend the Point prototype for dynamic methods + extend( + Point.prototype, + /** @lends Highcharts.Point.prototype */ { + /** + * Update point with new options (typically x/y data) and optionally redraw + * the series. + * + * @sample highcharts/members/point-update-column/ + * Update column value + * @sample highcharts/members/point-update-pie/ + * Update pie slice + * @sample maps/members/point-update/ + * Update map area value in Highmaps + * + * @function Highcharts.Point#update + * + * @param {Highcharts.PointOptionsType} options + * The point options. Point options are handled as described under + * the `series.type.data` item for each series type. For example + * for a line series, if options is a single number, the point will + * be given that number as the marin y value. If it is an array, it + * will be interpreted as x and y values respectively. If it is an + * object, advanced options are applied. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the point is updated. If doing + * more operations on the chart, it is best practice to set + * `redraw` to false and call `chart.redraw()` after. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true] + * Whether to apply animation, and optionally animation + * configuration. + * + * @return {void} + * + * @fires Highcharts.Point#event:update + */ + update: function (options, redraw, animation, runEvent) { + var point = this, + series = point.series, + graphic = point.graphic, + i, + chart = series.chart, + seriesOptions = series.options; + redraw = pick(redraw, true); + /** + * @private + */ + function update() { + point.applyOptions(options); + // Update visuals, #4146 + // Handle dummy graphic elements for a11y, #12718 + var hasDummyGraphic = graphic && point.hasDummyGraphic; + var shouldDestroyGraphic = + point.y === null ? !hasDummyGraphic : hasDummyGraphic; + if (graphic && shouldDestroyGraphic) { + point.graphic = graphic.destroy(); + delete point.hasDummyGraphic; + } + if (isObject(options, true)) { + // Destroy so we can get new elements + if (graphic && graphic.element) { + // "null" is also a valid symbol + if ( + options && + options.marker && + typeof options.marker.symbol !== "undefined" + ) { + point.graphic = graphic.destroy(); + } + } + if (options && options.dataLabels && point.dataLabel) { + point.dataLabel = point.dataLabel.destroy(); // #2468 + } + if (point.connector) { + point.connector = point.connector.destroy(); // #7243 + } + } + // record changes in the parallel arrays + i = point.index; + series.updateParallelArrays(point, i); + // Record the options to options.data. If the old or the new config + // is an object, use point options, otherwise use raw options + // (#4701, #4916). + seriesOptions.data[i] = + isObject(seriesOptions.data[i], true) || isObject(options, true) + ? point.options + : pick(options, seriesOptions.data[i]); + // redraw + series.isDirty = series.isDirtyData = true; + if (!series.fixedBox && series.hasCartesianSeries) { + // #1906, #2320 + chart.isDirtyBox = true; + } + if (seriesOptions.legendType === "point") { + // #1831, #1885 + chart.isDirtyLegend = true; + } + if (redraw) { + chart.redraw(animation); + } + } + // Fire the event with a default handler of doing the update + if (runEvent === false) { + // When called from setData + update(); + } else { + point.firePointEvent("update", { options: options }, update); + } + }, + /** + * Remove a point and optionally redraw the series and if necessary the axes + * + * @sample highcharts/plotoptions/series-point-events-remove/ + * Remove point and confirm + * @sample highcharts/members/point-remove/ + * Remove pie slice + * @sample maps/members/point-remove/ + * Remove selected points in Highmaps + * + * @function Highcharts.Point#remove + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart or wait for an explicit call. When + * doing more operations on the chart, for example running + * `point.remove()` in a loop, it is best practice to set `redraw` + * to false and call `chart.redraw()` after. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=false] + * Whether to apply animation, and optionally animation + * configuration. + * + * @return {void} + */ + remove: function (redraw, animation) { + this.series.removePoint( + this.series.data.indexOf(this), + redraw, + animation + ); + }, + } + ); + // Extend the series prototype for dynamic methods + extend( + LineSeries.prototype, + /** @lends Series.prototype */ { + /** + * Add a point to the series after render time. The point can be added at + * the end, or by giving it an X value, to the start or in the middle of the + * series. + * + * @sample highcharts/members/series-addpoint-append/ + * Append point + * @sample highcharts/members/series-addpoint-append-and-shift/ + * Append and shift + * @sample highcharts/members/series-addpoint-x-and-y/ + * Both X and Y values given + * @sample highcharts/members/series-addpoint-pie/ + * Append pie slice + * @sample stock/members/series-addpoint/ + * Append 100 points in Highstock + * @sample stock/members/series-addpoint-shift/ + * Append and shift in Highstock + * @sample maps/members/series-addpoint/ + * Add a point in Highmaps + * + * @function Highcharts.Series#addPoint + * + * @param {Highcharts.PointOptionsType} options + * The point options. If options is a single number, a point with + * that y value is appended to the series. If it is an array, it will + * be interpreted as x and y values respectively. If it is an + * object, advanced options as outlined under `series.data` are + * applied. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the point is added. When adding + * more than one point, it is highly recommended that the redraw + * option be set to false, and instead {@link Chart#redraw} is + * explicitly called after the adding of points is finished. + * Otherwise, the chart will redraw after adding each point. + * + * @param {boolean} [shift=false] + * If true, a point is shifted off the start of the series as one is + * appended to the end. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * Whether to apply animation, and optionally animation + * configuration. + * + * @param {boolean} [withEvent=true] + * Used internally, whether to fire the series `addPoint` event. + * + * @return {void} + * + * @fires Highcharts.Series#event:addPoint + */ + addPoint: function (options, redraw, shift, animation, withEvent) { + var series = this, + seriesOptions = series.options, + data = series.data, + chart = series.chart, + xAxis = series.xAxis, + names = xAxis && xAxis.hasNames && xAxis.names, + dataOptions = seriesOptions.data, + point, + xData = series.xData, + isInTheMiddle, + i, + x; + // Optional redraw, defaults to true + redraw = pick(redraw, true); + // Get options and push the point to xData, yData and series.options. In + // series.generatePoints the Point instance will be created on demand + // and pushed to the series.data array. + point = { series: series }; + series.pointClass.prototype.applyOptions.apply(point, [options]); + x = point.x; + // Get the insertion point + i = xData.length; + if (series.requireSorting && x < xData[i - 1]) { + isInTheMiddle = true; + while (i && xData[i - 1] > x) { + i--; + } + } + // Insert undefined item + series.updateParallelArrays(point, "splice", i, 0, 0); + // Update it + series.updateParallelArrays(point, i); + if (names && point.name) { + names[x] = point.name; + } + dataOptions.splice(i, 0, options); + if (isInTheMiddle) { + series.data.splice(i, 0, null); + series.processData(); + } + // Generate points to be added to the legend (#1329) + if (seriesOptions.legendType === "point") { + series.generatePoints(); + } + // Shift the first point off the parallel arrays + if (shift) { + if (data[0] && data[0].remove) { + data[0].remove(false); + } else { + data.shift(); + series.updateParallelArrays(point, "shift"); + dataOptions.shift(); + } + } + // Fire event + if (withEvent !== false) { + fireEvent(series, "addPoint", { point: point }); + } + // redraw + series.isDirty = true; + series.isDirtyData = true; + if (redraw) { + chart.redraw(animation); // Animation is set anyway on redraw, #5665 + } + }, + /** + * Remove a point from the series. Unlike the + * {@link Highcharts.Point#remove} method, this can also be done on a point + * that is not instanciated because it is outside the view or subject to + * Highstock data grouping. + * + * @sample highcharts/members/series-removepoint/ + * Remove cropped point + * + * @function Highcharts.Series#removePoint + * + * @param {number} i + * The index of the point in the {@link Highcharts.Series.data|data} + * array. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the point is added. When + * removing more than one point, it is highly recommended that the + * `redraw` option be set to `false`, and instead {@link + * Highcharts.Chart#redraw} is explicitly called after the adding of + * points is finished. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * Whether and optionally how the series should be animated. + * + * @return {void} + * + * @fires Highcharts.Point#event:remove + */ + removePoint: function (i, redraw, animation) { + var series = this, + data = series.data, + point = data[i], + points = series.points, + chart = series.chart, + remove = function () { + if (points && points.length === data.length) { + // #4935 + points.splice(i, 1); + } + data.splice(i, 1); + series.options.data.splice(i, 1); + series.updateParallelArrays( + point || { series: series }, + "splice", + i, + 1 + ); + if (point) { + point.destroy(); } + // redraw + series.isDirty = true; + series.isDirtyData = true; + if (redraw) { + chart.redraw(); + } + }; + setAnimation(animation, chart); + redraw = pick(redraw, true); + // Fire the event with a default handler of removing the point + if (point) { + point.firePointEvent("remove", null, remove); + } else { + remove(); + } + }, + /** + * Remove a series and optionally redraw the chart. + * + * @sample highcharts/members/series-remove/ + * Remove first series from a button + * + * @function Highcharts.Series#remove + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart or wait for an explicit call to + * {@link Highcharts.Chart#redraw}. + * + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation] + * Whether to apply animation, and optionally animation + * configuration. + * + * @param {boolean} [withEvent=true] + * Used internally, whether to fire the series `remove` event. + * + * @return {void} + * + * @fires Highcharts.Series#event:remove + */ + remove: function (redraw, animation, withEvent, keepEvents) { + var series = this, + chart = series.chart; + /** + * @private + */ + function remove() { + // Destroy elements + series.destroy(keepEvents); + series.remove = null; // Prevent from doing again (#9097) + // Redraw + chart.isDirtyLegend = chart.isDirtyBox = true; + chart.linkSeries(); + if (pick(redraw, true)) { + chart.redraw(animation); + } + } + // Fire the event with a default handler of removing the point + if (withEvent !== false) { + fireEvent(series, "remove", null, remove); + } else { + remove(); + } + }, + /** + * Update the series with a new set of options. For a clean and precise + * handling of new options, all methods and elements from the series are + * removed, and it is initialized from scratch. Therefore, this method is + * more performance expensive than some other utility methods like {@link + * Series#setData} or {@link Series#setVisible}. + * + * Note that `Series.update` may mutate the passed `data` options. + * + * @sample highcharts/members/series-update/ + * Updating series options + * @sample maps/members/series-update/ + * Update series options in Highmaps + * + * @function Highcharts.Series#update + * + * @param {Highcharts.SeriesOptionsType} options + * New options that will be merged with the series' existing options. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the series is altered. If doing + * more operations on the chart, it is a good idea to set redraw to + * false and call {@link Chart#redraw} after. + * + * @return {void} + * + * @fires Highcharts.Series#event:update + * @fires Highcharts.Series#event:afterUpdate + */ + update: function (options, redraw) { + options = H.cleanRecursively(options, this.userOptions); + fireEvent(this, "update", { options: options }); + var series = this, + chart = series.chart, + // must use user options when changing type because series.options + // is merged in with type specific plotOptions + oldOptions = series.userOptions, + seriesOptions, + initialType = series.initialType || series.type, + plotOptions = chart.options.plotOptions, + newType = + options.type || oldOptions.type || chart.options.chart.type, + keepPoints = !( + // Indicators, histograms etc recalculate the data. It should be + // possible to omit this. + ( + this.hasDerivedData || + // New type requires new point classes + (newType && newType !== this.type) || + // New options affecting how the data points are built + typeof options.pointStart !== "undefined" || + typeof options.pointInterval !== "undefined" || + // Changes to data grouping requires new points in new group + series.hasOptionChanged("dataGrouping") || + series.hasOptionChanged("pointStart") || + series.hasOptionChanged("pointInterval") || + series.hasOptionChanged("pointIntervalUnit") || + series.hasOptionChanged("keys") + ) + ), + initialSeriesProto = seriesTypes[initialType].prototype, + n, + groups = [ + "group", + "markerGroup", + "dataLabelsGroup", + "transformGroup", + ], + preserve = ["eventOptions", "navigatorSeries", "baseSeries"], + // Animation must be enabled when calling update before the initial + // animation has first run. This happens when calling update + // directly after chart initialization, or when applying responsive + // rules (#6912). + animation = series.finishedAnimating && { animation: false }, + kinds = {}; + if (keepPoints) { + preserve.push( + "data", + "isDirtyData", + "points", + "processedXData", + "processedYData", + "xIncrement", + "cropped", + "_hasPointMarkers", + "_hasPointLabels", + // Map specific, consider moving it to series-specific preserve- + // properties (#10617) + "mapMap", + "mapData", + "minY", + "maxY", + "minX", + "maxX" + ); + if (options.visible !== false) { + preserve.push("area", "graph"); + } + series.parallelArrays.forEach(function (key) { + preserve.push(key + "Data"); + }); + if (options.data) { + // setData uses dataSorting options so we need to update them + // earlier + if (options.dataSorting) { + extend(series.options.dataSorting, options.dataSorting); + } + this.setData(options.data, false); + } + } + // Do the merge, with some forced options + options = merge( + oldOptions, + animation, + { + // When oldOptions.index is null it should't be cleared. + // Otherwise navigator series will have wrong indexes (#10193). + index: + typeof oldOptions.index === "undefined" + ? series.index + : oldOptions.index, + pointStart: pick( + // when updating from blank (#7933) + plotOptions && + plotOptions.series && + plotOptions.series.pointStart, + oldOptions.pointStart, + // when updating after addPoint + series.xData[0] + ), + }, + !keepPoints && { data: series.options.data }, + options + ); + // Merge does not merge arrays, but replaces them. Since points were + // updated, `series.options.data` has correct merged options, use it: + if (keepPoints && options.data) { + options.data = series.options.data; + } + // Make sure preserved properties are not destroyed (#3094) + preserve = groups.concat(preserve); + preserve.forEach(function (prop) { + preserve[prop] = series[prop]; + delete series[prop]; }); - if (isLabelAffected) { - fireEvent(chart, 'afterHideAllOverlappingLabels'); + // Destroy the series and delete all properties. Reinsert all + // methods and properties from the new type prototype (#2270, + // #3719). + series.remove(false, null, false, true); + for (n in initialSeriesProto) { + // eslint-disable-line guard-for-in + series[n] = void 0; } - }; - - }); - _registerModule(_modules, 'Core/Interaction.js', [_modules['Core/Series/Series.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Legend.js'], _modules['Series/LineSeries.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Utilities.js']], function (BaseSeries, Chart, H, Legend, LineSeries, O, Point, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! - * - * */ - var seriesTypes = BaseSeries.seriesTypes; - var hasTouch = H.hasTouch, - svg = H.svg; - var defaultOptions = O.defaultOptions; - var addEvent = U.addEvent, - createElement = U.createElement, - css = U.css, - defined = U.defined, - extend = U.extend, - fireEvent = U.fireEvent, - isArray = U.isArray, - isFunction = U.isFunction, - isNumber = U.isNumber, - isObject = U.isObject, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick; - /** - * @interface Highcharts.PointEventsOptionsObject - */ /** - * Fires when the point is selected either programmatically or following a click - * on the point. One parameter, `event`, is passed to the function. Returning - * `false` cancels the operation. - * @name Highcharts.PointEventsOptionsObject#select - * @type {Highcharts.PointSelectCallbackFunction|undefined} - */ /** - * Fires when the point is unselected either programmatically or following a - * click on the point. One parameter, `event`, is passed to the function. - * Returning `false` cancels the operation. - * @name Highcharts.PointEventsOptionsObject#unselect - * @type {Highcharts.PointUnselectCallbackFunction|undefined} - */ - /** - * Information about the select/unselect event. - * - * @interface Highcharts.PointInteractionEventObject - * @extends global.Event - */ /** - * @name Highcharts.PointInteractionEventObject#accumulate - * @type {boolean} - */ + if (seriesTypes[newType || initialType]) { + extend(series, seriesTypes[newType || initialType].prototype); + } else { + error(17, true, chart, { + missingModuleFor: newType || initialType, + }); + } + // Re-register groups (#3094) and other preserved properties + preserve.forEach(function (prop) { + series[prop] = preserve[prop]; + }); + series.init(chart, options); + // Remove particular elements of the points. Check `series.options` + // because we need to consider the options being set on plotOptions as + // well. + if (keepPoints && this.points) { + seriesOptions = series.options; + // What kind of elements to destroy + if (seriesOptions.visible === false) { + kinds.graphic = 1; + kinds.dataLabel = 1; + } else if (!series._hasPointLabels) { + var marker = seriesOptions.marker, + dataLabels = seriesOptions.dataLabels; + if ( + marker && + (marker.enabled === false || "symbol" in marker) // #10870 + ) { + kinds.graphic = 1; + } + if (dataLabels && dataLabels.enabled === false) { + kinds.dataLabel = 1; + } + } + this.points.forEach(function (point) { + if (point && point.series) { + point.resolveColor(); + // Destroy elements in order to recreate based on updated + // series options. + if (Object.keys(kinds).length) { + point.destroyElements(kinds); + } + if ( + seriesOptions.showInLegend === false && + point.legendItem + ) { + chart.legend.destroyItem(point); + } + } + }, this); + } + series.initialType = initialType; + chart.linkSeries(); // Links are lost in series.remove (#3028) + fireEvent(this, "afterUpdate"); + if (pick(redraw, true)) { + chart.redraw(keepPoints ? void 0 : false); + } + }, + /** + * Used from within series.update + * + * @private + * @function Highcharts.Series#setName + * + * @param {string} name + * + * @return {void} + */ + setName: function (name) { + this.name = this.options.name = this.userOptions.name = name; + this.chart.isDirtyLegend = true; + }, + /** + * Check if the option has changed. + * + * @private + * @function Highcharts.Series#hasOptionChanged + * + * @param {string} option + * + * @return {boolean} + */ + hasOptionChanged: function (optionName) { + var chart = this.chart, + option = this.options[optionName], + plotOptions = chart.options.plotOptions, + oldOption = this.userOptions[optionName]; + if (oldOption) { + return option !== oldOption; + } + return ( + option !== + pick( + plotOptions && + plotOptions[this.type] && + plotOptions[this.type][optionName], + plotOptions && + plotOptions.series && + plotOptions.series[optionName], + option + ) + ); + }, + } + ); + // Extend the Axis.prototype for dynamic methods + extend( + Axis.prototype, + /** @lends Highcharts.Axis.prototype */ { + /** + * Update an axis object with a new set of options. The options are merged + * with the existing options, so only new or altered options need to be + * specified. + * + * @sample highcharts/members/axis-update/ + * Axis update demo + * + * @function Highcharts.Axis#update + * + * @param {Highcharts.AxisOptions} options + * The new options that will be merged in with existing options on + * the axis. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the axis is altered. If doing + * more operations on the chart, it is a good idea to set redraw to + * false and call {@link Chart#redraw} after. + * + * @return {void} + */ + update: function (options, redraw) { + var chart = this.chart, + newEvents = (options && options.events) || {}; + options = merge(this.userOptions, options); + // Color Axis is not an array, + // This change is applied in the ColorAxis wrapper + if (chart.options[this.coll].indexOf) { + // Don't use this.options.index, + // StockChart has Axes in navigator too + chart.options[this.coll][ + chart.options[this.coll].indexOf(this.userOptions) + ] = options; + } + // Remove old events, if no new exist (#8161) + objectEach(chart.options[this.coll].events, function (fn, ev) { + if (typeof newEvents[ev] === "undefined") { + newEvents[ev] = void 0; + } + }); + this.destroy(true); + this.init(chart, extend(options, { events: newEvents })); + chart.isDirtyBox = true; + if (pick(redraw, true)) { + chart.redraw(); + } + }, + /** + * Remove the axis from the chart. + * + * @sample highcharts/members/chart-addaxis/ + * Add and remove axes + * + * @function Highcharts.Axis#remove + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart following the remove. + * + * @return {void} + */ + remove: function (redraw) { + var chart = this.chart, + key = this.coll, // xAxis or yAxis + axisSeries = this.series, + i = axisSeries.length; + // Remove associated series (#2687) + while (i--) { + if (axisSeries[i]) { + axisSeries[i].remove(false); + } + } + // Remove the axis + erase(chart.axes, this); + erase(chart[key], this); + if (isArray(chart.options[key])) { + chart.options[key].splice(this.options.index, 1); + } else { + // color axis, #6488 + delete chart.options[key]; + } + chart[key].forEach(function (axis, i) { + // Re-index, #1706, #8075 + axis.options.index = axis.userOptions.index = i; + }); + this.destroy(); + chart.isDirtyBox = true; + if (pick(redraw, true)) { + chart.redraw(); + } + }, + /** + * Update the axis title by options after render time. + * + * @sample highcharts/members/axis-settitle/ + * Set a new Y axis title + * + * @function Highcharts.Axis#setTitle + * + * @param {Highcharts.AxisTitleOptions} titleOptions + * The additional title options. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after setting the title. + * + * @return {void} + */ + setTitle: function (titleOptions, redraw) { + this.update({ title: titleOptions }, redraw); + }, + /** + * Set new axis categories and optionally redraw. + * + * @sample highcharts/members/axis-setcategories/ + * Set categories by click on a button + * + * @function Highcharts.Axis#setCategories + * + * @param {Array<string>} categories + * The new categories. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart. + * + * @return {void} + */ + setCategories: function (categories, redraw) { + this.update({ categories: categories }, redraw); + }, + } + ); + } + ); + _registerModule( + _modules, + "Series/AreaSeries.js", + [ + _modules["Core/Series/Series.js"], + _modules["Core/Color/Color.js"], + _modules["Core/Globals.js"], + _modules["Mixins/LegendSymbol.js"], + _modules["Core/Utilities.js"], + ], + function (BaseSeries, Color, H, LegendSymbolMixin, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var color = Color.parse; + var objectEach = U.objectEach, + pick = U.pick; + var Series = H.Series; + /** + * Area series type. + * + * @private + * @class + * @name Highcharts.seriesTypes.area + * + * @augments Highcharts.Series + */ + BaseSeries.seriesType( + "area", + "line", /** - * Gets fired when the point is selected either programmatically or following a - * click on the point. - * - * @callback Highcharts.PointSelectCallbackFunction + * The area series type. * - * @param {Highcharts.Point} this - * Point where the event occured. + * @sample {highcharts} highcharts/demo/area-basic/ + * Area chart + * @sample {highstock} stock/demo/area/ + * Area chart * - * @param {Highcharts.PointInteractionEventObject} event - * Event that occured. + * @extends plotOptions.line + * @excluding useOhlcData + * @product highcharts highstock + * @optionparent plotOptions.area */ + { + /** + * @see [fillColor](#plotOptions.area.fillColor) + * @see [fillOpacity](#plotOptions.area.fillOpacity) + * + * @apioption plotOptions.area.color + */ + /** + * Fill color or gradient for the area. When `null`, the series' `color` + * is used with the series' `fillOpacity`. + * + * In styled mode, the fill color can be set with the `.highcharts-area` + * class name. + * + * @see [color](#plotOptions.area.color) + * @see [fillOpacity](#plotOptions.area.fillOpacity) + * + * @sample {highcharts} highcharts/plotoptions/area-fillcolor-default/ + * Null by default + * @sample {highcharts} highcharts/plotoptions/area-fillcolor-gradient/ + * Gradient + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @product highcharts highstock + * @apioption plotOptions.area.fillColor + */ + /** + * Fill opacity for the area. When you set an explicit `fillColor`, + * the `fillOpacity` is not applied. Instead, you should define the + * opacity in the `fillColor` with an rgba color definition. The + * `fillOpacity` setting, also the default setting, overrides the alpha + * component of the `color` setting. + * + * In styled mode, the fill opacity can be set with the + * `.highcharts-area` class name. + * + * @see [color](#plotOptions.area.color) + * @see [fillColor](#plotOptions.area.fillColor) + * + * @sample {highcharts} highcharts/plotoptions/area-fillopacity/ + * Automatic fill color and fill opacity of 0.1 + * + * @type {number} + * @default {highcharts} 0.75 + * @default {highstock} 0.75 + * @product highcharts highstock + * @apioption plotOptions.area.fillOpacity + */ + /** + * A separate color for the graph line. By default the line takes the + * `color` of the series, but the lineColor setting allows setting a + * separate color for the line without altering the `fillColor`. + * + * In styled mode, the line stroke can be set with the + * `.highcharts-graph` class name. + * + * @sample {highcharts} highcharts/plotoptions/area-linecolor/ + * Dark gray line + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @product highcharts highstock + * @apioption plotOptions.area.lineColor + */ + /** + * A separate color for the negative part of the area. + * + * In styled mode, a negative color is set with the + * `.highcharts-negative` class name. + * + * @see [negativeColor](#plotOptions.area.negativeColor) + * + * @sample {highcharts} highcharts/css/series-negative-color/ + * Negative color in styled mode + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 3.0 + * @product highcharts + * @apioption plotOptions.area.negativeFillColor + */ + /** + * Whether the whole area or just the line should respond to mouseover + * tooltips and other mouse or touch events. + * + * @sample {highcharts|highstock} highcharts/plotoptions/area-trackbyarea/ + * Display the tooltip when the area is hovered + * + * @type {boolean} + * @default false + * @since 1.1.6 + * @product highcharts highstock + * @apioption plotOptions.area.trackByArea + */ + /** + * The Y axis value to serve as the base for the area, for + * distinguishing between values above and below a threshold. The area + * between the graph and the threshold is filled. + * + * * If a number is given, the Y axis will scale to the threshold. + * * If `null`, the scaling behaves like a line series with fill between + * the graph and the Y axis minimum. + * * If `Infinity` or `-Infinity`, the area between the graph and the + * corresponding Y axis extreme is filled (since v6.1.0). + * + * @sample {highcharts} highcharts/plotoptions/area-threshold/ + * A threshold of 100 + * @sample {highcharts} highcharts/plotoptions/area-threshold-infinity/ + * A threshold of Infinity + * + * @type {number|null} + * @since 2.0 + * @product highcharts highstock + */ + threshold: 0, + }, + /* eslint-disable valid-jsdoc */ /** - * Fires when the point is unselected either programmatically or following a - * click on the point. - * - * @callback Highcharts.PointUnselectCallbackFunction + * @lends seriesTypes.area.prototype + */ + { + singleStacks: false, + /** + * Return an array of stacked points, where null and missing points are + * replaced by dummy points in order for gaps to be drawn correctly in + * stacks. + * @private + */ + getStackPoints: function (points) { + var series = this, + segment = [], + keys = [], + xAxis = this.xAxis, + yAxis = this.yAxis, + stack = yAxis.stacking.stacks[this.stackKey], + pointMap = {}, + seriesIndex = series.index, + yAxisSeries = yAxis.series, + seriesLength = yAxisSeries.length, + visibleSeries, + upOrDown = pick(yAxis.options.reversedStacks, true) ? 1 : -1, + i; + points = points || this.points; + if (this.options.stacking) { + for (i = 0; i < points.length; i++) { + // Reset after point update (#7326) + points[i].leftNull = points[i].rightNull = void 0; + // Create a map where we can quickly look up the points by + // their X values. + pointMap[points[i].x] = points[i]; + } + // Sort the keys (#1651) + objectEach(stack, function (stackX, x) { + // nulled after switching between + // grouping and not (#1651, #2336) + if (stackX.total !== null) { + keys.push(x); + } + }); + keys.sort(function (a, b) { + return a - b; + }); + visibleSeries = yAxisSeries.map(function (s) { + return s.visible; + }); + keys.forEach(function (x, idx) { + var y = 0, + stackPoint, + stackedValues; + if (pointMap[x] && !pointMap[x].isNull) { + segment.push(pointMap[x]); + // Find left and right cliff. -1 goes left, 1 goes + // right. + [-1, 1].forEach(function (direction) { + var nullName = direction === 1 ? "rightNull" : "leftNull", + cliffName = direction === 1 ? "rightCliff" : "leftCliff", + cliff = 0, + otherStack = stack[keys[idx + direction]]; + // If there is a stack next to this one, + // to the left or to the right... + if (otherStack) { + i = seriesIndex; + // Can go either up or down, + // depending on reversedStacks + while (i >= 0 && i < seriesLength) { + stackPoint = otherStack.points[i]; + if (!stackPoint) { + // If the next point in this series + // is missing, mark the point + // with point.leftNull or + // point.rightNull = true. + if (i === seriesIndex) { + pointMap[x][nullName] = true; + // If there are missing points in + // the next stack in any of the + // series below this one, we need + // to substract the missing values + // and add a hiatus to the left or + // right. + } else if (visibleSeries[i]) { + stackedValues = stack[x].points[i]; + if (stackedValues) { + cliff -= stackedValues[1] - stackedValues[0]; + } + } + } + // When reversedStacks is true, loop up, + // else loop down + i += upOrDown; + } + } + pointMap[x][cliffName] = cliff; + }); + // There is no point for this X value in this series, so we + // insert a dummy point in order for the areas to be drawn + // correctly. + } else { + // Loop down the stack to find the series below this + // one that has a value (#1991) + i = seriesIndex; + while (i >= 0 && i < seriesLength) { + stackPoint = stack[x].points[i]; + if (stackPoint) { + y = stackPoint[1]; + break; + } + // When reversedStacks is true, loop up, else loop + // down + i += upOrDown; + } + y = yAxis.translate( + // #6272 + y, + 0, + 1, + 0, + 1 + ); + segment.push({ + isNull: true, + plotX: xAxis.translate( + // #6272 + x, + 0, + 0, + 0, + 1 + ), + x: x, + plotY: y, + yBottom: y, + }); + } + }); + } + return segment; + }, + /** + * @private + */ + getGraphPath: function (points) { + var getGraphPath = Series.prototype.getGraphPath, + graphPath, + options = this.options, + stacking = options.stacking, + yAxis = this.yAxis, + topPath, + bottomPath, + bottomPoints = [], + graphPoints = [], + seriesIndex = this.index, + i, + areaPath, + plotX, + stacks = yAxis.stacking.stacks[this.stackKey], + threshold = options.threshold, + translatedThreshold = Math.round( + // #10909 + yAxis.getThreshold(options.threshold) + ), + isNull, + yBottom, + connectNulls = pick( + // #10574 + options.connectNulls, + stacking === "percent" + ), + // To display null points in underlying stacked series, this + // series graph must be broken, and the area also fall down to + // fill the gap left by the null point. #2069 + addDummyPoints = function (i, otherI, side) { + var point = points[i], + stackedValues = + stacking && stacks[point.x].points[seriesIndex], + nullVal = point[side + "Null"] || 0, + cliffVal = point[side + "Cliff"] || 0, + top, + bottom, + isNull = true; + if (cliffVal || nullVal) { + top = + (nullVal ? stackedValues[0] : stackedValues[1]) + cliffVal; + bottom = stackedValues[0] + cliffVal; + isNull = !!nullVal; + } else if ( + !stacking && + points[otherI] && + points[otherI].isNull + ) { + top = bottom = threshold; + } + // Add to the top and bottom line of the area + if (typeof top !== "undefined") { + graphPoints.push({ + plotX: plotX, + plotY: + top === null + ? translatedThreshold + : yAxis.getThreshold(top), + isNull: isNull, + isCliff: true, + }); + bottomPoints.push({ + plotX: plotX, + plotY: + bottom === null + ? translatedThreshold + : yAxis.getThreshold(bottom), + doCurve: false, // #1041, gaps in areaspline areas + }); + } + }; + // Find what points to use + points = points || this.points; + // Fill in missing points + if (stacking) { + points = this.getStackPoints(points); + } + for (i = 0; i < points.length; i++) { + // Reset after series.update of stacking property (#12033) + if (!stacking) { + points[i].leftCliff = + points[i].rightCliff = + points[i].leftNull = + points[i].rightNull = + void 0; + } + isNull = points[i].isNull; + plotX = pick(points[i].rectPlotX, points[i].plotX); + yBottom = stacking ? points[i].yBottom : translatedThreshold; + if (!isNull || connectNulls) { + if (!connectNulls) { + addDummyPoints(i, i - 1, "left"); + } + // Skip null point when stacking is false and connectNulls + // true + if (!(isNull && !stacking && connectNulls)) { + graphPoints.push(points[i]); + bottomPoints.push({ + x: i, + plotX: plotX, + plotY: yBottom, + }); + } + if (!connectNulls) { + addDummyPoints(i, i + 1, "right"); + } + } + } + topPath = getGraphPath.call(this, graphPoints, true, true); + bottomPoints.reversed = true; + bottomPath = getGraphPath.call(this, bottomPoints, true, true); + var firstBottomPoint = bottomPath[0]; + if (firstBottomPoint && firstBottomPoint[0] === "M") { + bottomPath[0] = ["L", firstBottomPoint[1], firstBottomPoint[2]]; + } + areaPath = topPath.concat(bottomPath); + // TODO: don't set leftCliff and rightCliff when connectNulls? + graphPath = getGraphPath.call( + this, + graphPoints, + false, + connectNulls + ); + areaPath.xMap = topPath.xMap; + this.areaPath = areaPath; + return graphPath; + }, + /** + * Draw the graph and the underlying area. This method calls the Series + * base function and adds the area. The areaPath is calculated in the + * getSegmentPath method called from Series.prototype.drawGraph. + * @private + */ + drawGraph: function () { + // Define or reset areaPath + this.areaPath = []; + // Call the base method + Series.prototype.drawGraph.apply(this); + // Define local variables + var series = this, + areaPath = this.areaPath, + options = this.options, + zones = this.zones, + props = [ + ["area", "highcharts-area", this.color, options.fillColor], + ]; // area name, main color, fill color + zones.forEach(function (zone, i) { + props.push([ + "zone-area-" + i, + "highcharts-area highcharts-zone-area-" + + i + + " " + + zone.className, + zone.color || series.color, + zone.fillColor || options.fillColor, + ]); + }); + props.forEach(function (prop) { + var areaKey = prop[0], + area = series[areaKey], + verb = area ? "animate" : "attr", + attribs = {}; + // Create or update the area + if (area) { + // update + area.endX = series.preventGraphAnimation ? null : areaPath.xMap; + area.animate({ d: areaPath }); + } else { + // create + attribs.zIndex = 0; // #1069 + area = series[areaKey] = series.chart.renderer + .path(areaPath) + .addClass(prop[1]) + .add(series.group); + area.isArea = true; + } + if (!series.chart.styledMode) { + attribs.fill = pick( + prop[3], + color(prop[2]) + .setOpacity(pick(options.fillOpacity, 0.75)) + .get() + ); + } + area[verb](attribs); + area.startX = areaPath.xMap; + area.shiftUnit = options.step ? 2 : 1; + }); + }, + drawLegendSymbol: LegendSymbolMixin.drawRectangle, + } + ); + /* eslint-enable valid-jsdoc */ + /** + * A `area` series. If the [type](#series.area.type) option is not + * specified, it is inherited from [chart.type](#chart.type). + * + * @extends series,plotOptions.area + * @excluding dataParser, dataURL, useOhlcData + * @product highcharts highstock + * @apioption series.area + */ + /** + * @see [fillColor](#series.area.fillColor) + * @see [fillOpacity](#series.area.fillOpacity) + * + * @apioption series.area.color + */ + /** + * An array of data points for the series. For the `area` series type, + * points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. The `x` values will be automatically + * calculated, either starting at 0 and incremented by 1, or from + * `pointStart` * and `pointInterval` given in the series options. If the + * axis has categories, these will be used. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 2. An array of arrays with 2 values. In this case, the values correspond to + * `x,y`. If the first value is a string, it is applied as the name of the + * point, and the `x` value is inferred. + * ```js + * data: [ + * [0, 9], + * [1, 7], + * [2, 6] + * ] + * ``` + * + * 3. 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.area.turboThreshold), this option is not + * available. + * ```js + * data: [{ + * x: 1, + * y: 9, + * name: "Point2", + * color: "#00FF00" + * }, { + * x: 1, + * y: 6, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @type {Array<number|Array<(number|string),(number|null)>|null|*>} + * @extends series.line.data + * @product highcharts highstock + * @apioption series.area.data + */ + /** + * @see [color](#series.area.color) + * @see [fillOpacity](#series.area.fillOpacity) + * + * @apioption series.area.fillColor + */ + /** + * @see [color](#series.area.color) + * @see [fillColor](#series.area.fillColor) + * + * @default {highcharts} 0.75 + * @default {highstock} 0.75 + * @apioption series.area.fillOpacity + */ + (""); // adds doclets above to transpilat + } + ); + _registerModule( + _modules, + "Series/SplineSeries.js", + [_modules["Core/Series/Series.js"], _modules["Core/Utilities.js"]], + function (BaseSeries, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var pick = U.pick; + /** + * Spline series type. + * + * @private + * @class + * @name Highcharts.seriesTypes.spline + * + * @augments Highcarts.Series + */ + BaseSeries.seriesType( + "spline", + "line", + /** + * A spline series is a special type of line series, where the segments + * between the data points are smoothed. * - * @param {Highcharts.Point} this - * Point where the event occured. + * @sample {highcharts} highcharts/demo/spline-irregular-time/ + * Spline chart + * @sample {highstock} stock/demo/spline/ + * Spline chart * - * @param {Highcharts.PointInteractionEventObject} event - * Event that occured. + * @extends plotOptions.series + * @excluding step, boostThreshold, boostBlending + * @product highcharts highstock + * @optionparent plotOptions.spline */ - ''; // detach doclets above - /* eslint-disable valid-jsdoc */ + {}, /** - * TrackerMixin for points and graphs. - * - * @private - * @mixin Highcharts.TrackerMixin + * @lends seriesTypes.spline.prototype */ - var TrackerMixin = H.TrackerMixin = { - /** - * Draw the tracker for a point. - * - * @private - * @function Highcharts.TrackerMixin.drawTrackerPoint - * @param {Highcharts.Series} this - * @fires Highcharts.Series#event:afterDrawTracker - */ - drawTrackerPoint: function () { - var series = this, - chart = series.chart, - pointer = chart.pointer, - onMouseOver = function (e) { - var point = pointer.getPointFromEvent(e); - // undefined on graph in scatterchart - if (typeof point !== 'undefined') { - pointer.isDirectTouch = true; - point.onMouseOver(e); - } - }, dataLabels; - // Add reference to the point - series.points.forEach(function (point) { - dataLabels = (isArray(point.dataLabels) ? - point.dataLabels : - (point.dataLabel ? [point.dataLabel] : [])); - if (point.graphic) { - point.graphic.element.point = point; - } - dataLabels.forEach(function (dataLabel) { - if (dataLabel.div) { - dataLabel.div.point = point; - } - else { - dataLabel.element.point = point; - } - }); - }); - // Add the event listeners, we need to do this only once - if (!series._hasTracking) { - series.trackerGroups.forEach(function (key) { - if (series[key]) { - // we don't always have dataLabelsGroup - series[key] - .addClass('highcharts-tracker') - .on('mouseover', onMouseOver) - .on('mouseout', function (e) { - pointer.onTrackerMouseOut(e); - }); - if (hasTouch) { - series[key].on('touchstart', onMouseOver); - } - if (!chart.styledMode && series.options.cursor) { - series[key] - .css(css) - .css({ cursor: series.options.cursor }); - } - } - }); - series._hasTracking = true; - } - fireEvent(this, 'afterDrawTracker'); - }, - /** - * Draw the tracker object that sits above all data labels and markers to - * track mouse events on the graph or points. For the line type charts - * the tracker uses the same graphPath, but with a greater stroke width - * for better control. - * - * @private - * @function Highcharts.TrackerMixin.drawTrackerGraph - * @param {Highcharts.Series} this - * @fires Highcharts.Series#event:afterDrawTracker - */ - drawTrackerGraph: function () { - var series = this, - options = series.options, - trackByArea = options.trackByArea, - trackerPath = [].concat(trackByArea ? - series.areaPath : - series.graphPath), - // trackerPathLength = trackerPath.length, - chart = series.chart, - pointer = chart.pointer, - renderer = chart.renderer, - snap = chart.options.tooltip.snap, - tracker = series.tracker, - i, - onMouseOver = function (e) { - if (chart.hoverSeries !== series) { - series.onMouseOver(); - } - }, - /* - * Empirical lowest possible opacities for TRACKER_FILL for an - * element to stay invisible but clickable - * IE6: 0.002 - * IE7: 0.002 - * IE8: 0.002 - * IE9: 0.00000000001 (unlimited) - * IE10: 0.0001 (exporting only) - * FF: 0.00000000001 (unlimited) - * Chrome: 0.000001 - * Safari: 0.000001 - * Opera: 0.00000000001 (unlimited) - */ - TRACKER_FILL = 'rgba(192,192,192,' + (svg ? 0.0001 : 0.002) + ')'; - // Draw the tracker - if (tracker) { - tracker.attr({ d: trackerPath }); - } - else if (series.graph) { // create - series.tracker = renderer.path(trackerPath) - .attr({ - visibility: series.visible ? 'visible' : 'hidden', - zIndex: 2 + { + /* eslint-disable valid-jsdoc */ + /** + * Get the spline segment from a given point's previous neighbour to the + * given point. + * + * @private + * @function Highcharts.seriesTypes.spline#getPointSpline + * + * @param {Array<Highcharts.Point>} + * + * @param {Highcharts.Point} point + * + * @param {number} i + * + * @return {Highcharts.SVGPathArray} + */ + getPointSpline: function (points, point, i) { + var // 1 means control points midway between points, 2 means 1/3 + // from the point, 3 is 1/4 etc + smoothing = 1.5, + denom = smoothing + 1, + plotX = point.plotX || 0, + plotY = point.plotY || 0, + lastPoint = points[i - 1], + nextPoint = points[i + 1], + leftContX, + leftContY, + rightContX, + rightContY, + ret; + /** + * @private + */ + function doCurve(otherPoint) { + return ( + otherPoint && + !otherPoint.isNull && + otherPoint.doCurve !== false && + // #6387, area splines next to null: + !point.isCliff + ); + } + // Find control points + if (doCurve(lastPoint) && doCurve(nextPoint)) { + var lastX = lastPoint.plotX || 0, + lastY = lastPoint.plotY || 0, + nextX = nextPoint.plotX || 0, + nextY = nextPoint.plotY || 0, + correction = 0; + leftContX = (smoothing * plotX + lastX) / denom; + leftContY = (smoothing * plotY + lastY) / denom; + rightContX = (smoothing * plotX + nextX) / denom; + rightContY = (smoothing * plotY + nextY) / denom; + // Have the two control points make a straight line through main + // point + if (rightContX !== leftContX) { + // #5016, division by zero + correction = + ((rightContY - leftContY) * (rightContX - plotX)) / + (rightContX - leftContX) + + plotY - + rightContY; + } + leftContY += correction; + rightContY += correction; + // to prevent false extremes, check that control points are + // between neighbouring points' y values + if (leftContY > lastY && leftContY > plotY) { + leftContY = Math.max(lastY, plotY); + // mirror of left control point + rightContY = 2 * plotY - leftContY; + } else if (leftContY < lastY && leftContY < plotY) { + leftContY = Math.min(lastY, plotY); + rightContY = 2 * plotY - leftContY; + } + if (rightContY > nextY && rightContY > plotY) { + rightContY = Math.max(nextY, plotY); + leftContY = 2 * plotY - rightContY; + } else if (rightContY < nextY && rightContY < plotY) { + rightContY = Math.min(nextY, plotY); + leftContY = 2 * plotY - rightContY; + } + // record for drawing in next point + point.rightContX = rightContX; + point.rightContY = rightContY; + } + // Visualize control points for debugging + /* + if (leftContX) { + this.chart.renderer.circle( + leftContX + this.chart.plotLeft, + leftContY + this.chart.plotTop, + 2 + ) + .attr({ + stroke: 'red', + 'stroke-width': 2, + fill: 'none', + zIndex: 9 }) - .addClass(trackByArea ? - 'highcharts-tracker-area' : - 'highcharts-tracker-line') - .add(series.group); - if (!chart.styledMode) { - series.tracker.attr({ - 'stroke-linecap': 'round', - 'stroke-linejoin': 'round', - stroke: TRACKER_FILL, - fill: trackByArea ? TRACKER_FILL : 'none', - 'stroke-width': series.graph.strokeWidth() + - (trackByArea ? 0 : 2 * snap) - }); - } - // The tracker is added to the series group, which is clipped, but - // is covered by the marker group. So the marker group also needs to - // capture events. - [series.tracker, series.markerGroup].forEach(function (tracker) { - tracker.addClass('highcharts-tracker') - .on('mouseover', onMouseOver) - .on('mouseout', function (e) { - pointer.onTrackerMouseOut(e); - }); - if (options.cursor && !chart.styledMode) { - tracker.css({ cursor: options.cursor }); - } - if (hasTouch) { - tracker.on('touchstart', onMouseOver); - } - }); - } - fireEvent(this, 'afterDrawTracker'); + .add(); + this.chart.renderer.path(['M', leftContX + this.chart.plotLeft, + leftContY + this.chart.plotTop, + 'L', plotX + this.chart.plotLeft, plotY + this.chart.plotTop]) + .attr({ + stroke: 'red', + 'stroke-width': 2, + zIndex: 9 + }) + .add(); } - }; - /* End TrackerMixin */ - // Add tracking event listener to the series group, so the point graphics - // themselves act as trackers - if (seriesTypes.column) { - /** - * @private - * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.column#drawTracker - */ - seriesTypes.column.prototype.drawTracker = TrackerMixin.drawTrackerPoint; - } - if (seriesTypes.pie) { - /** - * @private - * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.pie#drawTracker - */ - seriesTypes.pie.prototype.drawTracker = TrackerMixin.drawTrackerPoint; - } - if (seriesTypes.scatter) { - /** - * @private - * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.scatter#drawTracker - */ - seriesTypes.scatter.prototype.drawTracker = TrackerMixin.drawTrackerPoint; - } - // Extend Legend for item events. - extend(Legend.prototype, { - /** - * @private - * @function Highcharts.Legend#setItemEvents - * @param {Highcharts.BubbleLegend|Point|Highcharts.Series} item - * @param {Highcharts.SVGElement} legendItem - * @param {boolean} [useHTML=false] - * @fires Highcharts.Point#event:legendItemClick - * @fires Highcharts.Series#event:legendItemClick - */ - setItemEvents: function (item, legendItem, useHTML) { - var legend = this, - boxWrapper = legend.chart.renderer.boxWrapper, - isPoint = item instanceof Point, - activeClass = 'highcharts-legend-' + - (isPoint ? 'point' : 'series') + '-active', - styledMode = legend.chart.styledMode, - // When `useHTML`, the symbol is rendered in other group, so - // we need to apply events listeners to both places - legendItems = useHTML ? - [legendItem, - item.legendSymbol] : - [item.legendGroup]; - // Set the events on the item group, or in case of useHTML, the item - // itself (#1249) - legendItems.forEach(function (element) { - if (element) { - element - .on('mouseover', function () { - if (item.visible) { - legend.allItems.forEach(function (inactiveItem) { - if (item !== inactiveItem) { - inactiveItem.setState('inactive', !isPoint); - } - }); - } - item.setState('hover'); - // A CSS class to dim or hide other than the hovered - // series. - // Works only if hovered series is visible (#10071). - if (item.visible) { - boxWrapper.addClass(activeClass); - } - if (!styledMode) { - legendItem.css(legend.options.itemHoverStyle); - } - }) - .on('mouseout', function () { - if (!legend.chart.styledMode) { - legendItem.css(merge(item.visible ? - legend.itemStyle : - legend.itemHiddenStyle)); - } - legend.allItems.forEach(function (inactiveItem) { - if (item !== inactiveItem) { - inactiveItem.setState('', !isPoint); - } - }); - // A CSS class to dim or hide other than the hovered - // series. - boxWrapper.removeClass(activeClass); - item.setState(); - }) - .on('click', function (event) { - var strLegendItemClick = 'legendItemClick', - fnLegendItemClick = function () { - if (item.setVisible) { - item.setVisible(); - } - // Reset inactive state - legend.allItems.forEach(function (inactiveItem) { - if (item !== inactiveItem) { - inactiveItem.setState(item.visible ? 'inactive' : '', !isPoint); - } - }); - }; - // A CSS class to dim or hide other than the hovered - // series. Event handling in iOS causes the activeClass - // to be added prior to click in some cases (#7418). - boxWrapper.removeClass(activeClass); - // Pass over the click/touch event. #4. - event = { - browserEvent: event - }; - // click the name or symbol - if (item.firePointEvent) { // point - item.firePointEvent(strLegendItemClick, event, fnLegendItemClick); - } - else { - fireEvent(item, strLegendItemClick, event, fnLegendItemClick); - } - }); - } - }); - }, - /** - * @private - * @function Highcharts.Legend#createCheckboxForItem - * @param {Highcharts.BubbleLegend|Point|Highcharts.Series} item - * @fires Highcharts.Series#event:checkboxClick - */ - createCheckboxForItem: function (item) { - var legend = this; - item.checkbox = createElement('input', { - type: 'checkbox', - className: 'highcharts-legend-checkbox', - checked: item.selected, - defaultChecked: item.selected // required by IE7 - }, legend.options.itemCheckboxStyle, legend.chart.container); - addEvent(item.checkbox, 'click', function (event) { - var target = event.target; - fireEvent(item.series || item, 'checkboxClick', { - checked: target.checked, - item: item - }, function () { - item.select(); - }); - }); + if (rightContX) { + this.chart.renderer.circle( + rightContX + this.chart.plotLeft, + rightContY + this.chart.plotTop, + 2 + ) + .attr({ + stroke: 'green', + 'stroke-width': 2, + fill: 'none', + zIndex: 9 + }) + .add(); + this.chart.renderer.path(['M', rightContX + this.chart.plotLeft, + rightContY + this.chart.plotTop, + 'L', plotX + this.chart.plotLeft, plotY + this.chart.plotTop]) + .attr({ + stroke: 'green', + 'stroke-width': 2, + zIndex: 9 + }) + .add(); } - }); - // Extend the Chart object with interaction - extend(Chart.prototype, /** @lends Chart.prototype */ { - /** - * Display the zoom button, so users can reset zoom to the default view - * settings. - * - * @function Highcharts.Chart#showResetZoom - * - * @fires Highcharts.Chart#event:afterShowResetZoom - * @fires Highcharts.Chart#event:beforeShowResetZoom + // */ + ret = [ + "C", + pick(lastPoint.rightContX, lastPoint.plotX, 0), + pick(lastPoint.rightContY, lastPoint.plotY, 0), + pick(leftContX, plotX, 0), + pick(leftContY, plotY, 0), + plotX, + plotY, + ]; + // reset for updating series later + lastPoint.rightContX = lastPoint.rightContY = void 0; + return ret; + }, + /* eslint-enable valid-jsdoc */ + } + ); + /** + * A `spline` series. If the [type](#series.spline.type) option is + * not specified, it is inherited from [chart.type](#chart.type). + * + * @extends series,plotOptions.spline + * @excluding dataParser, dataURL, step, boostThreshold, boostBlending + * @product highcharts highstock + * @apioption series.spline + */ + /** + * An array of data points for the series. For the `spline` series type, + * points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. The `x` values will be automatically + * calculated, either starting at 0 and incremented by 1, or from + * `pointStart` and `pointInterval` given in the series options. If the axis + * has categories, these will be used. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 2. An array of arrays with 2 values. In this case, the values correspond to + * `x,y`. If the first value is a string, it is applied as the name of the + * point, and the `x` value is inferred. + * ```js + * data: [ + * [0, 9], + * [1, 2], + * [2, 8] + * ] + * ``` + * + * 3. 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.spline.turboThreshold), + * this option is not available. + * ```js + * data: [{ + * x: 1, + * y: 9, + * name: "Point2", + * color: "#00FF00" + * }, { + * x: 1, + * y: 0, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @type {Array<number|Array<(number|string),(number|null)>|null|*>} + * @extends series.line.data + * @product highcharts highstock + * @apioption series.spline.data + */ + (""); // adds doclets above intro transpilat + } + ); + _registerModule( + _modules, + "Series/AreaSplineSeries.js", + [ + _modules["Core/Series/Series.js"], + _modules["Mixins/LegendSymbol.js"], + _modules["Core/Options.js"], + ], + function (BaseSeries, LegendSymbolMixin, O) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var defaultOptions = O.defaultOptions; + var areaProto = BaseSeries.seriesTypes.area.prototype; + /** + * AreaSpline series type. + * + * @private + * @class + * @name Highcharts.seriesTypes.areaspline + * + * @augments Highcharts.Series + */ + BaseSeries.seriesType( + "areaspline", + "spline", + /** + * The area spline series is an area series where the graph between the + * points is smoothed into a spline. + * + * @sample {highcharts} highcharts/demo/areaspline/ + * Area spline chart + * @sample {highstock} stock/demo/areaspline/ + * Area spline chart + * + * @extends plotOptions.area + * @excluding step, boostThreshold, boostBlending + * @product highcharts highstock + * @apioption plotOptions.areaspline + */ + /** + * @see [fillColor](#plotOptions.areaspline.fillColor) + * @see [fillOpacity](#plotOptions.areaspline.fillOpacity) + * + * @apioption plotOptions.areaspline.color + */ + /** + * @see [color](#plotOptions.areaspline.color) + * @see [fillOpacity](#plotOptions.areaspline.fillOpacity) + * + * @apioption plotOptions.areaspline.fillColor + */ + /** + * @see [color](#plotOptions.areaspline.color) + * @see [fillColor](#plotOptions.areaspline.fillColor) + * + * @default {highcharts} 0.75 + * @default {highstock} 0.75 + * @apioption plotOptions.areaspline.fillOpacity + */ + defaultOptions.plotOptions.area, + { + getStackPoints: areaProto.getStackPoints, + getGraphPath: areaProto.getGraphPath, + drawGraph: areaProto.drawGraph, + drawLegendSymbol: LegendSymbolMixin.drawRectangle, + } + ); + /** + * A `areaspline` series. If the [type](#series.areaspline.type) option + * is not specified, it is inherited from [chart.type](#chart.type). + * + * + * @extends series,plotOptions.areaspline + * @excluding dataParser, dataURL, step, boostThreshold, boostBlending + * @product highcharts highstock + * @apioption series.areaspline + */ + /** + * @see [fillColor](#series.areaspline.fillColor) + * @see [fillOpacity](#series.areaspline.fillOpacity) + * + * @apioption series.areaspline.color + */ + /** + * An array of data points for the series. For the `areaspline` series + * type, points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. The `x` values will be automatically + * calculated, either starting at 0 and incremented by 1, or from + * `pointStart` and `pointInterval` given in the series options. If the axis + * has categories, these will be used. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 2. An array of arrays with 2 values. In this case, the values correspond to + * `x,y`. If the first value is a string, it is applied as the name of the + * point, and the `x` value is inferred. + * ```js + * data: [ + * [0, 10], + * [1, 9], + * [2, 3] + * ] + * ``` + * + * 3. 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.areaspline.turboThreshold), this option is not + * available. + * ```js + * data: [{ + * x: 1, + * y: 4, + * name: "Point2", + * color: "#00FF00" + * }, { + * x: 1, + * y: 4, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @type {Array<number|Array<(number|string),(number|null)>|null|*>} + * @extends series.line.data + * @product highcharts highstock + * @apioption series.areaspline.data + */ + /** + * @see [color](#series.areaspline.color) + * @see [fillOpacity](#series.areaspline.fillOpacity) + * + * @apioption series.areaspline.fillColor + */ + /** + * @see [color](#series.areaspline.color) + * @see [fillColor](#series.areaspline.fillColor) + * + * @default {highcharts} 0.75 + * @default {highstock} 0.75 + * @apioption series.areaspline.fillOpacity + */ + (""); // adds doclets above into transpilat + } + ); + _registerModule( + _modules, + "Series/ColumnSeries.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Series/Series.js"], + _modules["Core/Color/Color.js"], + _modules["Core/Globals.js"], + _modules["Mixins/LegendSymbol.js"], + _modules["Series/LineSeries.js"], + _modules["Core/Utilities.js"], + ], + function (A, BaseSeries, Color, H, LegendSymbolMixin, LineSeries, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var animObject = A.animObject; + var color = Color.parse; + var noop = H.noop; + var clamp = U.clamp, + defined = U.defined, + extend = U.extend, + isArray = U.isArray, + isNumber = U.isNumber, + merge = U.merge, + pick = U.pick, + objectEach = U.objectEach; + /** + * Adjusted width and x offset of the columns for grouping. + * + * @private + * @interface Highcharts.ColumnMetricsObject + */ /** + * Width of the columns. + * @name Highcharts.ColumnMetricsObject#width + * @type {number} + */ /** + * Offset of the columns. + * @name Highcharts.ColumnMetricsObject#offset + * @type {number} + */ + (""); // detach doclets above + /** + * The column series type. + * + * @private + * @class + * @name Highcharts.seriesTypes.column + * + * @augments Highcharts.Series + */ + var ColumnSeries = BaseSeries.seriesType( + "column", + "line", + /** + * Column series display one column per value along an X axis. + * + * @sample {highcharts} highcharts/demo/column-basic/ + * Column chart + * @sample {highstock} stock/demo/column/ + * Column chart + * + * @extends plotOptions.line + * @excluding connectEnds, connectNulls, gapSize, gapUnit, linecap, + * lineWidth, marker, step, useOhlcData + * @product highcharts highstock + * @optionparent plotOptions.column + */ + { + /** + * The corner radius of the border surrounding each column or bar. + * + * @sample {highcharts} highcharts/plotoptions/column-borderradius/ + * Rounded columns + * + * @product highcharts highstock gantt + * + * @private + */ + borderRadius: 0, + /** + * When using automatic point colors pulled from the global + * [colors](colors) or series-specific + * [plotOptions.column.colors](series.colors) collections, this option + * determines whether the chart should receive one color per series or + * one color per point. + * + * In styled mode, the `colors` or `series.colors` arrays are not + * supported, and instead this option gives the points individual color + * class names on the form `highcharts-color-{n}`. + * + * @see [series colors](#plotOptions.column.colors) + * + * @sample {highcharts} highcharts/plotoptions/column-colorbypoint-false/ + * False by default + * @sample {highcharts} highcharts/plotoptions/column-colorbypoint-true/ + * True + * + * @type {boolean} + * @default false + * @since 2.0 + * @product highcharts highstock gantt + * @apioption plotOptions.column.colorByPoint + */ + /** + * A series specific or series type specific color set to apply instead + * of the global [colors](#colors) when [colorByPoint]( + * #plotOptions.column.colorByPoint) is true. + * + * @type {Array<Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject>} + * @since 3.0 + * @product highcharts highstock gantt + * @apioption plotOptions.column.colors + */ + /** + * When `true`, the columns will center in the category, ignoring null + * or missing points. When `false`, space will be reserved for null or + * missing points. + * + * @sample {highcharts} highcharts/series-column/centerincategory/ + * Center in category + * + * @since 8.0.1 + * @product highcharts highstock gantt + * + * @private + */ + centerInCategory: false, + /** + * Padding between each value groups, in x axis units. + * + * @sample {highcharts} highcharts/plotoptions/column-grouppadding-default/ + * 0.2 by default + * @sample {highcharts} highcharts/plotoptions/column-grouppadding-none/ + * No group padding - all columns are evenly spaced + * + * @product highcharts highstock gantt + * + * @private + */ + groupPadding: 0.2, + /** + * Whether to group non-stacked columns or to let them render + * independent of each other. Non-grouped columns will be laid out + * individually and overlap each other. + * + * @sample {highcharts} highcharts/plotoptions/column-grouping-false/ + * Grouping disabled + * @sample {highstock} highcharts/plotoptions/column-grouping-false/ + * Grouping disabled + * + * @type {boolean} + * @default true + * @since 2.3.0 + * @product highcharts highstock gantt + * @apioption plotOptions.column.grouping + */ + /** + * @ignore-option + * @private + */ + marker: null, + /** + * The maximum allowed pixel width for a column, translated to the + * height of a bar in a bar chart. This prevents the columns from + * becoming too wide when there is a small number of points in the + * chart. + * + * @see [pointWidth](#plotOptions.column.pointWidth) + * + * @sample {highcharts} highcharts/plotoptions/column-maxpointwidth-20/ + * Limited to 50 + * @sample {highstock} highcharts/plotoptions/column-maxpointwidth-20/ + * Limited to 50 + * + * @type {number} + * @since 4.1.8 + * @product highcharts highstock gantt + * @apioption plotOptions.column.maxPointWidth + */ + /** + * Padding between each column or bar, in x axis units. + * + * @sample {highcharts} highcharts/plotoptions/column-pointpadding-default/ + * 0.1 by default + * @sample {highcharts} highcharts/plotoptions/column-pointpadding-025/ + * 0.25 + * @sample {highcharts} highcharts/plotoptions/column-pointpadding-none/ + * 0 for tightly packed columns + * + * @product highcharts highstock gantt + * + * @private + */ + pointPadding: 0.1, + /** + * A pixel value specifying a fixed width for each column or bar point. + * When `null`, the width is calculated from the `pointPadding` and + * `groupPadding`. The width effects the dimension that is not based on + * the point value. For column series it is the hoizontal length and for + * bar series it is the vertical length. + * + * @see [maxPointWidth](#plotOptions.column.maxPointWidth) + * + * @sample {highcharts} highcharts/plotoptions/column-pointwidth-20/ + * 20px wide columns regardless of chart width or the amount of + * data points + * + * @type {number} + * @since 1.2.5 + * @product highcharts highstock gantt + * @apioption plotOptions.column.pointWidth + */ + /** + * A pixel value specifying a fixed width for the column or bar. + * Overrides pointWidth on the series. + * + * @see [series.pointWidth](#plotOptions.column.pointWidth) + * + * @type {number} + * @default undefined + * @since 7.0.0 + * @product highcharts highstock gantt + * @apioption series.column.data.pointWidth + */ + /** + * The minimal height for a column or width for a bar. By default, + * 0 values are not shown. To visualize a 0 (or close to zero) point, + * set the minimal point length to a pixel value like 3\. In stacked + * column charts, minPointLength might not be respected for tightly + * packed values. + * + * @sample {highcharts} highcharts/plotoptions/column-minpointlength/ + * Zero base value + * @sample {highcharts} highcharts/plotoptions/column-minpointlength-pos-and-neg/ + * Positive and negative close to zero values + * + * @product highcharts highstock gantt + * + * @private + */ + minPointLength: 0, + /** + * When the series contains less points than the crop threshold, all + * points are drawn, event if the points fall outside the visible plot + * area at the current zoom. The advantage of drawing all points + * (including markers and columns), is that animation is performed on + * updates. On the other hand, when the series contains more points than + * the crop threshold, the series data is cropped to only contain points + * that fall within the plot area. The advantage of cropping away + * invisible points is to increase performance on large series. + * + * @product highcharts highstock gantt + * + * @private + */ + cropThreshold: 50, + /** + * The X axis range that each point is valid for. This determines the + * width of the column. On a categorized axis, the range will be 1 + * by default (one category unit). On linear and datetime axes, the + * range will be computed as the distance between the two closest data + * points. + * + * The default `null` means it is computed automatically, but this + * option can be used to override the automatic value. + * + * This option is set by default to 1 if data sorting is enabled. + * + * @sample {highcharts} highcharts/plotoptions/column-pointrange/ + * Set the point range to one day on a data set with one week + * between the points + * + * @type {number|null} + * @since 2.3 + * @product highcharts highstock gantt + * + * @private + */ + pointRange: null, + states: { + /** + * Options for the hovered point. These settings override the normal + * state options when a point is moused over or touched. + * + * @extends plotOptions.series.states.hover + * @excluding halo, lineWidth, lineWidthPlus, marker + * @product highcharts highstock gantt */ - showResetZoom: function () { - var chart = this, - lang = defaultOptions.lang, - btnOptions = chart.options.chart.resetZoomButton, - theme = btnOptions.theme, - states = theme.states, - alignTo = (btnOptions.relativeTo === 'chart' || - btnOptions.relativeTo === 'spaceBox' ? - null : - 'plotBox'); - /** - * @private - */ - function zoomOut() { - chart.zoomOut(); - } - fireEvent(this, 'beforeShowResetZoom', null, function () { - chart.resetZoomButton = chart.renderer - .button(lang.resetZoom, null, null, zoomOut, theme, states && states.hover) - .attr({ - align: btnOptions.position.align, - title: lang.resetZoomTitle - }) - .addClass('highcharts-reset-zoom') - .add() - .align(btnOptions.position, false, alignTo); - }); - fireEvent(this, 'afterShowResetZoom'); + hover: { + /** @ignore-option */ + halo: false, + /** + * A specific border color for the hovered point. Defaults to + * inherit the normal state border color. + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @product highcharts gantt + * @apioption plotOptions.column.states.hover.borderColor + */ + /** + * A specific color for the hovered point. + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @product highcharts gantt + * @apioption plotOptions.column.states.hover.color + */ + /** + * How much to brighten the point on interaction. Requires the + * main color to be defined in hex or rgb(a) format. + * + * In styled mode, the hover brightening is by default replaced + * with a fill-opacity set in the `.highcharts-point:hover` + * rule. + * + * @sample {highcharts} highcharts/plotoptions/column-states-hover-brightness/ + * Brighten by 0.5 + * + * @product highcharts highstock gantt + */ + brightness: 0.1, }, /** - * Zoom the chart out after a user has zoomed in. See also - * [Axis.setExtremes](/class-reference/Highcharts.Axis#setExtremes). - * - * @function Highcharts.Chart#zoomOut + * Options for the selected point. These settings override the + * normal state options when a point is selected. * - * @fires Highcharts.Chart#event:selection + * @extends plotOptions.series.states.select + * @excluding halo, lineWidth, lineWidthPlus, marker + * @product highcharts highstock gantt */ - zoomOut: function () { - fireEvent(this, 'selection', { resetSelection: true }, this.zoom); + select: { + /** + * A specific color for the selected point. + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @default #cccccc + * @product highcharts highstock gantt + */ + color: "#cccccc", + /** + * A specific border color for the selected point. + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @default #000000 + * @product highcharts highstock gantt + */ + borderColor: "#000000", }, + }, + dataLabels: { + align: void 0, + verticalAlign: void 0, /** - * Zoom into a given portion of the chart given by axis coordinates. + * The y position offset of the label relative to the point in + * pixels. * - * @private - * @function Highcharts.Chart#zoom - * @param {Highcharts.SelectEventObject} event - */ - zoom: function (event) { - var chart = this, - hasZoomed, - pointer = chart.pointer, - displayButton = false, - mouseDownPos = chart.inverted ? pointer.mouseDownX : pointer.mouseDownY, - resetZoomButton; - // If zoom is called with no arguments, reset the axes - if (!event || event.resetSelection) { - chart.axes.forEach(function (axis) { - hasZoomed = axis.zoom(); - }); - pointer.initiated = false; // #6804 - } - else { // else, zoom in on all axes - event.xAxis.concat(event.yAxis).forEach(function (axisData) { - var axis = axisData.axis, - axisStartPos = chart.inverted ? axis.left : axis.top, - axisEndPos = chart.inverted ? - axisStartPos + axis.width : axisStartPos + axis.height, - isXAxis = axis.isXAxis, - isWithinPane = false; - // Check if zoomed area is within the pane (#1289). - // In case of multiple panes only one pane should be zoomed. - if ((!isXAxis && - mouseDownPos >= axisStartPos && - mouseDownPos <= axisEndPos) || - isXAxis || - !defined(mouseDownPos)) { - isWithinPane = true; - } - // don't zoom more than minRange - if (pointer[isXAxis ? 'zoomX' : 'zoomY'] && isWithinPane) { - hasZoomed = axis.zoom(axisData.min, axisData.max); - if (axis.displayBtn) { - displayButton = true; - } - } + * @type {number} + */ + y: void 0, + }, + // false doesn't work well: https://jsfiddle.net/highcharts/hz8fopan/14/ + /** + * @ignore-option + * @private + */ + startFromThreshold: true, + stickyTracking: false, + tooltip: { + distance: 6, + }, + /** + * The Y axis value to serve as the base for the columns, for + * distinguishing between values above and below a threshold. If `null`, + * the columns extend from the padding Y axis minimum. + * + * @type {number|null} + * @since 2.0 + * @product highcharts + * + * @private + */ + threshold: 0, + /** + * The width of the border surrounding each column or bar. Defaults to + * `1` when there is room for a border, but to `0` when the columns are + * so dense that a border would cover the next column. + * + * In styled mode, the stroke width can be set with the + * `.highcharts-point` rule. + * + * @sample {highcharts} highcharts/plotoptions/column-borderwidth/ + * 2px black border + * + * @type {number} + * @default undefined + * @product highcharts highstock gantt + * @apioption plotOptions.column.borderWidth + */ + /** + * The color of the border surrounding each column or bar. + * + * In styled mode, the border stroke can be set with the + * `.highcharts-point` rule. + * + * @sample {highcharts} highcharts/plotoptions/column-bordercolor/ + * Dark gray border + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @default #ffffff + * @product highcharts highstock gantt + * + * @private + */ + borderColor: "#ffffff", + }, + /** + * @lends seriesTypes.column.prototype + */ + { + cropShoulder: 0, + // When tooltip is not shared, this series (and derivatives) requires + // direct touch/hover. KD-tree does not apply. + directTouch: true, + trackerGroups: ["group", "dataLabelsGroup"], + // use separate negative stacks, unlike area stacks where a negative + // point is substracted from previous (#1910) + negStacks: true, + /* eslint-disable valid-jsdoc */ + /** + * Initialize the series. Extends the basic Series.init method by + * marking other series of the same type as dirty. + * + * @private + * @function Highcharts.seriesTypes.column#init + * @return {void} + */ + init: function () { + LineSeries.prototype.init.apply(this, arguments); + var series = this, + chart = series.chart; + // if the series is added dynamically, force redraw of other + // series affected by a new column + if (chart.hasRendered) { + chart.series.forEach(function (otherSeries) { + if (otherSeries.type === series.type) { + otherSeries.isDirty = true; + } + }); + } + }, + /** + * Return the width and x offset of the columns adjusted for grouping, + * groupPadding, pointPadding, pointWidth etc. + * + * @private + * @function Highcharts.seriesTypes.column#getColumnMetrics + * @return {Highcharts.ColumnMetricsObject} + */ + getColumnMetrics: function () { + var series = this, + options = series.options, + xAxis = series.xAxis, + yAxis = series.yAxis, + reversedStacks = xAxis.options.reversedStacks, + // Keep backward compatibility: reversed xAxis had reversed + // stacks + reverseStacks = + (xAxis.reversed && !reversedStacks) || + (!xAxis.reversed && reversedStacks), + stackKey, + stackGroups = {}, + columnCount = 0; + // Get the total number of column type series. This is called on + // every series. Consider moving this logic to a chart.orderStacks() + // function and call it on init, addSeries and removeSeries + if (options.grouping === false) { + columnCount = 1; + } else { + series.chart.series.forEach(function (otherSeries) { + var otherYAxis = otherSeries.yAxis, + otherOptions = otherSeries.options, + columnIndex; + if ( + otherSeries.type === series.type && + (otherSeries.visible || + !series.chart.options.chart.ignoreHiddenSeries) && + yAxis.len === otherYAxis.len && + yAxis.pos === otherYAxis.pos + ) { + // #642, #2086 + if ( + otherOptions.stacking && + otherOptions.stacking !== "group" + ) { + stackKey = otherSeries.stackKey; + if (typeof stackGroups[stackKey] === "undefined") { + stackGroups[stackKey] = columnCount++; + } + columnIndex = stackGroups[stackKey]; + } else if (otherOptions.grouping !== false) { + // #1162 + columnIndex = columnCount++; + } + otherSeries.columnIndex = columnIndex; + } + }); + } + var categoryWidth = Math.min( + Math.abs(xAxis.transA) * + ((xAxis.ordinal && xAxis.ordinal.slope) || + options.pointRange || + xAxis.closestPointRange || + xAxis.tickInterval || + 1), // #2610 + xAxis.len // #1535 + ), + groupPadding = categoryWidth * options.groupPadding, + groupWidth = categoryWidth - 2 * groupPadding, + pointOffsetWidth = groupWidth / (columnCount || 1), + pointWidth = Math.min( + options.maxPointWidth || xAxis.len, + pick( + options.pointWidth, + pointOffsetWidth * (1 - 2 * options.pointPadding) + ) + ), + pointPadding = (pointOffsetWidth - pointWidth) / 2, + // #1251, #3737 + colIndex = (series.columnIndex || 0) + (reverseStacks ? 1 : 0), + pointXOffset = + pointPadding + + (groupPadding + + colIndex * pointOffsetWidth - + categoryWidth / 2) * + (reverseStacks ? -1 : 1); + // Save it for reading in linked series (Error bars particularly) + series.columnMetrics = { + width: pointWidth, + offset: pointXOffset, + paddedWidth: pointOffsetWidth, + columnCount: columnCount, + }; + return series.columnMetrics; + }, + /** + * Make the columns crisp. The edges are rounded to the nearest full + * pixel. + * + * @private + * @function Highcharts.seriesTypes.column#crispCol + * @param {number} x + * @param {number} y + * @param {number} w + * @param {number} h + * @return {Highcharts.BBoxObject} + */ + crispCol: function (x, y, w, h) { + var chart = this.chart, + borderWidth = this.borderWidth, + xCrisp = -(borderWidth % 2 ? 0.5 : 0), + yCrisp = borderWidth % 2 ? 0.5 : 1, + right, + bottom, + fromTop; + if (chart.inverted && chart.renderer.isVML) { + yCrisp += 1; + } + // Horizontal. We need to first compute the exact right edge, then + // round it and compute the width from there. + if (this.options.crisp) { + right = Math.round(x + w) + xCrisp; + x = Math.round(x) + xCrisp; + w = right - x; + } + // Vertical + bottom = Math.round(y + h) + yCrisp; + fromTop = Math.abs(y) <= 0.5 && bottom > 0.5; // #4504, #4656 + y = Math.round(y) + yCrisp; + h = bottom - y; + // Top edges are exceptions + if (fromTop && h) { + // #5146 + y -= 1; + h += 1; + } + return { + x: x, + y: y, + width: w, + height: h, + }; + }, + /** + * Adjust for missing columns, according to the `centerInCategory` + * option. Missing columns are either single points or stacks where the + * point or points are either missing or null. + * + * @private + * @function Highcharts.seriesTypes.column#adjustForMissingColumns + * @param {number} x + * The x coordinate of the column, left side + * @param {number} pointWidth + * The pointWidth, already computed upstream + * @param {Highcharts.ColumnPoint} point + * The point instance + * @param {Highcharts.ColumnMetricsObject} metrics + * The series-wide column metrics + * @return {number} + * The adjusted x position, or the original if not adjusted + */ + adjustForMissingColumns: function (x, pointWidth, point, metrics) { + var _this = this; + var stacking = this.options.stacking; + if (!point.isNull && metrics.columnCount > 1) { + var indexInCategory_1 = 0; + var totalInCategory_1 = 0; + // Loop over all the stacks on the Y axis. When stacking is + // enabled, these are real point stacks. When stacking is not + // enabled, but `centerInCategory` is true, there is one stack + // handling the grouping of points in each category. This is + // done in the `setGroupedPoints` function. + objectEach( + this.yAxis.stacking && this.yAxis.stacking.stacks, + function (stack) { + if (typeof point.x === "number") { + var stackItem = stack[point.x.toString()]; + if (stackItem) { + var pointValues = stackItem.points[_this.index], + total = stackItem.total; + // If true `stacking` is enabled, count the + // total number of non-null stacks in the + // category, and note which index this point is + // within those stacks. + if (stacking) { + if (pointValues) { + indexInCategory_1 = totalInCategory_1; + } + if (stackItem.hasValidPoints) { + totalInCategory_1++; + } + // If `stacking` is not enabled, look for the + // index and total of the `group` stack. + } else if (isArray(pointValues)) { + indexInCategory_1 = pointValues[1]; + totalInCategory_1 = total || 0; + } + } + } + } + ); + // Compute the adjusted x position + var boxWidth = + (totalInCategory_1 - 1) * metrics.paddedWidth + pointWidth; + x = + (point.plotX || 0) + + boxWidth / 2 - + pointWidth - + indexInCategory_1 * metrics.paddedWidth; + } + return x; + }, + /** + * Translate each point to the plot area coordinate system and find + * shape positions + * + * @private + * @function Highcharts.seriesTypes.column#translate + */ + translate: function () { + var series = this, + chart = series.chart, + options = series.options, + dense = (series.dense = + series.closestPointRange * series.xAxis.transA < 2), + borderWidth = (series.borderWidth = pick( + options.borderWidth, + dense ? 0 : 1 // #3635 + )), + xAxis = series.xAxis, + yAxis = series.yAxis, + threshold = options.threshold, + translatedThreshold = (series.translatedThreshold = + yAxis.getThreshold(threshold)), + minPointLength = pick(options.minPointLength, 5), + metrics = series.getColumnMetrics(), + seriesPointWidth = metrics.width, + // postprocessed for border width + seriesBarW = (series.barW = Math.max( + seriesPointWidth, + 1 + 2 * borderWidth + )), + seriesXOffset = (series.pointXOffset = metrics.offset), + dataMin = series.dataMin, + dataMax = series.dataMax; + if (chart.inverted) { + translatedThreshold -= 0.5; // #3355 + } + // When the pointPadding is 0, we want the columns to be packed + // tightly, so we allow individual columns to have individual sizes. + // When pointPadding is greater, we strive for equal-width columns + // (#2694). + if (options.pointPadding) { + seriesBarW = Math.ceil(seriesBarW); + } + LineSeries.prototype.translate.apply(series); + // Record the new values + series.points.forEach(function (point) { + var yBottom = pick(point.yBottom, translatedThreshold), + safeDistance = 999 + Math.abs(yBottom), + pointWidth = seriesPointWidth, + plotX = point.plotX || 0, + // Don't draw too far outside plot area (#1303, #2241, + // #4264) + plotY = clamp( + point.plotY, + -safeDistance, + yAxis.len + safeDistance + ), + barX = plotX + seriesXOffset, + barW = seriesBarW, + barY = Math.min(plotY, yBottom), + up, + barH = Math.max(plotY, yBottom) - barY; + // Handle options.minPointLength + if (minPointLength && Math.abs(barH) < minPointLength) { + barH = minPointLength; + up = + (!yAxis.reversed && !point.negative) || + (yAxis.reversed && point.negative); + // Reverse zeros if there's no positive value in the series + // in visible range (#7046) + if ( + isNumber(threshold) && + isNumber(dataMax) && + point.y === threshold && + dataMax <= threshold && + // and if there's room for it (#7311) + (yAxis.min || 0) < threshold && + // if all points are the same value (i.e zero) not draw + // as negative points (#10646) + dataMin !== dataMax + ) { + up = !up; + } + // If stacked... + barY = + Math.abs(barY - translatedThreshold) > minPointLength + ? // ...keep position + yBottom - minPointLength + : // #1485, #4051 + translatedThreshold - (up ? minPointLength : 0); + } + // Handle point.options.pointWidth + // @todo Handle grouping/stacking too. Calculate offset properly + if (defined(point.options.pointWidth)) { + pointWidth = barW = Math.ceil(point.options.pointWidth); + barX -= Math.round((pointWidth - seriesPointWidth) / 2); + } + // Adjust for null or missing points + if (options.centerInCategory) { + barX = series.adjustForMissingColumns( + barX, + pointWidth, + point, + metrics + ); + } + // Cache for access in polar + point.barX = barX; + point.pointWidth = pointWidth; + // Fix the tooltip on center of grouped columns (#1216, #424, + // #3648) + point.tooltipPos = chart.inverted + ? [ + yAxis.len + yAxis.pos - chart.plotLeft - plotY, + xAxis.len + + xAxis.pos - + chart.plotTop - + (plotX || 0) - + seriesXOffset - + barW / 2, + barH, + ] + : [barX + barW / 2, plotY + yAxis.pos - chart.plotTop, barH]; + // Register shape type and arguments to be used in drawPoints + // Allow shapeType defined on pointClass level + point.shapeType = series.pointClass.prototype.shapeType || "rect"; + point.shapeArgs = series.crispCol.apply( + series, + point.isNull + ? // #3169, drilldown from null must have a position to work + // from #6585, dataLabel should be placed on xAxis, not + // floating in the middle of the chart + [barX, translatedThreshold, barW, 0] + : [barX, barY, barW, barH] + ); + }); + }, + getSymbol: noop, + /** + * Use a solid rectangle like the area series types + * + * @private + * @function Highcharts.seriesTypes.column#drawLegendSymbol + * + * @param {Highcharts.Legend} legend + * The legend object + * + * @param {Highcharts.Series|Highcharts.Point} item + * The series (this) or point + */ + drawLegendSymbol: LegendSymbolMixin.drawRectangle, + /** + * Columns have no graph + * + * @private + * @function Highcharts.seriesTypes.column#drawGraph + */ + drawGraph: function () { + this.group[this.dense ? "addClass" : "removeClass"]( + "highcharts-dense-data" + ); + }, + /** + * Get presentational attributes + * + * @private + * @function Highcharts.seriesTypes.column#pointAttribs + * + * @param {Highcharts.ColumnPoint} point + * + * @param {string} state + * + * @return {Highcharts.SVGAttributes} + */ + pointAttribs: function (point, state) { + var options = this.options, + stateOptions, + ret, + p2o = this.pointAttrToOptions || {}, + strokeOption = p2o.stroke || "borderColor", + strokeWidthOption = p2o["stroke-width"] || "borderWidth", + fill = (point && point.color) || this.color, + // set to fill when borderColor null: + stroke = + (point && point[strokeOption]) || + options[strokeOption] || + this.color || + fill, + strokeWidth = + (point && point[strokeWidthOption]) || + options[strokeWidthOption] || + this[strokeWidthOption] || + 0, + dashstyle = + (point && point.options.dashStyle) || options.dashStyle, + opacity = pick(point && point.opacity, options.opacity, 1), + zone, + brightness; + // Handle zone colors + if (point && this.zones.length) { + zone = point.getZone(); + // When zones are present, don't use point.color (#4267). + // Changed order (#6527), added support for colorAxis (#10670) + fill = + point.options.color || + (zone && (zone.color || point.nonZonedColor)) || + this.color; + if (zone) { + stroke = zone.borderColor || stroke; + dashstyle = zone.dashStyle || dashstyle; + strokeWidth = zone.borderWidth || strokeWidth; + } + } + // Select or hover states + if (state && point) { + stateOptions = merge( + options.states[state], + // #6401 + (point.options.states && point.options.states[state]) || {} + ); + brightness = stateOptions.brightness; + fill = + stateOptions.color || + (typeof brightness !== "undefined" && + color(fill).brighten(stateOptions.brightness).get()) || + fill; + stroke = stateOptions[strokeOption] || stroke; + strokeWidth = stateOptions[strokeWidthOption] || strokeWidth; + dashstyle = stateOptions.dashStyle || dashstyle; + opacity = pick(stateOptions.opacity, opacity); + } + ret = { + fill: fill, + stroke: stroke, + "stroke-width": strokeWidth, + opacity: opacity, + }; + if (dashstyle) { + ret.dashstyle = dashstyle; + } + return ret; + }, + /** + * Draw the columns. For bars, the series.group is rotated, so the same + * coordinates apply for columns and bars. This method is inherited by + * scatter series. + * + * @private + * @function Highcharts.seriesTypes.column#drawPoints + */ + drawPoints: function () { + var series = this, + chart = this.chart, + options = series.options, + renderer = chart.renderer, + animationLimit = options.animationLimit || 250, + shapeArgs; + // draw the columns + series.points.forEach(function (point) { + var plotY = point.plotY, + graphic = point.graphic, + hasGraphic = !!graphic, + verb = + graphic && chart.pointCount < animationLimit + ? "animate" + : "attr"; + if (isNumber(plotY) && point.y !== null) { + shapeArgs = point.shapeArgs; + // When updating a series between 2d and 3d or cartesian and + // polar, the shape type changes. + if (graphic && point.hasNewShapeType()) { + graphic = graphic.destroy(); + } + // Set starting position for point sliding animation. + if (series.enabledDataSorting) { + point.startXPos = series.xAxis.reversed + ? -(shapeArgs ? shapeArgs.width : 0) + : series.xAxis.width; + } + if (!graphic) { + point.graphic = graphic = renderer[point.shapeType]( + shapeArgs + ).add(point.group || series.group); + if ( + graphic && + series.enabledDataSorting && + chart.hasRendered && + chart.pointCount < animationLimit + ) { + graphic.attr({ + x: point.startXPos, }); + hasGraphic = true; + verb = "animate"; + } } - // Show or hide the Reset zoom button - resetZoomButton = chart.resetZoomButton; - if (displayButton && !resetZoomButton) { - chart.showResetZoom(); - } - else if (!displayButton && isObject(resetZoomButton)) { - chart.resetZoomButton = resetZoomButton.destroy(); + if (graphic && hasGraphic) { + // update + graphic[verb](merge(shapeArgs)); } - // Redraw - if (hasZoomed) { - chart.redraw(pick(chart.options.chart.animation, event && event.animation, chart.pointCount < 100)); + // Border radius is not stylable (#6900) + if (options.borderRadius) { + graphic[verb]({ + r: options.borderRadius, + }); } - }, + // Presentational + if (!chart.styledMode) { + graphic[verb]( + series.pointAttribs(point, point.selected && "select") + ).shadow( + point.allowShadow !== false && options.shadow, + null, + options.stacking && !options.borderRadius + ); + } + graphic.addClass(point.getClassName(), true); + } else if (graphic) { + point.graphic = graphic.destroy(); // #1269 + } + }); + }, + /** + * Animate the column heights one by one from zero. + * + * @private + * @function Highcharts.seriesTypes.column#animate + * + * @param {boolean} init + * Whether to initialize the animation or run it + */ + animate: function (init) { + var series = this, + yAxis = this.yAxis, + options = series.options, + inverted = this.chart.inverted, + attr = {}, + translateProp = inverted ? "translateX" : "translateY", + translateStart, + translatedThreshold; + if (init) { + attr.scaleY = 0.001; + translatedThreshold = clamp( + yAxis.toPixels(options.threshold), + yAxis.pos, + yAxis.pos + yAxis.len + ); + if (inverted) { + attr.translateX = translatedThreshold - yAxis.len; + } else { + attr.translateY = translatedThreshold; + } + // apply finnal clipping (used in Highstock) (#7083) + // animation is done by scaleY, so cliping is for panes + if (series.clipBox) { + series.setClip(); + } + series.group.attr(attr); + } else { + // run the animation + translateStart = series.group.attr(translateProp); + series.group.animate( + { scaleY: 1 }, + extend(animObject(series.options.animation), { + // Do the scale synchronously to ensure smooth + // updating (#5030, #7228) + step: function (val, fx) { + if (series.group) { + attr[translateProp] = + translateStart + fx.pos * (yAxis.pos - translateStart); + series.group.attr(attr); + } + }, + }) + ); + } + }, + /** + * Remove this series from the chart + * + * @private + * @function Highcharts.seriesTypes.column#remove + */ + remove: function () { + var series = this, + chart = series.chart; + // column and bar series affects other series of the same type + // as they are either stacked or grouped + if (chart.hasRendered) { + chart.series.forEach(function (otherSeries) { + if (otherSeries.type === series.type) { + otherSeries.isDirty = true; + } + }); + } + LineSeries.prototype.remove.apply(series, arguments); + }, + } + ); + /* eslint-enable valid-jsdoc */ + /** + * A `column` series. If the [type](#series.column.type) option is + * not specified, it is inherited from [chart.type](#chart.type). + * + * @extends series,plotOptions.column + * @excluding connectNulls, dataParser, dataURL, gapSize, gapUnit, linecap, + * lineWidth, marker, connectEnds, step + * @product highcharts highstock + * @apioption series.column + */ + /** + * An array of data points for the series. For the `column` series type, + * points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. The `x` values will be automatically + * calculated, either starting at 0 and incremented by 1, or from + * `pointStart` and `pointInterval` given in the series options. If the axis + * has categories, these will be used. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 2. An array of arrays with 2 values. In this case, the values correspond to + * `x,y`. If the first value is a string, it is applied as the name of the + * point, and the `x` value is inferred. + * ```js + * data: [ + * [0, 6], + * [1, 2], + * [2, 6] + * ] + * ``` + * + * 3. 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.column.turboThreshold), this option is not + * available. + * ```js + * data: [{ + * x: 1, + * y: 9, + * name: "Point2", + * color: "#00FF00" + * }, { + * x: 1, + * y: 6, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @type {Array<number|Array<(number|string),(number|null)>|null|*>} + * @extends series.line.data + * @excluding marker + * @product highcharts highstock + * @apioption series.column.data + */ + /** + * The color of the border surrounding the column or bar. + * + * In styled mode, the border stroke can be set with the `.highcharts-point` + * rule. + * + * @sample {highcharts} highcharts/plotoptions/column-bordercolor/ + * Dark gray border + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @product highcharts highstock + * @apioption series.column.data.borderColor + */ + /** + * The width of the border surrounding the column or bar. + * + * In styled mode, the stroke width can be set with the `.highcharts-point` + * rule. + * + * @sample {highcharts} highcharts/plotoptions/column-borderwidth/ + * 2px black border + * + * @type {number} + * @product highcharts highstock + * @apioption series.column.data.borderWidth + */ + /** + * A name for the dash style to use for the column or bar. Overrides + * dashStyle on the series. + * + * In styled mode, the stroke dash-array can be set with the same classes as + * listed under [data.color](#series.column.data.color). + * + * @see [series.pointWidth](#plotOptions.column.dashStyle) + * + * @type {Highcharts.DashStyleValue} + * @apioption series.column.data.dashStyle + */ + /** + * A pixel value specifying a fixed width for the column or bar. Overrides + * pointWidth on the series. The width effects the dimension that is not based + * on the point value. + * + * @see [series.pointWidth](#plotOptions.column.pointWidth) + * + * @type {number} + * @apioption series.column.data.pointWidth + */ + /** + * @excluding halo, lineWidth, lineWidthPlus, marker + * @product highcharts highstock + * @apioption series.column.states.hover + */ + /** + * @excluding halo, lineWidth, lineWidthPlus, marker + * @product highcharts highstock + * @apioption series.column.states.select + */ + (""); // includes above doclets in transpilat + + return ColumnSeries; + } + ); + _registerModule( + _modules, + "Series/BarSeries.js", + [_modules["Core/Series/Series.js"]], + function (Series) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + /** + * Bar series type. + * + * @private + * @class + * @name Highcharts.seriesTypes.bar + * + * @augments Highcharts.Series + */ + Series.seriesType( + "bar", + "column", + /** + * A bar series is a special type of column series where the columns are + * horizontal. + * + * @sample highcharts/demo/bar-basic/ + * Bar chart + * + * @extends plotOptions.column + * @product highcharts + * @apioption plotOptions.bar + */ + /** + * @ignore + */ + null, + { + inverted: true, + } + ); + /** + * A `bar` series. If the [type](#series.bar.type) option is not specified, + * it is inherited from [chart.type](#chart.type). + * + * @extends series,plotOptions.bar + * @excluding connectNulls, dashStyle, dataParser, dataURL, gapSize, gapUnit, + * linecap, lineWidth, marker, connectEnds, step + * @product highcharts + * @apioption series.bar + */ + /** + * An array of data points for the series. For the `bar` series type, + * points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. The `x` values will be automatically + * calculated, either starting at 0 and incremented by 1, or from + * `pointStart` and `pointInterval` given in the series options. If the axis + * has categories, these will be used. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 2. An array of arrays with 2 values. In this case, the values correspond to + * `x,y`. If the first value is a string, it is applied as the name of the + * point, and the `x` value is inferred. + * ```js + * data: [ + * [0, 5], + * [1, 10], + * [2, 3] + * ] + * ``` + * + * 3. 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.bar.turboThreshold), this option is not + * available. + * ```js + * data: [{ + * x: 1, + * y: 1, + * name: "Point2", + * color: "#00FF00" + * }, { + * x: 1, + * y: 10, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @type {Array<number|Array<(number|string),(number|null)>|null|*>} + * @extends series.column.data + * @product highcharts + * @apioption series.bar.data + */ + /** + * @excluding halo,lineWidth,lineWidthPlus,marker + * @product highcharts highstock + * @apioption series.bar.states.hover + */ + /** + * @excluding halo,lineWidth,lineWidthPlus,marker + * @product highcharts highstock + * @apioption series.bar.states.select + */ + (""); // gets doclets above into transpilat + } + ); + _registerModule( + _modules, + "Series/ScatterSeries.js", + [ + _modules["Core/Series/Series.js"], + _modules["Core/Globals.js"], + _modules["Core/Utilities.js"], + ], + function (BaseSeries, H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var addEvent = U.addEvent; + var Series = H.Series; + /** + * Scatter series type. + * + * @private + * @class + * @name Highcharts.seriesTypes.scatter + * + * @augments Highcharts.Series + */ + BaseSeries.seriesType( + "scatter", + "line", + /** + * A scatter plot uses cartesian coordinates to display values for two + * variables for a set of data. + * + * @sample {highcharts} highcharts/demo/scatter/ + * Scatter plot + * + * @extends plotOptions.line + * @excluding cropThreshold, pointPlacement, shadow, useOhlcData + * @product highcharts highstock + * @optionparent plotOptions.scatter + */ + { + /** + * The width of the line connecting the data points. + * + * @sample {highcharts} highcharts/plotoptions/scatter-linewidth-none/ + * 0 by default + * @sample {highcharts} highcharts/plotoptions/scatter-linewidth-1/ + * 1px + * + * @product highcharts highstock + */ + lineWidth: 0, + findNearestPointBy: "xy", + /** + * Apply a jitter effect for the rendered markers. When plotting + * discrete values, a little random noise may help telling the points + * apart. The jitter setting applies a random displacement of up to `n` + * axis units in either direction. So for example on a horizontal X + * axis, setting the `jitter.x` to 0.24 will render the point in a + * random position between 0.24 units to the left and 0.24 units to the + * right of the true axis position. On a category axis, setting it to + * 0.5 will fill up the bin and make the data appear continuous. + * + * When rendered on top of a box plot or a column series, a jitter value + * of 0.24 will correspond to the underlying series' default + * [groupPadding]( + * https://api.highcharts.com/highcharts/plotOptions.column.groupPadding) + * and [pointPadding]( + * https://api.highcharts.com/highcharts/plotOptions.column.pointPadding) + * settings. + * + * @sample {highcharts} highcharts/series-scatter/jitter + * Jitter on a scatter plot + * + * @sample {highcharts} highcharts/series-scatter/jitter-boxplot + * Jittered scatter plot on top of a box plot + * + * @product highcharts highstock + * @since 7.0.2 + */ + jitter: { + /** + * The maximal X offset for the random jitter effect. + */ + x: 0, + /** + * The maximal Y offset for the random jitter effect. + */ + y: 0, + }, + marker: { + enabled: true, // Overrides auto-enabling in line series (#3647) + }, + /** + * Sticky tracking of mouse events. When true, the `mouseOut` event + * on a series isn't triggered until the mouse moves over another + * series, or out of the plot area. When false, the `mouseOut` event on + * a series is triggered when the mouse leaves the area around the + * series' graph or markers. This also implies the tooltip. When + * `stickyTracking` is false and `tooltip.shared` is false, the tooltip + * will be hidden when moving the mouse between series. + * + * @type {boolean} + * @default false + * @product highcharts highstock + * @apioption plotOptions.scatter.stickyTracking + */ + /** + * A configuration object for the tooltip rendering of each single + * series. Properties are inherited from [tooltip](#tooltip). + * Overridable properties are `headerFormat`, `pointFormat`, + * `yDecimals`, `xDateFormat`, `yPrefix` and `ySuffix`. Unlike other + * series, in a scatter plot the series.name by default shows in the + * headerFormat and point.x and point.y in the pointFormat. + * + * @product highcharts highstock + */ + tooltip: { + headerFormat: + '<span style="color:{point.color}">\u25CF</span> ' + + '<span style="font-size: 10px"> {series.name}</span><br/>', + pointFormat: "x: <b>{point.x}</b><br/>y: <b>{point.y}</b><br/>", + }, + // Prototype members + }, + { + sorted: false, + requireSorting: false, + noSharedTooltip: true, + trackerGroups: ["group", "markerGroup", "dataLabelsGroup"], + takeOrdinalPosition: false, + /* eslint-disable valid-jsdoc */ + /** + * @private + * @function Highcharts.seriesTypes.scatter#drawGraph + */ + drawGraph: function () { + if ( + this.options.lineWidth || + // In case we have a graph from before and we update the line + // width to 0 (#13816) + (this.options.lineWidth === 0 && + this.graph && + this.graph.strokeWidth()) + ) { + Series.prototype.drawGraph.call(this); + } + }, + // Optionally add the jitter effect + applyJitter: function () { + var series = this, + jitter = this.options.jitter, + len = this.points.length; /** - * Pan the chart by dragging the mouse across the pane. This function is - * called on mouse move, and the distance to pan is computed from chartX - * compared to the first chartX position in the dragging operation. - * + * Return a repeatable, pseudo-random number based on an integer + * seed. * @private - * @function Highcharts.Chart#pan - * @param {Highcharts.PointerEventObject} e - * @param {string} panning */ - pan: function (e, panning) { - var chart = this, - hoverPoints = chart.hoverPoints, - panningOptions, - chartOptions = chart.options.chart, - hasMapNavigation = chart.options.mapNavigation && - chart.options.mapNavigation.enabled, - doRedraw, - type; - if (typeof panning === 'object') { - panningOptions = panning; - } - else { - panningOptions = { - enabled: panning, - type: 'x' - }; - } - if (chartOptions && chartOptions.panning) { - chartOptions.panning = panningOptions; - } - type = panningOptions.type; - fireEvent(this, 'pan', { originalEvent: e }, function () { - // remove active points for shared tooltip - if (hoverPoints) { - hoverPoints.forEach(function (point) { - point.setState(); - }); - } - // panning axis mapping - var xy = [1]; // x - if (type === 'xy') { - xy = [1, 0]; - } - else if (type === 'y') { - xy = [0]; - } - xy.forEach(function (isX) { - var axis = chart[isX ? 'xAxis' : 'yAxis'][0], horiz = axis.horiz, mousePos = e[horiz ? 'chartX' : 'chartY'], mouseDown = horiz ? 'mouseDownX' : 'mouseDownY', startPos = chart[mouseDown], halfPointRange = (axis.pointRange || 0) / 2, pointRangeDirection = (axis.reversed && !chart.inverted) || - (!axis.reversed && chart.inverted) ? - -1 : - 1, extremes = axis.getExtremes(), panMin = axis.toValue(startPos - mousePos, true) + - halfPointRange * pointRangeDirection, panMax = axis.toValue(startPos + axis.len - mousePos, true) - - halfPointRange * pointRangeDirection, flipped = panMax < panMin, newMin = flipped ? panMax : panMin, newMax = flipped ? panMin : panMax, hasVerticalPanning = axis.hasVerticalPanning(), paddedMin, paddedMax, spill, panningState = axis.panningState; - // General calculations of panning state. - // This is related to using vertical panning. (#11315). - axis.series.forEach(function (series) { - if (hasVerticalPanning && - !isX && (!panningState || panningState.isDirty)) { - var processedData = series.getProcessedData(true), - dataExtremes = series.getExtremes(processedData.yData, - true); - if (!panningState) { - panningState = { - startMin: Number.MAX_VALUE, - startMax: -Number.MAX_VALUE - }; - } - if (isNumber(dataExtremes.dataMin) && - isNumber(dataExtremes.dataMax)) { - panningState.startMin = Math.min(dataExtremes.dataMin, panningState.startMin); - panningState.startMax = Math.max(dataExtremes.dataMax, panningState.startMax); - } - } - }); - paddedMin = Math.min(pick(panningState === null || panningState === void 0 ? void 0 : panningState.startMin, extremes.dataMin), halfPointRange ? - extremes.min : - axis.toValue(axis.toPixels(extremes.min) - - axis.minPixelPadding)); - paddedMax = Math.max(pick(panningState === null || panningState === void 0 ? void 0 : panningState.startMax, extremes.dataMax), halfPointRange ? - extremes.max : - axis.toValue(axis.toPixels(extremes.max) + - axis.minPixelPadding)); - axis.panningState = panningState; - // It is not necessary to calculate extremes on ordinal axis, - // because they are already calculated, so we don't want to - // override them. - if (!axis.isOrdinal) { - // If the new range spills over, either to the min or max, - // adjust the new range. - spill = paddedMin - newMin; - if (spill > 0) { - newMax += spill; - newMin = paddedMin; - } - spill = newMax - paddedMax; - if (spill > 0) { - newMax = paddedMax; - newMin -= spill; - } - // Set new extremes if they are actually new - if (axis.series.length && - newMin !== extremes.min && - newMax !== extremes.max && - newMin >= paddedMin && - newMax <= paddedMax) { - axis.setExtremes(newMin, newMax, false, false, { trigger: 'pan' }); - if (!chart.resetZoomButton && - !hasMapNavigation && - // Show reset zoom button only when both newMin and - // newMax values are between padded axis range. - newMin !== paddedMin && - newMax !== paddedMax && - type.match('y')) { - chart.showResetZoom(); - axis.displayBtn = false; - } - doRedraw = true; - } - // set new reference for next run: - chart[mouseDown] = mousePos; - } - }); - if (doRedraw) { - chart.redraw(false); - } - css(chart.container, { cursor: 'move' }); - }); + function unrandom(seed) { + var rand = Math.sin(seed) * 10000; + return rand - Math.floor(rand); } - }); - // Extend the Point object with interaction - extend(Point.prototype, /** @lends Highcharts.Point.prototype */ { - /** - * Toggle the selection status of a point. - * - * @see Highcharts.Chart#getSelectedPoints - * - * @sample highcharts/members/point-select/ - * Select a point from a button - * @sample highcharts/chart/events-selection-points/ - * Select a range of points through a drag selection - * @sample maps/series/data-id/ - * Select a point in Highmaps - * - * @function Highcharts.Point#select - * - * @param {boolean} [selected] - * When `true`, the point is selected. When `false`, the point is - * unselected. When `null` or `undefined`, the selection state is toggled. - * - * @param {boolean} [accumulate=false] - * When `true`, the selection is added to other selected points. - * When `false`, other selected points are deselected. Internally in - * Highcharts, when - * [allowPointSelect](https://api.highcharts.com/highcharts/plotOptions.series.allowPointSelect) - * is `true`, selected points are accumulated on Control, Shift or Cmd - * clicking the point. - * - * @fires Highcharts.Point#event:select - * @fires Highcharts.Point#event:unselect - */ - select: function (selected, accumulate) { - var point = this, - series = point.series, - chart = series.chart; - selected = pick(selected, !point.selected); - this.selectedStaging = selected; - // fire the event with the default handler - point.firePointEvent(selected ? 'select' : 'unselect', { accumulate: accumulate }, function () { - /** - * Whether the point is selected or not. - * - * @see Point#select - * @see Chart#getSelectedPoints - * - * @name Highcharts.Point#selected - * @type {boolean} - */ - point.selected = point.options.selected = selected; - series.options.data[series.data.indexOf(point)] = - point.options; - point.setState(selected && 'select'); - // unselect all other points unless Ctrl or Cmd + click - if (!accumulate) { - chart.getSelectedPoints().forEach(function (loopPoint) { - var loopSeries = loopPoint.series; - if (loopPoint.selected && loopPoint !== point) { - loopPoint.selected = loopPoint.options.selected = - false; - loopSeries.options.data[loopSeries.data.indexOf(loopPoint)] = loopPoint.options; - // Programatically selecting a point should restore - // normal state, but when click happened on other - // point, set inactive state to match other points - loopPoint.setState(chart.hoverPoints && - loopSeries.options.inactiveOtherPoints ? - 'inactive' : ''); - loopPoint.firePointEvent('unselect'); - } - }); - } + if (jitter) { + this.points.forEach(function (point, i) { + ["x", "y"].forEach(function (dim, j) { + var axis, + plotProp = "plot" + dim.toUpperCase(), + min, + max, + translatedJitter; + if (jitter[dim] && !point.isNull) { + axis = series[dim + "Axis"]; + translatedJitter = jitter[dim] * axis.transA; + if (axis && !axis.isLog) { + // Identify the outer bounds of the jitter range + min = Math.max(0, point[plotProp] - translatedJitter); + max = Math.min( + axis.len, + point[plotProp] + translatedJitter + ); + // Find a random position within this range + point[plotProp] = + min + (max - min) * unrandom(i + j * len); + // Update clientX for the tooltip k-d-tree + if (dim === "x") { + point.clientX = point.plotX; + } + } + } }); - delete this.selectedStaging; - }, - /** - * Runs on mouse over the point. Called internally from mouse and touch - * events. - * - * @function Highcharts.Point#onMouseOver - * - * @param {Highcharts.PointerEventObject} [e] - * The event arguments. - */ - onMouseOver: function (e) { - var point = this, - series = point.series, - chart = series.chart, - pointer = chart.pointer; - e = e ? - pointer.normalize(e) : - // In cases where onMouseOver is called directly without an event - pointer.getChartCoordinatesFromPoint(point, chart.inverted); - pointer.runPointActions(e, point); - }, - /** - * Runs on mouse out from the point. Called internally from mouse and touch - * events. - * - * @function Highcharts.Point#onMouseOut - * @fires Highcharts.Point#event:mouseOut - */ - onMouseOut: function () { - var point = this, - chart = point.series.chart; - point.firePointEvent('mouseOut'); - if (!point.series.options.inactiveOtherPoints) { - (chart.hoverPoints || []).forEach(function (p) { - p.setState(); - }); - } - chart.hoverPoints = chart.hoverPoint = null; - }, - /** - * Import events from the series' and point's options. Only do it on - * demand, to save processing time on hovering. - * - * @private - * @function Highcharts.Point#importEvents - */ - importEvents: function () { - if (!this.hasImportedEvents) { - var point = this, - options = merge(point.series.options.point, - point.options), - events = options.events; - point.events = events; - objectEach(events, function (event, eventType) { - if (isFunction(event)) { - addEvent(point, eventType, event); - } - }); - this.hasImportedEvents = true; - } - }, - /** - * Set the point's state. - * - * @function Highcharts.Point#setState - * - * @param {Highcharts.PointStateValue|""} [state] - * The new state, can be one of `'hover'`, `'select'`, `'inactive'`, - * or `''` (an empty string), `'normal'` or `undefined` to set to - * normal state. - * @param {boolean} [move] - * State for animation. - * - * @fires Highcharts.Point#event:afterSetState - */ - setState: function (state, move) { - var point = this, - series = point.series, - previousState = point.state, - stateOptions = (series.options.states[state || 'normal'] || - {}), - markerOptions = (defaultOptions.plotOptions[series.type].marker && - series.options.marker), - normalDisabled = (markerOptions && markerOptions.enabled === false), - markerStateOptions = ((markerOptions && - markerOptions.states && - markerOptions.states[state || 'normal']) || {}), - stateDisabled = markerStateOptions.enabled === false, - stateMarkerGraphic = series.stateMarkerGraphic, - pointMarker = point.marker || {}, - chart = series.chart, - halo = series.halo, - haloOptions, - markerAttribs, - pointAttribs, - pointAttribsAnimation, - hasMarkers = (markerOptions && series.markerAttribs), - newSymbol; - state = state || ''; // empty string - if ( - // already has this state - (state === point.state && !move) || - // selected points don't respond to hover - (point.selected && state !== 'select') || - // series' state options is disabled - (stateOptions.enabled === false) || - // general point marker's state options is disabled - (state && (stateDisabled || - (normalDisabled && - markerStateOptions.enabled === false))) || - // individual point marker's state options is disabled - (state && - pointMarker.states && - pointMarker.states[state] && - pointMarker.states[state].enabled === false) // #1610 - ) { - return; - } - point.state = state; - if (hasMarkers) { - markerAttribs = series.markerAttribs(point, state); - } - // Apply hover styles to the existing point - if (point.graphic) { - if (previousState) { - point.graphic.removeClass('highcharts-point-' + previousState); - } - if (state) { - point.graphic.addClass('highcharts-point-' + state); - } - if (!chart.styledMode) { - pointAttribs = series.pointAttribs(point, state); - pointAttribsAnimation = pick(chart.options.chart.animation, stateOptions.animation); - // Some inactive points (e.g. slices in pie) should apply - // oppacity also for it's labels - if (series.options.inactiveOtherPoints && pointAttribs.opacity) { - (point.dataLabels || []).forEach(function (label) { - if (label) { - label.animate({ - opacity: pointAttribs.opacity - }, pointAttribsAnimation); - } - }); - if (point.connector) { - point.connector.animate({ - opacity: pointAttribs.opacity - }, pointAttribsAnimation); - } - } - point.graphic.animate(pointAttribs, pointAttribsAnimation); - } - if (markerAttribs) { - point.graphic.animate(markerAttribs, pick( - // Turn off globally: - chart.options.chart.animation, markerStateOptions.animation, markerOptions.animation)); - } - // Zooming in from a range with no markers to a range with markers - if (stateMarkerGraphic) { - stateMarkerGraphic.hide(); - } - } - else { - // if a graphic is not applied to each point in the normal state, - // create a shared graphic for the hover state - if (state && markerStateOptions) { - newSymbol = pointMarker.symbol || series.symbol; - // If the point has another symbol than the previous one, throw - // away the state marker graphic and force a new one (#1459) - if (stateMarkerGraphic && - stateMarkerGraphic.currentSymbol !== newSymbol) { - stateMarkerGraphic = stateMarkerGraphic.destroy(); - } - // Add a new state marker graphic - if (markerAttribs) { - if (!stateMarkerGraphic) { - if (newSymbol) { - series.stateMarkerGraphic = stateMarkerGraphic = - chart.renderer - .symbol(newSymbol, markerAttribs.x, markerAttribs.y, markerAttribs.width, markerAttribs.height) - .add(series.markerGroup); - stateMarkerGraphic.currentSymbol = newSymbol; - } - // Move the existing graphic - } - else { - stateMarkerGraphic[move ? 'animate' : 'attr']({ - x: markerAttribs.x, - y: markerAttribs.y - }); - } - } - if (!chart.styledMode && stateMarkerGraphic) { - stateMarkerGraphic.attr(series.pointAttribs(point, state)); - } - } - if (stateMarkerGraphic) { - stateMarkerGraphic[state && point.isInside ? 'show' : 'hide'](); // #2450 - stateMarkerGraphic.element.point = point; // #4310 - } - } - // Show me your halo - haloOptions = stateOptions.halo; - var markerGraphic = (point.graphic || stateMarkerGraphic); - var markerVisibility = (markerGraphic && markerGraphic.visibility || 'inherit'); - if (haloOptions && - haloOptions.size && - markerGraphic && - markerVisibility !== 'hidden' && - !point.isCluster) { - if (!halo) { - series.halo = halo = chart.renderer.path() - // #5818, #5903, #6705 - .add(markerGraphic.parentGroup); - } - halo.show()[move ? 'animate' : 'attr']({ - d: point.haloPath(haloOptions.size) - }); - halo.attr({ - 'class': 'highcharts-halo highcharts-color-' + - pick(point.colorIndex, series.colorIndex) + - (point.className ? ' ' + point.className : ''), - 'visibility': markerVisibility, - 'zIndex': -1 // #4929, #8276 - }); - halo.point = point; // #6055 - if (!chart.styledMode) { - halo.attr(extend({ - 'fill': point.color || series.color, - 'fill-opacity': haloOptions.opacity - }, haloOptions.attributes)); - } - } - else if (halo && halo.point && halo.point.haloPath) { - // Animate back to 0 on the current halo point (#6055) - halo.animate({ d: halo.point.haloPath(0) }, null, - // Hide after unhovering. The `complete` callback runs in the - // halo's context (#7681). - halo.hide); - } - fireEvent(point, 'afterSetState'); - }, + }); + } + }, + /* eslint-enable valid-jsdoc */ + } + ); + /* eslint-disable no-invalid-this */ + addEvent(Series, "afterTranslate", function () { + if (this.applyJitter) { + this.applyJitter(); + } + }); + /* eslint-enable no-invalid-this */ + /** + * A `scatter` series. If the [type](#series.scatter.type) option is + * not specified, it is inherited from [chart.type](#chart.type). + * + * @extends series,plotOptions.scatter + * @excluding cropThreshold, dataParser, dataURL, useOhlcData + * @product highcharts highstock + * @apioption series.scatter + */ + /** + * An array of data points for the series. For the `scatter` series + * type, points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. The `x` values will be automatically + * calculated, either starting at 0 and incremented by 1, or from + * `pointStart` and `pointInterval` given in the series options. If the axis + * has categories, these will be used. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 2. An array of arrays with 2 values. In this case, the values correspond to + * `x,y`. If the first value is a string, it is applied as the name of the + * point, and the `x` value is inferred. + * ```js + * data: [ + * [0, 0], + * [1, 8], + * [2, 9] + * ] + * ``` + * + * 3. 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.scatter.turboThreshold), this option is not + * available. + * ```js + * data: [{ + * x: 1, + * y: 2, + * name: "Point2", + * color: "#00FF00" + * }, { + * x: 1, + * y: 4, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @type {Array<number|Array<(number|string),(number|null)>|null|*>} + * @extends series.line.data + * @product highcharts highstock + * @apioption series.scatter.data + */ + (""); // adds doclets above to transpilat + } + ); + _registerModule( + _modules, + "Mixins/CenteredSeries.js", + [_modules["Core/Globals.js"], _modules["Core/Utilities.js"]], + function (H, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + /** + * @private + * @interface Highcharts.RadianAngles + */ /** + * @name Highcharts.RadianAngles#end + * @type {number} + */ /** + * @name Highcharts.RadianAngles#start + * @type {number} + */ + var isNumber = U.isNumber, + pick = U.pick, + relativeLength = U.relativeLength; + var deg2rad = H.deg2rad; + /* eslint-disable valid-jsdoc */ + /** + * @private + * @mixin Highcharts.CenteredSeriesMixin + */ + var centeredSeriesMixin = (H.CenteredSeriesMixin = { + /** + * Get the center of the pie based on the size and center options relative + * to the plot area. Borrowed by the polar and gauge series types. + * + * @private + * @function Highcharts.CenteredSeriesMixin.getCenter + * + * @return {Array<number>} + */ + getCenter: function () { + var options = this.options, + chart = this.chart, + slicingRoom = 2 * (options.slicedOffset || 0), + handleSlicingRoom, + plotWidth = chart.plotWidth - 2 * slicingRoom, + plotHeight = chart.plotHeight - 2 * slicingRoom, + centerOption = options.center, + smallestSize = Math.min(plotWidth, plotHeight), + size = options.size, + innerSize = options.innerSize || 0, + positions, + i, + value; + if (typeof size === "string") { + size = parseFloat(size); + } + if (typeof innerSize === "string") { + innerSize = parseFloat(innerSize); + } + positions = [ + pick(centerOption[0], "50%"), + pick(centerOption[1], "50%"), + // Prevent from negative values + pick(size && size < 0 ? void 0 : options.size, "100%"), + pick( + innerSize && innerSize < 0 ? void 0 : options.innerSize || 0, + "0%" + ), + ]; + // No need for inner size in angular (gauges) series but still required + // for pie series + if (chart.angular && !(this instanceof H.Series)) { + positions[3] = 0; + } + for (i = 0; i < 4; ++i) { + value = positions[i]; + handleSlicingRoom = i < 2 || (i === 2 && /%$/.test(value)); + // i == 0: centerX, relative to width + // i == 1: centerY, relative to height + // i == 2: size, relative to smallestSize + // i == 3: innerSize, relative to size + positions[i] = + relativeLength( + value, + [plotWidth, plotHeight, smallestSize, positions[2]][i] + ) + (handleSlicingRoom ? slicingRoom : 0); + } + // innerSize cannot be larger than size (#3632) + if (positions[3] > positions[2]) { + positions[3] = positions[2]; + } + return positions; + }, + /** + * getStartAndEndRadians - Calculates start and end angles in radians. + * Used in series types such as pie and sunburst. + * + * @private + * @function Highcharts.CenteredSeriesMixin.getStartAndEndRadians + * + * @param {number} [start] + * Start angle in degrees. + * + * @param {number} [end] + * Start angle in degrees. + * + * @return {Highcharts.RadianAngles} + * Returns an object containing start and end angles as radians. + */ + getStartAndEndRadians: function (start, end) { + var startAngle = isNumber(start) ? start : 0, // must be a number + endAngle = + isNumber(end) && // must be a number + end > startAngle && // must be larger than the start angle + // difference must be less than 360 degrees + end - startAngle < 360 + ? end + : startAngle + 360, + correction = -90; + return { + start: deg2rad * (startAngle + correction), + end: deg2rad * (endAngle + correction), + }; + }, + }); + + return centeredSeriesMixin; + } + ); + _registerModule( + _modules, + "Series/PieSeries.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Series/Series.js"], + _modules["Mixins/CenteredSeries.js"], + _modules["Core/Globals.js"], + _modules["Mixins/LegendSymbol.js"], + _modules["Series/LineSeries.js"], + _modules["Core/Series/Point.js"], + _modules["Core/Renderer/SVG/SVGRenderer.js"], + _modules["Core/Utilities.js"], + ], + function ( + A, + BaseSeries, + CenteredSeriesMixin, + H, + LegendSymbolMixin, + LineSeries, + Point, + SVGRenderer, + U + ) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var setAnimation = A.setAnimation; + var getStartAndEndRadians = CenteredSeriesMixin.getStartAndEndRadians; + var noop = H.noop; + var addEvent = U.addEvent, + clamp = U.clamp, + defined = U.defined, + fireEvent = U.fireEvent, + isNumber = U.isNumber, + merge = U.merge, + pick = U.pick, + relativeLength = U.relativeLength; + /** + * Pie series type. + * + * @private + * @class + * @name Highcharts.seriesTypes.pie + * + * @augments Highcharts.Series + */ + BaseSeries.seriesType( + "pie", + "line", + /** + * A pie chart is a circular graphic which is divided into slices to + * illustrate numerical proportion. + * + * @sample highcharts/demo/pie-basic/ + * Pie chart + * + * @extends plotOptions.line + * @excluding animationLimit, boostThreshold, connectEnds, connectNulls, + * cropThreshold, dashStyle, dataSorting, dragDrop, + * findNearestPointBy, getExtremesFromAll, label, lineWidth, + * marker, negativeColor, pointInterval, pointIntervalUnit, + * pointPlacement, pointStart, softThreshold, stacking, step, + * threshold, turboThreshold, zoneAxis, zones, dataSorting, + * boostBlending + * @product highcharts + * @optionparent plotOptions.pie + */ + { + /** + * @excluding legendItemClick + * @apioption plotOptions.pie.events + */ + /** + * Fires when the checkbox next to the point name in the legend is + * clicked. One parameter, event, is passed to the function. The state + * of the checkbox is found by event.checked. The checked item is found + * by event.item. Return false to prevent the default action which is to + * toggle the select state of the series. + * + * @sample {highcharts} highcharts/plotoptions/series-events-checkboxclick/ + * Alert checkbox status + * + * @type {Function} + * @since 1.2.0 + * @product highcharts + * @context Highcharts.Point + * @apioption plotOptions.pie.events.checkboxClick + */ + /** + * Fires when the legend item belonging to the pie point (slice) is + * clicked. The `this` keyword refers to the point itself. One + * parameter, `event`, is passed to the function, containing common + * event information. The default action is to toggle the visibility of + * the point. This can be prevented by calling `event.preventDefault()`. + * + * @sample {highcharts} highcharts/plotoptions/pie-point-events-legenditemclick/ + * Confirm toggle visibility + * + * @type {Highcharts.PointLegendItemClickCallbackFunction} + * @since 1.2.0 + * @product highcharts + * @apioption plotOptions.pie.point.events.legendItemClick + */ + /** + * The center of the pie chart relative to the plot area. Can be + * percentages or pixel values. The default behaviour (as of 3.0) is to + * center the pie so that all slices and data labels are within the plot + * area. As a consequence, the pie may actually jump around in a chart + * with dynamic values, as the data labels move. In that case, the + * center should be explicitly set, for example to `["50%", "50%"]`. + * + * @sample {highcharts} highcharts/plotoptions/pie-center/ + * Centered at 100, 100 + * + * @type {Array<(number|string|null),(number|string|null)>} + * @default [null, null] + * @product highcharts + * + * @private + */ + center: [null, null], + /** + * The color of the pie series. A pie series is represented as an empty + * circle if the total sum of its values is 0. Use this property to + * define the color of its border. + * + * In styled mode, the color can be defined by the + * [colorIndex](#plotOptions.series.colorIndex) option. Also, the series + * color can be set with the `.highcharts-series`, + * `.highcharts-color-{n}`, `.highcharts-{type}-series` or + * `.highcharts-series-{n}` class, or individual classes given by the + * `className` option. + * + * @sample {highcharts} highcharts/plotoptions/pie-emptyseries/ + * Empty pie series + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @default #cccccc + * @apioption plotOptions.pie.color + */ + /** + * @product highcharts + * + * @private + */ + clip: false, + /** + * @ignore-option + * + * @private + */ + colorByPoint: true, + /** + * A series specific or series type specific color set to use instead + * of the global [colors](#colors). + * + * @sample {highcharts} highcharts/demo/pie-monochrome/ + * Set default colors for all pies + * + * @type {Array<Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject>} + * @since 3.0 + * @product highcharts + * @apioption plotOptions.pie.colors + */ + /** + * @declare Highcharts.SeriesPieDataLabelsOptionsObject + * @extends plotOptions.series.dataLabels + * @excluding align, allowOverlap, inside, staggerLines, step + * @private + */ + dataLabels: { + /** + * Alignment method for data labels. Possible values are: + * + * - `toPlotEdges`: Each label touches the nearest vertical edge of + * the plot area. + * + * - `connectors`: Connectors have the same x position and the + * widest label of each half (left & right) touches the nearest + * vertical edge of the plot area. + * + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-alignto-connectors/ + * alignTo: connectors + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-alignto-plotedges/ + * alignTo: plotEdges + * + * @type {string} + * @since 7.0.0 + * @product highcharts + * @apioption plotOptions.pie.dataLabels.alignTo + */ + allowOverlap: true, /** - * Get the path definition for the halo, which is usually a shadow-like - * circle around the currently hovered point. + * The color of the line connecting the data label to the pie slice. + * The default color is the same as the point's color. * - * @function Highcharts.Point#haloPath + * In styled mode, the connector stroke is given in the + * `.highcharts-data-label-connector` class. * - * @param {number} size - * The radius of the circular halo. + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorcolor/ + * Blue connectors + * @sample {highcharts} highcharts/css/pie-point/ + * Styled connectors * - * @return {Highcharts.SVGPathArray} - * The path definition. + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @since 2.1 + * @product highcharts + * @apioption plotOptions.pie.dataLabels.connectorColor */ - haloPath: function (size) { - var series = this.series, - chart = series.chart; - return chart.renderer.symbols.circle(Math.floor(this.plotX) - size, this.plotY - size, size * 2, size * 2); - } - }); - // Extend the Series object with interaction - extend(LineSeries.prototype, /** @lends Highcharts.Series.prototype */ { /** - * Runs on mouse over the series graphical items. + * The distance from the data label to the connector. Note that + * data labels also have a default `padding`, so in order for the + * connector to touch the text, the `padding` must also be 0. * - * @function Highcharts.Series#onMouseOver - * @fires Highcharts.Series#event:mouseOver + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorpadding/ + * No padding + * + * @since 2.1 + * @product highcharts */ - onMouseOver: function () { - var series = this, - chart = series.chart, - hoverSeries = chart.hoverSeries, - pointer = chart.pointer; - pointer.setHoverChartIndex(); - // set normal state to previous series - if (hoverSeries && hoverSeries !== series) { - hoverSeries.onMouseOut(); - } - // trigger the event, but to save processing time, - // only if defined - if (series.options.events.mouseOver) { - fireEvent(series, 'mouseOver'); - } - // hover this - series.setState('hover'); - /** - * Contains the original hovered series. - * - * @name Highcharts.Chart#hoverSeries - * @type {Highcharts.Series|null} - */ - chart.hoverSeries = series; - }, + connectorPadding: 5, /** - * Runs on mouse out of the series graphical items. + * Specifies the method that is used to generate the connector path. + * Highcharts provides 3 built-in connector shapes: `'fixedOffset'` + * (default), `'straight'` and `'crookedLine'`. Using + * `'crookedLine'` has the most sense (in most of the cases) when + * `'alignTo'` is set. * - * @function Highcharts.Series#onMouseOut + * Users can provide their own method by passing a function instead + * of a String. 3 arguments are passed to the callback: * - * @fires Highcharts.Series#event:mouseOut - */ - onMouseOut: function () { - // trigger the event only if listeners exist - var series = this, - options = series.options, - chart = series.chart, - tooltip = chart.tooltip, - hoverPoint = chart.hoverPoint; - // #182, set to null before the mouseOut event fires - chart.hoverSeries = null; - // trigger mouse out on the point, which must be in this series - if (hoverPoint) { - hoverPoint.onMouseOut(); - } - // fire the mouse out event - if (series && options.events.mouseOut) { - fireEvent(series, 'mouseOut'); - } - // hide the tooltip - if (tooltip && - !series.stickyTracking && - (!tooltip.shared || series.noSharedTooltip)) { - tooltip.hide(); - } - // Reset all inactive states - chart.series.forEach(function (s) { - s.setState('', true); - }); - }, - /** - * Set the state of the series. Called internally on mouse interaction - * operations, but it can also be called directly to visually - * highlight a series. - * - * @function Highcharts.Series#setState - * - * @param {Highcharts.SeriesStateValue|""} [state] - * The new state, can be either `'hover'`, `'inactive'`, `'select'`, - * or `''` (an empty string), `'normal'` or `undefined` to set to - * normal state. - * @param {boolean} [inherit] - * Determines if state should be inherited by points too. + * - Object that holds the information about the coordinates of the + * label (`x` & `y` properties) and how the label is located in + * relation to the pie (`alignment` property). `alignment` can by + * one of the following: + * `'left'` (pie on the left side of the data label), + * `'right'` (pie on the right side of the data label) or + * `'center'` (data label overlaps the pie). + * + * - Object that holds the information about the position of the + * connector. Its `touchingSliceAt` porperty tells the position + * of the place where the connector touches the slice. + * + * - Data label options + * + * The function has to return an SVG path definition in array form + * (see the example). + * + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorshape-string/ + * connectorShape is a String + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorshape-function/ + * connectorShape is a function + * + * @type {string|Function} + * @since 7.0.0 + * @product highcharts */ - setState: function (state, inherit) { - var series = this, - options = series.options, - graph = series.graph, - inactiveOtherPoints = options.inactiveOtherPoints, - stateOptions = options.states, - lineWidth = options.lineWidth, - opacity = options.opacity, - // By default a quick animation to hover/inactive, - // slower to un-hover - stateAnimation = pick((stateOptions[state || 'normal'] && - stateOptions[state || 'normal'].animation), - series.chart.options.chart.animation), - attribs, - i = 0; - state = state || ''; - if (series.state !== state) { - // Toggle class names - [ - series.group, - series.markerGroup, - series.dataLabelsGroup - ].forEach(function (group) { - if (group) { - // Old state - if (series.state) { - group.removeClass('highcharts-series-' + series.state); - } - // New state - if (state) { - group.addClass('highcharts-series-' + state); - } - } - }); - series.state = state; - if (!series.chart.styledMode) { - if (stateOptions[state] && - stateOptions[state].enabled === false) { - return; - } - if (state) { - lineWidth = (stateOptions[state].lineWidth || - lineWidth + (stateOptions[state].lineWidthPlus || 0)); // #4035 - opacity = pick(stateOptions[state].opacity, opacity); - } - if (graph && !graph.dashstyle) { - attribs = { - 'stroke-width': lineWidth - }; - // Animate the graph stroke-width. - graph.animate(attribs, stateAnimation); - while (series['zone-graph-' + i]) { - series['zone-graph-' + i].attr(attribs); - i = i + 1; - } - } - // For some types (pie, networkgraph, sankey) opacity is - // resolved on a point level - if (!inactiveOtherPoints) { - [ - series.group, - series.markerGroup, - series.dataLabelsGroup, - series.labelBySeries - ].forEach(function (group) { - if (group) { - group.animate({ - opacity: opacity - }, stateAnimation); - } - }); - } - } - } - // Don't loop over points on a series that doesn't apply inactive state - // to siblings markers (e.g. line, column) - if (inherit && inactiveOtherPoints && series.points) { - series.setAllPointsToState(state); - } - }, + connectorShape: "fixedOffset", /** - * Set the state for all points in the series. + * The width of the line connecting the data label to the pie slice. * - * @function Highcharts.Series#setAllPointsToState + * In styled mode, the connector stroke width is given in the + * `.highcharts-data-label-connector` class. * - * @private + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorwidth-disabled/ + * Disable the connector + * @sample {highcharts} highcharts/css/pie-point/ + * Styled connectors * - * @param {string} [state] - * Can be either `hover` or undefined to set to normal state. + * @type {number} + * @default 1 + * @since 2.1 + * @product highcharts + * @apioption plotOptions.pie.dataLabels.connectorWidth */ - setAllPointsToState: function (state) { - this.points.forEach(function (point) { - if (point.setState) { - point.setState(state); - } - }); - }, /** - * Show or hide the series. + * Works only if `connectorShape` is `'crookedLine'`. It defines how + * far from the vertical plot edge the coonnector path should be + * crooked. * - * @function Highcharts.Series#setVisible + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-crookdistance/ + * crookDistance set to 90% * - * @param {boolean} [visible] - * True to show the series, false to hide. If undefined, the visibility is - * toggled. + * @since 7.0.0 + * @product highcharts + */ + crookDistance: "70%", + /** + * The distance of the data label from the pie's edge. Negative + * numbers put the data label on top of the pie slices. Can also be + * defined as a percentage of pie's radius. Connectors are only + * shown for data labels outside the pie. * - * @param {boolean} [redraw=true] - * Whether to redraw the chart after the series is altered. If doing more - * operations on the chart, it is a good idea to set redraw to false and - * call {@link Chart#redraw|chart.redraw()} after. + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-distance/ + * Data labels on top of the pie * - * @fires Highcharts.Series#event:hide - * @fires Highcharts.Series#event:show + * @type {number|string} + * @since 2.1 + * @product highcharts */ - setVisible: function (vis, redraw) { - var series = this, - chart = series.chart, - legendItem = series.legendItem, - showOrHide, - ignoreHiddenSeries = chart.options.chart.ignoreHiddenSeries, - oldVisibility = series.visible; - // if called without an argument, toggle visibility - series.visible = - vis = - series.options.visible = - series.userOptions.visible = - typeof vis === 'undefined' ? !oldVisibility : vis; // #5618 - showOrHide = vis ? 'show' : 'hide'; - // show or hide elements - [ - 'group', - 'dataLabelsGroup', - 'markerGroup', - 'tracker', - 'tt' - ].forEach(function (key) { - if (series[key]) { - series[key][showOrHide](); - } - }); - // hide tooltip (#1361) - if (chart.hoverSeries === series || - (chart.hoverPoint && chart.hoverPoint.series) === series) { - series.onMouseOut(); - } - if (legendItem) { - chart.legend.colorizeItem(series, vis); - } - // rescale or adapt to resized chart - series.isDirty = true; - // in a stack, all other series are affected - if (series.options.stacking) { - chart.series.forEach(function (otherSeries) { - if (otherSeries.options.stacking && otherSeries.visible) { - otherSeries.isDirty = true; - } - }); - } - // show or hide linked series - series.linkedSeries.forEach(function (otherSeries) { - otherSeries.setVisible(vis, false); - }); - if (ignoreHiddenSeries) { - chart.isDirtyBox = true; - } - fireEvent(series, showOrHide); - if (redraw !== false) { - chart.redraw(); - } - }, + distance: 30, + enabled: true, /** - * Show the series if hidden. + * A + * [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting) + * for the data label. Available variables are the same as for + * `formatter`. * - * @sample highcharts/members/series-hide/ - * Toggle visibility from a button + * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/ + * Add a unit * - * @function Highcharts.Series#show - * @fires Highcharts.Series#event:show + * @type {string} + * @default undefined + * @since 3.0 + * @apioption plotOptions.pie.dataLabels.format */ - show: function () { - this.setVisible(true); - }, + // eslint-disable-next-line valid-jsdoc /** - * Hide the series if visible. If the - * [chart.ignoreHiddenSeries](https://api.highcharts.com/highcharts/chart.ignoreHiddenSeries) - * option is true, the chart is redrawn without this series. - * - * @sample highcharts/members/series-hide/ - * Toggle visibility from a button + * Callback JavaScript function to format the data label. Note that + * if a `format` is defined, the format takes precedence and the + * formatter is ignored. * - * @function Highcharts.Series#hide - * @fires Highcharts.Series#event:hide + * @type {Highcharts.DataLabelsFormatterCallbackFunction} + * @default function () { return this.point.isNull ? void 0 : this.point.name; } */ - hide: function () { - this.setVisible(false); + formatter: function () { + return this.point.isNull ? void 0 : this.point.name; }, /** - * Select or unselect the series. This means its - * {@link Highcharts.Series.selected|selected} - * property is set, the checkbox in the legend is toggled and when selected, - * the series is returned by the {@link Highcharts.Chart#getSelectedSeries} - * function. + * Whether to render the connector as a soft arc or a line with + * sharp break. Works only if `connectorShape` equals to + * `fixedOffset`. * - * @sample highcharts/members/series-select/ - * Select a series from a button + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-softconnector-true/ + * Soft + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-softconnector-false/ + * Non soft * - * @function Highcharts.Series#select - * - * @param {boolean} [selected] - * True to select the series, false to unselect. If undefined, the selection - * state is toggled. + * @since 2.1.7 + * @product highcharts + */ + softConnector: true, + /** + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow + * Long labels truncated with an ellipsis + * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow-wrap + * Long labels are wrapped * - * @fires Highcharts.Series#event:select - * @fires Highcharts.Series#event:unselect + * @type {Highcharts.CSSObject} + * @apioption plotOptions.pie.dataLabels.style + */ + x: 0, + }, + /** + * If the total sum of the pie's values is 0, the series is represented + * as an empty circle . The `fillColor` option defines the color of that + * circle. Use [pie.borderWidth](#plotOptions.pie.borderWidth) to set + * the border thickness. + * + * @sample {highcharts} highcharts/plotoptions/pie-emptyseries/ + * Empty pie series + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @private + */ + fillColor: void 0, + /** + * The end angle of the pie in degrees where 0 is top and 90 is right. + * Defaults to `startAngle` plus 360. + * + * @sample {highcharts} highcharts/demo/pie-semi-circle/ + * Semi-circle donut + * + * @type {number} + * @since 1.3.6 + * @product highcharts + * @apioption plotOptions.pie.endAngle + */ + /** + * Equivalent to [chart.ignoreHiddenSeries](#chart.ignoreHiddenSeries), + * this option tells whether the series shall be redrawn as if the + * hidden point were `null`. + * + * The default value changed from `false` to `true` with Highcharts + * 3.0. + * + * @sample {highcharts} highcharts/plotoptions/pie-ignorehiddenpoint/ + * True, the hiddden point is ignored + * + * @since 2.3.0 + * @product highcharts + * + * @private + */ + ignoreHiddenPoint: true, + /** + * @ignore-option + * + * @private + */ + inactiveOtherPoints: true, + /** + * The size of the inner diameter for the pie. A size greater than 0 + * renders a donut chart. Can be a percentage or pixel value. + * Percentages are relative to the pie size. Pixel values are given as + * integers. + * + * + * Note: in Highcharts < 4.1.2, the percentage was relative to the plot + * area, not the pie size. + * + * @sample {highcharts} highcharts/plotoptions/pie-innersize-80px/ + * 80px inner size + * @sample {highcharts} highcharts/plotoptions/pie-innersize-50percent/ + * 50% of the plot area + * @sample {highcharts} highcharts/demo/3d-pie-donut/ + * 3D donut + * + * @type {number|string} + * @default 0 + * @since 2.0 + * @product highcharts + * @apioption plotOptions.pie.innerSize + */ + /** + * @ignore-option + * + * @private + */ + legendType: "point", + /** + * @ignore-option + * + * @private + */ + marker: null, + /** + * The minimum size for a pie in response to auto margins. The pie will + * try to shrink to make room for data labels in side the plot area, + * but only to this size. + * + * @type {number|string} + * @default 80 + * @since 3.0 + * @product highcharts + * @apioption plotOptions.pie.minSize + */ + /** + * The diameter of the pie relative to the plot area. Can be a + * percentage or pixel value. Pixel values are given as integers. The + * default behaviour (as of 3.0) is to scale to the plot area and give + * room for data labels within the plot area. + * [slicedOffset](#plotOptions.pie.slicedOffset) is also included in the + * default size calculation. As a consequence, the size of the pie may + * vary when points are updated and data labels more around. In that + * case it is best to set a fixed value, for example `"75%"`. + * + * @sample {highcharts} highcharts/plotoptions/pie-size/ + * Smaller pie + * + * @type {number|string|null} + * @product highcharts + * + * @private + */ + size: null, + /** + * Whether to display this particular series or series type in the + * legend. Since 2.1, pies are not shown in the legend by default. + * + * @sample {highcharts} highcharts/plotoptions/series-showinlegend/ + * One series in the legend, one hidden + * + * @product highcharts + * + * @private + */ + showInLegend: false, + /** + * If a point is sliced, moved out from the center, how many pixels + * should it be moved?. + * + * @sample {highcharts} highcharts/plotoptions/pie-slicedoffset-20/ + * 20px offset + * + * @product highcharts + * + * @private + */ + slicedOffset: 10, + /** + * The start angle of the pie slices in degrees where 0 is top and 90 + * right. + * + * @sample {highcharts} highcharts/plotoptions/pie-startangle-90/ + * Start from right + * + * @type {number} + * @default 0 + * @since 2.3.4 + * @product highcharts + * @apioption plotOptions.pie.startAngle + */ + /** + * Sticky tracking of mouse events. When true, the `mouseOut` event + * on a series isn't triggered until the mouse moves over another + * series, or out of the plot area. When false, the `mouseOut` event on + * a series is triggered when the mouse leaves the area around the + * series' graph or markers. This also implies the tooltip. When + * `stickyTracking` is false and `tooltip.shared` is false, the tooltip + * will be hidden when moving the mouse between series. + * + * @product highcharts + * + * @private + */ + stickyTracking: false, + tooltip: { + followPointer: true, + }, + /** + * The color of the border surrounding each slice. When `null`, the + * border takes the same color as the slice fill. This can be used + * together with a `borderWidth` to fill drawing gaps created by + * antialiazing artefacts in borderless pies. + * + * In styled mode, the border stroke is given in the `.highcharts-point` + * class. + * + * @sample {highcharts} highcharts/plotoptions/pie-bordercolor-black/ + * Black border + * + * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} + * @default #ffffff + * @product highcharts + * + * @private + */ + borderColor: "#ffffff", + /** + * The width of the border surrounding each slice. + * + * When setting the border width to 0, there may be small gaps between + * the slices due to SVG antialiasing artefacts. To work around this, + * keep the border width at 0.5 or 1, but set the `borderColor` to + * `null` instead. + * + * In styled mode, the border stroke width is given in the + * `.highcharts-point` class. + * + * @sample {highcharts} highcharts/plotoptions/pie-borderwidth/ + * 3px border + * + * @product highcharts + * + * @private + */ + borderWidth: 1, + /** + * @ignore-options + * @private + */ + lineWidth: void 0, + states: { + /** + * @extends plotOptions.series.states.hover + * @excluding marker, lineWidth, lineWidthPlus + * @product highcharts */ - select: function (selected) { - var series = this; - series.selected = - selected = - this.options.selected = (typeof selected === 'undefined' ? - !series.selected : - selected); - if (series.checkbox) { - series.checkbox.checked = selected; + hover: { + /** + * How much to brighten the point on interaction. Requires the + * main color to be defined in hex or rgb(a) format. + * + * In styled mode, the hover brightness is by default replaced + * by a fill-opacity given in the `.highcharts-point-hover` + * class. + * + * @sample {highcharts} highcharts/plotoptions/pie-states-hover-brightness/ + * Brightened by 0.5 + * + * @product highcharts + */ + brightness: 0.1, + }, + }, + }, + /* eslint-disable valid-jsdoc */ + /** + * @lends seriesTypes.pie.prototype + */ + { + isCartesian: false, + requireSorting: false, + directTouch: true, + noSharedTooltip: true, + trackerGroups: ["group", "dataLabelsGroup"], + axisTypes: [], + pointAttribs: BaseSeries.seriesTypes.column.prototype.pointAttribs, + /** + * Animate the pies in + * + * @private + * @function Highcharts.seriesTypes.pie#animate + * + * @param {boolean} [init=false] + */ + animate: function (init) { + var series = this, + points = series.points, + startAngleRad = series.startAngleRad; + if (!init) { + points.forEach(function (point) { + var graphic = point.graphic, + args = point.shapeArgs; + if (graphic && args) { + // start values + graphic.attr({ + // animate from inner radius (#779) + r: pick( + point.startR, + series.center && series.center[3] / 2 + ), + start: startAngleRad, + end: startAngleRad, + }); + // animate + graphic.animate( + { + r: args.r, + start: args.start, + end: args.end, + }, + series.options.animation + ); } - fireEvent(series, selected ? 'select' : 'unselect'); + }); + } + }, + // Define hasData function for non-cartesian series. + // Returns true if the series has points at all. + hasData: function () { + return !!this.processedXData.length; // != 0 + }, + /** + * Recompute total chart sum and update percentages of points. + * + * @private + * @function Highcharts.seriesTypes.pie#updateTotals + * @return {void} + */ + updateTotals: function () { + var i, + total = 0, + points = this.points, + len = points.length, + point, + ignoreHiddenPoint = this.options.ignoreHiddenPoint; + // Get the total sum + for (i = 0; i < len; i++) { + point = points[i]; + total += + ignoreHiddenPoint && !point.visible + ? 0 + : point.isNull + ? 0 + : point.y; + } + this.total = total; + // Set each point's properties + for (i = 0; i < len; i++) { + point = points[i]; + point.percentage = + total > 0 && (point.visible || !ignoreHiddenPoint) + ? (point.y / total) * 100 + : 0; + point.total = total; + } + }, + /** + * Extend the generatePoints method by adding total and percentage + * properties to each point + * + * @private + * @function Highcharts.seriesTypes.pie#generatePoints + * @return {void} + */ + generatePoints: function () { + LineSeries.prototype.generatePoints.call(this); + this.updateTotals(); + }, + /** + * Utility for getting the x value from a given y, used for + * anticollision logic in data labels. Added point for using specific + * points' label distance. + * @private + */ + getX: function (y, left, point) { + var center = this.center, + // Variable pie has individual radius + radius = this.radii ? this.radii[point.index] : center[2] / 2, + angle, + x; + angle = Math.asin( + clamp((y - center[1]) / (radius + point.labelDistance), -1, 1) + ); + x = + center[0] + + (left ? -1 : 1) * + (Math.cos(angle) * (radius + point.labelDistance)) + + (point.labelDistance > 0 + ? (left ? -1 : 1) * this.options.dataLabels.padding + : 0); + return x; + }, + /** + * Do translation for pie slices + * + * @private + * @function Highcharts.seriesTypes.pie#translate + * @param {Array<number>} [positions] + * @return {void} + */ + translate: function (positions) { + this.generatePoints(); + var series = this, + cumulative = 0, + precision = 1000, // issue #172 + options = series.options, + slicedOffset = options.slicedOffset, + connectorOffset = slicedOffset + (options.borderWidth || 0), + finalConnectorOffset, + start, + end, + angle, + radians = getStartAndEndRadians( + options.startAngle, + options.endAngle + ), + startAngleRad = (series.startAngleRad = radians.start), + endAngleRad = (series.endAngleRad = radians.end), + circ = endAngleRad - startAngleRad, // 2 * Math.PI, + points = series.points, + // the x component of the radius vector for a given point + radiusX, + radiusY, + labelDistance = options.dataLabels.distance, + ignoreHiddenPoint = options.ignoreHiddenPoint, + i, + len = points.length, + point; + // Get positions - either an integer or a percentage string must be + // given. If positions are passed as a parameter, we're in a + // recursive loop for adjusting space for data labels. + if (!positions) { + series.center = positions = series.getCenter(); + } + // Calculate the geometry for each point + for (i = 0; i < len; i++) { + point = points[i]; + // set start and end angle + start = startAngleRad + cumulative * circ; + if (!ignoreHiddenPoint || point.visible) { + cumulative += point.percentage / 100; + } + end = startAngleRad + cumulative * circ; + // set the shape + point.shapeType = "arc"; + point.shapeArgs = { + x: positions[0], + y: positions[1], + r: positions[2] / 2, + innerR: positions[3] / 2, + start: Math.round(start * precision) / precision, + end: Math.round(end * precision) / precision, + }; + // Used for distance calculation for specific point. + point.labelDistance = pick( + point.options.dataLabels && point.options.dataLabels.distance, + labelDistance + ); + // Compute point.labelDistance if it's defined as percentage + // of slice radius (#8854) + point.labelDistance = relativeLength( + point.labelDistance, + point.shapeArgs.r + ); + // Saved for later dataLabels distance calculation. + series.maxLabelDistance = Math.max( + series.maxLabelDistance || 0, + point.labelDistance + ); + // The angle must stay within -90 and 270 (#2645) + angle = (end + start) / 2; + if (angle > 1.5 * Math.PI) { + angle -= 2 * Math.PI; + } else if (angle < -Math.PI / 2) { + angle += 2 * Math.PI; + } + // Center for the sliced out slice + point.slicedTranslation = { + translateX: Math.round(Math.cos(angle) * slicedOffset), + translateY: Math.round(Math.sin(angle) * slicedOffset), + }; + // set the anchor point for tooltips + radiusX = (Math.cos(angle) * positions[2]) / 2; + radiusY = (Math.sin(angle) * positions[2]) / 2; + point.tooltipPos = [ + positions[0] + radiusX * 0.7, + positions[1] + radiusY * 0.7, + ]; + point.half = angle < -Math.PI / 2 || angle > Math.PI / 2 ? 1 : 0; + point.angle = angle; + // Set the anchor point for data labels. Use point.labelDistance + // instead of labelDistance // #1174 + // finalConnectorOffset - not override connectorOffset value. + finalConnectorOffset = Math.min( + connectorOffset, + point.labelDistance / 5 + ); // #1678 + point.labelPosition = { + natural: { + // initial position of the data label - it's utilized for + // finding the final position for the label + x: + positions[0] + + radiusX + + Math.cos(angle) * point.labelDistance, + y: + positions[1] + + radiusY + + Math.sin(angle) * point.labelDistance, + }, + final: { + // used for generating connector path - + // initialized later in drawDataLabels function + // x: undefined, + // y: undefined + }, + // left - pie on the left side of the data label + // right - pie on the right side of the data label + // center - data label overlaps the pie + alignment: + point.labelDistance < 0 + ? "center" + : point.half + ? "right" + : "left", + connectorPosition: { + breakAt: { + x: + positions[0] + + radiusX + + Math.cos(angle) * finalConnectorOffset, + y: + positions[1] + + radiusY + + Math.sin(angle) * finalConnectorOffset, + }, + touchingSliceAt: { + x: positions[0] + radiusX, + y: positions[1] + radiusY, + }, + }, + }; + } + fireEvent(series, "afterTranslate"); + }, + /** + * Called internally to draw auxiliary graph in pie-like series in + * situtation when the default graph is not sufficient enough to present + * the data well. Auxiliary graph is saved in the same object as + * regular graph. + * + * @private + * @function Highcharts.seriesTypes.pie#drawEmpty + */ + drawEmpty: function () { + var centerX, + centerY, + start = this.startAngleRad, + end = this.endAngleRad, + options = this.options; + // Draw auxiliary graph if there're no visible points. + if (this.total === 0 && this.center) { + centerX = this.center[0]; + centerY = this.center[1]; + if (!this.graph) { + this.graph = this.chart.renderer + .arc(centerX, centerY, this.center[1] / 2, 0, start, end) + .addClass("highcharts-empty-series") + .add(this.group); + } + this.graph.attr({ + d: SVGRenderer.prototype.symbols.arc( + centerX, + centerY, + this.center[2] / 2, + 0, + { + start: start, + end: end, + innerR: this.center[3] / 2, + } + ), + }); + if (!this.chart.styledMode) { + this.graph.attr({ + "stroke-width": options.borderWidth, + fill: options.fillColor || "none", + stroke: options.color || "#cccccc", + }); + } + } else if (this.graph) { + // Destroy the graph object. + this.graph = this.graph.destroy(); + } + }, + /** + * Draw the data points + * + * @private + * @function Highcharts.seriesTypes.pie#drawPoints + * @return {void} + */ + redrawPoints: function () { + var series = this, + chart = series.chart, + renderer = chart.renderer, + groupTranslation, + graphic, + pointAttr, + shapeArgs, + shadow = series.options.shadow; + this.drawEmpty(); + if (shadow && !series.shadowGroup && !chart.styledMode) { + series.shadowGroup = renderer + .g("shadow") + .attr({ zIndex: -1 }) + .add(series.group); + } + // draw the slices + series.points.forEach(function (point) { + var animateTo = {}; + graphic = point.graphic; + if (!point.isNull && graphic) { + shapeArgs = point.shapeArgs; + // If the point is sliced, use special translation, else use + // plot area translation + groupTranslation = point.getTranslate(); + if (!chart.styledMode) { + // Put the shadow behind all points + var shadowGroup = point.shadowGroup; + if (shadow && !shadowGroup) { + shadowGroup = point.shadowGroup = renderer + .g("shadow") + .add(series.shadowGroup); + } + if (shadowGroup) { + shadowGroup.attr(groupTranslation); + } + pointAttr = series.pointAttribs( + point, + point.selected && "select" + ); + } + // Draw the slice + if (!point.delayedRendering) { + graphic.setRadialReference(series.center); + if (!chart.styledMode) { + merge(true, animateTo, pointAttr); + } + merge(true, animateTo, shapeArgs, groupTranslation); + graphic.animate(animateTo); + } else { + graphic + .setRadialReference(series.center) + .attr(shapeArgs) + .attr(groupTranslation); + if (!chart.styledMode) { + graphic + .attr(pointAttr) + .attr({ "stroke-linejoin": "round" }) + .shadow(shadow, shadowGroup); + } + point.delayedRendering = false; + } + graphic.attr({ + visibility: point.visible ? "inherit" : "hidden", + }); + graphic.addClass(point.getClassName()); + } else if (graphic) { + point.graphic = graphic.destroy(); + } + }); + }, + /** + * Slices in pie chart are initialized in DOM, but it's shapes and + * animations are normally run in `drawPoints()`. + * @private + */ + drawPoints: function () { + var renderer = this.chart.renderer; + this.points.forEach(function (point) { + // When updating a series between 2d and 3d or cartesian and + // polar, the shape type changes. + if (point.graphic && point.hasNewShapeType()) { + point.graphic = point.graphic.destroy(); + } + if (!point.graphic) { + point.graphic = renderer[point.shapeType](point.shapeArgs).add( + point.series.group + ); + point.delayedRendering = true; + } + }); + }, + /** + * @private + * @deprecated + * @function Highcharts.seriesTypes.pie#searchPoint + */ + searchPoint: noop, + /** + * Utility for sorting data labels + * + * @private + * @function Highcharts.seriesTypes.pie#sortByAngle + * @param {Array<Highcharts.Point>} points + * @param {number} sign + * @return {void} + */ + sortByAngle: function (points, sign) { + points.sort(function (a, b) { + return ( + typeof a.angle !== "undefined" && (b.angle - a.angle) * sign + ); + }); + }, + /** + * Use a simple symbol from LegendSymbolMixin. + * + * @private + * @borrows Highcharts.LegendSymbolMixin.drawRectangle as Highcharts.seriesTypes.pie#drawLegendSymbol + */ + drawLegendSymbol: LegendSymbolMixin.drawRectangle, + /** + * Use the getCenter method from drawLegendSymbol. + * + * @private + * @borrows Highcharts.CenteredSeriesMixin.getCenter as Highcharts.seriesTypes.pie#getCenter + */ + getCenter: CenteredSeriesMixin.getCenter, + /** + * Pies don't have point marker symbols. + * + * @deprecated + * @private + * @function Highcharts.seriesTypes.pie#getSymbol + */ + getSymbol: noop, + /** + * @private + * @type {null} + */ + drawGraph: null, + }, + /** + * @lends seriesTypes.pie.prototype.pointClass.prototype + */ + { + /** + * Initialize the pie slice + * + * @private + * @function Highcharts.seriesTypes.pie#pointClass#init + * @return {Highcharts.Point} + */ + init: function () { + Point.prototype.init.apply(this, arguments); + var point = this, + toggleSlice; + point.name = pick(point.name, "Slice"); + // add event listener for select + toggleSlice = function (e) { + point.slice(e.type === "select"); + }; + addEvent(point, "select", toggleSlice); + addEvent(point, "unselect", toggleSlice); + return point; + }, + /** + * Negative points are not valid (#1530, #3623, #5322) + * + * @private + * @function Highcharts.seriesTypes.pie#pointClass#isValid + * @return {boolean} + */ + isValid: function () { + return isNumber(this.y) && this.y >= 0; + }, + /** + * Toggle the visibility of the pie slice + * + * @private + * @function Highcharts.seriesTypes.pie#pointClass#setVisible + * @param {boolean} vis + * Whether to show the slice or not. If undefined, the visibility + * is toggled. + * @param {boolean} [redraw=false] + * @return {void} + */ + setVisible: function (vis, redraw) { + var point = this, + series = point.series, + chart = series.chart, + ignoreHiddenPoint = series.options.ignoreHiddenPoint; + redraw = pick(redraw, ignoreHiddenPoint); + if (vis !== point.visible) { + // If called without an argument, toggle visibility + point.visible = + point.options.visible = + vis = + typeof vis === "undefined" ? !point.visible : vis; + // update userOptions.data + series.options.data[series.data.indexOf(point)] = point.options; + // Show and hide associated elements. This is performed + // regardless of redraw or not, because chart.redraw only + // handles full series. + ["graphic", "dataLabel", "connector", "shadowGroup"].forEach( + function (key) { + if (point[key]) { + point[key][vis ? "show" : "hide"](true); + } + } + ); + if (point.legendItem) { + chart.legend.colorizeItem(point, vis); + } + // #4170, hide halo after hiding point + if (!vis && point.state === "hover") { + point.setState(""); + } + // Handle ignore hidden slices + if (ignoreHiddenPoint) { + series.isDirty = true; + } + if (redraw) { + chart.redraw(); + } + } + }, + /** + * Set or toggle whether the slice is cut out from the pie + * + * @private + * @function Highcharts.seriesTypes.pie#pointClass#slice + * @param {boolean} sliced + * When undefined, the slice state is toggled. + * @param {boolean} redraw + * Whether to redraw the chart. True by default. + * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} + * Animation options. + * @return {void} + */ + slice: function (sliced, redraw, animation) { + var point = this, + series = point.series, + chart = series.chart; + setAnimation(animation, chart); + // redraw is true by default + redraw = pick(redraw, true); + /** + * Pie series only. Whether to display a slice offset from the + * center. + * @name Highcharts.Point#sliced + * @type {boolean|undefined} + */ + // if called without an argument, toggle + point.sliced = + point.options.sliced = + sliced = + defined(sliced) ? sliced : !point.sliced; + // update userOptions.data + series.options.data[series.data.indexOf(point)] = point.options; + if (point.graphic) { + point.graphic.animate(this.getTranslate()); + } + if (point.shadowGroup) { + point.shadowGroup.animate(this.getTranslate()); + } + }, + /** + * @private + * @function Highcharts.seriesTypes.pie#pointClass#getTranslate + * @return {Highcharts.TranslationAttributes} + */ + getTranslate: function () { + return this.sliced + ? this.slicedTranslation + : { + translateX: 0, + translateY: 0, + }; + }, + /** + * @private + * @function Highcharts.seriesTypes.pie#pointClass#haloPath + * @param {number} size + * @return {Highcharts.SVGPathArray} + */ + haloPath: function (size) { + var shapeArgs = this.shapeArgs; + return this.sliced || !this.visible + ? [] + : this.series.chart.renderer.symbols.arc( + shapeArgs.x, + shapeArgs.y, + shapeArgs.r + size, + shapeArgs.r + size, + { + // Substract 1px to ensure the background is not bleeding + // through between the halo and the slice (#7495). + innerR: shapeArgs.r - 1, + start: shapeArgs.start, + end: shapeArgs.end, + } + ); + }, + connectorShapes: { + // only one available before v7.0.0 + fixedOffset: function (labelPosition, connectorPosition, options) { + var breakAt = connectorPosition.breakAt, + touchingSliceAt = connectorPosition.touchingSliceAt, + lineSegment = options.softConnector + ? [ + "C", + // 1st control point (of the curve) + labelPosition.x + + // 5 gives the connector a little horizontal bend + (labelPosition.alignment === "left" ? -5 : 5), + labelPosition.y, + 2 * breakAt.x - touchingSliceAt.x, + 2 * breakAt.y - touchingSliceAt.y, + breakAt.x, + breakAt.y, // + ] + : ["L", breakAt.x, breakAt.y]; + // assemble the path + return [ + ["M", labelPosition.x, labelPosition.y], + lineSegment, + ["L", touchingSliceAt.x, touchingSliceAt.y], + ]; }, - /** - * @private - * @borrows Highcharts.TrackerMixin.drawTrackerGraph as Highcharts.Series#drawTracker - */ - drawTracker: TrackerMixin.drawTrackerGraph + straight: function (labelPosition, connectorPosition) { + var touchingSliceAt = connectorPosition.touchingSliceAt; + // direct line to the slice + return [ + ["M", labelPosition.x, labelPosition.y], + ["L", touchingSliceAt.x, touchingSliceAt.y], + ]; + }, + crookedLine: function (labelPosition, connectorPosition, options) { + var touchingSliceAt = connectorPosition.touchingSliceAt, + series = this.series, + pieCenterX = series.center[0], + plotWidth = series.chart.plotWidth, + plotLeft = series.chart.plotLeft, + alignment = labelPosition.alignment, + radius = this.shapeArgs.r, + crookDistance = relativeLength( + // % to fraction + options.crookDistance, + 1 + ), + crookX = + alignment === "left" + ? pieCenterX + + radius + + (plotWidth + plotLeft - pieCenterX - radius) * + (1 - crookDistance) + : plotLeft + (pieCenterX - radius) * crookDistance, + segmentWithCrook = ["L", crookX, labelPosition.y], + useCrook = true; + // crookedLine formula doesn't make sense if the path overlaps + // the label - use straight line instead in that case + if ( + alignment === "left" + ? crookX > labelPosition.x || crookX < touchingSliceAt.x + : crookX < labelPosition.x || crookX > touchingSliceAt.x + ) { + useCrook = false; + } + // assemble the path + var path = [["M", labelPosition.x, labelPosition.y]]; + if (useCrook) { + path.push(segmentWithCrook); + } + path.push(["L", touchingSliceAt.x, touchingSliceAt.y]); + return path; + }, + }, + /** + * Extendable method for getting the path of the connector between the + * data label and the pie slice. + */ + getConnectorPath: function () { + var labelPosition = this.labelPosition, + options = this.series.options.dataLabels, + connectorShape = options.connectorShape, + predefinedShapes = this.connectorShapes; + // find out whether to use the predefined shape + if (predefinedShapes[connectorShape]) { + connectorShape = predefinedShapes[connectorShape]; + } + return connectorShape.call( + this, + { + // pass simplified label position object for user's convenience + x: labelPosition.final.x, + y: labelPosition.final.y, + alignment: labelPosition.alignment, + }, + labelPosition.connectorPosition, + options + ); + }, + } + /* eslint-enable valid-jsdoc */ + ); + /** + * A `pie` series. If the [type](#series.pie.type) option is not specified, + * it is inherited from [chart.type](#chart.type). + * + * @extends series,plotOptions.pie + * @excluding cropThreshold, dataParser, dataURL, stack, xAxis, yAxis, + * dataSorting, step, boostThreshold, boostBlending + * @product highcharts + * @apioption series.pie + */ + /** + * An array of data points for the series. For the `pie` series type, + * points can be given in the following ways: + * + * 1. An array of numerical values. In this case, the numerical values will be + * interpreted as `y` options. Example: + * ```js + * data: [0, 5, 3, 5] + * ``` + * + * 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.pie.turboThreshold), + * this option is not available. + * ```js + * data: [{ + * y: 1, + * name: "Point2", + * color: "#00FF00" + * }, { + * y: 7, + * name: "Point1", + * color: "#FF00FF" + * }] + * ``` + * + * @sample {highcharts} highcharts/chart/reflow-true/ + * Numerical values + * @sample {highcharts} highcharts/series/data-array-of-arrays/ + * Arrays of numeric x and y + * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/ + * Arrays of datetime x and y + * @sample {highcharts} highcharts/series/data-array-of-name-value/ + * Arrays of point.name and y + * @sample {highcharts} highcharts/series/data-array-of-objects/ + * Config objects + * + * @type {Array<number|Array<string,(number|null)>|null|*>} + * @extends series.line.data + * @excluding marker, x + * @product highcharts + * @apioption series.pie.data + */ + /** + * @type {Highcharts.SeriesPieDataLabelsOptionsObject} + * @product highcharts + * @apioption series.pie.data.dataLabels + */ + /** + * The sequential index of the data point in the legend. + * + * @type {number} + * @product highcharts + * @apioption series.pie.data.legendIndex + */ + /** + * Whether to display a slice offset from the center. + * + * @sample {highcharts} highcharts/point/sliced/ + * One sliced point + * + * @type {boolean} + * @product highcharts + * @apioption series.pie.data.sliced + */ + /** + * @excluding legendItemClick + * @product highcharts + * @apioption series.pie.events + */ + (""); // placeholder for transpiled doclets above + } + ); + _registerModule( + _modules, + "Core/Series/DataLabels.js", + [ + _modules["Core/Animation/AnimationUtilities.js"], + _modules["Core/Globals.js"], + _modules["Core/Series/CartesianSeries.js"], + _modules["Core/Utilities.js"], + ], + function (A, H, CartesianSeries, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var getDeferredAnimation = A.getDeferredAnimation; + var noop = H.noop, + seriesTypes = H.seriesTypes; + var arrayMax = U.arrayMax, + clamp = U.clamp, + defined = U.defined, + extend = U.extend, + fireEvent = U.fireEvent, + format = U.format, + isArray = U.isArray, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick, + relativeLength = U.relativeLength, + splat = U.splat, + stableSort = U.stableSort; + /** + * Callback JavaScript function to format the data label as a string. Note that + * if a `format` is defined, the format takes precedence and the formatter is + * ignored. + * + * @callback Highcharts.DataLabelsFormatterCallbackFunction + * + * @param {Highcharts.PointLabelObject} this + * Data label context to format + * + * @param {Highcharts.DataLabelsOptions} options + * [API options](/highcharts/plotOptions.series.dataLabels) of the data label + * + * @return {number|string|null|undefined} + * Formatted data label text + */ + /** + * Values for handling data labels that flow outside the plot area. + * + * @typedef {"allow"|"justify"} Highcharts.DataLabelsOverflowValue + */ + (""); // detach doclets above + /* eslint-disable valid-jsdoc */ + /** + * General distribution algorithm for distributing labels of differing size + * along a confined length in two dimensions. The algorithm takes an array of + * objects containing a size, a target and a rank. It will place the labels as + * close as possible to their targets, skipping the lowest ranked labels if + * necessary. + * + * @private + * @function Highcharts.distribute + * @param {Highcharts.DataLabelsBoxArray} boxes + * @param {number} len + * @param {number} [maxDistance] + * @return {void} + */ + H.distribute = function (boxes, len, maxDistance) { + var i, + overlapping = true, + origBoxes = boxes, // Original array will be altered with added .pos + restBoxes = [], // The outranked overshoot + box, + target, + total = 0, + reducedLen = origBoxes.reducedLen || len; + /** + * @private + */ + function sortByTarget(a, b) { + return a.target - b.target; + } + // If the total size exceeds the len, remove those boxes with the lowest + // rank + i = boxes.length; + while (i--) { + total += boxes[i].size; + } + // Sort by rank, then slice away overshoot + if (total > reducedLen) { + stableSort(boxes, function (a, b) { + return (b.rank || 0) - (a.rank || 0); + }); + i = 0; + total = 0; + while (total <= reducedLen) { + total += boxes[i].size; + i++; + } + restBoxes = boxes.splice(i - 1, boxes.length); + } + // Order by target + stableSort(boxes, sortByTarget); + // So far we have been mutating the original array. Now + // create a copy with target arrays + boxes = boxes.map(function (box) { + return { + size: box.size, + targets: [box.target], + align: pick(box.align, 0.5), + }; }); - - }); - _registerModule(_modules, 'Core/Responsive.js', [_modules['Core/Chart/Chart.js'], _modules['Core/Utilities.js']], function (Chart, U) { - /* * - * - * (c) 2010-2020 Torstein Honsi - * - * License: www.highcharts.com/license - * - * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + while (overlapping) { + // Initial positions: target centered in box + i = boxes.length; + while (i--) { + box = boxes[i]; + // Composite box, average of targets + target = + (Math.min.apply(0, box.targets) + + Math.max.apply(0, box.targets)) / + 2; + box.pos = clamp(target - box.size * box.align, 0, len - box.size); + } + // Detect overlap and join boxes + i = boxes.length; + overlapping = false; + while (i--) { + // Overlap + if (i > 0 && boxes[i - 1].pos + boxes[i - 1].size > boxes[i].pos) { + // Add this size to the previous box + boxes[i - 1].size += boxes[i].size; + boxes[i - 1].targets = boxes[i - 1].targets.concat( + boxes[i].targets + ); + boxes[i - 1].align = 0.5; + // Overlapping right, push left + if (boxes[i - 1].pos + boxes[i - 1].size > len) { + boxes[i - 1].pos = len - boxes[i - 1].size; + } + boxes.splice(i, 1); // Remove this item + overlapping = true; + } + } + } + // Add the rest (hidden boxes) + origBoxes.push.apply(origBoxes, restBoxes); + // Now the composite boxes are placed, we need to put the original boxes + // within them + i = 0; + boxes.some(function (box) { + var posInCompositeBox = 0; + if ( + box.targets.some(function () { + origBoxes[i].pos = box.pos + posInCompositeBox; + // If the distance between the position and the target exceeds + // maxDistance, abort the loop and decrease the length in increments + // of 10% to recursively reduce the number of visible boxes by + // rank. Once all boxes are within the maxDistance, we're good. + if ( + typeof maxDistance !== "undefined" && + Math.abs(origBoxes[i].pos - origBoxes[i].target) > maxDistance + ) { + // Reset the positions that are already set + origBoxes.slice(0, i + 1).forEach(function (box) { + delete box.pos; + }); + // Try with a smaller length + origBoxes.reducedLen = + (origBoxes.reducedLen || len) - len * 0.1; + // Recurse + if (origBoxes.reducedLen > len * 0.1) { + H.distribute(origBoxes, len, maxDistance); + } + // Exceeded maxDistance => abort + return true; + } + posInCompositeBox += origBoxes[i].size; + i++; + }) + ) { + // Exceeded maxDistance => abort + return true; + } + }); + // Add the rest (hidden) boxes and sort by target + stableSort(origBoxes, sortByTarget); + }; + /** + * Draw the data labels + * + * @private + * @function Highcharts.Series#drawDataLabels + * @return {void} + * @fires Highcharts.Series#event:afterDrawDataLabels + */ + CartesianSeries.prototype.drawDataLabels = function () { + var series = this, + chart = series.chart, + seriesOptions = series.options, + seriesDlOptions = seriesOptions.dataLabels, + points = series.points, + pointOptions, + hasRendered = series.hasRendered || 0, + dataLabelsGroup, + dataLabelAnim = seriesDlOptions.animation, + animationConfig = seriesDlOptions.defer + ? getDeferredAnimation(chart, dataLabelAnim, series) + : { defer: 0, duration: 0 }, + renderer = chart.renderer; + /** + * Handle the dataLabels.filter option. + * @private + */ + function applyFilter(point, options) { + var filter = options.filter, + op, + prop, + val; + if (filter) { + op = filter.operator; + prop = point[filter.property]; + val = filter.value; + if ( + (op === ">" && prop > val) || + (op === "<" && prop < val) || + (op === ">=" && prop >= val) || + (op === "<=" && prop <= val) || + (op === "==" && prop == val) || // eslint-disable-line eqeqeq + (op === "===" && prop === val) + ) { + return true; + } + return false; + } + return true; + } + /** + * Merge two objects that can be arrays. If one of them is an array, the + * other is merged into each element. If both are arrays, each element is + * merged by index. If neither are arrays, we use normal merge. + * @private + */ + function mergeArrays(one, two) { + var res = [], + i; + if (isArray(one) && !isArray(two)) { + res = one.map(function (el) { + return merge(el, two); + }); + } else if (isArray(two) && !isArray(one)) { + res = two.map(function (el) { + return merge(one, el); + }); + } else if (!isArray(one) && !isArray(two)) { + res = merge(one, two); + } else { + i = Math.max(one.length, two.length); + while (i--) { + res[i] = merge(one[i], two[i]); + } + } + return res; + } + // Merge in plotOptions.dataLabels for series + seriesDlOptions = mergeArrays( + mergeArrays( + chart.options.plotOptions && + chart.options.plotOptions.series && + chart.options.plotOptions.series.dataLabels, + chart.options.plotOptions && + chart.options.plotOptions[series.type] && + chart.options.plotOptions[series.type].dataLabels + ), + seriesDlOptions + ); + fireEvent(this, "drawDataLabels"); + if ( + isArray(seriesDlOptions) || + seriesDlOptions.enabled || + series._hasPointLabels + ) { + // Create a separate group for the data labels to avoid rotation + dataLabelsGroup = series.plotGroup( + "dataLabelsGroup", + "data-labels", + !hasRendered ? "hidden" : "inherit", // #5133, #10220 + seriesDlOptions.zIndex || 6 + ); + dataLabelsGroup.attr({ opacity: +hasRendered }); // #3300 + if (!hasRendered) { + var group = series.dataLabelsGroup; + if (group) { + if (series.visible) { + // #2597, #3023, #3024 + dataLabelsGroup.show(true); + } + group[seriesOptions.animation ? "animate" : "attr"]( + { opacity: 1 }, + animationConfig + ); + } + } + // Make the labels for each point + points.forEach(function (point) { + // Merge in series options for the point. + // @note dataLabelAttribs (like pointAttribs) would eradicate + // the need for dlOptions, and simplify the section below. + pointOptions = splat( + mergeArrays( + seriesDlOptions, + point.dlOptions || // dlOptions is used in treemaps + (point.options && point.options.dataLabels) + ) + ); + // Handle each individual data label for this point + pointOptions.forEach(function (labelOptions, i) { + // Options for one datalabel + var labelEnabled = + labelOptions.enabled && + // #2282, #4641, #7112, #10049 + (!point.isNull || point.dataLabelOnNull) && + applyFilter(point, labelOptions), + labelConfig, + formatString, + labelText, + style, + rotation, + attr, + dataLabel = point.dataLabels + ? point.dataLabels[i] + : point.dataLabel, + connector = point.connectors + ? point.connectors[i] + : point.connector, + labelDistance = pick( + labelOptions.distance, + point.labelDistance + ), + isNew = !dataLabel; + if (labelEnabled) { + // Create individual options structure that can be extended + // without affecting others + labelConfig = point.getLabelConfig(); + formatString = pick( + labelOptions[point.formatPrefix + "Format"], + labelOptions.format + ); + labelText = defined(formatString) + ? format(formatString, labelConfig, chart) + : ( + labelOptions[point.formatPrefix + "Formatter"] || + labelOptions.formatter + ).call(labelConfig, labelOptions); + style = labelOptions.style; + rotation = labelOptions.rotation; + if (!chart.styledMode) { + // Determine the color + style.color = pick( + labelOptions.color, + style.color, + series.color, + "#000000" + ); + // Get automated contrast color + if (style.color === "contrast") { + point.contrastColor = renderer.getContrast( + point.color || series.color + ); + style.color = + (!defined(labelDistance) && labelOptions.inside) || + labelDistance < 0 || + !!seriesOptions.stacking + ? point.contrastColor + : "#000000"; + } else { + delete point.contrastColor; + } + if (seriesOptions.cursor) { + style.cursor = seriesOptions.cursor; + } + } + attr = { + r: labelOptions.borderRadius || 0, + rotation: rotation, + padding: labelOptions.padding, + zIndex: 1, + }; + if (!chart.styledMode) { + attr.fill = labelOptions.backgroundColor; + attr.stroke = labelOptions.borderColor; + attr["stroke-width"] = labelOptions.borderWidth; + } + // Remove unused attributes (#947) + objectEach(attr, function (val, name) { + if (typeof val === "undefined") { + delete attr[name]; + } + }); + } + // If the point is outside the plot area, destroy it. #678, #820 + if (dataLabel && (!labelEnabled || !defined(labelText))) { + point.dataLabel = point.dataLabel && point.dataLabel.destroy(); + if (point.dataLabels) { + // Remove point.dataLabels if this was the last one + if (point.dataLabels.length === 1) { + delete point.dataLabels; + } else { + delete point.dataLabels[i]; + } + } + if (!i) { + delete point.dataLabel; + } + if (connector) { + point.connector = point.connector.destroy(); + if (point.connectors) { + // Remove point.connectors if this was the last one + if (point.connectors.length === 1) { + delete point.connectors; + } else { + delete point.connectors[i]; + } + } + } + // Individual labels are disabled if the are explicitly disabled + // in the point options, or if they fall outside the plot area. + } else if (labelEnabled && defined(labelText)) { + if (!dataLabel) { + // Create new label element + point.dataLabels = point.dataLabels || []; + dataLabel = point.dataLabels[i] = rotation + ? // Labels don't rotate, use text element + renderer + .text(labelText, 0, -9999, labelOptions.useHTML) + .addClass("highcharts-data-label") + : // We can use label + renderer.label( + labelText, + 0, + -9999, + labelOptions.shape, + null, + null, + labelOptions.useHTML, + null, + "data-label" + ); + // Store for backwards compatibility + if (!i) { + point.dataLabel = dataLabel; + } + dataLabel.addClass( + " highcharts-data-label-color-" + + point.colorIndex + + " " + + (labelOptions.className || "") + // #3398 + (labelOptions.useHTML ? " highcharts-tracker" : "") + ); + } else { + // Use old element and just update text + attr.text = labelText; + } + // Store data label options for later access + dataLabel.options = labelOptions; + dataLabel.attr(attr); + if (!chart.styledMode) { + // Styles must be applied before add in order to read + // text bounding box + dataLabel.css(style).shadow(labelOptions.shadow); + } + if (!dataLabel.added) { + dataLabel.add(dataLabelsGroup); + } + if (labelOptions.textPath && !labelOptions.useHTML) { + dataLabel.setTextPath( + (point.getDataLabelPath && + point.getDataLabelPath(dataLabel)) || + point.graphic, + labelOptions.textPath + ); + if (point.dataLabelPath && !labelOptions.textPath.enabled) { + // clean the DOM + point.dataLabelPath = point.dataLabelPath.destroy(); + } + } + // Now the data label is created and placed at 0,0, so we + // need to align it + series.alignDataLabel( + point, + dataLabel, + labelOptions, + null, + isNew + ); + } + }); + }); + } + fireEvent(this, "afterDrawDataLabels"); + }; + /** + * Align each individual data label. + * + * @private + * @function Highcharts.Series#alignDataLabel + * @param {Highcharts.Point} point + * @param {Highcharts.SVGElement} dataLabel + * @param {Highcharts.DataLabelsOptions} options + * @param {Highcharts.BBoxObject} alignTo + * @param {boolean} [isNew] + * @return {void} + */ + CartesianSeries.prototype.alignDataLabel = function ( + point, + dataLabel, + options, + alignTo, + isNew + ) { + var series = this, + chart = this.chart, + inverted = this.isCartesian && chart.inverted, + enabledDataSorting = this.enabledDataSorting, + plotX = pick(point.dlBox && point.dlBox.centerX, point.plotX, -9999), + plotY = pick(point.plotY, -9999), + bBox = dataLabel.getBBox(), + baseline, + rotation = options.rotation, + normRotation, + negRotation, + align = options.align, + rotCorr, // rotation correction + isInsidePlot = chart.isInsidePlot(plotX, Math.round(plotY), inverted), + // Math.round for rounding errors (#2683), alignTo to allow column + // labels (#2700) + alignAttr, // the final position; + justify = + pick(options.overflow, enabledDataSorting ? "none" : "justify") === + "justify", + visible = + this.visible && + point.visible !== false && + (point.series.forceDL || + (enabledDataSorting && !justify) || + isInsidePlot || + // If the data label is inside the align box, it is enough + // that parts of the align box is inside the plot area + // (#12370) + (options.inside && + alignTo && + chart.isInsidePlot( + plotX, + inverted ? alignTo.x + 1 : alignTo.y + alignTo.height - 1, + inverted + ))), + setStartPos = function (alignOptions) { + if (enabledDataSorting && series.xAxis && !justify) { + series.setDataLabelStartPos( + point, + dataLabel, + isNew, + isInsidePlot, + alignOptions + ); + } + }; + if (visible) { + baseline = chart.renderer.fontMetrics( + chart.styledMode ? void 0 : options.style.fontSize, + dataLabel + ).b; + // The alignment box is a singular point + alignTo = extend( + { + x: inverted ? this.yAxis.len - plotY : plotX, + y: Math.round(inverted ? this.xAxis.len - plotX : plotY), + width: 0, + height: 0, + }, + alignTo + ); + // Add the text size for alignment calculation + extend(options, { + width: bBox.width, + height: bBox.height, + }); + // Allow a hook for changing alignment in the last moment, then do the + // alignment + if (rotation) { + justify = false; // Not supported for rotated text + rotCorr = chart.renderer.rotCorr(baseline, rotation); // #3723 + alignAttr = { + x: alignTo.x + (options.x || 0) + alignTo.width / 2 + rotCorr.x, + y: + alignTo.y + + (options.y || 0) + + { top: 0, middle: 0.5, bottom: 1 }[options.verticalAlign] * + alignTo.height, + }; + setStartPos(alignAttr); // data sorting + dataLabel[isNew ? "attr" : "animate"](alignAttr).attr({ + align: align, + }); + // Compensate for the rotated label sticking out on the sides + normRotation = (rotation + 720) % 360; + negRotation = normRotation > 180 && normRotation < 360; + if (align === "left") { + alignAttr.y -= negRotation ? bBox.height : 0; + } else if (align === "center") { + alignAttr.x -= bBox.width / 2; + alignAttr.y -= bBox.height / 2; + } else if (align === "right") { + alignAttr.x -= bBox.width; + alignAttr.y -= negRotation ? 0 : bBox.height; + } + dataLabel.placed = true; + dataLabel.alignAttr = alignAttr; + } else { + setStartPos(alignTo); // data sorting + dataLabel.align(options, null, alignTo); + alignAttr = dataLabel.alignAttr; + } + // Handle justify or crop + if (justify && alignTo.height >= 0) { + // #8830 + this.justifyDataLabel( + dataLabel, + options, + alignAttr, + bBox, + alignTo, + isNew + ); + // Now check that the data label is within the plot area + } else if (pick(options.crop, true)) { + visible = + chart.isInsidePlot(alignAttr.x, alignAttr.y) && + chart.isInsidePlot( + alignAttr.x + bBox.width, + alignAttr.y + bBox.height + ); + } + // When we're using a shape, make it possible with a connector or an + // arrow pointing to thie point + if (options.shape && !rotation) { + dataLabel[isNew ? "attr" : "animate"]({ + anchorX: inverted ? chart.plotWidth - point.plotY : point.plotX, + anchorY: inverted ? chart.plotHeight - point.plotX : point.plotY, + }); + } + } + // To use alignAttr property in hideOverlappingLabels + if (isNew && enabledDataSorting) { + dataLabel.placed = false; + } + // Show or hide based on the final aligned position + if (!visible && (!enabledDataSorting || justify)) { + dataLabel.hide(true); + dataLabel.placed = false; // don't animate back in + } + }; + /** + * Set starting position for data label sorting animation. + * + * @private + * @function Highcharts.Series#setDataLabelStartPos + * @param {Highcharts.SVGElement} dataLabel + * @param {Highcharts.ColumnPoint} point + * @param {boolean | undefined} [isNew] + * @param {boolean} [isInside] + * @param {Highcharts.AlignObject} [alignOptions] + * + * @return {void} + */ + CartesianSeries.prototype.setDataLabelStartPos = function ( + point, + dataLabel, + isNew, + isInside, + alignOptions + ) { + var chart = this.chart, + inverted = chart.inverted, + xAxis = this.xAxis, + reversed = xAxis.reversed, + labelCenter = inverted ? dataLabel.height / 2 : dataLabel.width / 2, + pointWidth = point.pointWidth, + halfWidth = pointWidth ? pointWidth / 2 : 0, + startXPos, + startYPos; + startXPos = inverted + ? alignOptions.x + : reversed + ? -labelCenter - halfWidth + : xAxis.width - labelCenter + halfWidth; + startYPos = inverted + ? reversed + ? this.yAxis.height - labelCenter + halfWidth + : -labelCenter - halfWidth + : alignOptions.y; + dataLabel.startXPos = startXPos; + dataLabel.startYPos = startYPos; + // We need to handle visibility in case of sorting point outside plot area + if (!isInside) { + dataLabel + .attr({ opacity: 1 }) + .animate({ opacity: 0 }, void 0, dataLabel.hide); + } else if (dataLabel.visibility === "hidden") { + dataLabel.show(); + dataLabel.attr({ opacity: 0 }).animate({ opacity: 1 }); + } + // Save start position on first render, but do not change position + if (!chart.hasRendered) { + return; + } + // Set start position + if (isNew) { + dataLabel.attr({ x: dataLabel.startXPos, y: dataLabel.startYPos }); + } + dataLabel.placed = true; + }; + /** + * If data labels fall partly outside the plot area, align them back in, in a + * way that doesn't hide the point. + * + * @private + * @function Highcharts.Series#justifyDataLabel + * @param {Highcharts.SVGElement} dataLabel + * @param {Highcharts.DataLabelsOptions} options + * @param {Highcharts.SVGAttributes} alignAttr + * @param {Highcharts.BBoxObject} bBox + * @param {Highcharts.BBoxObject} [alignTo] + * @param {boolean} [isNew] + * @return {boolean|undefined} + */ + CartesianSeries.prototype.justifyDataLabel = function ( + dataLabel, + options, + alignAttr, + bBox, + alignTo, + isNew + ) { + var chart = this.chart, + align = options.align, + verticalAlign = options.verticalAlign, + off, + justified, + padding = dataLabel.box ? 0 : dataLabel.padding || 0; + var _a = options.x, + x = _a === void 0 ? 0 : _a, + _b = options.y, + y = _b === void 0 ? 0 : _b; + // Off left + off = alignAttr.x + padding; + if (off < 0) { + if (align === "right" && x >= 0) { + options.align = "left"; + options.inside = true; + } else { + x -= off; + } + justified = true; + } + // Off right + off = alignAttr.x + bBox.width - padding; + if (off > chart.plotWidth) { + if (align === "left" && x <= 0) { + options.align = "right"; + options.inside = true; + } else { + x += chart.plotWidth - off; + } + justified = true; + } + // Off top + off = alignAttr.y + padding; + if (off < 0) { + if (verticalAlign === "bottom" && y >= 0) { + options.verticalAlign = "top"; + options.inside = true; + } else { + y -= off; + } + justified = true; + } + // Off bottom + off = alignAttr.y + bBox.height - padding; + if (off > chart.plotHeight) { + if (verticalAlign === "top" && y <= 0) { + options.verticalAlign = "bottom"; + options.inside = true; + } else { + y += chart.plotHeight - off; + } + justified = true; + } + if (justified) { + options.x = x; + options.y = y; + dataLabel.placed = !isNew; + dataLabel.align(options, void 0, alignTo); + } + return justified; + }; + if (seriesTypes.pie) { + seriesTypes.pie.prototype.dataLabelPositioners = { + // Based on the value computed in Highcharts' distribute algorithm. + radialDistributionY: function (point) { + return point.top + point.distributeBox.pos; + }, + // get the x - use the natural x position for labels near the + // top and bottom, to prevent the top and botton slice + // connectors from touching each other on either side + // Based on the value computed in Highcharts' distribute algorithm. + radialDistributionX: function (series, point, y, naturalY) { + return series.getX( + y < point.top + 2 || y > point.bottom - 2 ? naturalY : y, + point.half, + point + ); + }, + // dataLabels.distance determines the x position of the label + justify: function (point, radius, seriesCenter) { + return ( + seriesCenter[0] + + (point.half ? -1 : 1) * (radius + point.labelDistance) + ); + }, + // Left edges of the left-half labels touch the left edge of the plot + // area. Right edges of the right-half labels touch the right edge of + // the plot area. + alignToPlotEdges: function (dataLabel, half, plotWidth, plotLeft) { + var dataLabelWidth = dataLabel.getBBox().width; + return half + ? dataLabelWidth + plotLeft + : plotWidth - dataLabelWidth - plotLeft; + }, + // Connectors of each side end in the same x position. Labels are + // aligned to them. Left edge of the widest left-half label touches the + // left edge of the plot area. Right edge of the widest right-half label + // touches the right edge of the plot area. + alignToConnectors: function (points, half, plotWidth, plotLeft) { + var maxDataLabelWidth = 0, + dataLabelWidth; + // find widest data label + points.forEach(function (point) { + dataLabelWidth = point.dataLabel.getBBox().width; + if (dataLabelWidth > maxDataLabelWidth) { + maxDataLabelWidth = dataLabelWidth; + } + }); + return half + ? maxDataLabelWidth + plotLeft + : plotWidth - maxDataLabelWidth - plotLeft; + }, + }; + /** + * Override the base drawDataLabels method by pie specific functionality * - * */ - var find = U.find, - isArray = U.isArray, - isObject = U.isObject, - merge = U.merge, - objectEach = U.objectEach, - pick = U.pick, - splat = U.splat, - uniqueKey = U.uniqueKey; + * @private + * @function Highcharts.seriesTypes.pie#drawDataLabels + * @return {void} + */ + seriesTypes.pie.prototype.drawDataLabels = function () { + var series = this, + data = series.data, + point, + chart = series.chart, + options = series.options.dataLabels || {}, + connectorPadding = options.connectorPadding, + connectorWidth, + plotWidth = chart.plotWidth, + plotHeight = chart.plotHeight, + plotLeft = chart.plotLeft, + maxWidth = Math.round(chart.chartWidth / 3), + connector, + seriesCenter = series.center, + radius = seriesCenter[2] / 2, + centerY = seriesCenter[1], + dataLabel, + dataLabelWidth, + // labelPos, + labelPosition, + labelHeight, + // divide the points into right and left halves for anti collision + halves = [ + [], + [], // left + ], + x, + y, + visibility, + j, + overflow = [0, 0, 0, 0], // top, right, bottom, left + dataLabelPositioners = series.dataLabelPositioners, + pointDataLabelsOptions; + // get out if not enabled + if ( + !series.visible || + (!options.enabled && !series._hasPointLabels) + ) { + return; + } + // Reset all labels that have been shortened + data.forEach(function (point) { + if (point.dataLabel && point.visible && point.dataLabel.shortened) { + point.dataLabel + .attr({ + width: "auto", + }) + .css({ + width: "auto", + textOverflow: "clip", + }); + point.dataLabel.shortened = false; + } + }); + // run parent method + CartesianSeries.prototype.drawDataLabels.apply(series); + data.forEach(function (point) { + if (point.dataLabel) { + if (point.visible) { + // #407, #2510 + // Arrange points for detection collision + halves[point.half].push(point); + // Reset positions (#4905) + point.dataLabel._pos = null; + // Avoid long labels squeezing the pie size too far down + if ( + !defined(options.style.width) && + !defined( + point.options.dataLabels && + point.options.dataLabels.style && + point.options.dataLabels.style.width + ) + ) { + if (point.dataLabel.getBBox().width > maxWidth) { + point.dataLabel.css({ + // Use a fraction of the maxWidth to avoid + // wrapping close to the end of the string. + width: Math.round(maxWidth * 0.7) + "px", + }); + point.dataLabel.shortened = true; + } + } + } else { + point.dataLabel = point.dataLabel.destroy(); + // Workaround to make pies destroy multiple datalabels + // correctly. This logic needs rewriting to support multiple + // datalabels fully. + if (point.dataLabels && point.dataLabels.length === 1) { + delete point.dataLabels; + } + } + } + }); + /* Loop over the points in each half, starting from the top and bottom + * of the pie to detect overlapping labels. + */ + halves.forEach(function (points, i) { + var top, + bottom, + length = points.length, + positions = [], + naturalY, + sideOverflow, + size, + distributionLength; + if (!length) { + return; + } + // Sort by angle + series.sortByAngle(points, i - 0.5); + // Only do anti-collision when we have dataLabels outside the pie + // and have connectors. (#856) + if (series.maxLabelDistance > 0) { + top = Math.max(0, centerY - radius - series.maxLabelDistance); + bottom = Math.min( + centerY + radius + series.maxLabelDistance, + chart.plotHeight + ); + points.forEach(function (point) { + // check if specific points' label is outside the pie + if (point.labelDistance > 0 && point.dataLabel) { + // point.top depends on point.labelDistance value + // Used for calculation of y value in getX method + point.top = Math.max( + 0, + centerY - radius - point.labelDistance + ); + point.bottom = Math.min( + centerY + radius + point.labelDistance, + chart.plotHeight + ); + size = point.dataLabel.getBBox().height || 21; + // point.positionsIndex is needed for getting index of + // parameter related to specific point inside positions + // array - not every point is in positions array. + point.distributeBox = { + target: + point.labelPosition.natural.y - point.top + size / 2, + size: size, + rank: point.y, + }; + positions.push(point.distributeBox); + } + }); + distributionLength = bottom + size - top; + H.distribute( + positions, + distributionLength, + distributionLength / 5 + ); + } + // Now the used slots are sorted, fill them up sequentially + for (j = 0; j < length; j++) { + point = points[j]; + // labelPos = point.labelPos; + labelPosition = point.labelPosition; + dataLabel = point.dataLabel; + visibility = point.visible === false ? "hidden" : "inherit"; + naturalY = labelPosition.natural.y; + y = naturalY; + if (positions && defined(point.distributeBox)) { + if (typeof point.distributeBox.pos === "undefined") { + visibility = "hidden"; + } else { + labelHeight = point.distributeBox.size; + // Find label's y position + y = dataLabelPositioners.radialDistributionY(point); + } + } + // It is needed to delete point.positionIndex for + // dynamically added points etc. + delete point.positionIndex; // @todo unused + // Find label's x position + // justify is undocumented in the API - preserve support for it + if (options.justify) { + x = dataLabelPositioners.justify(point, radius, seriesCenter); + } else { + switch (options.alignTo) { + case "connectors": + x = dataLabelPositioners.alignToConnectors( + points, + i, + plotWidth, + plotLeft + ); + break; + case "plotEdges": + x = dataLabelPositioners.alignToPlotEdges( + dataLabel, + i, + plotWidth, + plotLeft + ); + break; + default: + x = dataLabelPositioners.radialDistributionX( + series, + point, + y, + naturalY + ); + } + } + // Record the placement and visibility + dataLabel._attr = { + visibility: visibility, + align: labelPosition.alignment, + }; + pointDataLabelsOptions = point.options.dataLabels || {}; + dataLabel._pos = { + x: + x + + pick(pointDataLabelsOptions.x, options.x) + // (#12985) + ({ + left: connectorPadding, + right: -connectorPadding, + }[labelPosition.alignment] || 0), + // 10 is for the baseline (label vs text) + y: + y + + pick(pointDataLabelsOptions.y, options.y) - // (#12985) + 10, + }; + // labelPos.x = x; + // labelPos.y = y; + labelPosition.final.x = x; + labelPosition.final.y = y; + // Detect overflowing data labels + if (pick(options.crop, true)) { + dataLabelWidth = dataLabel.getBBox().width; + sideOverflow = null; + // Overflow left + if ( + x - dataLabelWidth < connectorPadding && + i === 1 // left half + ) { + sideOverflow = Math.round( + dataLabelWidth - x + connectorPadding + ); + overflow[3] = Math.max(sideOverflow, overflow[3]); + // Overflow right + } else if ( + x + dataLabelWidth > plotWidth - connectorPadding && + i === 0 // right half + ) { + sideOverflow = Math.round( + x + dataLabelWidth - plotWidth + connectorPadding + ); + overflow[1] = Math.max(sideOverflow, overflow[1]); + } + // Overflow top + if (y - labelHeight / 2 < 0) { + overflow[0] = Math.max( + Math.round(-y + labelHeight / 2), + overflow[0] + ); + // Overflow left + } else if (y + labelHeight / 2 > plotHeight) { + overflow[2] = Math.max( + Math.round(y + labelHeight / 2 - plotHeight), + overflow[2] + ); + } + dataLabel.sideOverflow = sideOverflow; + } + } // for each point + }); // for each half + // Do not apply the final placement and draw the connectors until we + // have verified that labels are not spilling over. + if ( + arrayMax(overflow) === 0 || + this.verifyDataLabelOverflow(overflow) + ) { + // Place the labels in the final position + this.placeDataLabels(); + this.points.forEach(function (point) { + // #8864: every connector can have individual options + pointDataLabelsOptions = merge(options, point.options.dataLabels); + connectorWidth = pick(pointDataLabelsOptions.connectorWidth, 1); + // Draw the connector + if (connectorWidth) { + var isNew; + connector = point.connector; + dataLabel = point.dataLabel; + if ( + dataLabel && + dataLabel._pos && + point.visible && + point.labelDistance > 0 + ) { + visibility = dataLabel._attr.visibility; + isNew = !connector; + if (isNew) { + point.connector = connector = chart.renderer + .path() + .addClass( + "highcharts-data-label-connector " + + " highcharts-color-" + + point.colorIndex + + (point.className ? " " + point.className : "") + ) + .add(series.dataLabelsGroup); + if (!chart.styledMode) { + connector.attr({ + "stroke-width": connectorWidth, + stroke: + pointDataLabelsOptions.connectorColor || + point.color || + "#666666", + }); + } + } + connector[isNew ? "attr" : "animate"]({ + d: point.getConnectorPath(), + }); + connector.attr("visibility", visibility); + } else if (connector) { + point.connector = connector.destroy(); + } + } + }); + } + }; /** - * A callback function to gain complete control on when the responsive rule - * applies. + * Extendable method for getting the path of the connector between the data + * label and the pie slice. * - * @callback Highcharts.ResponsiveCallbackFunction + * @private + * @function Highcharts.seriesTypes.pie#connectorPath * - * @param {Highcharts.Chart} this - * Chart context. + * @param {*} labelPos * - * @return {boolean} - * Return `true` if it applies. + * @return {Highcharts.SVGPathArray} */ + // TODO: depracated - remove it + /* + seriesTypes.pie.prototype.connectorPath = function (labelPos) { + var x = labelPos.x, + y = labelPos.y; + return pick(this.options.dataLabels.softConnector, true) ? [ + 'M', + // end of the string at the label + x + (labelPos[6] === 'left' ? 5 : -5), y, + 'C', + x, y, // first break, next to the label + 2 * labelPos[2] - labelPos[4], 2 * labelPos[3] - labelPos[5], + labelPos[2], labelPos[3], // second break + 'L', + labelPos[4], labelPos[5] // base + ] : [ + 'M', + // end of the string at the label + x + (labelPos[6] === 'left' ? 5 : -5), y, + 'L', + labelPos[2], labelPos[3], // second break + 'L', + labelPos[4], labelPos[5] // base + ]; + }; + */ /** - * Allows setting a set of rules to apply for different screen or chart - * sizes. Each rule specifies additional chart options. + * Perform the final placement of the data labels after we have verified + * that they fall within the plot area. * - * @sample {highstock} stock/demo/responsive/ - * Stock chart - * @sample highcharts/responsive/axis/ - * Axis - * @sample highcharts/responsive/legend/ - * Legend - * @sample highcharts/responsive/classname/ - * Class name - * - * @since 5.0.0 - * @apioption responsive + * @private + * @function Highcharts.seriesTypes.pie#placeDataLabels + * @return {void} */ + seriesTypes.pie.prototype.placeDataLabels = function () { + this.points.forEach(function (point) { + var dataLabel = point.dataLabel, + _pos; + if (dataLabel && point.visible) { + _pos = dataLabel._pos; + if (_pos) { + // Shorten data labels with ellipsis if they still overflow + // after the pie has reached minSize (#223). + if (dataLabel.sideOverflow) { + dataLabel._attr.width = Math.max( + dataLabel.getBBox().width - dataLabel.sideOverflow, + 0 + ); + dataLabel.css({ + width: dataLabel._attr.width + "px", + textOverflow: + (this.options.dataLabels.style || {}).textOverflow || + "ellipsis", + }); + dataLabel.shortened = true; + } + dataLabel.attr(dataLabel._attr); + dataLabel[dataLabel.moved ? "animate" : "attr"](_pos); + dataLabel.moved = true; + } else if (dataLabel) { + dataLabel.attr({ y: -9999 }); + } + } + // Clear for update + delete point.distributeBox; + }, this); + }; + seriesTypes.pie.prototype.alignDataLabel = noop; /** - * A set of rules for responsive settings. The rules are executed from - * the top down. + * Verify whether the data labels are allowed to draw, or we should run more + * translation and data label positioning to keep them inside the plot area. + * Returns true when data labels are ready to draw. * - * @sample {highcharts} highcharts/responsive/axis/ - * Axis changes - * @sample {highstock} highcharts/responsive/axis/ - * Axis changes - * @sample {highmaps} highcharts/responsive/axis/ - * Axis changes - * - * @type {Array<*>} - * @since 5.0.0 - * @apioption responsive.rules - */ - /** - * A full set of chart options to apply as overrides to the general - * chart options. The chart options are applied when the given rule - * is active. - * - * A special case is configuration objects that take arrays, for example - * [xAxis](#xAxis), [yAxis](#yAxis) or [series](#series). For these - * collections, an `id` option is used to map the new option set to - * an existing object. If an existing object of the same id is not found, - * the item of the same indexupdated. So for example, setting `chartOptions` - * with two series items without an `id`, will cause the existing chart's - * two series to be updated with respective options. - * - * @sample {highstock} stock/demo/responsive/ - * Stock chart - * @sample highcharts/responsive/axis/ - * Axis - * @sample highcharts/responsive/legend/ - * Legend - * @sample highcharts/responsive/classname/ - * Class name - * - * @type {Highcharts.Options} - * @since 5.0.0 - * @apioption responsive.rules.chartOptions + * @private + * @function Highcharts.seriesTypes.pie#verifyDataLabelOverflow + * @param {Array<number>} overflow + * @return {boolean} */ + seriesTypes.pie.prototype.verifyDataLabelOverflow = function ( + overflow + ) { + var center = this.center, + options = this.options, + centerOption = options.center, + minSize = options.minSize || 80, + newSize = minSize, + // If a size is set, return true and don't try to shrink the pie + // to fit the labels. + ret = options.size !== null; + if (!ret) { + // Handle horizontal size and center + if (centerOption[0] !== null) { + // Fixed center + newSize = Math.max( + center[2] - Math.max(overflow[1], overflow[3]), + minSize + ); + } else { + // Auto center + newSize = Math.max( + // horizontal overflow + center[2] - overflow[1] - overflow[3], + minSize + ); + // horizontal center + center[0] += (overflow[3] - overflow[1]) / 2; + } + // Handle vertical size and center + if (centerOption[1] !== null) { + // Fixed center + newSize = clamp( + newSize, + minSize, + center[2] - Math.max(overflow[0], overflow[2]) + ); + } else { + // Auto center + newSize = clamp( + newSize, + minSize, + // vertical overflow + center[2] - overflow[0] - overflow[2] + ); + // vertical center + center[1] += (overflow[0] - overflow[2]) / 2; + } + // If the size must be decreased, we need to run translate and + // drawDataLabels again + if (newSize < center[2]) { + center[2] = newSize; + center[3] = Math.min( + // #3632 + relativeLength(options.innerSize || 0, newSize), + newSize + ); + this.translate(center); + if (this.drawDataLabels) { + this.drawDataLabels(); + } + // Else, return true to indicate that the pie and its labels is + // within the plot area + } else { + ret = true; + } + } + return ret; + }; + } + if (seriesTypes.column) { /** - * Under which conditions the rule applies. + * Override the basic data label alignment by adjusting for the position of + * the column. * - * @since 5.0.0 - * @apioption responsive.rules.condition + * @private + * @function Highcharts.seriesTypes.column#alignDataLabel + * @param {Highcharts.Point} point + * @param {Highcharts.SVGElement} dataLabel + * @param {Highcharts.DataLabelsOptions} options + * @param {Highcharts.BBoxObject} alignTo + * @param {boolean} [isNew] + * @return {void} */ - /** - * A callback function to gain complete control on when the responsive - * rule applies. Return `true` if it applies. This opens for checking - * against other metrics than the chart size, for example the document - * size or other elements. + seriesTypes.column.prototype.alignDataLabel = function ( + point, + dataLabel, + options, + alignTo, + isNew + ) { + var inverted = this.chart.inverted, + series = point.series, + // data label box for alignment + dlBox = point.dlBox || point.shapeArgs, + below = pick( + point.below, // range series + point.plotY > pick(this.translatedThreshold, series.yAxis.len) + ), + // draw it inside the box? + inside = pick(options.inside, !!this.options.stacking), + overshoot; + // Align to the column itself, or the top of it + if (dlBox) { + // Area range uses this method but not alignTo + alignTo = merge(dlBox); + if (alignTo.y < 0) { + alignTo.height += alignTo.y; + alignTo.y = 0; + } + // If parts of the box overshoots outside the plot area, modify the + // box to center the label inside + overshoot = alignTo.y + alignTo.height - series.yAxis.len; + if (overshoot > 0 && overshoot < alignTo.height) { + alignTo.height -= overshoot; + } + if (inverted) { + alignTo = { + x: series.yAxis.len - alignTo.y - alignTo.height, + y: series.xAxis.len - alignTo.x - alignTo.width, + width: alignTo.height, + height: alignTo.width, + }; + } + // Compute the alignment box + if (!inside) { + if (inverted) { + alignTo.x += below ? 0 : alignTo.width; + alignTo.width = 0; + } else { + alignTo.y += below ? alignTo.height : 0; + alignTo.height = 0; + } + } + } + // When alignment is undefined (typically columns and bars), display the + // individual point below or above the point depending on the threshold + options.align = pick( + options.align, + !inverted || inside ? "center" : below ? "right" : "left" + ); + options.verticalAlign = pick( + options.verticalAlign, + inverted || inside ? "middle" : below ? "top" : "bottom" + ); + // Call the parent method + CartesianSeries.prototype.alignDataLabel.call( + this, + point, + dataLabel, + options, + alignTo, + isNew + ); + // If label was justified and we have contrast, set it: + if (options.inside && point.contrastColor) { + dataLabel.css({ + color: point.contrastColor, + }); + } + }; + } + } + ); + _registerModule( + _modules, + "Extensions/OverlappingDataLabels.js", + [_modules["Core/Chart/Chart.js"], _modules["Core/Utilities.js"]], + function (Chart, U) { + /* * + * + * Highcharts module to hide overlapping data labels. + * This module is included in Highcharts. + * + * (c) 2009-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var addEvent = U.addEvent, + fireEvent = U.fireEvent, + isArray = U.isArray, + isNumber = U.isNumber, + objectEach = U.objectEach, + pick = U.pick; + /** + * Internal type + * @private + */ + /* eslint-disable no-invalid-this */ + // Collect potensial overlapping data labels. Stack labels probably don't need + // to be considered because they are usually accompanied by data labels that lie + // inside the columns. + addEvent(Chart, "render", function collectAndHide() { + var labels = []; + // Consider external label collectors + (this.labelCollectors || []).forEach(function (collector) { + labels = labels.concat(collector()); + }); + (this.yAxis || []).forEach(function (yAxis) { + if ( + yAxis.stacking && + yAxis.options.stackLabels && + !yAxis.options.stackLabels.allowOverlap + ) { + objectEach(yAxis.stacking.stacks, function (stack) { + objectEach(stack, function (stackItem) { + labels.push(stackItem.label); + }); + }); + } + }); + (this.series || []).forEach(function (series) { + var dlOptions = series.options.dataLabels; + if ( + series.visible && + !(dlOptions.enabled === false && !series._hasPointLabels) + ) { + // #3866 + (series.nodes || series.points).forEach(function (point) { + if (point.visible) { + var dataLabels = isArray(point.dataLabels) + ? point.dataLabels + : point.dataLabel + ? [point.dataLabel] + : []; + dataLabels.forEach(function (label) { + var options = label.options; + label.labelrank = pick( + options.labelrank, + point.labelrank, + point.shapeArgs && point.shapeArgs.height + ); // #4118 + if (!options.allowOverlap) { + labels.push(label); + } + }); + } + }); + } + }); + this.hideOverlappingLabels(labels); + }); + /** + * Hide overlapping labels. Labels are moved and faded in and out on zoom to + * provide a smooth visual imression. + * + * @private + * @function Highcharts.Chart#hideOverlappingLabels + * @param {Array<Highcharts.SVGElement>} labels + * Rendered data labels + * @requires modules/overlapping-datalabels + */ + Chart.prototype.hideOverlappingLabels = function (labels) { + var chart = this, + len = labels.length, + ren = chart.renderer, + label, + i, + j, + label1, + label2, + box1, + box2, + isLabelAffected = false, + isIntersectRect = function (box1, box2) { + return !( + box2.x >= box1.x + box1.width || + box2.x + box2.width <= box1.x || + box2.y >= box1.y + box1.height || + box2.y + box2.height <= box1.y + ); + }, + // Get the box with its position inside the chart, as opposed to getBBox + // that only reports the position relative to the parent. + getAbsoluteBox = function (label) { + var pos, + parent, + bBox, + // Substract the padding if no background or border (#4333) + padding = label.box ? 0 : label.padding || 0, + lineHeightCorrection = 0, + xOffset = 0, + boxWidth, + alignValue; + if (label && (!label.alignAttr || label.placed)) { + pos = label.alignAttr || { + x: label.attr("x"), + y: label.attr("y"), + }; + parent = label.parentGroup; + // Get width and height if pure text nodes (stack labels) + if (!label.width) { + bBox = label.getBBox(); + label.width = bBox.width; + label.height = bBox.height; + // Labels positions are computed from top left corner, so + // we need to substract the text height from text nodes too. + lineHeightCorrection = ren.fontMetrics(null, label.element).h; + } + boxWidth = label.width - 2 * padding; + alignValue = { + left: "0", + center: "0.5", + right: "1", + }[label.alignValue]; + if (alignValue) { + xOffset = +alignValue * boxWidth; + } else if ( + isNumber(label.x) && + Math.round(label.x) !== label.translateX + ) { + xOffset = label.x - label.translateX; + } + return { + x: pos.x + (parent.translateX || 0) + padding - (xOffset || 0), + y: + pos.y + + (parent.translateY || 0) + + padding - + lineHeightCorrection, + width: label.width - 2 * padding, + height: label.height - 2 * padding, + }; + } + }; + for (i = 0; i < len; i++) { + label = labels[i]; + if (label) { + // Mark with initial opacity + label.oldOpacity = label.opacity; + label.newOpacity = 1; + label.absoluteBox = getAbsoluteBox(label); + } + } + // Prevent a situation in a gradually rising slope, that each label will + // hide the previous one because the previous one always has lower rank. + labels.sort(function (a, b) { + return (b.labelrank || 0) - (a.labelrank || 0); + }); + // Detect overlapping labels + for (i = 0; i < len; i++) { + label1 = labels[i]; + box1 = label1 && label1.absoluteBox; + for (j = i + 1; j < len; ++j) { + label2 = labels[j]; + box2 = label2 && label2.absoluteBox; + if ( + box1 && + box2 && + label1 !== label2 && // #6465, polar chart with connectEnds + label1.newOpacity !== 0 && + label2.newOpacity !== 0 + ) { + if (isIntersectRect(box1, box2)) { + (label1.labelrank < label2.labelrank + ? label1 + : label2 + ).newOpacity = 0; + } + } + } + } + // Hide or show + labels.forEach(function (label) { + var complete, newOpacity; + if (label) { + newOpacity = label.newOpacity; + if (label.oldOpacity !== newOpacity) { + // Make sure the label is completely hidden to avoid catching + // clicks (#4362) + if (label.alignAttr && label.placed) { + // data labels + label[newOpacity ? "removeClass" : "addClass"]( + "highcharts-data-label-hidden" + ); + complete = function () { + if (!chart.styledMode) { + label.css({ pointerEvents: newOpacity ? "auto" : "none" }); + } + label.visibility = newOpacity ? "inherit" : "hidden"; + }; + isLabelAffected = true; + // Animate or set the opacity + label.alignAttr.opacity = newOpacity; + label[label.isOld ? "animate" : "attr"]( + label.alignAttr, + null, + complete + ); + fireEvent(chart, "afterHideOverlappingLabel"); + } else { + // other labels, tick labels + label.attr({ + opacity: newOpacity, + }); + } + } + label.isOld = true; + } + }); + if (isLabelAffected) { + fireEvent(chart, "afterHideAllOverlappingLabels"); + } + }; + } + ); + _registerModule( + _modules, + "Core/Interaction.js", + [ + _modules["Core/Series/Series.js"], + _modules["Core/Chart/Chart.js"], + _modules["Core/Globals.js"], + _modules["Core/Legend.js"], + _modules["Series/LineSeries.js"], + _modules["Core/Options.js"], + _modules["Core/Series/Point.js"], + _modules["Core/Utilities.js"], + ], + function (BaseSeries, Chart, H, Legend, LineSeries, O, Point, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var seriesTypes = BaseSeries.seriesTypes; + var hasTouch = H.hasTouch, + svg = H.svg; + var defaultOptions = O.defaultOptions; + var addEvent = U.addEvent, + createElement = U.createElement, + css = U.css, + defined = U.defined, + extend = U.extend, + fireEvent = U.fireEvent, + isArray = U.isArray, + isFunction = U.isFunction, + isNumber = U.isNumber, + isObject = U.isObject, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick; + /** + * @interface Highcharts.PointEventsOptionsObject + */ /** + * Fires when the point is selected either programmatically or following a click + * on the point. One parameter, `event`, is passed to the function. Returning + * `false` cancels the operation. + * @name Highcharts.PointEventsOptionsObject#select + * @type {Highcharts.PointSelectCallbackFunction|undefined} + */ /** + * Fires when the point is unselected either programmatically or following a + * click on the point. One parameter, `event`, is passed to the function. + * Returning `false` cancels the operation. + * @name Highcharts.PointEventsOptionsObject#unselect + * @type {Highcharts.PointUnselectCallbackFunction|undefined} + */ + /** + * Information about the select/unselect event. + * + * @interface Highcharts.PointInteractionEventObject + * @extends global.Event + */ /** + * @name Highcharts.PointInteractionEventObject#accumulate + * @type {boolean} + */ + /** + * Gets fired when the point is selected either programmatically or following a + * click on the point. + * + * @callback Highcharts.PointSelectCallbackFunction + * + * @param {Highcharts.Point} this + * Point where the event occured. + * + * @param {Highcharts.PointInteractionEventObject} event + * Event that occured. + */ + /** + * Fires when the point is unselected either programmatically or following a + * click on the point. + * + * @callback Highcharts.PointUnselectCallbackFunction + * + * @param {Highcharts.Point} this + * Point where the event occured. + * + * @param {Highcharts.PointInteractionEventObject} event + * Event that occured. + */ + (""); // detach doclets above + /* eslint-disable valid-jsdoc */ + /** + * TrackerMixin for points and graphs. + * + * @private + * @mixin Highcharts.TrackerMixin + */ + var TrackerMixin = (H.TrackerMixin = { + /** + * Draw the tracker for a point. * - * @type {Highcharts.ResponsiveCallbackFunction} - * @since 5.0.0 - * @context Highcharts.Chart - * @apioption responsive.rules.condition.callback + * @private + * @function Highcharts.TrackerMixin.drawTrackerPoint + * @param {Highcharts.Series} this + * @fires Highcharts.Series#event:afterDrawTracker */ + drawTrackerPoint: function () { + var series = this, + chart = series.chart, + pointer = chart.pointer, + onMouseOver = function (e) { + var point = pointer.getPointFromEvent(e); + // undefined on graph in scatterchart + if (typeof point !== "undefined") { + pointer.isDirectTouch = true; + point.onMouseOver(e); + } + }, + dataLabels; + // Add reference to the point + series.points.forEach(function (point) { + dataLabels = isArray(point.dataLabels) + ? point.dataLabels + : point.dataLabel + ? [point.dataLabel] + : []; + if (point.graphic) { + point.graphic.element.point = point; + } + dataLabels.forEach(function (dataLabel) { + if (dataLabel.div) { + dataLabel.div.point = point; + } else { + dataLabel.element.point = point; + } + }); + }); + // Add the event listeners, we need to do this only once + if (!series._hasTracking) { + series.trackerGroups.forEach(function (key) { + if (series[key]) { + // we don't always have dataLabelsGroup + series[key] + .addClass("highcharts-tracker") + .on("mouseover", onMouseOver) + .on("mouseout", function (e) { + pointer.onTrackerMouseOut(e); + }); + if (hasTouch) { + series[key].on("touchstart", onMouseOver); + } + if (!chart.styledMode && series.options.cursor) { + series[key].css(css).css({ cursor: series.options.cursor }); + } + } + }); + series._hasTracking = true; + } + fireEvent(this, "afterDrawTracker"); + }, /** - * The responsive rule applies if the chart height is less than this. + * Draw the tracker object that sits above all data labels and markers to + * track mouse events on the graph or points. For the line type charts + * the tracker uses the same graphPath, but with a greater stroke width + * for better control. * - * @type {number} - * @since 5.0.0 - * @apioption responsive.rules.condition.maxHeight - */ + * @private + * @function Highcharts.TrackerMixin.drawTrackerGraph + * @param {Highcharts.Series} this + * @fires Highcharts.Series#event:afterDrawTracker + */ + drawTrackerGraph: function () { + var series = this, + options = series.options, + trackByArea = options.trackByArea, + trackerPath = [].concat( + trackByArea ? series.areaPath : series.graphPath + ), + // trackerPathLength = trackerPath.length, + chart = series.chart, + pointer = chart.pointer, + renderer = chart.renderer, + snap = chart.options.tooltip.snap, + tracker = series.tracker, + i, + onMouseOver = function (e) { + if (chart.hoverSeries !== series) { + series.onMouseOver(); + } + }, + /* + * Empirical lowest possible opacities for TRACKER_FILL for an + * element to stay invisible but clickable + * IE6: 0.002 + * IE7: 0.002 + * IE8: 0.002 + * IE9: 0.00000000001 (unlimited) + * IE10: 0.0001 (exporting only) + * FF: 0.00000000001 (unlimited) + * Chrome: 0.000001 + * Safari: 0.000001 + * Opera: 0.00000000001 (unlimited) + */ + TRACKER_FILL = "rgba(192,192,192," + (svg ? 0.0001 : 0.002) + ")"; + // Draw the tracker + if (tracker) { + tracker.attr({ d: trackerPath }); + } else if (series.graph) { + // create + series.tracker = renderer + .path(trackerPath) + .attr({ + visibility: series.visible ? "visible" : "hidden", + zIndex: 2, + }) + .addClass( + trackByArea + ? "highcharts-tracker-area" + : "highcharts-tracker-line" + ) + .add(series.group); + if (!chart.styledMode) { + series.tracker.attr({ + "stroke-linecap": "round", + "stroke-linejoin": "round", + stroke: TRACKER_FILL, + fill: trackByArea ? TRACKER_FILL : "none", + "stroke-width": + series.graph.strokeWidth() + (trackByArea ? 0 : 2 * snap), + }); + } + // The tracker is added to the series group, which is clipped, but + // is covered by the marker group. So the marker group also needs to + // capture events. + [series.tracker, series.markerGroup].forEach(function (tracker) { + tracker + .addClass("highcharts-tracker") + .on("mouseover", onMouseOver) + .on("mouseout", function (e) { + pointer.onTrackerMouseOut(e); + }); + if (options.cursor && !chart.styledMode) { + tracker.css({ cursor: options.cursor }); + } + if (hasTouch) { + tracker.on("touchstart", onMouseOver); + } + }); + } + fireEvent(this, "afterDrawTracker"); + }, + }); + /* End TrackerMixin */ + // Add tracking event listener to the series group, so the point graphics + // themselves act as trackers + if (seriesTypes.column) { /** - * The responsive rule applies if the chart width is less than this. - * - * @sample highcharts/responsive/axis/ - * Max width is 500 - * - * @type {number} - * @since 5.0.0 - * @apioption responsive.rules.condition.maxWidth + * @private + * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.column#drawTracker */ + seriesTypes.column.prototype.drawTracker = + TrackerMixin.drawTrackerPoint; + } + if (seriesTypes.pie) { /** - * The responsive rule applies if the chart height is greater than this. - * - * @type {number} - * @default 0 - * @since 5.0.0 - * @apioption responsive.rules.condition.minHeight + * @private + * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.pie#drawTracker */ + seriesTypes.pie.prototype.drawTracker = TrackerMixin.drawTrackerPoint; + } + if (seriesTypes.scatter) { /** - * The responsive rule applies if the chart width is greater than this. - * - * @type {number} - * @default 0 - * @since 5.0.0 - * @apioption responsive.rules.condition.minWidth + * @private + * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.scatter#drawTracker */ - /* eslint-disable no-invalid-this, valid-jsdoc */ + seriesTypes.scatter.prototype.drawTracker = + TrackerMixin.drawTrackerPoint; + } + // Extend Legend for item events. + extend(Legend.prototype, { /** - * Update the chart based on the current chart/document size and options for - * responsiveness. - * * @private - * @function Highcharts.Chart#setResponsive - * @param {boolean} [redraw=true] - * @param {boolean} [reset=false] - * Reset by un-applying all rules. Chart.update resets all rules before applying - * updated options. - */ - Chart.prototype.setResponsive = function (redraw, reset) { - var options = this.options.responsive, - ruleIds = [], - currentResponsive = this.currentResponsive, - currentRuleIds, - undoOptions; - if (!reset && options && options.rules) { - options.rules.forEach(function (rule) { - if (typeof rule._id === 'undefined') { - rule._id = uniqueKey(); - } - this.matchResponsiveRule(rule, ruleIds /* , redraw */); - }, this); - } - // Merge matching rules - var mergedOptions = merge.apply(0, - ruleIds.map(function (ruleId) { - return find(options.rules, - function (rule) { - return rule._id === ruleId; - }).chartOptions; - })); - mergedOptions.isResponsiveOptions = true; - // Stringified key for the rules that currently apply. - ruleIds = (ruleIds.toString() || void 0); - currentRuleIds = currentResponsive && currentResponsive.ruleIds; - // Changes in what rules apply - if (ruleIds !== currentRuleIds) { - // Undo previous rules. Before we apply a new set of rules, we need to - // roll back completely to base options (#6291). - if (currentResponsive) { - this.update(currentResponsive.undoOptions, redraw, true); - } - if (ruleIds) { - // Get undo-options for matching rules - undoOptions = this.currentOptions(mergedOptions); - undoOptions.isResponsiveOptions = true; - this.currentResponsive = { - ruleIds: ruleIds, - mergedOptions: mergedOptions, - undoOptions: undoOptions + * @function Highcharts.Legend#setItemEvents + * @param {Highcharts.BubbleLegend|Point|Highcharts.Series} item + * @param {Highcharts.SVGElement} legendItem + * @param {boolean} [useHTML=false] + * @fires Highcharts.Point#event:legendItemClick + * @fires Highcharts.Series#event:legendItemClick + */ + setItemEvents: function (item, legendItem, useHTML) { + var legend = this, + boxWrapper = legend.chart.renderer.boxWrapper, + isPoint = item instanceof Point, + activeClass = + "highcharts-legend-" + (isPoint ? "point" : "series") + "-active", + styledMode = legend.chart.styledMode, + // When `useHTML`, the symbol is rendered in other group, so + // we need to apply events listeners to both places + legendItems = useHTML + ? [legendItem, item.legendSymbol] + : [item.legendGroup]; + // Set the events on the item group, or in case of useHTML, the item + // itself (#1249) + legendItems.forEach(function (element) { + if (element) { + element + .on("mouseover", function () { + if (item.visible) { + legend.allItems.forEach(function (inactiveItem) { + if (item !== inactiveItem) { + inactiveItem.setState("inactive", !isPoint); + } + }); + } + item.setState("hover"); + // A CSS class to dim or hide other than the hovered + // series. + // Works only if hovered series is visible (#10071). + if (item.visible) { + boxWrapper.addClass(activeClass); + } + if (!styledMode) { + legendItem.css(legend.options.itemHoverStyle); + } + }) + .on("mouseout", function () { + if (!legend.chart.styledMode) { + legendItem.css( + merge( + item.visible ? legend.itemStyle : legend.itemHiddenStyle + ) + ); + } + legend.allItems.forEach(function (inactiveItem) { + if (item !== inactiveItem) { + inactiveItem.setState("", !isPoint); + } + }); + // A CSS class to dim or hide other than the hovered + // series. + boxWrapper.removeClass(activeClass); + item.setState(); + }) + .on("click", function (event) { + var strLegendItemClick = "legendItemClick", + fnLegendItemClick = function () { + if (item.setVisible) { + item.setVisible(); + } + // Reset inactive state + legend.allItems.forEach(function (inactiveItem) { + if (item !== inactiveItem) { + inactiveItem.setState( + item.visible ? "inactive" : "", + !isPoint + ); + } + }); }; - this.update(mergedOptions, redraw, true); - } - else { - this.currentResponsive = void 0; - } - } - }; - /** - * Handle a single responsiveness rule. - * - * @private - * @function Highcharts.Chart#matchResponsiveRule - * @param {Highcharts.ResponsiveRulesOptions} rule - * @param {Array<string>} matches - */ - Chart.prototype.matchResponsiveRule = function (rule, matches) { - var condition = rule.condition, - fn = condition.callback || function () { - return (this.chartWidth <= pick(condition.maxWidth, - Number.MAX_VALUE) && - this.chartHeight <= - pick(condition.maxHeight, - Number.MAX_VALUE) && - this.chartWidth >= pick(condition.minWidth, 0) && - this.chartHeight >= pick(condition.minHeight, 0)); - }; - if (fn.call(this)) { - matches.push(rule._id); + // A CSS class to dim or hide other than the hovered + // series. Event handling in iOS causes the activeClass + // to be added prior to click in some cases (#7418). + boxWrapper.removeClass(activeClass); + // Pass over the click/touch event. #4. + event = { + browserEvent: event, + }; + // click the name or symbol + if (item.firePointEvent) { + // point + item.firePointEvent( + strLegendItemClick, + event, + fnLegendItemClick + ); + } else { + fireEvent( + item, + strLegendItemClick, + event, + fnLegendItemClick + ); + } + }); } - }; + }); + }, /** - * Get the current values for a given set of options. Used before we update - * the chart with a new responsiveness rule. - * - * @todo Restore axis options (by id?). The matching of items in collections - * bears resemblance to the oneToOne matching in Chart.update. Probably we can - * refactor out that matching and reuse it in both functions. - * * @private - * @function Highcharts.Chart#currentOptions - * @param {Highcharts.Options} options - * @return {Highcharts.Options} - */ - Chart.prototype.currentOptions = function (options) { + * @function Highcharts.Legend#createCheckboxForItem + * @param {Highcharts.BubbleLegend|Point|Highcharts.Series} item + * @fires Highcharts.Series#event:checkboxClick + */ + createCheckboxForItem: function (item) { + var legend = this; + item.checkbox = createElement( + "input", + { + type: "checkbox", + className: "highcharts-legend-checkbox", + checked: item.selected, + defaultChecked: item.selected, // required by IE7 + }, + legend.options.itemCheckboxStyle, + legend.chart.container + ); + addEvent(item.checkbox, "click", function (event) { + var target = event.target; + fireEvent( + item.series || item, + "checkboxClick", + { + checked: target.checked, + item: item, + }, + function () { + item.select(); + } + ); + }); + }, + }); + // Extend the Chart object with interaction + extend( + Chart.prototype, + /** @lends Chart.prototype */ { + /** + * Display the zoom button, so users can reset zoom to the default view + * settings. + * + * @function Highcharts.Chart#showResetZoom + * + * @fires Highcharts.Chart#event:afterShowResetZoom + * @fires Highcharts.Chart#event:beforeShowResetZoom + */ + showResetZoom: function () { var chart = this, - ret = {}; + lang = defaultOptions.lang, + btnOptions = chart.options.chart.resetZoomButton, + theme = btnOptions.theme, + states = theme.states, + alignTo = + btnOptions.relativeTo === "chart" || + btnOptions.relativeTo === "spaceBox" + ? null + : "plotBox"; /** - * Recurse over a set of options and its current values, - * and store the current values in the ret object. + * @private */ - function getCurrent(options, curr, ret, depth) { - var i; - objectEach(options, function (val, key) { - if (!depth && - chart.collectionsWithUpdate.indexOf(key) > -1) { - val = splat(val); - ret[key] = []; - // Iterate over collections like series, xAxis or yAxis and map - // the items by index. - for (i = 0; i < Math.max(val.length, curr[key].length); i++) { - // Item exists in current data (#6347) - if (curr[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 (val[i] === void 0) { - ret[key][i] = curr[key][i]; - // Otherwise, proceed - } - else { - ret[key][i] = {}; - getCurrent(val[i], curr[key][i], ret[key][i], depth + 1); - } - } - } - } - else if (isObject(val)) { - ret[key] = isArray(val) ? [] : {}; - getCurrent(val, curr[key] || {}, ret[key], depth + 1); - } - else if (typeof curr[key] === 'undefined') { // #10286 - ret[key] = null; - } - else { - ret[key] = curr[key]; - } + function zoomOut() { + chart.zoomOut(); + } + fireEvent(this, "beforeShowResetZoom", null, function () { + chart.resetZoomButton = chart.renderer + .button( + lang.resetZoom, + null, + null, + zoomOut, + theme, + states && states.hover + ) + .attr({ + align: btnOptions.position.align, + title: lang.resetZoomTitle, + }) + .addClass("highcharts-reset-zoom") + .add() + .align(btnOptions.position, false, alignTo); + }); + fireEvent(this, "afterShowResetZoom"); + }, + /** + * Zoom the chart out after a user has zoomed in. See also + * [Axis.setExtremes](/class-reference/Highcharts.Axis#setExtremes). + * + * @function Highcharts.Chart#zoomOut + * + * @fires Highcharts.Chart#event:selection + */ + zoomOut: function () { + fireEvent(this, "selection", { resetSelection: true }, this.zoom); + }, + /** + * Zoom into a given portion of the chart given by axis coordinates. + * + * @private + * @function Highcharts.Chart#zoom + * @param {Highcharts.SelectEventObject} event + */ + zoom: function (event) { + var chart = this, + hasZoomed, + pointer = chart.pointer, + displayButton = false, + mouseDownPos = chart.inverted + ? pointer.mouseDownX + : pointer.mouseDownY, + resetZoomButton; + // If zoom is called with no arguments, reset the axes + if (!event || event.resetSelection) { + chart.axes.forEach(function (axis) { + hasZoomed = axis.zoom(); + }); + pointer.initiated = false; // #6804 + } else { + // else, zoom in on all axes + event.xAxis.concat(event.yAxis).forEach(function (axisData) { + var axis = axisData.axis, + axisStartPos = chart.inverted ? axis.left : axis.top, + axisEndPos = chart.inverted + ? axisStartPos + axis.width + : axisStartPos + axis.height, + isXAxis = axis.isXAxis, + isWithinPane = false; + // Check if zoomed area is within the pane (#1289). + // In case of multiple panes only one pane should be zoomed. + if ( + (!isXAxis && + mouseDownPos >= axisStartPos && + mouseDownPos <= axisEndPos) || + isXAxis || + !defined(mouseDownPos) + ) { + isWithinPane = true; + } + // don't zoom more than minRange + if (pointer[isXAxis ? "zoomX" : "zoomY"] && isWithinPane) { + hasZoomed = axis.zoom(axisData.min, axisData.max); + if (axis.displayBtn) { + displayButton = true; + } + } + }); + } + // Show or hide the Reset zoom button + resetZoomButton = chart.resetZoomButton; + if (displayButton && !resetZoomButton) { + chart.showResetZoom(); + } else if (!displayButton && isObject(resetZoomButton)) { + chart.resetZoomButton = resetZoomButton.destroy(); + } + // Redraw + if (hasZoomed) { + chart.redraw( + pick( + chart.options.chart.animation, + event && event.animation, + chart.pointCount < 100 + ) + ); + } + }, + /** + * Pan the chart by dragging the mouse across the pane. This function is + * called on mouse move, and the distance to pan is computed from chartX + * compared to the first chartX position in the dragging operation. + * + * @private + * @function Highcharts.Chart#pan + * @param {Highcharts.PointerEventObject} e + * @param {string} panning + */ + pan: function (e, panning) { + var chart = this, + hoverPoints = chart.hoverPoints, + panningOptions, + chartOptions = chart.options.chart, + hasMapNavigation = + chart.options.mapNavigation && + chart.options.mapNavigation.enabled, + doRedraw, + type; + if (typeof panning === "object") { + panningOptions = panning; + } else { + panningOptions = { + enabled: panning, + type: "x", + }; + } + if (chartOptions && chartOptions.panning) { + chartOptions.panning = panningOptions; + } + type = panningOptions.type; + fireEvent(this, "pan", { originalEvent: e }, function () { + // remove active points for shared tooltip + if (hoverPoints) { + hoverPoints.forEach(function (point) { + point.setState(); }); + } + // panning axis mapping + var xy = [1]; // x + if (type === "xy") { + xy = [1, 0]; + } else if (type === "y") { + xy = [0]; + } + xy.forEach(function (isX) { + var axis = chart[isX ? "xAxis" : "yAxis"][0], + horiz = axis.horiz, + mousePos = e[horiz ? "chartX" : "chartY"], + mouseDown = horiz ? "mouseDownX" : "mouseDownY", + startPos = chart[mouseDown], + halfPointRange = (axis.pointRange || 0) / 2, + pointRangeDirection = + (axis.reversed && !chart.inverted) || + (!axis.reversed && chart.inverted) + ? -1 + : 1, + extremes = axis.getExtremes(), + panMin = + axis.toValue(startPos - mousePos, true) + + halfPointRange * pointRangeDirection, + panMax = + axis.toValue(startPos + axis.len - mousePos, true) - + halfPointRange * pointRangeDirection, + flipped = panMax < panMin, + newMin = flipped ? panMax : panMin, + newMax = flipped ? panMin : panMax, + hasVerticalPanning = axis.hasVerticalPanning(), + paddedMin, + paddedMax, + spill, + panningState = axis.panningState; + // General calculations of panning state. + // This is related to using vertical panning. (#11315). + axis.series.forEach(function (series) { + if ( + hasVerticalPanning && + !isX && + (!panningState || panningState.isDirty) + ) { + var processedData = series.getProcessedData(true), + dataExtremes = series.getExtremes( + processedData.yData, + true + ); + if (!panningState) { + panningState = { + startMin: Number.MAX_VALUE, + startMax: -Number.MAX_VALUE, + }; + } + if ( + isNumber(dataExtremes.dataMin) && + isNumber(dataExtremes.dataMax) + ) { + panningState.startMin = Math.min( + dataExtremes.dataMin, + panningState.startMin + ); + panningState.startMax = Math.max( + dataExtremes.dataMax, + panningState.startMax + ); + } + } + }); + paddedMin = Math.min( + pick( + panningState === null || panningState === void 0 + ? void 0 + : panningState.startMin, + extremes.dataMin + ), + halfPointRange + ? extremes.min + : axis.toValue( + axis.toPixels(extremes.min) - axis.minPixelPadding + ) + ); + paddedMax = Math.max( + pick( + panningState === null || panningState === void 0 + ? void 0 + : panningState.startMax, + extremes.dataMax + ), + halfPointRange + ? extremes.max + : axis.toValue( + axis.toPixels(extremes.max) + axis.minPixelPadding + ) + ); + axis.panningState = panningState; + // It is not necessary to calculate extremes on ordinal axis, + // because they are already calculated, so we don't want to + // override them. + if (!axis.isOrdinal) { + // If the new range spills over, either to the min or max, + // adjust the new range. + spill = paddedMin - newMin; + if (spill > 0) { + newMax += spill; + newMin = paddedMin; + } + spill = newMax - paddedMax; + if (spill > 0) { + newMax = paddedMax; + newMin -= spill; + } + // Set new extremes if they are actually new + if ( + axis.series.length && + newMin !== extremes.min && + newMax !== extremes.max && + newMin >= paddedMin && + newMax <= paddedMax + ) { + axis.setExtremes(newMin, newMax, false, false, { + trigger: "pan", + }); + if ( + !chart.resetZoomButton && + !hasMapNavigation && + // Show reset zoom button only when both newMin and + // newMax values are between padded axis range. + newMin !== paddedMin && + newMax !== paddedMax && + type.match("y") + ) { + chart.showResetZoom(); + axis.displayBtn = false; + } + doRedraw = true; + } + // set new reference for next run: + chart[mouseDown] = mousePos; + } + }); + if (doRedraw) { + chart.redraw(false); + } + css(chart.container, { cursor: "move" }); + }); + }, + } + ); + // Extend the Point object with interaction + extend( + Point.prototype, + /** @lends Highcharts.Point.prototype */ { + /** + * Toggle the selection status of a point. + * + * @see Highcharts.Chart#getSelectedPoints + * + * @sample highcharts/members/point-select/ + * Select a point from a button + * @sample highcharts/chart/events-selection-points/ + * Select a range of points through a drag selection + * @sample maps/series/data-id/ + * Select a point in Highmaps + * + * @function Highcharts.Point#select + * + * @param {boolean} [selected] + * When `true`, the point is selected. When `false`, the point is + * unselected. When `null` or `undefined`, the selection state is toggled. + * + * @param {boolean} [accumulate=false] + * When `true`, the selection is added to other selected points. + * When `false`, other selected points are deselected. Internally in + * Highcharts, when + * [allowPointSelect](https://api.highcharts.com/highcharts/plotOptions.series.allowPointSelect) + * is `true`, selected points are accumulated on Control, Shift or Cmd + * clicking the point. + * + * @fires Highcharts.Point#event:select + * @fires Highcharts.Point#event:unselect + */ + select: function (selected, accumulate) { + var point = this, + series = point.series, + chart = series.chart; + selected = pick(selected, !point.selected); + this.selectedStaging = selected; + // fire the event with the default handler + point.firePointEvent( + selected ? "select" : "unselect", + { accumulate: accumulate }, + function () { + /** + * Whether the point is selected or not. + * + * @see Point#select + * @see Chart#getSelectedPoints + * + * @name Highcharts.Point#selected + * @type {boolean} + */ + point.selected = point.options.selected = selected; + series.options.data[series.data.indexOf(point)] = point.options; + point.setState(selected && "select"); + // unselect all other points unless Ctrl or Cmd + click + if (!accumulate) { + chart.getSelectedPoints().forEach(function (loopPoint) { + var loopSeries = loopPoint.series; + if (loopPoint.selected && loopPoint !== point) { + loopPoint.selected = loopPoint.options.selected = false; + loopSeries.options.data[ + loopSeries.data.indexOf(loopPoint) + ] = loopPoint.options; + // Programatically selecting a point should restore + // normal state, but when click happened on other + // point, set inactive state to match other points + loopPoint.setState( + chart.hoverPoints && + loopSeries.options.inactiveOtherPoints + ? "inactive" + : "" + ); + loopPoint.firePointEvent("unselect"); + } + }); + } + } + ); + delete this.selectedStaging; + }, + /** + * Runs on mouse over the point. Called internally from mouse and touch + * events. + * + * @function Highcharts.Point#onMouseOver + * + * @param {Highcharts.PointerEventObject} [e] + * The event arguments. + */ + onMouseOver: function (e) { + var point = this, + series = point.series, + chart = series.chart, + pointer = chart.pointer; + e = e + ? pointer.normalize(e) + : // In cases where onMouseOver is called directly without an event + pointer.getChartCoordinatesFromPoint(point, chart.inverted); + pointer.runPointActions(e, point); + }, + /** + * Runs on mouse out from the point. Called internally from mouse and touch + * events. + * + * @function Highcharts.Point#onMouseOut + * @fires Highcharts.Point#event:mouseOut + */ + onMouseOut: function () { + var point = this, + chart = point.series.chart; + point.firePointEvent("mouseOut"); + if (!point.series.options.inactiveOtherPoints) { + (chart.hoverPoints || []).forEach(function (p) { + p.setState(); + }); } - getCurrent(options, this.options, ret, 0); - return ret; - }; - - }); - _registerModule(_modules, 'masters/highcharts.src.js', [_modules['Core/Globals.js']], function (Highcharts) { - - - return Highcharts; - }); - _modules['masters/highcharts.src.js']._modules = _modules; - return _modules['masters/highcharts.src.js']; -})); \ No newline at end of file + chart.hoverPoints = chart.hoverPoint = null; + }, + /** + * Import events from the series' and point's options. Only do it on + * demand, to save processing time on hovering. + * + * @private + * @function Highcharts.Point#importEvents + */ + importEvents: function () { + if (!this.hasImportedEvents) { + var point = this, + options = merge(point.series.options.point, point.options), + events = options.events; + point.events = events; + objectEach(events, function (event, eventType) { + if (isFunction(event)) { + addEvent(point, eventType, event); + } + }); + this.hasImportedEvents = true; + } + }, + /** + * Set the point's state. + * + * @function Highcharts.Point#setState + * + * @param {Highcharts.PointStateValue|""} [state] + * The new state, can be one of `'hover'`, `'select'`, `'inactive'`, + * or `''` (an empty string), `'normal'` or `undefined` to set to + * normal state. + * @param {boolean} [move] + * State for animation. + * + * @fires Highcharts.Point#event:afterSetState + */ + setState: function (state, move) { + var point = this, + series = point.series, + previousState = point.state, + stateOptions = series.options.states[state || "normal"] || {}, + markerOptions = + defaultOptions.plotOptions[series.type].marker && + series.options.marker, + normalDisabled = markerOptions && markerOptions.enabled === false, + markerStateOptions = + (markerOptions && + markerOptions.states && + markerOptions.states[state || "normal"]) || + {}, + stateDisabled = markerStateOptions.enabled === false, + stateMarkerGraphic = series.stateMarkerGraphic, + pointMarker = point.marker || {}, + chart = series.chart, + halo = series.halo, + haloOptions, + markerAttribs, + pointAttribs, + pointAttribsAnimation, + hasMarkers = markerOptions && series.markerAttribs, + newSymbol; + state = state || ""; // empty string + if ( + // already has this state + (state === point.state && !move) || + // selected points don't respond to hover + (point.selected && state !== "select") || + // series' state options is disabled + stateOptions.enabled === false || + // general point marker's state options is disabled + (state && + (stateDisabled || + (normalDisabled && markerStateOptions.enabled === false))) || + // individual point marker's state options is disabled + (state && + pointMarker.states && + pointMarker.states[state] && + pointMarker.states[state].enabled === false) // #1610 + ) { + return; + } + point.state = state; + if (hasMarkers) { + markerAttribs = series.markerAttribs(point, state); + } + // Apply hover styles to the existing point + if (point.graphic) { + if (previousState) { + point.graphic.removeClass("highcharts-point-" + previousState); + } + if (state) { + point.graphic.addClass("highcharts-point-" + state); + } + if (!chart.styledMode) { + pointAttribs = series.pointAttribs(point, state); + pointAttribsAnimation = pick( + chart.options.chart.animation, + stateOptions.animation + ); + // Some inactive points (e.g. slices in pie) should apply + // oppacity also for it's labels + if ( + series.options.inactiveOtherPoints && + pointAttribs.opacity + ) { + (point.dataLabels || []).forEach(function (label) { + if (label) { + label.animate( + { + opacity: pointAttribs.opacity, + }, + pointAttribsAnimation + ); + } + }); + if (point.connector) { + point.connector.animate( + { + opacity: pointAttribs.opacity, + }, + pointAttribsAnimation + ); + } + } + point.graphic.animate(pointAttribs, pointAttribsAnimation); + } + if (markerAttribs) { + point.graphic.animate( + markerAttribs, + pick( + // Turn off globally: + chart.options.chart.animation, + markerStateOptions.animation, + markerOptions.animation + ) + ); + } + // Zooming in from a range with no markers to a range with markers + if (stateMarkerGraphic) { + stateMarkerGraphic.hide(); + } + } else { + // if a graphic is not applied to each point in the normal state, + // create a shared graphic for the hover state + if (state && markerStateOptions) { + newSymbol = pointMarker.symbol || series.symbol; + // If the point has another symbol than the previous one, throw + // away the state marker graphic and force a new one (#1459) + if ( + stateMarkerGraphic && + stateMarkerGraphic.currentSymbol !== newSymbol + ) { + stateMarkerGraphic = stateMarkerGraphic.destroy(); + } + // Add a new state marker graphic + if (markerAttribs) { + if (!stateMarkerGraphic) { + if (newSymbol) { + series.stateMarkerGraphic = stateMarkerGraphic = + chart.renderer + .symbol( + newSymbol, + markerAttribs.x, + markerAttribs.y, + markerAttribs.width, + markerAttribs.height + ) + .add(series.markerGroup); + stateMarkerGraphic.currentSymbol = newSymbol; + } + // Move the existing graphic + } else { + stateMarkerGraphic[move ? "animate" : "attr"]({ + x: markerAttribs.x, + y: markerAttribs.y, + }); + } + } + if (!chart.styledMode && stateMarkerGraphic) { + stateMarkerGraphic.attr(series.pointAttribs(point, state)); + } + } + if (stateMarkerGraphic) { + stateMarkerGraphic[state && point.isInside ? "show" : "hide"](); // #2450 + stateMarkerGraphic.element.point = point; // #4310 + } + } + // Show me your halo + haloOptions = stateOptions.halo; + var markerGraphic = point.graphic || stateMarkerGraphic; + var markerVisibility = + (markerGraphic && markerGraphic.visibility) || "inherit"; + if ( + haloOptions && + haloOptions.size && + markerGraphic && + markerVisibility !== "hidden" && + !point.isCluster + ) { + if (!halo) { + series.halo = halo = chart.renderer + .path() + // #5818, #5903, #6705 + .add(markerGraphic.parentGroup); + } + halo.show()[move ? "animate" : "attr"]({ + d: point.haloPath(haloOptions.size), + }); + halo.attr({ + class: + "highcharts-halo highcharts-color-" + + pick(point.colorIndex, series.colorIndex) + + (point.className ? " " + point.className : ""), + visibility: markerVisibility, + zIndex: -1, // #4929, #8276 + }); + halo.point = point; // #6055 + if (!chart.styledMode) { + halo.attr( + extend( + { + fill: point.color || series.color, + "fill-opacity": haloOptions.opacity, + }, + haloOptions.attributes + ) + ); + } + } else if (halo && halo.point && halo.point.haloPath) { + // Animate back to 0 on the current halo point (#6055) + halo.animate( + { d: halo.point.haloPath(0) }, + null, + // Hide after unhovering. The `complete` callback runs in the + // halo's context (#7681). + halo.hide + ); + } + fireEvent(point, "afterSetState"); + }, + /** + * Get the path definition for the halo, which is usually a shadow-like + * circle around the currently hovered point. + * + * @function Highcharts.Point#haloPath + * + * @param {number} size + * The radius of the circular halo. + * + * @return {Highcharts.SVGPathArray} + * The path definition. + */ + haloPath: function (size) { + var series = this.series, + chart = series.chart; + return chart.renderer.symbols.circle( + Math.floor(this.plotX) - size, + this.plotY - size, + size * 2, + size * 2 + ); + }, + } + ); + // Extend the Series object with interaction + extend( + LineSeries.prototype, + /** @lends Highcharts.Series.prototype */ { + /** + * Runs on mouse over the series graphical items. + * + * @function Highcharts.Series#onMouseOver + * @fires Highcharts.Series#event:mouseOver + */ + onMouseOver: function () { + var series = this, + chart = series.chart, + hoverSeries = chart.hoverSeries, + pointer = chart.pointer; + pointer.setHoverChartIndex(); + // set normal state to previous series + if (hoverSeries && hoverSeries !== series) { + hoverSeries.onMouseOut(); + } + // trigger the event, but to save processing time, + // only if defined + if (series.options.events.mouseOver) { + fireEvent(series, "mouseOver"); + } + // hover this + series.setState("hover"); + /** + * Contains the original hovered series. + * + * @name Highcharts.Chart#hoverSeries + * @type {Highcharts.Series|null} + */ + chart.hoverSeries = series; + }, + /** + * Runs on mouse out of the series graphical items. + * + * @function Highcharts.Series#onMouseOut + * + * @fires Highcharts.Series#event:mouseOut + */ + onMouseOut: function () { + // trigger the event only if listeners exist + var series = this, + options = series.options, + chart = series.chart, + tooltip = chart.tooltip, + hoverPoint = chart.hoverPoint; + // #182, set to null before the mouseOut event fires + chart.hoverSeries = null; + // trigger mouse out on the point, which must be in this series + if (hoverPoint) { + hoverPoint.onMouseOut(); + } + // fire the mouse out event + if (series && options.events.mouseOut) { + fireEvent(series, "mouseOut"); + } + // hide the tooltip + if ( + tooltip && + !series.stickyTracking && + (!tooltip.shared || series.noSharedTooltip) + ) { + tooltip.hide(); + } + // Reset all inactive states + chart.series.forEach(function (s) { + s.setState("", true); + }); + }, + /** + * Set the state of the series. Called internally on mouse interaction + * operations, but it can also be called directly to visually + * highlight a series. + * + * @function Highcharts.Series#setState + * + * @param {Highcharts.SeriesStateValue|""} [state] + * The new state, can be either `'hover'`, `'inactive'`, `'select'`, + * or `''` (an empty string), `'normal'` or `undefined` to set to + * normal state. + * @param {boolean} [inherit] + * Determines if state should be inherited by points too. + */ + setState: function (state, inherit) { + var series = this, + options = series.options, + graph = series.graph, + inactiveOtherPoints = options.inactiveOtherPoints, + stateOptions = options.states, + lineWidth = options.lineWidth, + opacity = options.opacity, + // By default a quick animation to hover/inactive, + // slower to un-hover + stateAnimation = pick( + stateOptions[state || "normal"] && + stateOptions[state || "normal"].animation, + series.chart.options.chart.animation + ), + attribs, + i = 0; + state = state || ""; + if (series.state !== state) { + // Toggle class names + [ + series.group, + series.markerGroup, + series.dataLabelsGroup, + ].forEach(function (group) { + if (group) { + // Old state + if (series.state) { + group.removeClass("highcharts-series-" + series.state); + } + // New state + if (state) { + group.addClass("highcharts-series-" + state); + } + } + }); + series.state = state; + if (!series.chart.styledMode) { + if ( + stateOptions[state] && + stateOptions[state].enabled === false + ) { + return; + } + if (state) { + lineWidth = + stateOptions[state].lineWidth || + lineWidth + (stateOptions[state].lineWidthPlus || 0); // #4035 + opacity = pick(stateOptions[state].opacity, opacity); + } + if (graph && !graph.dashstyle) { + attribs = { + "stroke-width": lineWidth, + }; + // Animate the graph stroke-width. + graph.animate(attribs, stateAnimation); + while (series["zone-graph-" + i]) { + series["zone-graph-" + i].attr(attribs); + i = i + 1; + } + } + // For some types (pie, networkgraph, sankey) opacity is + // resolved on a point level + if (!inactiveOtherPoints) { + [ + series.group, + series.markerGroup, + series.dataLabelsGroup, + series.labelBySeries, + ].forEach(function (group) { + if (group) { + group.animate( + { + opacity: opacity, + }, + stateAnimation + ); + } + }); + } + } + } + // Don't loop over points on a series that doesn't apply inactive state + // to siblings markers (e.g. line, column) + if (inherit && inactiveOtherPoints && series.points) { + series.setAllPointsToState(state); + } + }, + /** + * Set the state for all points in the series. + * + * @function Highcharts.Series#setAllPointsToState + * + * @private + * + * @param {string} [state] + * Can be either `hover` or undefined to set to normal state. + */ + setAllPointsToState: function (state) { + this.points.forEach(function (point) { + if (point.setState) { + point.setState(state); + } + }); + }, + /** + * Show or hide the series. + * + * @function Highcharts.Series#setVisible + * + * @param {boolean} [visible] + * True to show the series, false to hide. If undefined, the visibility is + * toggled. + * + * @param {boolean} [redraw=true] + * Whether to redraw the chart after the series is altered. If doing more + * operations on the chart, it is a good idea to set redraw to false and + * call {@link Chart#redraw|chart.redraw()} after. + * + * @fires Highcharts.Series#event:hide + * @fires Highcharts.Series#event:show + */ + setVisible: function (vis, redraw) { + var series = this, + chart = series.chart, + legendItem = series.legendItem, + showOrHide, + ignoreHiddenSeries = chart.options.chart.ignoreHiddenSeries, + oldVisibility = series.visible; + // if called without an argument, toggle visibility + series.visible = + vis = + series.options.visible = + series.userOptions.visible = + typeof vis === "undefined" ? !oldVisibility : vis; // #5618 + showOrHide = vis ? "show" : "hide"; + // show or hide elements + [ + "group", + "dataLabelsGroup", + "markerGroup", + "tracker", + "tt", + ].forEach(function (key) { + if (series[key]) { + series[key][showOrHide](); + } + }); + // hide tooltip (#1361) + if ( + chart.hoverSeries === series || + (chart.hoverPoint && chart.hoverPoint.series) === series + ) { + series.onMouseOut(); + } + if (legendItem) { + chart.legend.colorizeItem(series, vis); + } + // rescale or adapt to resized chart + series.isDirty = true; + // in a stack, all other series are affected + if (series.options.stacking) { + chart.series.forEach(function (otherSeries) { + if (otherSeries.options.stacking && otherSeries.visible) { + otherSeries.isDirty = true; + } + }); + } + // show or hide linked series + series.linkedSeries.forEach(function (otherSeries) { + otherSeries.setVisible(vis, false); + }); + if (ignoreHiddenSeries) { + chart.isDirtyBox = true; + } + fireEvent(series, showOrHide); + if (redraw !== false) { + chart.redraw(); + } + }, + /** + * Show the series if hidden. + * + * @sample highcharts/members/series-hide/ + * Toggle visibility from a button + * + * @function Highcharts.Series#show + * @fires Highcharts.Series#event:show + */ + show: function () { + this.setVisible(true); + }, + /** + * Hide the series if visible. If the + * [chart.ignoreHiddenSeries](https://api.highcharts.com/highcharts/chart.ignoreHiddenSeries) + * option is true, the chart is redrawn without this series. + * + * @sample highcharts/members/series-hide/ + * Toggle visibility from a button + * + * @function Highcharts.Series#hide + * @fires Highcharts.Series#event:hide + */ + hide: function () { + this.setVisible(false); + }, + /** + * Select or unselect the series. This means its + * {@link Highcharts.Series.selected|selected} + * property is set, the checkbox in the legend is toggled and when selected, + * the series is returned by the {@link Highcharts.Chart#getSelectedSeries} + * function. + * + * @sample highcharts/members/series-select/ + * Select a series from a button + * + * @function Highcharts.Series#select + * + * @param {boolean} [selected] + * True to select the series, false to unselect. If undefined, the selection + * state is toggled. + * + * @fires Highcharts.Series#event:select + * @fires Highcharts.Series#event:unselect + */ + select: function (selected) { + var series = this; + series.selected = + selected = + this.options.selected = + typeof selected === "undefined" ? !series.selected : selected; + if (series.checkbox) { + series.checkbox.checked = selected; + } + fireEvent(series, selected ? "select" : "unselect"); + }, + /** + * @private + * @borrows Highcharts.TrackerMixin.drawTrackerGraph as Highcharts.Series#drawTracker + */ + drawTracker: TrackerMixin.drawTrackerGraph, + } + ); + } + ); + _registerModule( + _modules, + "Core/Responsive.js", + [_modules["Core/Chart/Chart.js"], _modules["Core/Utilities.js"]], + function (Chart, U) { + /* * + * + * (c) 2010-2020 Torstein Honsi + * + * License: www.highcharts.com/license + * + * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!! + * + * */ + var find = U.find, + isArray = U.isArray, + isObject = U.isObject, + merge = U.merge, + objectEach = U.objectEach, + pick = U.pick, + splat = U.splat, + uniqueKey = U.uniqueKey; + /** + * A callback function to gain complete control on when the responsive rule + * applies. + * + * @callback Highcharts.ResponsiveCallbackFunction + * + * @param {Highcharts.Chart} this + * Chart context. + * + * @return {boolean} + * Return `true` if it applies. + */ + /** + * Allows setting a set of rules to apply for different screen or chart + * sizes. Each rule specifies additional chart options. + * + * @sample {highstock} stock/demo/responsive/ + * Stock chart + * @sample highcharts/responsive/axis/ + * Axis + * @sample highcharts/responsive/legend/ + * Legend + * @sample highcharts/responsive/classname/ + * Class name + * + * @since 5.0.0 + * @apioption responsive + */ + /** + * A set of rules for responsive settings. The rules are executed from + * the top down. + * + * @sample {highcharts} highcharts/responsive/axis/ + * Axis changes + * @sample {highstock} highcharts/responsive/axis/ + * Axis changes + * @sample {highmaps} highcharts/responsive/axis/ + * Axis changes + * + * @type {Array<*>} + * @since 5.0.0 + * @apioption responsive.rules + */ + /** + * A full set of chart options to apply as overrides to the general + * chart options. The chart options are applied when the given rule + * is active. + * + * A special case is configuration objects that take arrays, for example + * [xAxis](#xAxis), [yAxis](#yAxis) or [series](#series). For these + * collections, an `id` option is used to map the new option set to + * an existing object. If an existing object of the same id is not found, + * the item of the same indexupdated. So for example, setting `chartOptions` + * with two series items without an `id`, will cause the existing chart's + * two series to be updated with respective options. + * + * @sample {highstock} stock/demo/responsive/ + * Stock chart + * @sample highcharts/responsive/axis/ + * Axis + * @sample highcharts/responsive/legend/ + * Legend + * @sample highcharts/responsive/classname/ + * Class name + * + * @type {Highcharts.Options} + * @since 5.0.0 + * @apioption responsive.rules.chartOptions + */ + /** + * Under which conditions the rule applies. + * + * @since 5.0.0 + * @apioption responsive.rules.condition + */ + /** + * A callback function to gain complete control on when the responsive + * rule applies. Return `true` if it applies. This opens for checking + * against other metrics than the chart size, for example the document + * size or other elements. + * + * @type {Highcharts.ResponsiveCallbackFunction} + * @since 5.0.0 + * @context Highcharts.Chart + * @apioption responsive.rules.condition.callback + */ + /** + * The responsive rule applies if the chart height is less than this. + * + * @type {number} + * @since 5.0.0 + * @apioption responsive.rules.condition.maxHeight + */ + /** + * The responsive rule applies if the chart width is less than this. + * + * @sample highcharts/responsive/axis/ + * Max width is 500 + * + * @type {number} + * @since 5.0.0 + * @apioption responsive.rules.condition.maxWidth + */ + /** + * The responsive rule applies if the chart height is greater than this. + * + * @type {number} + * @default 0 + * @since 5.0.0 + * @apioption responsive.rules.condition.minHeight + */ + /** + * The responsive rule applies if the chart width is greater than this. + * + * @type {number} + * @default 0 + * @since 5.0.0 + * @apioption responsive.rules.condition.minWidth + */ + /* eslint-disable no-invalid-this, valid-jsdoc */ + /** + * Update the chart based on the current chart/document size and options for + * responsiveness. + * + * @private + * @function Highcharts.Chart#setResponsive + * @param {boolean} [redraw=true] + * @param {boolean} [reset=false] + * Reset by un-applying all rules. Chart.update resets all rules before applying + * updated options. + */ + Chart.prototype.setResponsive = function (redraw, reset) { + var options = this.options.responsive, + ruleIds = [], + currentResponsive = this.currentResponsive, + currentRuleIds, + undoOptions; + if (!reset && options && options.rules) { + options.rules.forEach(function (rule) { + if (typeof rule._id === "undefined") { + rule._id = uniqueKey(); + } + this.matchResponsiveRule(rule, ruleIds /* , redraw */); + }, this); + } + // Merge matching rules + var mergedOptions = merge.apply( + 0, + ruleIds.map(function (ruleId) { + return find(options.rules, function (rule) { + return rule._id === ruleId; + }).chartOptions; + }) + ); + mergedOptions.isResponsiveOptions = true; + // Stringified key for the rules that currently apply. + ruleIds = ruleIds.toString() || void 0; + currentRuleIds = currentResponsive && currentResponsive.ruleIds; + // Changes in what rules apply + if (ruleIds !== currentRuleIds) { + // Undo previous rules. Before we apply a new set of rules, we need to + // roll back completely to base options (#6291). + if (currentResponsive) { + this.update(currentResponsive.undoOptions, redraw, true); + } + if (ruleIds) { + // Get undo-options for matching rules + undoOptions = this.currentOptions(mergedOptions); + undoOptions.isResponsiveOptions = true; + this.currentResponsive = { + ruleIds: ruleIds, + mergedOptions: mergedOptions, + undoOptions: undoOptions, + }; + this.update(mergedOptions, redraw, true); + } else { + this.currentResponsive = void 0; + } + } + }; + /** + * Handle a single responsiveness rule. + * + * @private + * @function Highcharts.Chart#matchResponsiveRule + * @param {Highcharts.ResponsiveRulesOptions} rule + * @param {Array<string>} matches + */ + Chart.prototype.matchResponsiveRule = function (rule, matches) { + var condition = rule.condition, + fn = + condition.callback || + function () { + return ( + this.chartWidth <= pick(condition.maxWidth, Number.MAX_VALUE) && + this.chartHeight <= + pick(condition.maxHeight, Number.MAX_VALUE) && + this.chartWidth >= pick(condition.minWidth, 0) && + this.chartHeight >= pick(condition.minHeight, 0) + ); + }; + if (fn.call(this)) { + matches.push(rule._id); + } + }; + /** + * Get the current values for a given set of options. Used before we update + * the chart with a new responsiveness rule. + * + * @todo Restore axis options (by id?). The matching of items in collections + * bears resemblance to the oneToOne matching in Chart.update. Probably we can + * refactor out that matching and reuse it in both functions. + * + * @private + * @function Highcharts.Chart#currentOptions + * @param {Highcharts.Options} options + * @return {Highcharts.Options} + */ + Chart.prototype.currentOptions = function (options) { + var chart = this, + ret = {}; + /** + * Recurse over a set of options and its current values, + * and store the current values in the ret object. + */ + function getCurrent(options, curr, ret, depth) { + var i; + objectEach(options, function (val, key) { + if (!depth && chart.collectionsWithUpdate.indexOf(key) > -1) { + val = splat(val); + ret[key] = []; + // Iterate over collections like series, xAxis or yAxis and map + // the items by index. + for (i = 0; i < Math.max(val.length, curr[key].length); i++) { + // Item exists in current data (#6347) + if (curr[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 (val[i] === void 0) { + ret[key][i] = curr[key][i]; + // Otherwise, proceed + } else { + ret[key][i] = {}; + getCurrent(val[i], curr[key][i], ret[key][i], depth + 1); + } + } + } + } else if (isObject(val)) { + ret[key] = isArray(val) ? [] : {}; + getCurrent(val, curr[key] || {}, ret[key], depth + 1); + } else if (typeof curr[key] === "undefined") { + // #10286 + ret[key] = null; + } else { + ret[key] = curr[key]; + } + }); + } + getCurrent(options, this.options, ret, 0); + return ret; + }; + } + ); + _registerModule( + _modules, + "masters/highcharts.src.js", + [_modules["Core/Globals.js"]], + function (Highcharts) { + return Highcharts; + } + ); + _modules["masters/highcharts.src.js"]._modules = _modules; + return _modules["masters/highcharts.src.js"]; +}); diff --git a/notemyprogress/lang/en/local_notemyprogress.php b/notemyprogress/lang/en/local_notemyprogress.php index 23a84b33ac66017a05d991c771b69f267c44262e..fb06870c7d449aa95a06139b3c3ac401cb847132 100644 --- a/notemyprogress/lang/en/local_notemyprogress.php +++ b/notemyprogress/lang/en/local_notemyprogress.php @@ -715,9 +715,10 @@ $string['fml_settings_bddname_description'] = 'This parameter designates the nam $string['fml_settings_bddname_default'] = 'Empty'; /* Gamification */ -$string['studentRanking1']="¿Quieres aparecer en el ranking ? "; -$string['studentRanking2']="/!\ Te verán todos los inscritos en este curso que hayan aceptado. Aceptar es definitivo: "; -$string['studentRanking3']="Acceptar"; +$string['EnableGame']="Disable/Enable :"; +$string['studentRanking1']="Do you want to appear in the Ranking Board ?"; +$string['studentRanking2']=" /!\ All the registered in this course having accepted you will see. Accepting is final:"; +$string['studentRanking3']="Accept"; $string['gamification_denied'] = 'gamification has been desactivated by your teacher'; $string['tg_colon'] = '{$a->a}: {$a->b}'; $string['tg_section_title'] = 'Course gamification settings'; @@ -751,7 +752,7 @@ $string['tg_section_settings_appearance_title'] = 'Title'; $string['tg_section_settings_levels'] = 'Level settings'; $string['tg_section_settings_levels_quantity'] = 'Levels'; $string['tg_section_settings_levels_base_points'] = 'Base Points'; -$string['tg_section_settings_levels_calculated'] = 'Calculated'; +$string['tg_section_settings_levels_calculated'] = 'Automatic'; $string['tg_section_settings_levels_manually'] = 'Manually'; $string['tg_section_settings_levels_name'] = 'Name'; $string['tg_section_settings_levels_required'] = 'Points required'; @@ -759,6 +760,8 @@ $string['tg_section_settings_rules'] = 'Rules setting'; $string['tg_section_settings_add_rule'] = 'Add new rule'; $string['tg_section_settings_earn'] = 'For this event win:'; $string['tg_section_settings_select_event'] = 'Select an event'; +$string['gm_Chart_Title']='Spreading chart'; +$string['gm_Chart_Y']='Students'; $string['tg_timenow'] = 'Just now'; $string['tg_timeseconds'] = '{$a}s ago'; @@ -778,3 +781,9 @@ $string['fml_api_invalid_transaction'] = 'The request is incorrect'; $string['fml_api_invalid_profile'] = 'You cannot do this action, your profile is incorrect'; $string['fml_api_save_successful'] = 'The data has been successfully saved on the server'; $string['fml_api_cancel_action'] = 'You have canceled the action'; +$string['overview']='Overview : '; +$string['game_point_error']='Points are required'; +$string['game_event_error']='Event is required'; +$string['game_name_error']='Name required'; + + diff --git a/notemyprogress/lang/es/local_notemyprogress.php b/notemyprogress/lang/es/local_notemyprogress.php index 48b7d7c101a7f47f52512271899e26eb4334bca8..2269b8bc9dfe1ef99949cd7f8648d6a82fac248e 100644 --- a/notemyprogress/lang/es/local_notemyprogress.php +++ b/notemyprogress/lang/es/local_notemyprogress.php @@ -715,68 +715,73 @@ $string['fml_settings_bddname_label'] = 'Nombre de la base de datos'; $string['fml_settings_bddname_description'] = 'Este parámetro designa el nombre de la base de datos MongoDB en la que se guardará la información.'; $string['fml_settings_bddname_default'] = 'Vacío'; /* Gamification */ -$string['studentRanking1']="Do you want to appear in the Ranking Board ?"; -$string['studentRanking2']=" /!\ All the registered in this course having accepted you will see. Accepting is final:"; -$string['studentRanking3']="Accept"; -$string['gamification_denied'] = 'gamification has been desactivated by your teacher'; +$string['EnableGame']="Desactivar/Activar :"; +$string['studentRanking1']="¿Quieres aparecer en el ranking ? "; +$string['studentRanking2']="/!\ Te verán todos los inscritos en este curso que hayan aceptado. Aceptar es definitivo: "; +$string['studentRanking3']="Acceptar"; +$string['gamification_denied'] = 'La gamificación ha sido desactivada por su maestro'; $string['tg_colon'] = '{$a->a}: {$a->b}'; -$string['tg_section_title'] = 'Course gamification settings'; -$string['tg_section_help_title'] = 'Course gamification settings'; -$string['tg_section_help_description'] = 'You can give the option of gamification to the students of the course.'; -$string['tg_save_warning_title'] = "Are you sure you want to save the changes?"; -$string['tg_save_warning_content'] = "If you change the settings for the gamification when the course has already started, data may be lost..."; -$string['tg_confirm_ok'] = "Save"; -$string['tg_confirm_cancel'] = "Cancel"; -$string['tg_save'] = "Save configuration"; -$string['tg_section_preview'] = 'Preview'; -$string['tg_section_experience'] = 'Experience points'; -$string['tg_section_information'] = 'Information'; -$string['tg_section_ranking'] = 'Ranking/Report'; -$string['tg_section_settings'] = 'Settings'; - -$string['tg_section_points'] = 'points'; -$string['tg_section_description'] = 'Description'; -$string['tg_section_no_description'] = 'No description'; -$string['tg_section_no_name'] = 'No name'; - -$string['tg_section_preview_next_level'] = 'to the next level'; -$string['tg_section_ranking_ranking_text'] = 'Ranking'; -$string['tg_section_ranking_level'] = 'Level'; -$string['tg_section_ranking_student'] = 'Student'; +$string['tg_section_title'] = 'Configuración de gamificación del curso'; +$string['tg_section_help_title'] = 'Configuración de gamificación del curso'; +$string['tg_section_help_description'] = 'Puede dar la opción de gamificación a los estudiantes del curso.'; +$string['tg_save_warning_title'] = "¿Estás seguro de que quieres guardar los cambios?"; +$string['tg_save_warning_content'] = "Si cambia la configuración de la gamificación cuando el curso ya ha comenzado, los datos pueden perderse ..."; +$string['tg_confirm_ok'] = "Guardar"; +$string['tg_confirm_cancel'] = "Cancelar"; +$string['tg_save'] = "Guardar configuración"; +$string['tg_section_preview'] = 'Avance'; +$string['tg_section_experience'] = 'Puntos de experiencia'; +$string['tg_section_information'] = 'Información'; +$string['tg_section_ranking'] = 'Clasificación/informe'; +$string['tg_section_settings'] = 'Ajustes'; + +$string['tg_section_points'] = 'puntos'; +$string['tg_section_description'] = 'Descripción'; +$string['tg_section_no_description'] = 'Sin descripción'; +$string['tg_section_no_name'] = 'Sin nombre'; + +$string['tg_section_preview_next_level'] = 'al siguiente nivel'; +$string['tg_section_ranking_ranking_text'] = 'Clasificación'; +$string['tg_section_ranking_level'] = 'Nivel'; +$string['tg_section_ranking_student'] = 'Alumna'; $string['tg_section_ranking_total'] = 'Total'; -$string['tg_section_ranking_progress'] = 'Progress %'; - -$string['tg_section_settings_appearance'] = 'Appearance'; -$string['tg_section_settings_appearance_title'] = 'Title'; -$string['tg_section_settings_levels'] = 'Level settings'; -$string['tg_section_settings_levels_quantity'] = 'Levels'; -$string['tg_section_settings_levels_base_points'] = 'Base Points'; -$string['tg_section_settings_levels_calculated'] = 'Calculated'; -$string['tg_section_settings_levels_manually'] = 'Manually'; -$string['tg_section_settings_levels_name'] = 'Name'; -$string['tg_section_settings_levels_required'] = 'Points required'; -$string['tg_section_settings_rules'] = 'Rules setting'; -$string['tg_section_settings_add_rule'] = 'Add new rule'; -$string['tg_section_settings_earn'] = 'For this event win:'; -$string['tg_section_settings_select_event'] = 'Select an event'; - -$string['tg_timenow'] = 'Just now'; -$string['tg_timeseconds'] = '{$a}s ago'; -$string['tg_timeminutes'] = '{$a}m ago'; -$string['tg_timehours'] = '{$a}h ago'; -$string['tg_timedays'] = '{$a}d ago'; -$string['tg_timeweeks'] = '{$a}w ago'; +$string['tg_section_ranking_progress'] = 'Progreso %'; + +$string['tg_section_settings_appearance'] = 'Apariencia'; +$string['tg_section_settings_appearance_title'] = 'Título'; +$string['tg_section_settings_levels'] = 'Configuración de nivel'; +$string['tg_section_settings_levels_quantity'] = 'Nivel'; +$string['tg_section_settings_levels_base_points'] = 'Puntos base'; +$string['tg_section_settings_levels_calculated'] = 'automático'; +$string['tg_section_settings_levels_manually'] = 'A mano'; +$string['tg_section_settings_levels_name'] = 'Nombre'; +$string['tg_section_settings_levels_required'] = 'Se requieren puntos'; +$string['tg_section_settings_rules'] = 'Establecimiento de reglas'; +$string['tg_section_settings_add_rule'] = 'Agregar nueva regla'; +$string['tg_section_settings_earn'] = 'Para este evento, gane:'; +$string['tg_section_settings_select_event'] = 'Seleccione un evento'; +$string['gm_Chart_Title']='Tabla de esparcimiento'; +$string['gm_Chart_Y']='Alumno'; + +$string['tg_timenow'] = 'Justo ahora'; +$string['tg_timeseconds'] = '{$a}s atrás'; +$string['tg_timeminutes'] = '{$a}m atrás'; +$string['tg_timehours'] = '{$a}h atrás'; +$string['tg_timedays'] = '{$a}d atrás'; +$string['tg_timeweeks'] = '{$a}w atrás'; $string['tg_timewithinayearformat'] = '%b %e'; $string['tg_timeolderyearformat'] = '%b %Y'; /* General Errors */ -$string['fml_api_error_network'] = "An error has occurred in communication with the server."; -$string['fml_api_invalid_data'] = 'Incorrect data'; -$string['fml_api_json_decode_error'] = 'Error Decoding sent data'; -$string['fml_api_invalid_token_error'] = 'The token sent in the transaction is invalid, please refresh the page'; -$string['fml_api_invalid_transaction'] = 'The request is incorrect'; -$string['fml_api_invalid_profile'] = 'You cannot do this action, your profile is incorrect'; -$string['fml_api_save_successful'] = 'The data has been successfully saved on the server'; -$string['fml_api_cancel_action'] = 'You have canceled the action'; - - +$string['fml_api_error_network'] = "Se ha producido un error en la comunicación con el servidor."; +$string['fml_api_invalid_data'] = 'Datos Incorrectos'; +$string['fml_api_json_decode_error'] = 'Error en decodificación de datos enviados'; +$string['fml_api_invalid_token_error'] = 'El token enviado en la transacción no es válido, actualice la página'; +$string['fml_api_invalid_transaction'] = 'La solicitud es incorrecta'; +$string['fml_api_invalid_profile'] = 'No puedes hacer esta acción, tu perfil es incorrecto'; +$string['fml_api_save_successful'] = 'Los datos se han guardado correctamente en el servidor'; +$string['fml_api_cancel_action'] = 'Has cancelado la acción'; +$string['overview']='Visión de conjunto : '; +$string['game_point_error']='Los puntos son requeridos'; +$string['game_event_error']='El evento es requerido'; +$string['game_name_error']='Se requiere el nombre'; diff --git a/notemyprogress/lang/fr/local_notemyprogress.php b/notemyprogress/lang/fr/local_notemyprogress.php index 1e23153d967686d30b4bfac08934d68f79628535..e1430c57874519e1e1d4624de6789f5d4fd860c1 100644 --- a/notemyprogress/lang/fr/local_notemyprogress.php +++ b/notemyprogress/lang/fr/local_notemyprogress.php @@ -720,66 +720,73 @@ $string['fml_settings_bddname_label'] = 'Nom de la base de données'; $string['fml_settings_bddname_description'] = 'Ce paramètre désigne le nom de la base de données MongoDB dans laquelle vont être enregistrées les informations.'; /* Gamification */ +$string['EnableGame']="Désactiver/Activer :"; $string['studentRanking1']="Voulez vous apparaitre dans le ranking board ? "; -$string['studentRanking2']= "/!\ tout les inscrits à ce cours ayant accepté vous verrons) accepter est définitif:"; +$string['studentRanking2']= "/!\ Tout les inscrits à ce cours ayant accepté vous verrons. Accepter est définitif:"; $string['studentRanking3']="Accepter"; -$string['gamification_denied'] = 'gamification has been desactivated by your teacher'; +$string['gamification_denied'] = 'La gamification a été désactivée par votre professeur'; $string['tg_colon'] = '{$a->a}: {$a->b}'; -$string['tg_section_title'] = 'Course gamification settings'; -$string['tg_section_help_title'] = 'Course gamification settings'; -$string['tg_section_help_description'] = 'You can give the option of gamification to the students of the course.'; -$string['tg_save_warning_title'] = "Are you sure you want to save the changes?"; -$string['tg_save_warning_content'] = "If you change the settings for the gamification when the course has already started, data may be lost..."; -$string['tg_confirm_ok'] = "Save"; -$string['tg_confirm_cancel'] = "Cancel"; -$string['tg_save'] = "Save configuration"; +$string['tg_section_title'] = 'Gamification :'; +$string['tg_section_help_title'] = 'Paramètres de gamification des cours'; +$string['tg_section_help_description'] = 'Vous pouvez donner la possibilité de gamification aux étudiants du cours.'; +$string['tg_save_warning_title'] = "Êtes-vous sûr de vouloir enregistrer les modifications?"; +$string['tg_save_warning_content'] = "Si vous modifiez les paramètres de la gamification au début du cours, les données peuvent être perdues ..."; +$string['tg_confirm_ok'] = "Sauver"; +$string['tg_confirm_cancel'] = "Annuler"; +$string['tg_save'] = "Enregistrer la configuration"; $string['tg_section_preview'] = 'Point de vue étudiant'; -$string['tg_section_experience'] = 'Experience points'; +$string['tg_section_experience'] = 'Points d\'experience'; $string['tg_section_information'] = 'Information'; -$string['tg_section_ranking'] = 'Ranking/Report'; -$string['tg_section_settings'] = 'Settings'; +$string['tg_section_ranking'] = 'Classement / rapport'; +$string['tg_section_settings'] = 'Réglages'; $string['tg_section_points'] = 'points'; $string['tg_section_description'] = 'Description'; -$string['tg_section_no_description'] = 'No description'; -$string['tg_section_no_name'] = 'No name'; +$string['tg_section_no_description'] = 'Pas de description'; +$string['tg_section_no_name'] = 'Sans nom'; -$string['tg_section_preview_next_level'] = 'to the next level'; -$string['tg_section_ranking_ranking_text'] = 'Ranking'; -$string['tg_section_ranking_level'] = 'Level'; -$string['tg_section_ranking_student'] = 'Student'; +$string['tg_section_preview_next_level'] = 'au niveau supérieur'; +$string['tg_section_ranking_ranking_text'] = 'Classement'; +$string['tg_section_ranking_level'] = 'Niveau'; +$string['tg_section_ranking_student'] = 'Élève'; $string['tg_section_ranking_total'] = 'Total'; -$string['tg_section_ranking_progress'] = 'Progress %'; - -$string['tg_section_settings_appearance'] = 'Appearance'; -$string['tg_section_settings_appearance_title'] = 'Title'; -$string['tg_section_settings_levels'] = 'Level settings'; -$string['tg_section_settings_levels_quantity'] = 'Levels'; -$string['tg_section_settings_levels_base_points'] = 'Base Points'; -$string['tg_section_settings_levels_calculated'] = 'Calculated'; -$string['tg_section_settings_levels_manually'] = 'Manually'; -$string['tg_section_settings_levels_name'] = 'Name'; -$string['tg_section_settings_levels_required'] = 'Points required'; -$string['tg_section_settings_rules'] = 'Rules setting'; -$string['tg_section_settings_add_rule'] = 'Add new rule'; -$string['tg_section_settings_earn'] = 'For this event win:'; -$string['tg_section_settings_select_event'] = 'Select an event'; - -$string['tg_timenow'] = 'Just now'; -$string['tg_timeseconds'] = '{$a}s ago'; -$string['tg_timeminutes'] = '{$a}m ago'; -$string['tg_timehours'] = '{$a}h ago'; -$string['tg_timedays'] = '{$a}d ago'; -$string['tg_timeweeks'] = '{$a}w ago'; +$string['tg_section_ranking_progress'] ='Progréssion %'; + +$string['tg_section_settings_appearance'] = 'Apparence'; +$string['tg_section_settings_appearance_title'] = 'Titre'; +$string['tg_section_settings_levels'] = 'Paramètres de niveau'; +$string['tg_section_settings_levels_quantity'] = 'Niveaux'; +$string['tg_section_settings_levels_base_points'] = 'Points de base'; +$string['tg_section_settings_levels_calculated'] = 'Automatique'; +$string['tg_section_settings_levels_manually'] = 'Manuellement'; +$string['tg_section_settings_levels_name'] = 'Nom'; +$string['tg_section_settings_levels_required'] = 'Points requis'; +$string['tg_section_settings_rules'] = 'Liste des régles'; +$string['tg_section_settings_add_rule'] = 'Nouvelle règle'; +$string['tg_section_settings_earn'] = 'Pour cet événement, victoire:'; +$string['tg_section_settings_select_event'] = 'Sélectionnez un événement'; +$string['gm_Chart_Title']='Graphique de répartion'; +$string['gm_Chart_Y']='Etudiants'; + +$string['tg_timenow'] = 'Juste maintenant'; +$string['tg_timeseconds'] = 'il y a {$a}s '; +$string['tg_timeminutes'] = 'il y a {$a}m '; +$string['tg_timehours'] = 'il y a {$a}h '; +$string['tg_timedays'] = 'il y a {$a}d '; +$string['tg_timeweeks'] = 'il y a {$a}w '; $string['tg_timewithinayearformat'] = '%b %e'; $string['tg_timeolderyearformat'] = '%b %Y'; /* General Errors */ -$string['fml_api_error_network'] = "An error has occurred in communication with the server."; -$string['fml_api_invalid_data'] = 'Incorrect data'; -$string['fml_api_json_decode_error'] = 'Error Decoding sent data'; -$string['fml_api_invalid_token_error'] = 'The token sent in the transaction is invalid, please refresh the page'; -$string['fml_api_invalid_transaction'] = 'The request is incorrect'; -$string['fml_api_invalid_profile'] = 'You cannot do this action, your profile is incorrect'; -$string['fml_api_save_successful'] = 'The data has been successfully saved on the server'; -$string['fml_api_cancel_action'] = 'You have canceled the action'; +$string['fml_api_error_network'] = "Une erreur s'est produite dans la communication avec le serveur."; +$string['fml_api_invalid_data'] = 'Données incorrectes'; +$string['fml_api_json_decode_error'] = 'Décodage d erreur de données envoyées'; +$string['fml_api_invalid_token_error'] = 'Le jeton envoyé dans la transaction n\'est pas valide, veuillez actualiser la page'; +$string['fml_api_invalid_transaction'] = 'La demande est incorrecte'; +$string['fml_api_invalid_profile'] = 'Vous ne pouvez pas faire cette action, votre profil est incorrect'; +$string['fml_api_save_successful'] = 'Les données ont été enregistrées avec succès sur le serveur'; +$string['fml_api_cancel_action'] = 'Vous avez annulé l\'action'; +$string['overview']='Vue d\'ensemble : '; +$string['game_point_error']='Veuillez saisir les points'; +$string['game_event_error']='Veuillez saisir l\'évènement'; +$string['game_name_error']='Le nom est requis'; \ No newline at end of file diff --git a/notemyprogress/locallib.php b/notemyprogress/locallib.php index d329f48140b9d09945b09b2d69c1809992c11c43..5e207f8df4b71f357b68219ecf43d7d58bfbaeea 100644 --- a/notemyprogress/locallib.php +++ b/notemyprogress/locallib.php @@ -185,16 +185,18 @@ function local_notemyprogress_validate_token($token){ local_notemyprogress_ajax_response(null, $message, false, 403); } function local_notemyprogress_save_gamification_config($courseid, $userid, $rules, $levels, $settings, $url,$enable){ - // \local_notemyprogress\logs::create( - // "setgamification", - // "configweeks", - // "saved", - // "weeks_settings", - // $url, - // 4, - // $userid, - // $courseid - // ); + // \local_notemyprogress\logs::create( + // "setgamification", + // "configweeks", + // "saved", + // "weeks_settings", + // $url, + // 4, + // $userid, + // $courseid + // ); + $logs = new \local_notemyprogress\logs($courseid, $userid); + $logs->addLogsNMP("Saved", "section", "CONFIGURATION_GAMIFICATION", "configuration_gamification", $url, "GamificationSaved"); $configLevels = new \local_notemyprogress\configgamification($courseid, $userid); $configLevels->save_levels($levels, $settings, $rules); diff --git a/notemyprogress/pix/rankImage.jpg b/notemyprogress/pix/rankImage.jpg new file mode 100644 index 0000000000000000000000000000000000000000..d76465c8405a7cbe5825e39e070d0709e5d4b493 Binary files /dev/null and b/notemyprogress/pix/rankImage.jpg differ diff --git a/notemyprogress/student_gamification.php b/notemyprogress/student_gamification.php index 78307ad3d2e2f7bdc6a85b7d193a22719659115a..0ebb905fc2ae58fef0871d6bb6171302f16129ea 100644 --- a/notemyprogress/student_gamification.php +++ b/notemyprogress/student_gamification.php @@ -75,15 +75,17 @@ $content = [ 'studentRanking1'=>get_string("studentRanking1","local_notemyprogress"), 'studentRanking2'=>get_string("studentRanking2","local_notemyprogress"), 'studentRanking3'=>get_string("studentRanking3","local_notemyprogress"), + 'overview'=>get_string("overview","local_notemyprogress"), ], 'levels_data' => $configgamification->get_levels_data(), 'indicators' => $es->get_user_info(), - 'ranking' => $es->get_ranking(), + 'ranking' => $es->get_ranking(1), ]; $templatecontext = [ - 'image' => $OUTPUT->image_url('badge', 'local_notemyprogress') + 'image' => $OUTPUT->image_url('badge', 'local_notemyprogress'), + 'rankImage' => $OUTPUT->image_url('rankImage', 'local_notemyprogress') ]; $maxTimeSql = "SELECT MAX(timecreated) as maximum from {notemyprogress_gamification} where courseid=? "; diff --git a/notemyprogress/templates/gamification.mustache b/notemyprogress/templates/gamification.mustache index 0b750072fdf2c086949f029245db1528e26264b1..c40a8ac215b5265c43df7a50cc3a46dba30f9185 100644 --- a/notemyprogress/templates/gamification.mustache +++ b/notemyprogress/templates/gamification.mustache @@ -5,250 +5,253 @@ :exitbutton="strings.exitbutton" :helpcontents="get_help_content()" :token="token" - @open_help_section_modal="openHelpSectionModalEvent" - > + @open_help_section_modal="openHelpSectionModalEvent"> </pageheader> <template> - <!-- Server message --> - <v-row class="justify-center" v-if="notifications.length > 0"> - <v-col cols="8"> - <v-alert v-for="(value, index, key) in notifications" :key="key" - :type="value.type" border="left" class="text-center" dense text dismissible> - <span v-text="value.message"></span> - </v-alert> - </v-col> - </v-row> - <v-row> - <v-card - class="align-center text-center flex" - height="100" - > - <h4>Disable / Enable : <h4> - <v-switch - :input-value="false" - v-model ="swDisableEnable" - inset - color="red darken-3" - class="pt-800 px-800 justify-center" - @Change = disableEnable(swDisableEnable); - ></v-switch> - </v-card> - <v-row> - <v-card - width = '100%'> - <v-row> - <v-col> - <v-tabs v-model="tab" background-color="transparent"> - <v-tab v-text="strings.settings_level"></v-tab> - <v-tab v-text="strings.preview"></v-tab> - <v-tab v-text="strings.settings"></v-tab> - </v-tabs> - <v-tabs-items v-model="tab"> - <v-tab-item> <!-- Level and settings --> - <v-row class="pt-8 px-8 justify-center"> - <v-row> - <v-col cols="6" sm="4" md="3" lg="5"> - <v-text-field - type="number" - readonly - :label="strings.quantity" - v-model="levelsData.length" - append-outer-icon="mdi-plus-circle" - prepend-icon="mdi-minus-circle" - @click:append-outer="addLevel()" - @click:prepend="removeLevel()" - ></v-text-field> - </v-col> - <v-col cols="6" sm="4" md="3" lg="2"> - <v-text-field :label="strings.base_points" v-model="pointsBase" type="number"> - </v-text-field> - </v-col> - <v-col cols="12" sm="4" md="4" lg="3"> - <v-btn-toggle v-model="setPointsOption" tile group> - <v-btn value="calculated" v-text="strings.calculated"></v-btn> - <v-btn value="manually" v-text="strings.manually"></v-btn> - </v-btn-toggle> - </v-col> - </v-row> - </v-row> - <v-row class="pb-5 px-8"> - <v-row v-for="(level, index) in levelsData" :key="level.lvl" cols="12" sm="6" md="4" lg="3"> - <v-card class="pt-5 align-center text-center"> - <v-col :content="level.lvl" bottom offset-x="20" offset-y="20"> - <v-avatar size="100"> - <v-img src="{{{image}}}"></v-img> - </v-avatar> - </v-col> - <v-card-text class="text--primary text-center"> - <v-text-field :label="strings.name" :rules="[v => !!v || 'Name is required']" required v-model="level.nam" type="text"> - </v-text-field> - <v-text-field :label="strings.description" v-model="level.des" type="text"> - </v-text-field> - <v-text-field :disabled="setPointsOption === 'calculated'" :label="strings.levels_required" v-model="calculatePoints(index)" type="number"> - </v-text-field> - </v-card-text> - </v-card> + <v-container pa-8> + <!-- Server message --> + <v-row class="justify-center" v-if="notifications.length > 0"> + <v-col cols="8"> + <v-alert v-for="(value, index, key) in notifications" + :type="value.type" + class="text-center" dense text dismissible> + <span v-text="value.message"></span> + </v-alert> + </v-col> + </v-row> + <v-card + class="align-center text-center flex" + height="100" + elevation ="1"> + <h4 v-text="strings.enable"><h4> + <v-switch + v-model ="swDisableEnable" + inset + color="red darken-3" + class="pt-800 px-800 justify-center" + @Change = disableEnable(swDisableEnable); + ></v-switch> + </v-card> + <v-row> + <v-col> + <v-tabs v-model="tab" background-color="transparent"> + <v-tab v-text="strings.settings_level"></v-tab> + <v-tab v-text="strings.preview"></v-tab> + <v-tab v-text="strings.settings"></v-tab> + </v-tabs> + <v-tabs-items v-model="tab"> + <v-tab-item> <!-- Level and settings --> + <v-row class="pt-8 px-8 justify-center"> + <v-row> + <v-col cols="6" sm="4" md="3" lg="5"> + <v-text-field + type="number" + readonly + :label="strings.quantity" + v-model="levelsData.length" + append-outer-icon="mdi-plus-circle" + prepend-icon="mdi-minus-circle" + @click:append-outer="addLevel()" + @click:prepend="removeLevel()" + ></v-text-field> + </v-col> + <v-col cols="6" sm="4" md="3" lg="2"> + <v-text-field :label="strings.base_points" v-model="pointsBase" type="number"> + </v-text-field> + </v-col> + <v-col cols="12" sm="4" md="4" lg="3"> + <v-btn-toggle v-model="setPointsOption" tile group> + <v-btn value="calculated" v-text="strings.calculated"></v-btn> + <v-btn value="manually" v-text="strings.manually"></v-btn> + </v-btn-toggle> + </v-col> + </v-row> + </v-row> + <v-row class="pb-5 px-8"> + <v-row v-for="(level, index) in levelsData" cols="12" sm="6" md="4" lg="3"> + <v-card class="pt-5 align-center text-center"> + <v-col :content="level.lvl" bottom offset-x="20" offset-y="20"> + <v-avatar size="100"> + <v-img src="{{{image}}}"></v-img> + </v-avatar> + </v-col> + <v-card-text class="text--primary text-center"> + <v-text-field :label="strings.name" :rules="[v => !!v || strings.nameError]" required v-model="level.nam" type="text"> + </v-text-field> + <v-text-field :label="strings.description" v-model="level.des" type="text"> + </v-text-field> + <v-text-field :disabled="setPointsOption === 'calculated'" :label="strings.levels_required" v-model="calculatePoints(index)" type="number"> + </v-text-field> + </v-card-text> + </v-card> + </v-row> + </v-row> + <v-row class="pb-10 justify-center"> + <v-btn @click="save_changes('levelsEdit')" v-text="strings.save" class="white--text" color="#118AB2"></v-btn> + </v-row> + </v-tab-item> + <v-tab-item> <!-- Student preview tab --> + <v-row class="justify-center"> + <v-col cols="12" sm="11" md="7" lg="6" class="d-flex py-10"> + <v-card class="align-center text-center flex"> + <h4 class="py-5" v-text="settings.tit"></h4> + <p v-if="levelsData[0].nam" v-text="levelsData[0].nam"></p> + <v-badge + :content="levelsData[0].lvl" + bottom + offset-x="20" + offset-y="20" + > + <v-avatar size="100"> + <v-img src="{{{image}}}"></v-img> + </v-avatar> + </v-badge> + <v-card-text class="py-5"> + <small class="d-inline" v-text="levelsData[1].points"></small> + <sup>xp</sup> + <small v-text="strings.to_next_level"></small> + <v-progress-linear + color="#118AB2" + :value="0" + height="8" + > + </v-progress-linear> + <p v-if="levelsData[0].des" class="pt-5" v-text="levelsData[0].des"></p> + <h3 :class="[ !levelsData[0].des ? 'pt-5' : 'pt-0']" + v-text="levelsData[0].points +' '+strings.points"></h3> + </v-card-text> + <v-card-actions> + <p class="py-0 px-2" v-text="settings.des"></p> + </v-card-actions> + </v-card> + </v-col> + </v-row> <!-- Change the preview --> + <v-row class="pt-5 px-8 justify-center"> + <v-col cols="12" md="12"> + <h4 v-text="strings.appearance"></h4> + </v-col> + <v-col cols="12" md="12" class=""> + <v-row class="justify-center"> + <v-col cols="12" xs="10" md="6"> + <v-text-field :label="strings.appearance_title" v-model="settings.tit"></v-text-field> + </v-col> + </v-row> + <v-row class="justify-center"> + <v-col cols="12" xs="10" md="6"> + <v-textarea :label="strings.description" auto-grow v-model="settings.des"></v-textarea> + </v-col> + </v-row> + </v-col> + </v-row> + <v-row class="pb-10 justify-center"> + <v-btn @click="save_changes('studentSideEdit')" v-text="strings.save" class="white--text" color="#118AB2"></v-btn> + </v-row> + </v-tab-item> + <v-tab-item> + <v-row class="pt-5 px-8 justify-center justify-sm-space-between "> + <v-col cols="12" sm="6" md="5" lg="5"> + <h4 v-text="strings.rules"></h4> + </v-col> + <v-col cols="6" sm="5" md="4" lg="4"> + <v-btn + block v-text="strings.add_rule" + @click="addRule()" + class="white--text" + color="#118AB2"> + </v-btn> + </v-col> + </v-row> + <v-row class="pt-5 pb-5 px-8"> + <v-col v-for="(rule, index) in rulesData" cols="12" md="12"> + <v-card class="align-center text-center"> + <v-card-text class="text--primary text-center"> + <v-row class="align-center justify-space-around"> + <v-col cols="12" sm="4" md="3"> + <v-text-field + v-model="rule.points" + :label="strings.earn" + :rules="[v => !!v || strings.pointError]" + :suffix="strings.points" + type="number"> + </v-text-field> + </v-col> + <v-col cols="12" sm="6" md="7"> + <v-select + v-model="rule.rule" + attach + :menu-props="{ top: true, offsetY: true }" + :items="events" + item-text='name' + item-value='event' + :label="strings.select_event" + :rules="[v => !!v || strings.eventError]" + required> + </v-select> + </v-col> + <v-col cols="12" sm="2" md="1"> + <v-btn :disabled="rulesData.length < 3" icon @click="removeRule(index)"> + <v-icon>mdi-delete</v-icon> + </v-btn> + </v-col> + </v-row> + </v-card-text> + </v-card> + </v-col> + + </v-row> + <v-row class="pb-10 justify-center"> + <v-btn @click="save_changes('rulesEdit')" v-text="strings.save" class="white--text" color="#118AB2"></v-btn> </v-row> - </v-row> - <v-row class="pb-10 justify-center"> - <v-btn @click="save_changes" v-text="strings.save" class="white--text" color="#118AB2"></v-btn> - </v-row> - </v-tab-item> - <v-tab-item> <!-- Student preview tab --> - <v-row class="justify-center"> - <v-col cols="12" sm="11" md="7" lg="6" class="d-flex py-10"> - <v-card class="align-center text-center flex"> - <h4 class="py-5" v-text="settings.tit"></h4> - <p v-if="levelsData[0].nam" v-text="levelsData[0].nam"></p> + </v-tab-item> + </v-tabs-items> + </v-col> + + <v-col> + <v-row class="py-5 px-8"> + <v-col cols="12" md="12"> + <v-data-table + :headers="table_headers()" + :items="ranking" + item-key="id" + :items-per-page="7" + > + <template v-slot:item.level="{ item }"> <v-badge - :content="levelsData[0].lvl" - bottom - offset-x="20" - offset-y="20" + inline + bordered + :content="item.level" > - <v-avatar size="100"> - <v-img src="{{{image}}}"></v-img> + <v-avatar size="30"> + <img src="{{{image}}}"> </v-avatar> </v-badge> - <v-card-text class="py-5"> - <small class="d-inline" v-text="levelsData[1].points"></small> - <sup>xp</sup> - <small v-text="strings.to_next_level"></small> - <v-progress-linear - color="#118AB2" - :value="0" - height="8" - > - </v-progress-linear> - <p v-if="levelsData[0].des" class="pt-5" v-text="levelsData[0].des"></p> - <h3 :class="[ !levelsData[0].des ? 'pt-5' : 'pt-0']" - v-text="levelsData[0].points +' '+strings.points"></h3> - </v-card-text> - <v-card-actions> - <p class="py-0 px-2" v-text="settings.des"></p> - </v-card-actions> - </v-card> - </v-col> - </v-row> <!-- Change the preview --> - <v-row class="pt-5 px-8 justify-center"> - <v-col cols="12" md="12"> - <h4 v-text="strings.appearance"></h4> - </v-col> - <v-col cols="12" md="12" class=""> - <v-row class="justify-center"> - <v-col cols="12" xs="10" md="6"> - <v-text-field :label="strings.appearance_title" v-model="settings.tit"></v-text-field> - </v-col> - </v-row> - <v-row class="justify-center"> - <v-col cols="12" xs="10" md="6"> - <v-textarea :label="strings.description" auto-grow v-model="settings.des"></v-textarea> - </v-col> - </v-row> - </v-col> - </v-row> - <v-row class="pb-10 justify-center"> - <v-btn @click="save_changes" v-text="strings.save" class="white--text" color="#118AB2"></v-btn> - </v-row> - </v-tab-item> - <v-tab-item> - <v-row class="pt-5 px-8 justify-center justify-sm-space-between "> - <v-col cols="12" sm="6" md="5" lg="5"> - <h4 v-text="strings.rules"></h4> - </v-col> - <v-col cols="6" sm="5" md="4" lg="4"> - <v-btn - block v-text="strings.add_rule" - @click="addRule()" - class="white--text" - color="#118AB2"> - </v-btn> - </v-col> - </v-row> - <v-row class="pt-5 pb-5 px-8"> - <v-col v-for="(rule, index) in rulesData" :key="rule" cols="12" md="12"> - <v-card class="align-center text-center"> - <v-card-text class="text--primary text-center"> - <v-row class="align-center justify-space-around"> - <v-col cols="12" sm="4" md="3"> - <v-text-field - v-model="rule.points" - :label="strings.earn" - :rules="[v => !!v || 'Los puntos son requeridos']" - :suffix="strings.points" - type="number"> - </v-text-field> - </v-col> - <v-col cols="12" sm="6" md="7"> - <v-select - v-model="rule.rule" - attach - :menu-props="{ top: true, offsetY: true }" - :items="events" - item-text='name' - item-value='event' - :label="strings.select_event" - :rules="[v => !!v || 'El evento es requerido']" - required> - </v-select> - </v-col> - <v-col cols="12" sm="2" md="1"> - <v-btn :disabled="rulesData.length < 3" icon @click="removeRule(index)"> - <v-icon>mdi-delete</v-icon> - </v-btn> - </v-col> - </v-row> - </v-card-text> - </v-card> - </v-col> - - </v-row> - <v-row class="pb-10 justify-center"> - <v-btn @click="save_changes" v-text="strings.save" class="white--text" color="#118AB2"></v-btn> - </v-row> - </v-tab-item> - </v-tabs-items> - </v-col> - - <v-col> - <v-row class="py-5 px-8"> - <v-col cols="12" md="12"> - <v-data-table - :headers="table_headers()" - :items="ranking" - item-key="id" - :items-per-page="7" - > - <template v-slot:item.level="{ item }"> - <v-badge - inline - bordered - :content="item.level" - > - <v-avatar size="30"> - <img src="{{{image}}}"> - </v-avatar> - </v-badge> - </template> - <template v-slot:item.progress_percentage="{ item }"> - <v-progress-linear - color="#118AB2" - :value="item.progress_percentage" - height="8" - > - </v-progress-linear> - </template> - <template v-slot:item.total="{ item }"> - <p class="d-inline" v-text="item.total"></p> - <sup>xp</sup> - </template> - </v-data-table> - </v-col> - </v-row> - - </v-col> - </v-row> - </v-card> + </template> + <template v-slot:item.progress_percentage="{ item }"> + <v-progress-linear + color="#118AB2" + :value="item.progress_percentage" + height="8" + > + </v-progress-linear> + </template> + <template v-slot:item.total="{ item }"> + <p class="d-inline" v-text="item.total"></p> + <sup>xp</sup> + </template> + </v-data-table> + </v-col> + </v-row> + <v-row> + <chart + :container="'week_resourcess'" + :chart="chart_spread()" + :lang="strings.chart" + ></chart> + </v-row> + </v-col> + </v-row> + + </v-container> </template> </v-main> </v-app> \ No newline at end of file diff --git a/notemyprogress/templates/student_gamification.mustache b/notemyprogress/templates/student_gamification.mustache index de5c49cac4441119be99981f0cd49b4954c4260b..e464216de1cc646ed7a5d23197bbb90db37e66a2 100644 --- a/notemyprogress/templates/student_gamification.mustache +++ b/notemyprogress/templates/student_gamification.mustache @@ -10,7 +10,7 @@ <template> <v-card> <v-row> - <v-col cols=12 lg="7"> + <v-col cols=12 lg="6"> <v-row class="justify-center"> <v-col cols="12" sm="11" md="7" lg="6" class="d-flex py-10"> <v-card class="align-center text-center flex"> @@ -38,7 +38,7 @@ <p v-if="levelInfo.des" class="pt-5" v-text="levelInfo.des"></p> <h3 :class="[ !levelInfo.des ? 'pt-5' : 'pt-0']" v-text="indicators.points +' '+strings.points"></h3> - <v-row class="justify-center text-left" v-for="action in activity" :key="action"> + <v-row class="justify-center text-left" v-for="action in activity" > <v-col cols="5"> <small v-text="action.description"></small> </v-col> @@ -49,7 +49,6 @@ <small v-text="action.points + ' '+strings.points"></small> </v-col> </v-row> - <small v-text="rankable"></small> </v-card-text> <v-card-actions> <p class="py-0 px-2" v-text="settings.des"></p> @@ -57,8 +56,32 @@ </v-card> </v-col> </v-row> + <v-col> + <h4 v-text=strings.overview></h4> + <v-row class="py-5 px-8"> + <v-col v-for="(level, index) in levelsData" cols="12" sm="6" md="3" class="d-flex"> + <v-card class="pt-5 align-center text-center flex"> + <v-badge + :content="level.lvl" + bottom + offset-x="20" + offset-y="20" + > + <v-avatar size="100"> + <v-img src="{{{image}}}"></v-img> + </v-avatar> + </v-badge> + <v-card-text class="text--primary text-center"> + <h6 v-if="level.nam" v-text="level.nam"></h6> + <p v-if="level.des" class="pt-4 text-left" v-text="level.des"></p> + <p class="mb-0 pb-0" v-text="level.points+' '+strings.points"></p> + </v-card-text> + </v-card> + </v-col> + </v-row> + </v-col> </v-col> - <v-col cols=12 lg="5"> + <v-col cols=12 lg="6"> <v-row v-if="strings.rankable"class="py-5 px-8"> <v-col cols="12" md="12"> <v-data-table @@ -84,7 +107,6 @@ :value="item.progress_percentage" height="8" > - </v-progress-linear> </template> <template v-slot:item.total="{ item }"> @@ -92,20 +114,22 @@ <sup>xp</sup> </template> </v-data-table> + + + </v-col> </v-row> <v-row v-else> <v-card class="mx-auto" - max-width="100%" + max-width="90%" outlined > <v-list-item three-line> <v-list-item-content> - <v-list-item-title class="text-h5 mb-1"> - <v-text v-text=strings.studentRanking1></v-text> + <v-list-item-title class="text-h5 mb-1" v-text=strings.studentRanking1> </v-list-item-title> - <v-list-item-subtitle><v-text v-text=strings.studentRanking2></v-text> </v-list-item-subtitle> + <v-list-item-subtitle v-text=strings.studentRanking2></v-list-item-subtitle> </v-list-item-content> <v-list-item-avatar @@ -113,7 +137,7 @@ size="80" color="red" - ><v-img src="{{{image}}}"></v-img></v-list-item-avatar> + ><v-img src="{{{rankImage}}}"></v-img></v-list-item-avatar> </v-list-item> <v-card-actions> @@ -122,8 +146,8 @@ rounded text @click=activateRanking() + v-text=strings.studentRanking3 > - <v-text v-text=strings.studentRanking3></v-text> </v-btn> </v-card-actions> </v-card> @@ -131,31 +155,7 @@ </v-col> </v-row> </v-card> - <v-card> - <v-col> - <v-row class="py-5 px-8"> - <v-col v-for="(level, index) in levelsData" :key="level.level" cols="12" sm="6" md="3" class="d-flex"> - <v-card class="pt-5 align-center text-center flex"> - <v-badge - :content="level.lvl" - bottom - offset-x="20" - offset-y="20" - > - <v-avatar size="100"> - <v-img src="{{{image}}}"></v-img> - </v-avatar> - </v-badge> - <v-card-text class="text--primary text-center"> - <h6 v-if="level.nam" v-text="level.nam"></h6> - <p v-if="level.des" class="pt-4 text-left" v-text="level.des"></p> - <p class="mb-0 pb-0" v-text="level.points+' '+strings.points"></p> - </v-card-text> - </v-card> - </v-col> - </v-row> - </v-col> - </v-card> + </template> </v-main> </v-app> \ No newline at end of file