diff --git a/packages/@ourworldindata/grapher/src/axis/Axis.test.ts b/packages/@ourworldindata/grapher/src/axis/Axis.test.ts index a24e55fbf5..2d135aa335 100755 --- a/packages/@ourworldindata/grapher/src/axis/Axis.test.ts +++ b/packages/@ourworldindata/grapher/src/axis/Axis.test.ts @@ -7,7 +7,7 @@ import { SynthesizeGDPTable, } from "@ourworldindata/core-table" import { AxisConfig } from "./AxisConfig" -import { AxisAlign } from "@ourworldindata/utils" +import { AxisAlign, last } from "@ourworldindata/utils" it("can create an axis", () => { const axisConfig = new AxisConfig({ @@ -88,7 +88,21 @@ it("respects nice parameter", () => { axis.range = [0, 300] const tickValues = axis.getTickValues() expect(tickValues[0].value).toEqual(0) - expect(tickValues[tickValues.length - 1].value).toEqual(100) + expect(last(tickValues)?.value).toEqual(100) +}) + +it("fine-tunes d3's nice implementation", () => { + const config: AxisConfigInterface = { + min: 0.0001, + max: 90.0001, + maxTicks: 10, + nice: true, + } + const axis = new AxisConfig(config).toVerticalAxis() + axis.range = [0, 300] + const tickValues = axis.getTickValues() + expect(tickValues[0].value).toEqual(0) + expect(last(tickValues)?.value).toEqual(90) }) it("creates compact labels", () => { diff --git a/packages/@ourworldindata/grapher/src/axis/Axis.ts b/packages/@ourworldindata/grapher/src/axis/Axis.ts index eb9f2e9159..aa7227f60f 100644 --- a/packages/@ourworldindata/grapher/src/axis/Axis.ts +++ b/packages/@ourworldindata/grapher/src/axis/Axis.ts @@ -21,6 +21,7 @@ import { ValueRange, cloneDeep, OwidVariableRoundingMode, + last, } from "@ourworldindata/utils" import { AxisConfig, AxisManager } from "./AxisConfig" import { MarkdownTextWrap } from "@ourworldindata/components" @@ -209,11 +210,44 @@ abstract class AbstractAxis { }) } + @observable private niceTicks?: number[] + private makeScaleNice( + scale: ScaleLinear, + totalTicksTarget: number + ): ScaleLinear { + const ticks = scale.ticks(totalTicksTarget) + + // use d3's nice function when there is only one tick + if (ticks.length < 2) return scale.nice(totalTicksTarget) + + const tickStep = ticks[1] - ticks[0] + const firstTick = ticks[0] + const lastTick = last(ticks)! + + this.niceTicks = ticks + + // if the the max or min value exceeds the last grid line by more than 10%, + // expand the domain to include an additional grid line + const [minValue, maxValue] = scale.domain() + if (maxValue > lastTick + 0.25 * tickStep) { + scale.domain([scale.domain()[0], lastTick + tickStep]) + this.niceTicks = [...this.niceTicks, lastTick + tickStep] + } + if (minValue < firstTick - 0.25 * tickStep) { + scale.domain([firstTick - tickStep, scale.domain()[1]]) + this.niceTicks = [firstTick - tickStep, ...this.niceTicks] + } + + return scale + } + @computed private get d3_scale(): Scale { - const d3Scale = - this.scaleType === ScaleType.log ? scaleLog : scaleLinear + const isLogScale = this.scaleType === ScaleType.log + const d3Scale = isLogScale ? scaleLog : scaleLinear let scale = d3Scale().domain(this.domain).range(this.range) - scale = this.nice ? scale.nice(this.totalTicksTarget) : scale + if (this.nice && !isLogScale) { + scale = this.makeScaleNice(scale, this.totalTicksTarget) + } if (this.config.domainValues) { // compute bandwidth and adjust the scale @@ -356,7 +390,9 @@ abstract class AbstractAxis { } else { // Only use priority 2 here because we want the start / end ticks // to be priority 1 - ticks = d3_scale.ticks(this.totalTicksTarget).map((tickValue) => ({ + const d3_ticks = + this.niceTicks ?? d3_scale.ticks(this.totalTicksTarget) + ticks = d3_ticks.map((tickValue) => ({ value: tickValue, priority: 2, })) diff --git a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx index 3369bbbfa2..6b84258212 100644 --- a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx +++ b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx @@ -1418,9 +1418,9 @@ export class LineChart } @computed private get yAxisConfig(): AxisConfig { - // TODO: enable nice axis ticks for linear scales return new AxisConfig( { + nice: true, // if we only have a single y value (probably 0), we want the // horizontal axis to be at the bottom of the chart. // see https://github.com/owid/owid-grapher/pull/975#issuecomment-890798547 diff --git a/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx b/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx index 5b40a3086b..812da2c381 100644 --- a/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx +++ b/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx @@ -253,8 +253,13 @@ export class AbstractStackedChart } @computed private get yAxisConfig(): AxisConfig { - // TODO: enable nice axis ticks for linear scales - return new AxisConfig(this.manager.yAxisConfig, this) + return new AxisConfig( + { + nice: true, + ...this.manager.yAxisConfig, + }, + this + ) } // implemented in subclasses