UNPKG

apexcharts

Version:

A JavaScript Chart Library

421 lines (367 loc) 12.5 kB
// @ts-check import YAxis from '../axes/YAxis' import Helpers from './Helpers' import DimXAxis from './XAxis' import DimYAxis from './YAxis' import Grid from './Grid' import { LINE_HEIGHT_RATIO } from '../../utils/Constants' /** * ApexCharts Dimensions Class for calculating rects of all elements that are drawn and will be drawn. * * @module Dimensions **/ export default class Dimensions { /** * @param {import('../../types/internal').ChartStateW} w * @param {import('../../types/internal').ChartContext} ctx */ constructor(w, ctx) { this.w = w this.ctx = ctx // needed: new XAxis(w, ctx) for xAxisLabelClick event callback this.theme = ctx.theme this.timeScale = ctx.timeScale this.lgRect = /** @type {any} */ ({}) this.yAxisWidth = 0 this.yAxisWidthLeft = 0 this.yAxisWidthRight = 0 this.xAxisHeight = 0 this.isSparkline = this.w.config.chart.sparkline.enabled this.dimHelpers = new Helpers(this) this.dimYAxis = new DimYAxis(this) this.dimXAxis = new DimXAxis(this) this.dimGrid = new Grid(this) this.lgWidthForSideLegends = 0 this.gridPad = this.w.config.grid.padding this.xPadRight = 0 this.xPadLeft = 0 this.datalabelsCoords = { width: 0, height: 0 } /** @type {number} */ this.xAxisWidth = 0 /** @type {any[]} */ this.timescaleLabels = [] } /** * @memberof Dimensions **/ plotCoords() { const w = this.w const gl = w.globals this.lgRect = this.dimHelpers.getLegendsRect() this.datalabelsCoords = { width: 0, height: 0 } const maxStrokeWidth = Array.isArray(w.config.stroke.width) ? Math.max(...w.config.stroke.width) : w.config.stroke.width if (this.isSparkline) { if (w.config.markers.discrete.length > 0 || w.config.markers.size > 0) { Object.entries(this.gridPad).forEach(([k, v]) => { this.gridPad[k] = Math.max( v, this.w.globals.markers.largestSize / 1.5, ) }) } this.gridPad.top = Math.max(maxStrokeWidth / 2, this.gridPad.top) this.gridPad.bottom = Math.max(maxStrokeWidth / 2, this.gridPad.bottom) } if (gl.axisCharts) { // for line / area / scatter / column this.setDimensionsForAxisCharts() } else { // for pie / donuts / circle this.setDimensionsForNonAxisCharts() } this.dimGrid.gridPadFortitleSubtitle() // after calculating everything, apply padding set by user w.layout.gridHeight = w.layout.gridHeight - this.gridPad.top - this.gridPad.bottom w.layout.gridWidth = w.layout.gridWidth - this.gridPad.left - this.gridPad.right - this.xPadRight - this.xPadLeft const barWidth = this.dimGrid.gridPadForColumnsInNumericAxis( w.layout.gridWidth, ) w.layout.gridWidth = w.layout.gridWidth - barWidth * 2 w.layout.translateX = w.layout.translateX + this.gridPad.left + this.xPadLeft + (barWidth > 0 ? barWidth : 0) w.layout.translateY = w.layout.translateY + this.gridPad.top // Return a snapshot of all computed layout state grouped by future w.layout slice destination. // Phase 1: callers use named writer stubs (no-ops — mutations above already wrote to gl). // Phase 2: writers will assign to typed slices instead of gl.*. return { // w.layout (future slice) layout: { gridHeight: w.layout.gridHeight, gridWidth: w.layout.gridWidth, translateX: w.layout.translateX, translateY: w.layout.translateY, translateXAxisX: w.layout.translateXAxisX, translateXAxisY: w.layout.translateXAxisY, rotateXLabels: w.layout.rotateXLabels, xAxisHeight: w.layout.xAxisHeight, xAxisLabelsHeight: w.layout.xAxisLabelsHeight, xAxisGroupLabelsHeight: w.layout.xAxisGroupLabelsHeight, xAxisLabelsWidth: w.layout.xAxisLabelsWidth, yLabelsCoords: w.layout.yLabelsCoords, yTitleCoords: w.layout.yTitleCoords, }, } } setDimensionsForAxisCharts() { const w = this.w const gl = w.globals const yaxisLabelCoords = this.dimYAxis.getyAxisLabelsCoords() const yTitleCoords = this.dimYAxis.getyAxisTitleCoords() if (gl.isSlopeChart) { this.datalabelsCoords = this.dimHelpers.getDatalabelsRect() } w.layout.yLabelsCoords = [] w.layout.yTitleCoords = [] /** * @param {ApexYAxis} yaxe * @param {number} index */ w.config.yaxis.map((yaxe, index) => { // store the labels and titles coords in global vars w.layout.yLabelsCoords.push({ width: yaxisLabelCoords[index].width, index, }) w.layout.yTitleCoords.push( /** @type {any} */ ({ width: yTitleCoords[index].width, index, }), ) }) this.yAxisWidth = this.dimYAxis.getTotalYAxisWidth() const xaxisLabelCoords = this.dimXAxis.getxAxisLabelsCoords() const xaxisGroupLabelCoords = this.dimXAxis.getxAxisGroupLabelsCoords() const xtitleCoords = this.dimXAxis.getxAxisTitleCoords() this.conditionalChecksForAxisCoords( xaxisLabelCoords, xtitleCoords, xaxisGroupLabelCoords, ) w.layout.translateXAxisY = w.layout.rotateXLabels ? this.xAxisHeight / 8 : -4 w.layout.translateXAxisX = w.layout.rotateXLabels && w.axisFlags.isXNumeric && w.config.xaxis.labels.rotate <= -45 ? -this.xAxisWidth / 4 : 0 if (w.globals.isBarHorizontal) { w.layout.rotateXLabels = false w.layout.translateXAxisY = -1 * (parseInt(w.config.xaxis.labels.style.fontSize, 10) / 1.5) } w.layout.translateXAxisY = w.layout.translateXAxisY + w.config.xaxis.labels.offsetY w.layout.translateXAxisX = w.layout.translateXAxisX + w.config.xaxis.labels.offsetX let yAxisWidth = this.yAxisWidth let xAxisHeight = this.xAxisHeight w.layout.xAxisLabelsHeight = this.xAxisHeight - xtitleCoords.height w.layout.xAxisGroupLabelsHeight = w.layout.xAxisLabelsHeight - xaxisLabelCoords.height w.layout.xAxisLabelsWidth = this.xAxisWidth w.layout.xAxisHeight = this.xAxisHeight let translateY = 10 if (w.config.chart.type === 'radar' || this.isSparkline) { yAxisWidth = 0 xAxisHeight = 0 } if (this.isSparkline) { this.lgRect = { height: 0, width: 0, } } if (this.isSparkline || w.config.chart.type === 'treemap') { yAxisWidth = 0 xAxisHeight = 0 translateY = 0 } if (!this.isSparkline && w.config.chart.type !== 'treemap') { this.dimXAxis.additionalPaddingXLabels(xaxisLabelCoords) } const legendTopBottom = () => { w.layout.translateX = yAxisWidth + this.datalabelsCoords.width w.layout.gridHeight = gl.svgHeight - this.lgRect.height - xAxisHeight - (!this.isSparkline && w.config.chart.type !== 'treemap' ? w.layout.rotateXLabels ? 10 : 15 : 0) w.layout.gridWidth = gl.svgWidth - yAxisWidth - this.datalabelsCoords.width * 2 } if (w.config.xaxis.position === 'top') translateY = w.layout.xAxisHeight - w.config.xaxis.axisTicks.height - 5 switch (w.config.legend.position) { case 'bottom': w.layout.translateY = translateY legendTopBottom() break case 'top': w.layout.translateY = this.lgRect.height + translateY legendTopBottom() break case 'left': w.layout.translateY = translateY w.layout.translateX = this.lgRect.width + yAxisWidth + this.datalabelsCoords.width w.layout.gridHeight = gl.svgHeight - xAxisHeight - 12 w.layout.gridWidth = gl.svgWidth - this.lgRect.width - yAxisWidth - this.datalabelsCoords.width * 2 break case 'right': w.layout.translateY = translateY w.layout.translateX = yAxisWidth + this.datalabelsCoords.width w.layout.gridHeight = gl.svgHeight - xAxisHeight - 12 w.layout.gridWidth = gl.svgWidth - this.lgRect.width - yAxisWidth - this.datalabelsCoords.width * 2 - 5 break default: throw new Error('Legend position not supported') } this.dimGrid.setGridXPosForDualYAxis(yTitleCoords, yaxisLabelCoords) // after drawing everything, set the Y axis positions const objyAxis = new YAxis(this.w, { theme: this.theme, timeScale: this.timeScale, }) objyAxis.setYAxisXPosition(yaxisLabelCoords, yTitleCoords) } setDimensionsForNonAxisCharts() { const w = this.w const gl = w.globals const cnf = w.config let xPad = 0 if (w.config.legend.show && !w.config.legend.floating) { xPad = 20 } const type = cnf.chart.type === 'pie' || cnf.chart.type === 'polarArea' || cnf.chart.type === 'donut' ? 'pie' : 'radialBar' const offY = cnf.plotOptions[type].offsetY const offX = cnf.plotOptions[type].offsetX if (!cnf.legend.show || cnf.legend.floating) { w.layout.gridHeight = gl.svgHeight const maxWidth = w.dom.elWrap.getBoundingClientRect().width w.layout.gridWidth = Math.min(maxWidth, w.layout.gridHeight) w.layout.translateY = offY w.layout.translateX = offX + (gl.svgWidth - w.layout.gridWidth) / 2 return } switch (cnf.legend.position) { case 'bottom': w.layout.gridHeight = gl.svgHeight - this.lgRect.height w.layout.gridWidth = gl.svgWidth w.layout.translateY = offY - 10 w.layout.translateX = offX + (gl.svgWidth - w.layout.gridWidth) / 2 break case 'top': w.layout.gridHeight = gl.svgHeight - this.lgRect.height w.layout.gridWidth = gl.svgWidth w.layout.translateY = this.lgRect.height + offY + 10 w.layout.translateX = offX + (gl.svgWidth - w.layout.gridWidth) / 2 break case 'left': w.layout.gridWidth = gl.svgWidth - this.lgRect.width - xPad w.layout.gridHeight = cnf.chart.height !== 'auto' ? gl.svgHeight : w.layout.gridWidth w.layout.translateY = offY w.layout.translateX = offX + this.lgRect.width + xPad break case 'right': w.layout.gridWidth = gl.svgWidth - this.lgRect.width - xPad - 5 w.layout.gridHeight = cnf.chart.height !== 'auto' ? gl.svgHeight : w.layout.gridWidth w.layout.translateY = offY w.layout.translateX = offX + 10 break default: throw new Error('Legend position not supported') } } /** * @param {any} xaxisLabelCoords * @param {any} xtitleCoords * @param {any} xaxisGroupLabelCoords */ conditionalChecksForAxisCoords( xaxisLabelCoords, xtitleCoords, xaxisGroupLabelCoords, ) { const w = this.w const xAxisNum = w.labelData.hasXaxisGroups ? 2 : 1 const baseXAxisHeight = xaxisGroupLabelCoords.height + xaxisLabelCoords.height + xtitleCoords.height const xAxisHeightMultiplicate = w.axisFlags.isMultiLineX ? 1.2 : LINE_HEIGHT_RATIO const rotatedXAxisOffset = w.layout.rotateXLabels ? 22 : 10 const rotatedXAxisLegendOffset = w.layout.rotateXLabels && w.config.legend.position === 'bottom' const additionalOffset = rotatedXAxisLegendOffset ? 10 : 0 this.xAxisHeight = baseXAxisHeight * xAxisHeightMultiplicate + xAxisNum * rotatedXAxisOffset + additionalOffset this.xAxisWidth = xaxisLabelCoords.width if ( this.xAxisHeight - xtitleCoords.height > w.config.xaxis.labels.maxHeight ) { this.xAxisHeight = w.config.xaxis.labels.maxHeight } if ( w.config.xaxis.labels.minHeight && this.xAxisHeight < w.config.xaxis.labels.minHeight ) { this.xAxisHeight = w.config.xaxis.labels.minHeight } if (w.config.xaxis.floating) { this.xAxisHeight = 0 } let minYAxisWidth = 0 let maxYAxisWidth = 0 /** * @param {number} y */ w.config.yaxis.forEach((y) => { minYAxisWidth += y.labels.minWidth maxYAxisWidth += y.labels.maxWidth }) if (this.yAxisWidth < minYAxisWidth) { this.yAxisWidth = minYAxisWidth } if (this.yAxisWidth > maxYAxisWidth) { this.yAxisWidth = maxYAxisWidth } } }