apexcharts
Version:
A JavaScript Chart Library
361 lines (308 loc) • 10.3 kB
JavaScript
import YAxis from '../axes/YAxis'
import Helpers from './Helpers'
import DimXAxis from './XAxis'
import DimYAxis from './YAxis'
import Grid from './Grid'
/**
* ApexCharts Dimensions Class for calculating rects of all elements that are drawn and will be drawn.
*
* @module Dimensions
**/
export default class Dimensions {
constructor(ctx) {
this.ctx = ctx
this.w = ctx.w
this.lgRect = {}
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
}
/**
* @memberof Dimensions
* @param {object} w - chart context
**/
plotCoords() {
let w = this.w
let 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
gl.gridHeight = gl.gridHeight - this.gridPad.top - this.gridPad.bottom
gl.gridWidth =
gl.gridWidth -
this.gridPad.left -
this.gridPad.right -
this.xPadRight -
this.xPadLeft
let barWidth = this.dimGrid.gridPadForColumnsInNumericAxis(gl.gridWidth)
gl.gridWidth = gl.gridWidth - barWidth * 2
gl.translateX =
gl.translateX +
this.gridPad.left +
this.xPadLeft +
(barWidth > 0 ? barWidth : 0)
gl.translateY = gl.translateY + this.gridPad.top
}
setDimensionsForAxisCharts() {
let w = this.w
let gl = w.globals
let yaxisLabelCoords = this.dimYAxis.getyAxisLabelsCoords()
let yTitleCoords = this.dimYAxis.getyAxisTitleCoords()
if (gl.isSlopeChart) {
this.datalabelsCoords = this.dimHelpers.getDatalabelsRect()
}
w.globals.yLabelsCoords = []
w.globals.yTitleCoords = []
w.config.yaxis.map((yaxe, index) => {
// store the labels and titles coords in global vars
w.globals.yLabelsCoords.push({
width: yaxisLabelCoords[index].width,
index,
})
w.globals.yTitleCoords.push({
width: yTitleCoords[index].width,
index,
})
})
this.yAxisWidth = this.dimYAxis.getTotalYAxisWidth()
let xaxisLabelCoords = this.dimXAxis.getxAxisLabelsCoords()
let xaxisGroupLabelCoords = this.dimXAxis.getxAxisGroupLabelsCoords()
let xtitleCoords = this.dimXAxis.getxAxisTitleCoords()
this.conditionalChecksForAxisCoords(
xaxisLabelCoords,
xtitleCoords,
xaxisGroupLabelCoords
)
gl.translateXAxisY = w.globals.rotateXLabels ? this.xAxisHeight / 8 : -4
gl.translateXAxisX =
w.globals.rotateXLabels &&
w.globals.isXNumeric &&
w.config.xaxis.labels.rotate <= -45
? -this.xAxisWidth / 4
: 0
if (w.globals.isBarHorizontal) {
gl.rotateXLabels = false
gl.translateXAxisY =
-1 * (parseInt(w.config.xaxis.labels.style.fontSize, 10) / 1.5)
}
gl.translateXAxisY = gl.translateXAxisY + w.config.xaxis.labels.offsetY
gl.translateXAxisX = gl.translateXAxisX + w.config.xaxis.labels.offsetX
let yAxisWidth = this.yAxisWidth
let xAxisHeight = this.xAxisHeight
gl.xAxisLabelsHeight = this.xAxisHeight - xtitleCoords.height
gl.xAxisGroupLabelsHeight = gl.xAxisLabelsHeight - xaxisLabelCoords.height
gl.xAxisLabelsWidth = this.xAxisWidth
gl.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 = () => {
gl.translateX = yAxisWidth + this.datalabelsCoords.width
gl.gridHeight =
gl.svgHeight -
this.lgRect.height -
xAxisHeight -
(!this.isSparkline && w.config.chart.type !== 'treemap'
? w.globals.rotateXLabels
? 10
: 15
: 0)
gl.gridWidth = gl.svgWidth - yAxisWidth - this.datalabelsCoords.width * 2
}
if (w.config.xaxis.position === 'top')
translateY = gl.xAxisHeight - w.config.xaxis.axisTicks.height - 5
switch (w.config.legend.position) {
case 'bottom':
gl.translateY = translateY
legendTopBottom()
break
case 'top':
gl.translateY = this.lgRect.height + translateY
legendTopBottom()
break
case 'left':
gl.translateY = translateY
gl.translateX =
this.lgRect.width + yAxisWidth + this.datalabelsCoords.width
gl.gridHeight = gl.svgHeight - xAxisHeight - 12
gl.gridWidth =
gl.svgWidth -
this.lgRect.width -
yAxisWidth -
this.datalabelsCoords.width * 2
break
case 'right':
gl.translateY = translateY
gl.translateX = yAxisWidth + this.datalabelsCoords.width
gl.gridHeight = gl.svgHeight - xAxisHeight - 12
gl.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
let objyAxis = new YAxis(this.ctx)
objyAxis.setYAxisXPosition(yaxisLabelCoords, yTitleCoords)
}
setDimensionsForNonAxisCharts() {
let w = this.w
let gl = w.globals
let 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'
let offY = cnf.plotOptions[type].offsetY
let offX = cnf.plotOptions[type].offsetX
if (!cnf.legend.show || cnf.legend.floating) {
gl.gridHeight = gl.svgHeight
const maxWidth = gl.dom.elWrap.getBoundingClientRect().width
gl.gridWidth = Math.min(maxWidth, gl.gridHeight)
gl.translateY = offY
gl.translateX = offX + (gl.svgWidth - gl.gridWidth) / 2
return
}
switch (cnf.legend.position) {
case 'bottom':
gl.gridHeight = gl.svgHeight - this.lgRect.height
gl.gridWidth = gl.svgWidth
gl.translateY = offY - 10
gl.translateX = offX + (gl.svgWidth - gl.gridWidth) / 2
break
case 'top':
gl.gridHeight = gl.svgHeight - this.lgRect.height
gl.gridWidth = gl.svgWidth
gl.translateY = this.lgRect.height + offY + 10
gl.translateX = offX + (gl.svgWidth - gl.gridWidth) / 2
break
case 'left':
gl.gridWidth = gl.svgWidth - this.lgRect.width - xPad
gl.gridHeight =
cnf.chart.height !== 'auto' ? gl.svgHeight : gl.gridWidth
gl.translateY = offY
gl.translateX = offX + this.lgRect.width + xPad
break
case 'right':
gl.gridWidth = gl.svgWidth - this.lgRect.width - xPad - 5
gl.gridHeight =
cnf.chart.height !== 'auto' ? gl.svgHeight : gl.gridWidth
gl.translateY = offY
gl.translateX = offX + 10
break
default:
throw new Error('Legend position not supported')
}
}
conditionalChecksForAxisCoords(
xaxisLabelCoords,
xtitleCoords,
xaxisGroupLabelCoords
) {
const w = this.w
const xAxisNum = w.globals.hasXaxisGroups ? 2 : 1
const baseXAxisHeight =
xaxisGroupLabelCoords.height +
xaxisLabelCoords.height +
xtitleCoords.height
const xAxisHeightMultiplicate = w.globals.isMultiLineX
? 1.2
: w.globals.LINE_HEIGHT_RATIO
const rotatedXAxisOffset = w.globals.rotateXLabels ? 22 : 10
const rotatedXAxisLegendOffset =
w.globals.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
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
}
}
}