Skip to content

Commit

Permalink
✨ use 'nice' axis ticks for linear scales
Browse files Browse the repository at this point in the history
  • Loading branch information
sophiamersmann committed Jan 10, 2025
1 parent 98092b8 commit d6d2059
Show file tree
Hide file tree
Showing 4 changed files with 56 additions and 8 deletions.
18 changes: 16 additions & 2 deletions packages/@ourworldindata/grapher/src/axis/Axis.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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({
Expand Down Expand Up @@ -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", () => {
Expand Down
35 changes: 32 additions & 3 deletions packages/@ourworldindata/grapher/src/axis/Axis.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import {
ValueRange,
cloneDeep,
OwidVariableRoundingMode,
last,
} from "@ourworldindata/utils"
import { AxisConfig, AxisManager } from "./AxisConfig"
import { MarkdownTextWrap } from "@ourworldindata/components"
Expand Down Expand Up @@ -209,11 +210,39 @@ abstract class AbstractAxis {
})
}

private static makeScaleNice(
scale: ScaleLinear<number, number>,
totalTicksTarget: number
): ScaleLinear<number, number> {
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)!

// 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.1 * tickStep) {
scale.domain([minValue, lastTick + tickStep])
}
if (minValue < firstTick - 0.1 * tickStep) {
scale.domain([firstTick - tickStep, maxValue])
}

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 = AbstractAxis.makeScaleNice(scale, this.totalTicksTarget)
}

if (this.config.domainValues) {
// compute bandwidth and adjust the scale
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down

0 comments on commit d6d2059

Please sign in to comment.