diff --git a/CHANGELOG-2021.md b/CHANGELOG-2021.md
index 581acbc615..3b57dc9101 100644
--- a/CHANGELOG-2021.md
+++ b/CHANGELOG-2021.md
@@ -1,6 +1,6 @@
# Observable Plot - Changelog [2021]
-Year: [Current (2023)](./CHANGELOG.md) · [2022](./CHANGELOG-2022.md) · **2021**
+Year: [Current (2024)](./CHANGELOG.md) · [2023](./CHANGELOG-2023.md) · [2022](./CHANGELOG-2022.md) · **2021**
## 0.3.2
diff --git a/CHANGELOG-2022.md b/CHANGELOG-2022.md
index 76976e0590..fcc2fc6e4f 100644
--- a/CHANGELOG-2022.md
+++ b/CHANGELOG-2022.md
@@ -1,6 +1,6 @@
# Observable Plot - Changelog [2022]
-Year: [Current (2023)](./CHANGELOG.md) · **2022** · [2021](./CHANGELOG-2021.md)
+Year: [Current (2024)](./CHANGELOG.md) · [2023](./CHANGELOG-2023.md) · **2022** · [2021](./CHANGELOG-2021.md)
## 0.6.1
diff --git a/CHANGELOG-2023.md b/CHANGELOG-2023.md
new file mode 100644
index 0000000000..7fc18fb69d
--- /dev/null
+++ b/CHANGELOG-2023.md
@@ -0,0 +1,855 @@
+# Observable Plot - Changelog [2023]
+
+Year: [Current (2024)](./CHANGELOG.md) · **2023** · [2022](./CHANGELOG-2022.md) · [2021](./CHANGELOG-2021.md)
+
+## 0.6.13
+
+[Released December 13, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.13)
+
+Fixes the *observable10* categorical color scheme. In 0.6.12, we inadvertently published an earlier draft; this fixed version better aligns the order of colors with *tableau10*.
+
+We reduced the specificity of selectors in the generated stylesheets to 0, allowing consumers of Plot to much more easily override these styles using their own stylesheets while still enjoying the default styling. Thanks, @mstade!
+
+## 0.6.12
+
+[Released December 7, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.12)
+
+To better support dark mode, we’ve made a breaking change to default styles: the background color is now `unset` (transparent) instead of `white`. Additionally, marks that expect to use the same color as the background (for instance, the tip mark) now use the CSS custom property `--plot-background` instead of `white`.
+
+As since [version 0.6.6](#066), you can override Plot’s styles via the `plot-d6a7b5` class. For example, the following stylesheet applies a dark background and white foreground:
+
+```css
+svg.plot-d6a7b5 {
+ --plot-background: #333;
+ background: var(--plot-background);
+ color: white;
+}
+```
+
+Previously Plot forced you to choose between [rect](https://observablehq.com/plot/marks/rect), [bar](https://observablehq.com/plot/marks/bar), and [cell](https://observablehq.com/plot/marks/cell) based on whether the *x* and *y* scales were ordinal or quantitative. Rejoice: the [rect mark](https://observablehq.com/plot/marks/rect) now supports *band* scales so you can simply use rect in all cases! 🎉 (The bar and cell marks still offer conveniences for ordinal scales, so use them if you like.)
+
+
+
+```js
+Plot.rectY(alphabet, {x: "letter", y: "frequency"}).plot()
+```
+
+Categorical color scales now default to the new *observable10* color scheme by [Jeff Pettiross](https://github.com/pettiross). These colors are intended as a drop-in replacement for *tableau10* with similar ease of discrimination and ordering, but with a slightly more saturated vibe that helps charts pop.
+
+
+
+```js
+Plot.cellX(d3.range(10)).plot({color: {type: "categorical"}})
+```
+
+The new [difference mark](https://observablehq.com/plot/marks/difference) puts a metric in context by comparing it to another metric or constant value. Like the [area mark](https://observablehq.com/plot/marks/area), the region between two lines is filled; unlike the area mark, alternating color shows when the metric is above or below the comparison.
+
+
+
+```js
+Plot.differenceY(gistemp, Plot.windowY(28, {x: "Date", y: "Anomaly"})).plot()
+```
+
+The chart above shows a moving average of global temperature anomaly; above-average temperatures are shown in green and below-average temperatures are shown in blue.
+
+The difference mark can also compare a metric to its earlier self, showing change over time. The chart below shows the year-over-year change in Apple stock price; the gray region represents an increase over last year while the red region represents a decrease.
+
+
+
+```js
+Plot.differenceY(
+ aapl,
+ Plot.shiftX("year", {
+ x: "Date",
+ y: "Close",
+ positiveFillOpacity: 0.2,
+ positiveFill: "currentColor",
+ negativeFillOpacity: 0.8,
+ negativeFill: "red"
+ })
+).plot()
+```
+
+The chart above is constructed using the new [shift transform](https://observablehq.com/plot/transforms/shift), which derives a time-shifted copy of a metric for the given time interval, enabling year–over-year, month-over-month, or any other period-over-period comparison.
+
+The new [find reducer](https://observablehq.com/plot/transforms/group#find) allows you to pivot data with the [bin](https://observablehq.com/plot/transforms/bin) and [group](https://observablehq.com/plot/transforms/group) transforms, effectively turning “tall” data (with fewer columns) into “wide” data (with more columns). For example, say you have time-series data that has separate rows for observed daily temperatures in San Francisco (SF) and San Jose (SJ):
+
+```csv
+date,station,tmax,tmin
+2020-12-31,SJ,59,43
+2020-12-31,SF,60,47
+2020-12-30,SJ,58,33
+2020-12-30,SF,57,40
+2020-12-29,SJ,62,38
+2020-12-29,SF,61,41
+2020-12-28,SJ,57,43
+2020-12-28,SF,56,47
+2020-12-27,SJ,62,43
+2020-12-27,SF,57,46
+2020-12-26,SJ,60,46
+2020-12-26,SF,61,49
+```
+
+To compare the average minimum temperatures of San Francisco and San Jose as another difference chart:
+
+
+
+```js
+Plot.plot({
+ x: {tickFormat: "%b"},
+ y: {grid: true},
+ marks: [
+ Plot.ruleY([32]),
+ Plot.differenceY(
+ temperature,
+ Plot.windowY(
+ 14,
+ Plot.groupX(
+ {y1: Plot.find((d) => d.station === "SJ"), y2: Plot.find((d) => d.station === "SF")},
+ {x: "date", y: "tmin"}
+ )
+ )
+ )
+ ]
+})
+```
+
+The green region above represents when San Francisco was warmer than San Jose, and the blue region when San Francisco was cooler than San Jose.
+
+The [tip mark](https://observablehq.com/plot/marks/tip) now supports a **preferredAnchor** option, providing greater control over tip placement: if the tip fits within the frame at the preferred anchor, this anchor will be used. (In contrast, the **anchor** option uses the specified anchor regardless of whether it will fit.) The tip mark now also prefers the *bottom* anchor by default for a more traditional, less comic-like appearance.
+
+The [**marker** option](https://observablehq.com/plot/features/markers) now supports new marker types: *tick*, *tick-x*, and *tick-y*. The [bin](https://observablehq.com/plot/transforms/bin) and [group](https://observablehq.com/plot/transforms/group) transforms now pass _data_ to reducers. The [group](https://observablehq.com/plot/transforms/group) and [hexbin](https://observablehq.com/plot/transforms/hexbin) transforms now support _x_ and _y_ reducers.
+
+This release includes several additional fixes:
+
+* The default axis for a *time* scale now uses local time.
+* The text mark’s **lineWidth** option is now more accurate when **monospace** is true.
+* The tip mark no longer truncates **title** text.
+* The scale **type** option is now case-insensitive.
+* Transform type definitions have correct overload precedence.
+
+## 0.6.11
+
+[Released September 20, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.11)
+
+The **tip** mark option can now pass options to the derived [tip mark](https://observablehq.com/plot/marks/tip); the options object can also specify the **pointer** option to control the derived tip’s pointer mode (_x_, _y_, or _xy_). The new **format** tip mark option enables greater control over order and formatting of channels.
+
+
+
+```js
+Plot.dot(olympians, {
+ x: "weight",
+ y: "height",
+ stroke: "sex",
+ channels: {
+ name: "name",
+ nationality: "nationality",
+ sport: "sport"
+ },
+ tip: {
+ format: {
+ name: true, // show name first
+ y: (d) => `${d}m`, // units in meters
+ x: (d) => `${d}kg`, // units in kilograms
+ stroke: false // suppress stroke channel
+ }
+ }
+}).plot()
+```
+
+Axes for ordinal scales now generalize the scale’s temporal or quantitative **interval** if any, resulting in more readable ticks. For instance, the bar chart below of monthly values now sports multi-line tick labels.
+
+
+
+```js
+Plot.plot({
+ x: {interval: "month"},
+ marks: [
+ Plot.barY(aapl, Plot.groupX({y: "median"}, {x: "Date", y: "Close"}))
+ ]
+})
+```
+
+Plot now recognizes CSS Color Module [Level 4](https://www.w3.org/TR/css-color-4/) and [Level 5](https://www.w3.org/TR/css-color-5/) syntax as literal colors, making it easier to use modern color syntax such as [oklab()](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/oklab), [color-mix()](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/color-mix), and alternative color spaces such as [display-p3](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/color).
+
+A channel value can now be given a label by specifying it as a {value, label} object; this may affect the label used in axes, legends, and tips.
+
+This release also includes numerous bug fixes:
+- exposed ordinal domains are now correctly deduplicated;
+- the default symbol set is now inferred correctly when **fill** is *currentColor*;
+- the **fontVariant** axis option now applies to the axis label in addition to ticks;
+- the tip mark is no longer briefly visible before asynchronous rendering;
+- the bin transform no longer generates undefined colors for empty bins;
+- the bin transform now uses the **interval** option to reduce *x1* & *x2* (and *y1* & *y2*);
+- the stack transform now correctly handles the *exclude* **facet** option;
+- the tree transform now correctly handles escaping with the **delimiter** option.
+
+## 0.6.10
+
+[Released August 14, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.10)
+
+The new **title** and **subtitle** [plot options](https://observablehq.com/plot/features/plots#other-options) specify a primary and secondary heading. Headings are implemented as h2 and h3 elements by default, but you can provide existing elements instead of text for greater control. Like the existing **caption** option, headings add context and assist interpretation.
+
+
+
+```js
+Plot.plot({
+ title: "For charts, an informative title",
+ subtitle: "Subtitle to follow with additional context",
+ caption: "Figure 1. A chart with a title, subtitle, and caption.",
+ marks: [
+ Plot.frame(),
+ Plot.text(["Titles, subtitles, captions, and annotations assist interpretation by telling the reader what’s interesting. Don’t make the reader work to find what you already know."], {lineWidth: 30, frameAnchor: "middle"})
+ ]
+})
+```
+
+When a chart has a title, subtitle, caption, or legend, Plot automatically wraps the chart’s SVG element with an HTML figure element. The new **figure** plot option, if true, wraps the chart in a figure even if it doesn’t have these other elements; likewise, if false, Plot ignores these other elements and returns a bare SVG element. The figure element now has an associated class (`plot-d6a7b5-figure`).
+
+The new **clip** plot option determines the default clipping behavior if the [**clip** mark option](https://observablehq.com/plot/features/marks#mark-options) is not specified; set it to true to enable clipping. This option does not affect axis, grid, and frame marks, whose **clip** option defaults to false.
+
+
+
+```js
+Plot.plot({
+ clip: true,
+ x: {domain: [new Date(2015, 0, 1), new Date(2015, 3, 1)]},
+ y: {grid: true},
+ marks: [
+ Plot.areaY(aapl, {x: "Date", y: "Close", fillOpacity: 0.1}),
+ Plot.lineY(aapl, {x: "Date", y: "Close"}),
+ Plot.ruleY([0], {clip: false})
+ ]
+});
+```
+
+The new [bollinger mark](https://observablehq.com/plot/marks/bollinger) composes a line representing a moving average and an area representing volatility as a band; the band thickness is proportional to the deviation of nearby values. The bollinger mark is built on the new [bollinger map method](https://observablehq.com/plot/marks/bollinger#bollinger), and is often used to analyze the price of financial instruments such as stocks.
+
+
+
+```js
+Plot.bollingerY(aapl, {x: "Date", y: "Close", n: 20, k: 2}).plot()
+```
+
+The [arrow mark](https://observablehq.com/plot/marks/arrow) supports a new **sweep** option to control the bend orientation. Below, we set this option to *-y* to draw arrows bulging right, independent of the relative vertical positions of its source and target.
+
+[](https://observablehq.com/@observablehq/plot-arc-diagram?intent=fork)
+
+```js
+Plot.plot({
+ height: 1080,
+ marginLeft: 100,
+ axis: null,
+ x: {domain: [0, 1]}, // see https://github.com/observablehq/plot/issues/1541
+ color: {domain: d3.range(10), unknown: "#ccc"},
+ marks: [
+ Plot.dot(miserables.nodes, {x: 0, y: "id", fill: "group", sort: {y: "fill"}}),
+ Plot.text(miserables.nodes, {x: 0, y: "id", text: "id", textAnchor: "end", dx: -6, fill: "group"}),
+ Plot.arrow(miserables.links, {x: 0, y1: "source", y2: "target", sweep: "-y", bend: 90, headLength: 0, stroke: samegroup, sort: samegroup, reverse: true})
+ ]
+})
+```
+
+The [auto mark](https://observablehq.com/plot/marks/auto) now does a better job determining the appropriate bar mark implementation, such as with ordinal time series bar charts.
+
+
+
+```js
+Plot.auto(timeSeries, {x: "date", y: {value: "value", reduce: "sum"}, color: "type", mark: "bar"}).plot()
+```
+
+The [pointerX and pointerY transform](https://observablehq.com/plot/interactions/pointer) now use unscaled distance to decide the closest point across facets, preventing points from distant facets from being considered closest. The pointer transform now correctly reports the closest point when moving between facets, and no longer reports multiple closest points if they are the same distance across facets.
+
+Plot’s documentation now has an [API index](https://observablehq.com/plot/api), version badges pointing to the release notes for a particular feature (or to the pull request for a prerelease feature), and shorter anchors.
+
+The [tip mark](https://observablehq.com/plot/marks/tip) now shows both labels for paired channels such as *y1*–*y2* or *x1*–*y2* when the channel labels differ. When the **tip** option is set to true on a [geo mark](https://observablehq.com/plot/marks/geo) without a projection, as when using preprojected planar geometry, the display no longer collapses.
+
+The [stack transform](https://observablehq.com/plot/transforms/stack) now emits a friendlier error message when the supplied value is null.
+
+## 0.6.9
+
+[Released June 27, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.9)
+
+Time [axes](https://observablehq.com/plot/marks/axis) now default to multi-line ticks, greatly improving readability. When a tick has the same second field value as the previous tick (*e.g.*, “19 Jan” after “17 Jan”), only the first field (“19”) is shown for brevity. The tick format is now based on the tick interval and hence is always consistent, whereas the prior “multi-scale” format varied based on the date, such as “Jan 29” (for Sunday, January 29) and “Tue 31” (for Tuesday, January 31). The new ticks are similar to [Datawrapper](https://blog.datawrapper.de/new-axis-ticks/).
+
+Before:
+
+
+After:
+
+
+It is now easier to construct a “piecewise” [continuous scale](https://observablehq.com/plot/features/scales#continuous-scales) with more than two elements in the **domain** or **range**. This is most often used for a custom color scheme interpolating through fixed colors, such as this pleasing rainbow (sometimes used by artist [Dave Whyte](https://beesandbombs.com/), *a.k.a.* beesandbombs).
+
+
+
+```js
+Plot.plot({
+ color: {
+ type: "linear",
+ range: ["#d70441", "#f4e904", "#009978", "#5e3688"]
+ },
+ marks: [
+ Plot.cellX(d3.range(40), {fill: Plot.identity})
+ ]
+})
+```
+
+The [tree mark](https://observablehq.com/plot/marks/tree) now supports a **textLayout** option, which defaults to *mirrored* to alternate the orientation of labels for internal (non-leaf) *vs.* external (leaf) nodes. The treeNode and treeLink marks now also support a new **treeFilter** option, allowing these marks to be filtered without affecting the tree layout.
+
+
+
+```js
+Plot.plot({
+ axis: null,
+ height: 100,
+ margin: 10,
+ marginLeft: 40,
+ marginRight: 120,
+ marks: [
+ Plot.tree(gods, {textStroke: "white"})
+ ]
+})
+```
+
+The [barycentric interpolator](https://observablehq.com/plot/marks/raster#interpolatorBarycentric) used by the [raster](https://observablehq.com/plot/marks/raster) and [contour](https://observablehq.com/plot/marks/contour) marks now behaves correctly outside the convex hull of samples. The new algorithm (below right) radiates outwards from the hull, ensuring a continuous image; the old algorithm (below left) radiated inwards from values imputed on the frame’s edges, producing discontinuities.
+
+
+
+The [tip mark](https://observablehq.com/plot/marks/tip) now automatically sets the pointer-events attribute to *none* when associated with the [pointer transform](https://observablehq.com/plot/interactions/pointer) when the the pointer is not sticky, as when hovering a chart without clicking to lock the pointer. This prevents the tip mark from interfering with interaction on other marks, such as clickable links.
+
+The [auto mark](https://observablehq.com/plot/marks/auto) now renders as a cell, instead of a degenerate invisible rect, when **x** and **y** are both ordinal and the **mark** option is set to *bar*. The [tree mark](https://observablehq.com/plot/marks/tree) no longer produces duplicate tips with the **tip** option. The [rule mark](https://observablehq.com/plot/marks/rule) now respects the top-level **document** option, if any, when using the **clip** option. The [axis mark](https://observablehq.com/plot/marks/axis) now correctly handles the **sort**, **filter**, **reverse**, and **initializer** options.
+
+The **title**, **ariaLabel**, and **href** channels no longer filter by default; these channels may now be sparsely defined and the associated mark instance will still render.
+
+The [pointer transform](https://observablehq.com/plot/interactions/pointer) now handles non-faceted marks in faceted plots. The [window transform](https://observablehq.com/plot/transforms/window)’s *median*, *deviation*, *variance*, and percentile reducers have been fixed.
+
+## 0.6.8
+
+[Released June 2, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.8)
+
+The *x* and *y* scale default domains now incorporate geometry. This allows arbitrary polygonal geometry to be defined in abstract coordinates using the [geo mark](https://observablehq.com/plot/marks/geo) and then displayed using scales.
+
+
+
+```js
+Plot.geo({type: "LineString", coordinates: aapl.map((d) => [d.Date, d.Close])}).plot()
+```
+
+The stack transform’s **order** option now supports *-order* descending shorthand or a two-argument comparator. For example, you can use *-appearance* to sort by descending appearance. Below, a custom comparator is used to sort by ascending *group* and then descending *revenue*.
+
+
+
+```js
+Plot.plot({
+ y: {
+ grid: true,
+ label: "Annual revenue (billions, adj.)",
+ transform: (d) => d / 1000
+ },
+ marks: [
+ Plot.areaY(
+ riaa,
+ Plot.stackY({
+ x: "year",
+ y: "revenue",
+ z: "format",
+ order: (a, b) => d3.ascending(a.group, b.group) || d3.descending(a.revenue, b.revenue),
+ fill: "group",
+ stroke: "white"
+ })
+ ),
+ Plot.ruleY([0])
+ ]
+})
+```
+
+Fix the default **stroke** with the hexbin transform when used with the tip mark. Fix spurious high-cardinality warning with an odd number of elements when using varying aesthetics with the area or line mark.
+
+Fix *color* legends when the **domain** and **range** have different lengths: extra elements in the range are now ignored by the color legend, and a warning is issued.
+
+Fix duplicate application of scale transforms with the **tip** mark option. Fix erroneous implicit **title** channel with the **tip** mark option.
+
+## 0.6.7
+
+[Released May 24, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.7)
+
+The new [tip mark](https://observablehq.com/plot/marks/tip) displays text, or name-value pairs derived from channels, in a floating box anchored to a given position in **x** and **y**. The tip mark is often paired with the new [pointer interaction](https://observablehq.com/plot/interactions/pointer) such that only the point closest to the pointer is rendered, allowing the tip mark to reveal details interactively by hovering the chart.
+
+
+
+```js
+Plot.lineY(aapl, {x: "Date", y: "Close", tip: true}).plot()
+```
+
+The new **tip** mark option adds an implicit tip mark with pointer interaction derived from the current mark. The line chart above can be written more explicitly as:
+
+```js
+Plot.plot({
+ marks: [
+ Plot.lineY(aapl, {x: "Date", y: "Close"}),
+ Plot.tip(aapl, Plot.pointerX({x: "Date", y: "Close"}))
+ ]
+})
+```
+
+The tip mark can be also used to draw attention to points of interest and add commentary. When used with the **title** channel, the tip mark supports text wrapping and multi-line text.
+
+
+
+```js
+Plot.plot({
+ y: {grid: true},
+ marks: [
+ Plot.lineY(aapl, {x: "Date", y: "Close"}),
+ Plot.tip(
+ [`Apple stock reaches a new high of $133 on Feb. 23, 2015. The release of the first Apple Watch, slated for April, is hotly anticipated.`],
+ {x: new Date("2015-02-23"), y: 133, dy: -3, anchor: "bottom"}
+ ),
+ Plot.tip(
+ [`Apple stock drops 8% after the company misses Q2 revenue targets and reports declining iPhone sales. It reaches a two-year low of $90.34 on May 12.`],
+ {x: new Date("2016-05-12"), y: 90.34, dy: 3, anchor: "top"}
+ )
+ ]
+})
+```
+
+The pointer interaction can be paired with any mark, not just a tip: a red dot, say, to emphasize the focused point, or a rule to highlight its *x* or *y* position, or text to show a value. You can independently control the target position from the display using the **px** and **py** channels, say to show the currently-focused point’s value in the top-left corner.
+
+
+
+```js
+Plot.plot({
+ height: 160,
+ y: {axis: "right", grid: true, nice: true},
+ marks: [
+ Plot.lineY(aapl, {x: "Date", y: "Close"}),
+ Plot.ruleX(aapl, Plot.pointerX({x: "Date", py: "Close", stroke: "red"})),
+ Plot.dot(aapl, Plot.pointerX({x: "Date", y: "Close", stroke: "red"})),
+ Plot.text(aapl, Plot.pointerX({px: "Date", py: "Close", dy: -17, frameAnchor: "top-left", fontVariant: "tabular-nums", text: (d) => [`Date ${Plot.formatIsoDate(d.Date)}`, `Close ${d.Close.toFixed(2)}`].join(" ")}))
+ ]
+})
+```
+
+The pointer interaction supports both two-dimensional (pointer) and one-dimensional (pointerX and pointerY) pointing modes. Above, one-dimensional pointing is used for a time-series chart to find the closest *x*-value; below, two-dimensional pointing is used for a scatterplot to find the closest point in *x* and *y*.
+
+The pointer interaction also powers the new [crosshair mark](https://observablehq.com/plot/interactions/crosshair) which shows the *x* (horizontal↔︎ position) and *y* (vertical↕︎ position) value of the point closest to the pointer on the bottom and left sides of the frame, respectively.
+
+
+
+```js
+Plot.plot({
+ marks: [
+ Plot.dot(penguins, {x: "culmen_length_mm", y: "culmen_depth_mm", stroke: "sex"}),
+ Plot.crosshair(penguins, {x: "culmen_length_mm", y: "culmen_depth_mm"})
+ ]
+})
+```
+
+The pointer interaction (and by extension the crosshair mark and **tip** mark option) supports “click-to-stick”: if you click on the chart, the currently-focused point will remain locked until you click again. By temporarily locking the pointer, you can select text from the tip for copy and paste.
+
+The pointer interaction also emits an [input event](https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement/input_event) when the focused point changes, and assigns the generated plot’s value to the corresponding data. This allows event listeners to react to pointing, and supports [Observable views](https://observablehq.com/@observablehq/views).
+
+In addition to these exciting new interaction features, Plot 0.6.7 includes a variety of improvements and bug fixes.
+
+The **sort** mark option now supports *-channel* descending shorthand for [imputed scales domains](https://observablehq.com/plot/features/scales#sort-mark-option) and the [sort transform](https://observablehq.com/plot/transforms/sort). For example, sorting *x* by *-y* orders a bar chart by descending value:
+
+
+
+```js
+Plot.barY(alphabet, {x: "letter", y: "frequency", sort: {x: "-y"}}).plot()
+```
+
+This is equivalent to:
+
+```js
+Plot.barY(alphabet, {x: "letter", y: "frequency", sort: {x: "y", order: "descending"}}).plot()
+```
+
+The **sort** option for imputed channel domains now also takes an **order** option which can be *ascending*, *descending*, a custom comparator, or null. (This more closely matches the **order** option for the sort transform.) The **reverse** option now reverses the order rather than using descending natural order, placing nulls first instead of last.
+
+Previously setting the scale **label** option would disable Plot’s automatic directional arrow in axis labels (↑, →, ↓, or ←); now Plot implicitly adds an arrow to the label you provide, unless the label already has an arrow or you set the **labelArrow** option to *none* or false. You can also force an arrow by setting this option to true, or *up*, *right*, *down*, or *left*. In Plot’s code base, we were able to remove 161 Unicode arrows thanks to this change! 😌
+
+Color scales now default to a *categorical* scale when a categorical color scheme is specified. For example, this no longer throws a “unknown quantitative scheme” error:
+
+
+
+```js
+Plot.cellX(d3.range(10)).plot({color: {scheme: "Tableau10"}})
+```
+
+Time intervals can now be specified as integer multiples of a base time interval, such as *3 months* or *10 years*. Ordinal scales are now smarter about choosing a default time format when the **interval** option is a yearly interval: the four-digit year is (YYYY) shown instead of year, month, and day (YYYY-01-01).
+
+Mark transforms are now passed the plot’s *options* as a third argument; this allows the group, bin, and stack transforms (and other transforms) to check if the scale has a declared **interval** option, and if so, apply that interval before grouping. For example, to count athletes by age at 5-year intervals:
+
+
+
+```js
+Plot.barY(olympians, Plot.groupX({y: "count"}, {x: "date_of_birth"})).plot({x: {interval: "5 years"}})
+```
+
+The new **imageFilter** mark option applies a [CSS filter effect](https://developer.mozilla.org/en-US/docs/Web/CSS/filter) to the mark, such as a drop shadow or blur. The [rule](https://observablehq.com/plot/marks/rule) and [tick](https://observablehq.com/plot/marks/tick) marks now support **marker** options.
+
+The [normalize](https://observablehq.com/plot/transforms/normalize) and [window](https://observablehq.com/plot/transforms/window) transforms now accept two-argument “index-aware” reducers; the [map](https://observablehq.com/plot/transforms/map) transform now also accepts a two-argument index-aware map method. The mapX transform now defaults **x** to identity if none of **x**, **x1**, and **x2** are specified; the mapY transform does the same for **y**.
+
+When faceting, the plot dimensions now includes **facet**.**width** and **facet**.**height** for the plot’s outer dimensions. The plot context now includes the ownerSVGElement.
+
+Fix a bug where the *opacity* channel didn’t automatically opt-out of the *opacity* scale when all values are in the interval [0,1].
+
+Fix a bug where the frame mark would crash if a channel-eligible option such as **fill** were specified as an invalid CSS color string, such as *bleu*. The frame mark now supports channel-eligible options to be specified as abstract values, in which case the abstract values will be encoded using a scale; for example, setting **fill** to *Biscoe* would apply an *ordinal* *color* scale.
+
+Fix a bug where the position defaults for the rectX, rectY, barX, and barY marks were only applied if the *options* object was strictly undefined; now the position defaults apply if the relevant position options are undefined.
+
+Fix a bug when filtering facets and computing facet anchors with non-primitive facet domains, such as dates. Fix *z*-order across facets: each mark now draws atop all earlier marks across facets. Fix *z*-order of facet axes, which should be drawn below rather than above other marks, as other axes do.
+
+Fix the auto mark to choose the rect mark instead of rectX or rectY when appropriate, and likewise choose correctly between line, lineX, and lineY, and areaX and areaY. Also, the autoSpec method now returns the name of the explicit mark and transform implementations as **markImpl** and **transformImpl** respectively, along with the **markOptions** and **transformOptions** needed to instantiate them.
+
+
+
+
+
+
+
+## 0.6.6
+
+[Released April 26, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.6)
+
+**Plot has a new documentation website!**
+
+👉 https://observablehq.com/plot 👈
+
+The [image mark](https://observablehq.com/plot/marks/image) can now generate circular images with the **r** channel, and rotate images with the **rotate** channel.
+
+The [axis mark](https://observablehq.com/plot/marks/axis) now properly respects the **margin** shorthand option, changing the default for **marginTop**, **marginRight**, **marginBottom**, and **marginLeft**. The axis mark now correctly renders the axis label when the **href** option is used, or any other option that may be interpreted as a channel.
+
+Facet scale domains are now imputed correctly when the **sort** mark option is used with a **limit**, or otherwise causing the facet domain to be truncated. Plot no longer generates a spurious warning when faceting and using non-array data, such as an Arquero table. The **interval** scale option, when expressed as a fractional number such as 0.2, now has better floating point precision.
+
+The [Plot.indexOf](https://observablehq.com/plot/features/transforms#indexOf) channel transform, used internally by some mark shorthand, is now exported.
+
+Plot has a few improvements for server-side rendering. Plot now assumes a high pixel density display when headless. The default class name for plots is now deterministically generated (`plot-d6a7b5`) rather than randomly generated; this makes it easier to apply overrides to Plot’s default styles with an external stylesheet. (The default class name will change if Plot’s default styles change in a future release.) The **className** plot option is now inherited by a plot’s legends, if any. The density mark now respects the Plot’s **document** option, and the **caption** option now uses a duck test instead of testing against the global Node.
+
+## 0.6.5
+
+[Released April 1, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.5)
+
+**TypeScript!** The feature you asked for most is here. 🎉 Plot now exports TypeScript type declarations (.d.ts) with precise types and documentation for all of Plot’s public API. This allows code editors such as VS Code to offer better suggestions, complete with inline documentation. For example, if you type “Plot.g”, Plot.geo is now suggested:
+
+
+
+Function calls also have hints, showing their expected arguments.
+
+
+
+Types are especially useful for suggesting well-known names, such as projections, scale types, and color schemes.
+
+
+
+TypeScript also helps for reading code; hover any symbol and immediately see the relevant documentation.
+
+
+
+Lastly, if you’re using Plot in a TypeScript project, you can now benefit from additional static type checking to find errors.
+
+
+
+If you have suggestions or clarifications on how to improve the documentation, please open an issue or discussion, or send us a pull request.
+
+The **strokeOpacity**, **fillOpacity**, and **opacity** channels now opt-out of the *opacity* scale by default when all values are numbers in the range [0, 1]; to opt-in to the *opacity* scale, specify the channel as {value, scale: true}. The raster mark now also supports scale overrides.
+
+The scale **nice** option now accepts named time intervals such as *day* and *week*. The new *quarter* and *half* time intervals represent quarters (three months) and half-years (six months). The quantile scale now supports the **unknown** option. BigInt channels are now supported; values are coerced to numbers. Color legends now support the **opacity** option.
+
+The text mark no longer crashes when a NaN tick value is specified for an axis. The **className** option now allows mixed-case class names. Channel-derived scale domains now correctly handle non-orderable values, duplicate values, and mark transforms. Plot.auto now renders correctly when a non-zero **reduce** is used in conjunction with the *bar* **mark**, and in some cases when the **zero** option is specified. The **zero** option no longer changes the default mark type, and Plot.autoSpec now materializes the **zero** option.
+
+The interfaces for reduce and map implementations have changed. To better disambiguate from arrays, reduce implementations now have a *reduceIndex* method instead of a *reduce* method, and map implementations now have a *mapIndex* method instead of a *map* method. The old interfaces are still supported, but now deprecated.
+
+## 0.6.4
+
+[Released February 28, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.4)
+
+The new top-level [**aspectRatio** option](https://observablehq.com/plot/features/plots#aspectRatio) changes the default plot **height** such that, assuming both *x* and *y* are *linear* scales, a scaled unit distance in *x* divided by a scaled unit distance in *y* is the given aspect ratio. For example, if *x* and *y* represent the same units (say, degrees Fahrenheit), and if the **aspectRatio** is one, then scaled distances in *x* and *y* will be equivalent.
+
+
+
+```js
+Plot.plot({
+ aspectRatio: 1,
+ grid: true,
+ x: {label: "Daily low temperature (°F) →", tickSpacing: 40},
+ y: {label: "↑ Daily temperature variation (Δ°F)", tickSpacing: 40},
+ color: {scheme: "rainbow", nice: true, legend: true, tickFormat: "%b"},
+ marks: [
+ Plot.ruleY([0]),
+ Plot.dot(weather, {
+ x: (d) => d.temp_min,
+ y: (d) => d.temp_max - d.temp_min,
+ fill: (d) => (d = new Date(d.date), d.setUTCFullYear(2020), d)
+ })
+ ]
+})
+```
+
+The new **textOverflow** option for the [text mark](https://observablehq.com/plot/marks/text) allows text to be truncated when a line of text is longer than the specified **lineWidth**. Overflowing characters can either be clipped (*clip*) or replaced with an ellipsis (*ellipsis*), either at the start, middle, or end of each line.
+
+
+
+When wrapping or truncating, the text mark now more accurately estimates the width of ellipses and emojis, and no longer separates combining marks or emoji character sequences such as 👨👩👧👦.
+
+The [link mark](https://observablehq.com/plot/marks/link) now respects the current [**projection**](https://observablehq.com/plot/features/projections), if any, given the default [**curve**](https://observablehq.com/plot/features/curves) of *auto*. This matches the behavior of the line mark. To opt-out of the projection and draw a straight line, set the **curve** to *linear*.
+
+The [image mark](https://observablehq.com/plot/marks/image) now supports the **imageRendering** option. (Note: Safari currently ignores the SVG image-rendering attribute.)
+
+You can now override the scale for a given [mark channel](https://observablehq.com/plot/features/marks#marks-have-channels) by specifying the corresponding option as a {value, scale} object. For example, to force the **stroke** channel to be unscaled, interpreting the associated values as literal color strings:
+
+```js
+Plot.dot(data, {stroke: {value: "foo", scale: null}})
+```
+
+To instead force the **stroke** channel to be bound to the *color* scale regardless of the provided values, say:
+
+```js
+Plot.dot(data, {stroke: {value: "foo", scale: "color"}})
+```
+
+Color channels (**fill** and **stroke**) are bound to the *color* scale by default, unless the provided values are all valid CSS color strings or nullish, in which case the values are interpreted literally and unscaled. Likewise, if the dot mark’s **symbol** channel values are all symbols, symbol names, or nullish, values are interpreted literally and unscaled; otherwise, the channel is bound to the *symbol* scale. (If some color channels are literal values while other color channels are not, the channels with literal values will now automatically opt-out of the color scale; the same goes for symbol channels. This deviates from the previous behavior, where *all* channels associated with a scale were required to be literal values in order to have the scale default to an *identity* scale.)
+
+The mark [**facetAnchor** option](https://observablehq.com/plot/features/facets#facetAnchor) can now be set to *empty* such that a mark is only rendered on empty facets. This is typically used for annotation.
+
+The new Plot.autoSpec method takes *data* and *options* suitable for [Plot.auto](https://observablehq.com/plot/marks/auto) and returns a corresponding *options* object with default options realized. While intended primarily as an internal helper, Plot.autoSpec may be useful for debugging by letting you inspect which mark and reducers are chosen by Plot.auto.
+
+Fix Plot.auto to only default to the *bar* mark if *x* or *y* is zeroed. Fix Plot.auto’s support for the *area* mark. Fix Plot.auto’s use of the *bar* mark with possibly ordinal reducers. Fix a bug where arrays of values could be erroneously interpreted as reducers. Fix a crash when the mark **facet** option is set to *exclude*, but the mark is not faceted; the option is now ignored. Fix a crash coercing BigInt values to numbers.
+
+## 0.6.3
+
+[Released February 6, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.3)
+
+The new [auto mark](https://observablehq.com/plot/marks/auto) ([Plot.auto](https://observablehq.com/plot/marks/auto#auto)) automatically selects a mark type that best represents the given dimensions of data according to some simple heuristics. For example,
+
+[](https://observablehq.com/plot/marks/auto)
+
+```js
+Plot.auto(olympians, {x: "height", y: "weight"}).plot()
+```
+
+makes a scatterplot (equivalent to [dot](https://observablehq.com/plot/marks/dot)); adding **color** as
+
+[](https://observablehq.com/plot/marks/auto)
+
+```js
+Plot.auto(olympians, {x: "height", y: "weight", color: "count"}).plot()
+```
+
+makes a heatmap (equivalent to [rect](https://observablehq.com/plot/marks/rect) and [bin](https://observablehq.com/plot/transforms/bin); chosen since _height_ and _weight_ are quantitative); switching to
+
+[](https://observablehq.com/plot/marks/auto)
+
+```js
+Plot.auto(aapl, {x: "Date", y: "Close"}).plot()
+```
+
+makes a line chart (equivalent to [lineY](https://observablehq.com/plot/marks/line#lineY); chosen because the selected *x* dimension *Date* is temporal and monotonic, _i.e._, the data is in chronological order);
+
+[](https://observablehq.com/plot/marks/auto)
+
+```js
+Plot.auto(penguins, {x: "body_mass_g"}).plot()
+```
+
+makes a histogram (equivalent to [rectY](https://observablehq.com/plot/marks/rect#rectY) and [binX](https://observablehq.com/plot/transforms/bin#binX); chosen because the _body_mass_g_ column is quantitative); and
+
+[](https://observablehq.com/plot/marks/auto)
+
+```js
+Plot.auto(penguins, {x: "island"}).plot()
+```
+
+makes a bar chart (equivalent to [barY](https://observablehq.com/plot/marks/bar#barY) and [groupX](https://observablehq.com/plot/transforms/group#groupX); chosen because the _island_ column is categorical). The auto mark is intended to support fast exploratory analysis where the goal is to get a useful plot as quickly as possible. It’s also great if you’re new to Plot, since you can get started with a minimal API.
+
+Plot’s new [axis](https://observablehq.com/plot/marks/axis) and [grid](https://observablehq.com/plot/marks/grid) marks allow customization and styling of axes. This has been one of our most asked-for features, closing more than a dozen feature requests (see [#1197](https://github.com/observablehq/plot/pull/1197))! The new axis mark composes a [vector](https://observablehq.com/plot/marks/vector) for tick marks and a [text](https://observablehq.com/plot/marks/text) for tick and axis labels. As such, you can use the rich capabilities of these marks, such the **lineWidth** option to wrap long text labels.
+
+[](https://observablehq.com/plot/marks/auto)
+
+```js
+Plot.plot({
+ y: {percent: true},
+ marks: [
+ Plot.axisX({label: null, lineWidth: 8, marginBottom: 40}),
+ Plot.axisY({label: "↑ Responses (%)"}),
+ Plot.barY(responses, {x: "name", y: "value"}),
+ Plot.ruleY([0])
+ ]
+})
+```
+
+And since axes and grids are now proper marks, you can interleave them with other marks, for example to produce ggplot2-style axes with a gray background and white grid lines.
+
+[](https://observablehq.com/plot/marks/auto)
+
+```js
+Plot.plot({
+ inset: 10,
+ marks: [
+ Plot.frame({fill: "#eaeaea"}),
+ Plot.gridY({stroke: "#fff", strokeOpacity: 1}),
+ Plot.gridX({stroke: "#fff", strokeOpacity: 1}),
+ Plot.line(aapl, {x: "Date", y: "Close"})
+ ]
+})
+```
+
+The *x* and *y* axes are now automatically repeated in empty facets, improving readability by reducing eye travel to read tick values. Below, note that the *x* axis for culmen depth (with ticks at 15 and 20 mm) is rendered below the Adelie/null-sex facet in the top-right.
+
+[](ttps://observablehq.com/plot/marks/axis)
+
+```js
+Plot.plot({
+ facet: {marginRight: 80},
+ marks: [
+ Plot.dot(penguins, {x: "culmen_depth_mm", y: "culmen_length_mm", stroke: "#ddd"}),
+ Plot.frame(),
+ Plot.gridX(),
+ Plot.gridY(),
+ Plot.dot(penguins, {x: "culmen_depth_mm", y: "culmen_length_mm", fx: "sex", fy: "species"})
+ ]
+})
+```
+
+See [Plot: Axes](https://observablehq.com/plot/marks/axis) for more examples, including the new _both_ **axis** option to repeat axes on both sides of the plot, dashed grid lines via the **strokeDasharray** option, data-driven tick placement, and layering axes to show hierarchical time intervals (years, months, weeks).
+
+Marks can now declare default margins via the **marginTop**, **marginRight**, **marginBottom**, and **marginLeft** options, and the **margin** shorthand. For each side, the maximum corresponding margin across marks becomes the plot’s default. While most marks default to zero margins (because they are drawn inside the chart area), Plot‘s axis mark provides default margins depending on their anchor. The facet margin options (*e.g.*, facet.**marginRight**) now correctly affect the positioning of the *x* and *y* axis labels.
+
+The new [*mark*.**facetAnchor**](https://observablehq.com/plot/features/facets#facetAnchor) mark option controls the facets in which the mark will appear when faceting. It defaults to null for all marks except for axis marks, where it defaults to *top-empty* if the axis anchor is *top*, *right-empty* if anchor is *right*, *bottom-empty* if anchor is *bottom*, and *left-empty* if anchor is *left*. This ensures the proper positioning of the axes with respect to empty facets.
+
+The [frame mark](https://observablehq.com/plot/marks/frame)’s new **anchor** option allows you to draw a line on one side of the frame (as opposed to the default behavior where a rect is drawn around all four sides); this feature is now used by the *scale*.**line** option for *x* and *y* scales. The [text mark](https://observablehq.com/plot/marks/text) now supports soft hyphens (`\xad`); lines are now eligible to break at soft hyphens, in which case a hyphen (-) will appear at the end of the line before the break. The [raster mark](https://observablehq.com/plot/marks/raster) no longer crashes with an _identity_ color scale. The [voronoi mark](https://observablehq.com/plot/marks/delaunay#voronoi) now correctly respects the **target**, **mixBlendMode**, and **opacity** options.
+
+## 0.6.2
+
+[Released January 18, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.2)
+
+The new [raster mark](https://observablehq.com/plot/marks/raster) and [contour mark](https://observablehq.com/plot/marks/contour) generate a raster image and smooth contours, respectively, from spatial samples. For example, the plot below shows a gridded digital elevation model of Maungawhau (R’s [`volcano` dataset](./test/data/volcano.json)) with contours every 10 meters:
+
+[](https://observablehq.com/plot/marks/raster)
+
+```js
+Plot.plot({
+ color: {legend: true, label: "Height (m)"},
+ marks: [
+ Plot.raster(volcano.values, {width: volcano.width, height: volcano.height}),
+ Plot.contour(volcano.values, {width: volcano.width, height: volcano.height, interval: 10})
+ ]
+})
+```
+
+For non-gridded or sparse data, the raster and contour marks implement a variety of [spatial interpolation methods](https://observablehq.com/plot/marks/raster#spatial-interpolators) to populate the raster grid. The *barycentric* interpolation method, shown below with data from the [Great Britain aeromagnetic survey](https://www.bgs.ac.uk/datasets/gb-aeromagnetic-survey/), uses barycentric coordinates from a Delaunay triangulation of the samples (small black dots).
+
+[](https://observablehq.com/plot/marks/raster)
+
+```js
+Plot.plot({
+ width: 640,
+ height: 484,
+ inset: 4,
+ x: {tickFormat: "s"},
+ y: {tickFormat: "s", ticks: 5},
+ color: {type: "diverging", legend: true},
+ marks: [
+ Plot.raster(ca55, {x: "LONGITUDE", y: "LATITUDE", fill: "MAG_IGRF90", interpolate: "barycentric"}),
+ Plot.dot(ca55, {x: "LONGITUDE", y: "LATITUDE", r: 0.75, fill: "currentColor"})
+ ]
+})
+```
+
+The same data, with a smidge of blur, as filled contours in projected coordinates:
+
+[](https://observablehq.com/plot/marks/contour)
+
+```js
+Plot.plot({
+ width: 640,
+ height: 484,
+ color: {type: "diverging", legend: true},
+ projection: {type: "reflect-y", domain: {type: "MultiPoint", coordinates: ca55.map((d) => [d.GRID_EAST, d.GRID_NORTH])}},
+ marks: [Plot.contour(ca55, {x: "GRID_EAST", y: "GRID_NORTH", fill: "MAG_IGRF90", stroke: "currentColor", blur: 2})]
+})
+```
+
+Naturally, the raster and contour mark are compatible with Plot’s [projection system](https://observablehq.com/plot/features/projections), allowing spatial samples to be shown in any geographic projection and in conjunction with other geographic data. The *equirectangular* projection is the natural choice for this gridded global water vapor dataset from [NASA Earth Observations](https://neo.gsfc.nasa.gov/view.php?datasetId=MYDAL2_M_SKY_WV&date=2022-11-01).
+
+[](https://observablehq.com/plot/marks/raster)
+
+```js
+Plot.plot({
+ projection: "equirectangular",
+ color: {
+ scheme: "ylgnbu",
+ unknown: "#ccc",
+ legend: true,
+ label: "Water vapor (cm)"
+ },
+ marks: [
+ Plot.raster(vapor, {
+ x1: -180,
+ y1: 90,
+ x2: 180,
+ y2: -90,
+ width: 720,
+ height: 360
+ }),
+ Plot.graticule(),
+ Plot.frame()
+ ]
+})
+```
+
+The raster and contour mark also support sampling continuous spatial functions *f*(*x*, *y*). For example, here is the famous Mandelbrot set, with color encoding the number of iterations before the point escapes:
+
+[](https://observablehq.com/plot/marks/raster)
+
+```js
+Plot.plot({
+ height: 500,
+ marks: [
+ Plot.raster({
+ fill: (x, y) => {
+ for (let n = 0, zr = 0, zi = 0; n < 80; ++n) {
+ [zr, zi] = [zr * zr - zi * zi + x, 2 * zr * zi + y];
+ if (zr * zr + zi * zi > 4) return n;
+ }
+ },
+ x1: -2,
+ y1: -1.164,
+ x2: 1,
+ y2: 1.164,
+ pixelSize: 0.5
+ })
+ ]
+})
+```
+
+The [vector mark](https://observablehq.com/plot/marks/vector) now supports the **shape** constant option; the built-in shapes are *arrow* (default) and *spike*. A custom shape can also be implemented, returning the corresponding SVG path data for the desired shape. The new [spike convenience constructor](https://observablehq.com/plot/marks/vector#spike) creates a vector suitable for spike maps. The vector mark also now supports an **r** constant option to set the shape radius.
+
+[](https://observablehq.com/@observablehq/plot-spike)
+
+```js
+Plot.plot({
+ width: 960,
+ height: 600,
+ projection: "albers-usa",
+ length: {
+ range: [0, 200]
+ },
+ marks: [
+ Plot.geo(nation, {fill: "#e0e0e0"}),
+ Plot.geo(statemesh, {stroke: "white"}),
+ Plot.spike(counties.features, Plot.geoCentroid({stroke: "red", length: (d) => population.get(d.id)}))
+ ]
+});
+```
+
+The new [geoCentroid transform](https://observablehq.com/plot/transforms/centroid#geoCentroid) and [centroid initializer](https://observablehq.com/plot/transforms/centroid#centroid) compute the spherical and projected planar centroids of geometry, respectively. The new [identity](https://observablehq.com/plot/features/transforms#identity) channel helper returns a source array as-is, avoiding an extra copy.
+
+The **interval** option now supports named time intervals such as “sunday” and “hour”, equivalent to the corresponding d3-time interval (_e.g._, d3.utcSunday and d3.utcHour). The [bin transform](https://observablehq.com/plot/transforms/bin) is now many times faster, especially when there are many bins and when binning temporal data.
+
+Diverging scales now correctly handle descending domains. When the stack **order** option is used without a *z* channel, a helpful error message is now thrown. The **clip** option *frame* now correctly handles band scales. Using D3 7.8, generated SVG path data is now rounded to three decimal points to reduce output size. Fix a crash when a facet scale’s domain includes a value for which there is no corresponding facet data. The bin, group, and hexbin transforms now correctly ignore undefined outputs. Upgrade D3 to 7.8.2.
+
+---
+
+For earlier changes, continue to the [2022 CHANGELOG](./CHANGELOG-2022.md).
diff --git a/CHANGELOG.md b/CHANGELOG.md
index a8cdc36caa..9cd26467c2 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,855 +1,19 @@
# Observable Plot - Changelog
-Year: **Current (2023)** · [2022](./CHANGELOG-2022.md) · [2021](./CHANGELOG-2021.md)
+Year: **Current (2024)** · [2023](./CHANGELOG-2023.md) · [2022](./CHANGELOG-2022.md) · [2021](./CHANGELOG-2021.md)
-## 0.6.13
+## 0.6.14
-[Released December 13, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.13)
+[Released March 12, 2024.](https://github.com/observablehq/plot/releases/tag/v0.6.14)
-Fixes the *observable10* categorical color scheme. In 0.6.12, we inadvertently published an earlier draft; this fixed version better aligns the order of colors with *tableau10*.
+Changes the default categorical color scheme to *Observable10*.
-We reduced the specificity of selectors in the generated stylesheets to 0, allowing consumers of Plot to much more easily override these styles using their own stylesheets while still enjoying the default styling. Thanks, @mstade!
+The group transform now preserves the input order of groups by default, making it easier to sort groups by using the **sort** option. The group and bin transforms now support the *z* reducer.
-## 0.6.12
+Improves the accessibility of axes by hidding tick marks and grid lines from the accessibility tree.
-[Released December 7, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.12)
-
-To better support dark mode, we’ve made a breaking change to default styles: the background color is now `unset` (transparent) instead of `white`. Additionally, marks that expect to use the same color as the background (for instance, the tip mark) now use the CSS custom property `--plot-background` instead of `white`.
-
-As since [version 0.6.6](#066), you can override Plot’s styles via the `plot-d6a7b5` class. For example, the following stylesheet applies a dark background and white foreground:
-
-```css
-svg.plot-d6a7b5 {
- --plot-background: #333;
- background: var(--plot-background);
- color: white;
-}
-```
-
-Previously Plot forced you to choose between [rect](https://observablehq.com/plot/marks/rect), [bar](https://observablehq.com/plot/marks/bar), and [cell](https://observablehq.com/plot/marks/cell) based on whether the *x* and *y* scales were ordinal or quantitative. Rejoice: the [rect mark](https://observablehq.com/plot/marks/rect) now supports *band* scales so you can simply use rect in all cases! 🎉 (The bar and cell marks still offer conveniences for ordinal scales, so use them if you like.)
-
-
-
-```js
-Plot.rectY(alphabet, {x: "letter", y: "frequency"}).plot()
-```
-
-Categorical color scales now default to the new *observable10* color scheme by [Jeff Pettiross](https://github.com/pettiross). These colors are intended as a drop-in replacement for *tableau10* with similar ease of discrimination and ordering, but with a slightly more saturated vibe that helps charts pop.
-
-
-
-```js
-Plot.cellX(d3.range(10)).plot({color: {type: "categorical"}})
-```
-
-The new [difference mark](https://observablehq.com/plot/marks/difference) puts a metric in context by comparing it to another metric or constant value. Like the [area mark](https://observablehq.com/plot/marks/area), the region between two lines is filled; unlike the area mark, alternating color shows when the metric is above or below the comparison.
-
-
-
-```js
-Plot.differenceY(gistemp, Plot.windowY(28, {x: "Date", y: "Anomaly"})).plot()
-```
-
-The chart above shows a moving average of global temperature anomaly; above-average temperatures are shown in green and below-average temperatures are shown in blue.
-
-The difference mark can also compare a metric to its earlier self, showing change over time. The chart below shows the year-over-year change in Apple stock price; the gray region represents an increase over last year while the red region represents a decrease.
-
-
-
-```js
-Plot.differenceY(
- aapl,
- Plot.shiftX("year", {
- x: "Date",
- y: "Close",
- positiveFillOpacity: 0.2,
- positiveFill: "currentColor",
- negativeFillOpacity: 0.8,
- negativeFill: "red"
- })
-).plot()
-```
-
-The chart above is constructed using the new [shift transform](https://observablehq.com/plot/transforms/shift), which derives a time-shifted copy of a metric for the given time interval, enabling year–over-year, month-over-month, or any other period-over-period comparison.
-
-The new [find reducer](https://observablehq.com/plot/transforms/group#find) allows you to pivot data with the [bin](https://observablehq.com/plot/transforms/bin) and [group](https://observablehq.com/plot/transforms/group) transforms, effectively turning “tall” data (with fewer columns) into “wide” data (with more columns). For example, say you have time-series data that has separate rows for observed daily temperatures in San Francisco (SF) and San Jose (SJ):
-
-```csv
-date,station,tmax,tmin
-2020-12-31,SJ,59,43
-2020-12-31,SF,60,47
-2020-12-30,SJ,58,33
-2020-12-30,SF,57,40
-2020-12-29,SJ,62,38
-2020-12-29,SF,61,41
-2020-12-28,SJ,57,43
-2020-12-28,SF,56,47
-2020-12-27,SJ,62,43
-2020-12-27,SF,57,46
-2020-12-26,SJ,60,46
-2020-12-26,SF,61,49
-```
-
-To compare the average minimum temperatures of San Francisco and San Jose as another difference chart:
-
-
-
-```js
-Plot.plot({
- x: {tickFormat: "%b"},
- y: {grid: true},
- marks: [
- Plot.ruleY([32]),
- Plot.differenceY(
- temperature,
- Plot.windowY(
- 14,
- Plot.groupX(
- {y1: Plot.find((d) => d.station === "SJ"), y2: Plot.find((d) => d.station === "SF")},
- {x: "date", y: "tmin"}
- )
- )
- )
- ]
-})
-```
-
-The green region above represents when San Francisco was warmer than San Jose, and the blue region when San Francisco was cooler than San Jose.
-
-The [tip mark](https://observablehq.com/plot/marks/tip) now supports a **preferredAnchor** option, providing greater control over tip placement: if the tip fits within the frame at the preferred anchor, this anchor will be used. (In contrast, the **anchor** option uses the specified anchor regardless of whether it will fit.) The tip mark now also prefers the *bottom* anchor by default for a more traditional, less comic-like appearance.
-
-The [**marker** option](https://observablehq.com/plot/features/markers) now supports new marker types: *tick*, *tick-x*, and *tick-y*. The [bin](https://observablehq.com/plot/transforms/bin) and [group](https://observablehq.com/plot/transforms/group) transforms now pass _data_ to reducers. The [group](https://observablehq.com/plot/transforms/group) and [hexbin](https://observablehq.com/plot/transforms/hexbin) transforms now support _x_ and _y_ reducers.
-
-This release includes several additional fixes:
-
-* The default axis for a *time* scale now uses local time.
-* The text mark’s **lineWidth** option is now more accurate when **monospace** is true.
-* The tip mark no longer truncates **title** text.
-* The scale **type** option is now case-insensitive.
-* Transform type definitions have correct overload precedence.
-
-## 0.6.11
-
-[Released September 20, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.11)
-
-The **tip** mark option can now pass options to the derived [tip mark](https://observablehq.com/plot/marks/tip); the options object can also specify the **pointer** option to control the derived tip’s pointer mode (_x_, _y_, or _xy_). The new **format** tip mark option enables greater control over order and formatting of channels.
-
-
-
-```js
-Plot.dot(olympians, {
- x: "weight",
- y: "height",
- stroke: "sex",
- channels: {
- name: "name",
- nationality: "nationality",
- sport: "sport"
- },
- tip: {
- format: {
- name: true, // show name first
- y: (d) => `${d}m`, // units in meters
- x: (d) => `${d}kg`, // units in kilograms
- stroke: false // suppress stroke channel
- }
- }
-}).plot()
-```
-
-Axes for ordinal scales now generalize the scale’s temporal or quantitative **interval** if any, resulting in more readable ticks. For instance, the bar chart below of monthly values now sports multi-line tick labels.
-
-
-
-```js
-Plot.plot({
- x: {interval: "month"},
- marks: [
- Plot.barY(aapl, Plot.groupX({y: "median"}, {x: "Date", y: "Close"}))
- ]
-})
-```
-
-Plot now recognizes CSS Color Module [Level 4](https://www.w3.org/TR/css-color-4/) and [Level 5](https://www.w3.org/TR/css-color-5/) syntax as literal colors, making it easier to use modern color syntax such as [oklab()](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/oklab), [color-mix()](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/color-mix), and alternative color spaces such as [display-p3](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value/color).
-
-A channel value can now be given a label by specifying it as a {value, label} object; this may affect the label used in axes, legends, and tips.
-
-This release also includes numerous bug fixes:
-- exposed ordinal domains are now correctly deduplicated;
-- the default symbol set is now inferred correctly when **fill** is *currentColor*;
-- the **fontVariant** axis option now applies to the axis label in addition to ticks;
-- the tip mark is no longer briefly visible before asynchronous rendering;
-- the bin transform no longer generates undefined colors for empty bins;
-- the bin transform now uses the **interval** option to reduce *x1* & *x2* (and *y1* & *y2*);
-- the stack transform now correctly handles the *exclude* **facet** option;
-- the tree transform now correctly handles escaping with the **delimiter** option.
-
-## 0.6.10
-
-[Released August 14, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.10)
-
-The new **title** and **subtitle** [plot options](https://observablehq.com/plot/features/plots#other-options) specify a primary and secondary heading. Headings are implemented as h2 and h3 elements by default, but you can provide existing elements instead of text for greater control. Like the existing **caption** option, headings add context and assist interpretation.
-
-
-
-```js
-Plot.plot({
- title: "For charts, an informative title",
- subtitle: "Subtitle to follow with additional context",
- caption: "Figure 1. A chart with a title, subtitle, and caption.",
- marks: [
- Plot.frame(),
- Plot.text(["Titles, subtitles, captions, and annotations assist interpretation by telling the reader what’s interesting. Don’t make the reader work to find what you already know."], {lineWidth: 30, frameAnchor: "middle"})
- ]
-})
-```
-
-When a chart has a title, subtitle, caption, or legend, Plot automatically wraps the chart’s SVG element with an HTML figure element. The new **figure** plot option, if true, wraps the chart in a figure even if it doesn’t have these other elements; likewise, if false, Plot ignores these other elements and returns a bare SVG element. The figure element now has an associated class (`plot-d6a7b5-figure`).
-
-The new **clip** plot option determines the default clipping behavior if the [**clip** mark option](https://observablehq.com/plot/features/marks#mark-options) is not specified; set it to true to enable clipping. This option does not affect axis, grid, and frame marks, whose **clip** option defaults to false.
-
-
-
-```js
-Plot.plot({
- clip: true,
- x: {domain: [new Date(2015, 0, 1), new Date(2015, 3, 1)]},
- y: {grid: true},
- marks: [
- Plot.areaY(aapl, {x: "Date", y: "Close", fillOpacity: 0.1}),
- Plot.lineY(aapl, {x: "Date", y: "Close"}),
- Plot.ruleY([0], {clip: false})
- ]
-});
-```
-
-The new [bollinger mark](https://observablehq.com/plot/marks/bollinger) composes a line representing a moving average and an area representing volatility as a band; the band thickness is proportional to the deviation of nearby values. The bollinger mark is built on the new [bollinger map method](https://observablehq.com/plot/marks/bollinger#bollinger), and is often used to analyze the price of financial instruments such as stocks.
-
-
-
-```js
-Plot.bollingerY(aapl, {x: "Date", y: "Close", n: 20, k: 2}).plot()
-```
-
-The [arrow mark](https://observablehq.com/plot/marks/arrow) supports a new **sweep** option to control the bend orientation. Below, we set this option to *-y* to draw arrows bulging right, independent of the relative vertical positions of its source and target.
-
-[](https://observablehq.com/@observablehq/plot-arc-diagram?intent=fork)
-
-```js
-Plot.plot({
- height: 1080,
- marginLeft: 100,
- axis: null,
- x: {domain: [0, 1]}, // see https://github.com/observablehq/plot/issues/1541
- color: {domain: d3.range(10), unknown: "#ccc"},
- marks: [
- Plot.dot(miserables.nodes, {x: 0, y: "id", fill: "group", sort: {y: "fill"}}),
- Plot.text(miserables.nodes, {x: 0, y: "id", text: "id", textAnchor: "end", dx: -6, fill: "group"}),
- Plot.arrow(miserables.links, {x: 0, y1: "source", y2: "target", sweep: "-y", bend: 90, headLength: 0, stroke: samegroup, sort: samegroup, reverse: true})
- ]
-})
-```
-
-The [auto mark](https://observablehq.com/plot/marks/auto) now does a better job determining the appropriate bar mark implementation, such as with ordinal time series bar charts.
-
-
-
-```js
-Plot.auto(timeSeries, {x: "date", y: {value: "value", reduce: "sum"}, color: "type", mark: "bar"}).plot()
-```
-
-The [pointerX and pointerY transform](https://observablehq.com/plot/interactions/pointer) now use unscaled distance to decide the closest point across facets, preventing points from distant facets from being considered closest. The pointer transform now correctly reports the closest point when moving between facets, and no longer reports multiple closest points if they are the same distance across facets.
-
-Plot’s documentation now has an [API index](https://observablehq.com/plot/api), version badges pointing to the release notes for a particular feature (or to the pull request for a prerelease feature), and shorter anchors.
-
-The [tip mark](https://observablehq.com/plot/marks/tip) now shows both labels for paired channels such as *y1*–*y2* or *x1*–*y2* when the channel labels differ. When the **tip** option is set to true on a [geo mark](https://observablehq.com/plot/marks/geo) without a projection, as when using preprojected planar geometry, the display no longer collapses.
-
-The [stack transform](https://observablehq.com/plot/transforms/stack) now emits a friendlier error message when the supplied value is null.
-
-## 0.6.9
-
-[Released June 27, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.9)
-
-Time [axes](https://observablehq.com/plot/marks/axis) now default to multi-line ticks, greatly improving readability. When a tick has the same second field value as the previous tick (*e.g.*, “19 Jan” after “17 Jan”), only the first field (“19”) is shown for brevity. The tick format is now based on the tick interval and hence is always consistent, whereas the prior “multi-scale” format varied based on the date, such as “Jan 29” (for Sunday, January 29) and “Tue 31” (for Tuesday, January 31). The new ticks are similar to [Datawrapper](https://blog.datawrapper.de/new-axis-ticks/).
-
-Before:
-
-
-After:
-
-
-It is now easier to construct a “piecewise” [continuous scale](https://observablehq.com/plot/features/scales#continuous-scales) with more than two elements in the **domain** or **range**. This is most often used for a custom color scheme interpolating through fixed colors, such as this pleasing rainbow (sometimes used by artist [Dave Whyte](https://beesandbombs.com/), *a.k.a.* beesandbombs).
-
-
-
-```js
-Plot.plot({
- color: {
- type: "linear",
- range: ["#d70441", "#f4e904", "#009978", "#5e3688"]
- },
- marks: [
- Plot.cellX(d3.range(40), {fill: Plot.identity})
- ]
-})
-```
-
-The [tree mark](https://observablehq.com/plot/marks/tree) now supports a **textLayout** option, which defaults to *mirrored* to alternate the orientation of labels for internal (non-leaf) *vs.* external (leaf) nodes. The treeNode and treeLink marks now also support a new **treeFilter** option, allowing these marks to be filtered without affecting the tree layout.
-
-
-
-```js
-Plot.plot({
- axis: null,
- height: 100,
- margin: 10,
- marginLeft: 40,
- marginRight: 120,
- marks: [
- Plot.tree(gods, {textStroke: "white"})
- ]
-})
-```
-
-The [barycentric interpolator](https://observablehq.com/plot/marks/raster#interpolatorBarycentric) used by the [raster](https://observablehq.com/plot/marks/raster) and [contour](https://observablehq.com/plot/marks/contour) marks now behaves correctly outside the convex hull of samples. The new algorithm (below right) radiates outwards from the hull, ensuring a continuous image; the old algorithm (below left) radiated inwards from values imputed on the frame’s edges, producing discontinuities.
-
-
-
-The [tip mark](https://observablehq.com/plot/marks/tip) now automatically sets the pointer-events attribute to *none* when associated with the [pointer transform](https://observablehq.com/plot/interactions/pointer) when the the pointer is not sticky, as when hovering a chart without clicking to lock the pointer. This prevents the tip mark from interfering with interaction on other marks, such as clickable links.
-
-The [auto mark](https://observablehq.com/plot/marks/auto) now renders as a cell, instead of a degenerate invisible rect, when **x** and **y** are both ordinal and the **mark** option is set to *bar*. The [tree mark](https://observablehq.com/plot/marks/tree) no longer produces duplicate tips with the **tip** option. The [rule mark](https://observablehq.com/plot/marks/rule) now respects the top-level **document** option, if any, when using the **clip** option. The [axis mark](https://observablehq.com/plot/marks/axis) now correctly handles the **sort**, **filter**, **reverse**, and **initializer** options.
-
-The **title**, **ariaLabel**, and **href** channels no longer filter by default; these channels may now be sparsely defined and the associated mark instance will still render.
-
-The [pointer transform](https://observablehq.com/plot/interactions/pointer) now handles non-faceted marks in faceted plots. The [window transform](https://observablehq.com/plot/transforms/window)’s *median*, *deviation*, *variance*, and percentile reducers have been fixed.
-
-## 0.6.8
-
-[Released June 2, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.8)
-
-The *x* and *y* scale default domains now incorporate geometry. This allows arbitrary polygonal geometry to be defined in abstract coordinates using the [geo mark](https://observablehq.com/plot/marks/geo) and then displayed using scales.
-
-
-
-```js
-Plot.geo({type: "LineString", coordinates: aapl.map((d) => [d.Date, d.Close])}).plot()
-```
-
-The stack transform’s **order** option now supports *-order* descending shorthand or a two-argument comparator. For example, you can use *-appearance* to sort by descending appearance. Below, a custom comparator is used to sort by ascending *group* and then descending *revenue*.
-
-
-
-```js
-Plot.plot({
- y: {
- grid: true,
- label: "Annual revenue (billions, adj.)",
- transform: (d) => d / 1000
- },
- marks: [
- Plot.areaY(
- riaa,
- Plot.stackY({
- x: "year",
- y: "revenue",
- z: "format",
- order: (a, b) => d3.ascending(a.group, b.group) || d3.descending(a.revenue, b.revenue),
- fill: "group",
- stroke: "white"
- })
- ),
- Plot.ruleY([0])
- ]
-})
-```
-
-Fix the default **stroke** with the hexbin transform when used with the tip mark. Fix spurious high-cardinality warning with an odd number of elements when using varying aesthetics with the area or line mark.
-
-Fix *color* legends when the **domain** and **range** have different lengths: extra elements in the range are now ignored by the color legend, and a warning is issued.
-
-Fix duplicate application of scale transforms with the **tip** mark option. Fix erroneous implicit **title** channel with the **tip** mark option.
-
-## 0.6.7
-
-[Released May 24, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.7)
-
-The new [tip mark](https://observablehq.com/plot/marks/tip) displays text, or name-value pairs derived from channels, in a floating box anchored to a given position in **x** and **y**. The tip mark is often paired with the new [pointer interaction](https://observablehq.com/plot/interactions/pointer) such that only the point closest to the pointer is rendered, allowing the tip mark to reveal details interactively by hovering the chart.
-
-
-
-```js
-Plot.lineY(aapl, {x: "Date", y: "Close", tip: true}).plot()
-```
-
-The new **tip** mark option adds an implicit tip mark with pointer interaction derived from the current mark. The line chart above can be written more explicitly as:
-
-```js
-Plot.plot({
- marks: [
- Plot.lineY(aapl, {x: "Date", y: "Close"}),
- Plot.tip(aapl, Plot.pointerX({x: "Date", y: "Close"}))
- ]
-})
-```
-
-The tip mark can be also used to draw attention to points of interest and add commentary. When used with the **title** channel, the tip mark supports text wrapping and multi-line text.
-
-
-
-```js
-Plot.plot({
- y: {grid: true},
- marks: [
- Plot.lineY(aapl, {x: "Date", y: "Close"}),
- Plot.tip(
- [`Apple stock reaches a new high of $133 on Feb. 23, 2015. The release of the first Apple Watch, slated for April, is hotly anticipated.`],
- {x: new Date("2015-02-23"), y: 133, dy: -3, anchor: "bottom"}
- ),
- Plot.tip(
- [`Apple stock drops 8% after the company misses Q2 revenue targets and reports declining iPhone sales. It reaches a two-year low of $90.34 on May 12.`],
- {x: new Date("2016-05-12"), y: 90.34, dy: 3, anchor: "top"}
- )
- ]
-})
-```
-
-The pointer interaction can be paired with any mark, not just a tip: a red dot, say, to emphasize the focused point, or a rule to highlight its *x* or *y* position, or text to show a value. You can independently control the target position from the display using the **px** and **py** channels, say to show the currently-focused point’s value in the top-left corner.
-
-
-
-```js
-Plot.plot({
- height: 160,
- y: {axis: "right", grid: true, nice: true},
- marks: [
- Plot.lineY(aapl, {x: "Date", y: "Close"}),
- Plot.ruleX(aapl, Plot.pointerX({x: "Date", py: "Close", stroke: "red"})),
- Plot.dot(aapl, Plot.pointerX({x: "Date", y: "Close", stroke: "red"})),
- Plot.text(aapl, Plot.pointerX({px: "Date", py: "Close", dy: -17, frameAnchor: "top-left", fontVariant: "tabular-nums", text: (d) => [`Date ${Plot.formatIsoDate(d.Date)}`, `Close ${d.Close.toFixed(2)}`].join(" ")}))
- ]
-})
-```
-
-The pointer interaction supports both two-dimensional (pointer) and one-dimensional (pointerX and pointerY) pointing modes. Above, one-dimensional pointing is used for a time-series chart to find the closest *x*-value; below, two-dimensional pointing is used for a scatterplot to find the closest point in *x* and *y*.
-
-The pointer interaction also powers the new [crosshair mark](https://observablehq.com/plot/interactions/crosshair) which shows the *x* (horizontal↔︎ position) and *y* (vertical↕︎ position) value of the point closest to the pointer on the bottom and left sides of the frame, respectively.
-
-
-
-```js
-Plot.plot({
- marks: [
- Plot.dot(penguins, {x: "culmen_length_mm", y: "culmen_depth_mm", stroke: "sex"}),
- Plot.crosshair(penguins, {x: "culmen_length_mm", y: "culmen_depth_mm"})
- ]
-})
-```
-
-The pointer interaction (and by extension the crosshair mark and **tip** mark option) supports “click-to-stick”: if you click on the chart, the currently-focused point will remain locked until you click again. By temporarily locking the pointer, you can select text from the tip for copy and paste.
-
-The pointer interaction also emits an [input event](https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement/input_event) when the focused point changes, and assigns the generated plot’s value to the corresponding data. This allows event listeners to react to pointing, and supports [Observable views](https://observablehq.com/@observablehq/views).
-
-In addition to these exciting new interaction features, Plot 0.6.7 includes a variety of improvements and bug fixes.
-
-The **sort** mark option now supports *-channel* descending shorthand for [imputed scales domains](https://observablehq.com/plot/features/scales#sort-mark-option) and the [sort transform](https://observablehq.com/plot/transforms/sort). For example, sorting *x* by *-y* orders a bar chart by descending value:
-
-
-
-```js
-Plot.barY(alphabet, {x: "letter", y: "frequency", sort: {x: "-y"}}).plot()
-```
-
-This is equivalent to:
-
-```js
-Plot.barY(alphabet, {x: "letter", y: "frequency", sort: {x: "y", order: "descending"}}).plot()
-```
-
-The **sort** option for imputed channel domains now also takes an **order** option which can be *ascending*, *descending*, a custom comparator, or null. (This more closely matches the **order** option for the sort transform.) The **reverse** option now reverses the order rather than using descending natural order, placing nulls first instead of last.
-
-Previously setting the scale **label** option would disable Plot’s automatic directional arrow in axis labels (↑, →, ↓, or ←); now Plot implicitly adds an arrow to the label you provide, unless the label already has an arrow or you set the **labelArrow** option to *none* or false. You can also force an arrow by setting this option to true, or *up*, *right*, *down*, or *left*. In Plot’s code base, we were able to remove 161 Unicode arrows thanks to this change! 😌
-
-Color scales now default to a *categorical* scale when a categorical color scheme is specified. For example, this no longer throws a “unknown quantitative scheme” error:
-
-
-
-```js
-Plot.cellX(d3.range(10)).plot({color: {scheme: "Tableau10"}})
-```
-
-Time intervals can now be specified as integer multiples of a base time interval, such as *3 months* or *10 years*. Ordinal scales are now smarter about choosing a default time format when the **interval** option is a yearly interval: the four-digit year is (YYYY) shown instead of year, month, and day (YYYY-01-01).
-
-Mark transforms are now passed the plot’s *options* as a third argument; this allows the group, bin, and stack transforms (and other transforms) to check if the scale has a declared **interval** option, and if so, apply that interval before grouping. For example, to count athletes by age at 5-year intervals:
-
-
-
-```js
-Plot.barY(olympians, Plot.groupX({y: "count"}, {x: "date_of_birth"})).plot({x: {interval: "5 years"}})
-```
-
-The new **imageFilter** mark option applies a [CSS filter effect](https://developer.mozilla.org/en-US/docs/Web/CSS/filter) to the mark, such as a drop shadow or blur. The [rule](https://observablehq.com/plot/marks/rule) and [tick](https://observablehq.com/plot/marks/tick) marks now support **marker** options.
-
-The [normalize](https://observablehq.com/plot/transforms/normalize) and [window](https://observablehq.com/plot/transforms/window) transforms now accept two-argument “index-aware” reducers; the [map](https://observablehq.com/plot/transforms/map) transform now also accepts a two-argument index-aware map method. The mapX transform now defaults **x** to identity if none of **x**, **x1**, and **x2** are specified; the mapY transform does the same for **y**.
-
-When faceting, the plot dimensions now includes **facet**.**width** and **facet**.**height** for the plot’s outer dimensions. The plot context now includes the ownerSVGElement.
-
-Fix a bug where the *opacity* channel didn’t automatically opt-out of the *opacity* scale when all values are in the interval [0,1].
-
-Fix a bug where the frame mark would crash if a channel-eligible option such as **fill** were specified as an invalid CSS color string, such as *bleu*. The frame mark now supports channel-eligible options to be specified as abstract values, in which case the abstract values will be encoded using a scale; for example, setting **fill** to *Biscoe* would apply an *ordinal* *color* scale.
-
-Fix a bug where the position defaults for the rectX, rectY, barX, and barY marks were only applied if the *options* object was strictly undefined; now the position defaults apply if the relevant position options are undefined.
-
-Fix a bug when filtering facets and computing facet anchors with non-primitive facet domains, such as dates. Fix *z*-order across facets: each mark now draws atop all earlier marks across facets. Fix *z*-order of facet axes, which should be drawn below rather than above other marks, as other axes do.
-
-Fix the auto mark to choose the rect mark instead of rectX or rectY when appropriate, and likewise choose correctly between line, lineX, and lineY, and areaX and areaY. Also, the autoSpec method now returns the name of the explicit mark and transform implementations as **markImpl** and **transformImpl** respectively, along with the **markOptions** and **transformOptions** needed to instantiate them.
-
-
-
-
-
-
-
-## 0.6.6
-
-[Released April 26, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.6)
-
-**Plot has a new documentation website!**
-
-👉 https://observablehq.com/plot 👈
-
-The [image mark](https://observablehq.com/plot/marks/image) can now generate circular images with the **r** channel, and rotate images with the **rotate** channel.
-
-The [axis mark](https://observablehq.com/plot/marks/axis) now properly respects the **margin** shorthand option, changing the default for **marginTop**, **marginRight**, **marginBottom**, and **marginLeft**. The axis mark now correctly renders the axis label when the **href** option is used, or any other option that may be interpreted as a channel.
-
-Facet scale domains are now imputed correctly when the **sort** mark option is used with a **limit**, or otherwise causing the facet domain to be truncated. Plot no longer generates a spurious warning when faceting and using non-array data, such as an Arquero table. The **interval** scale option, when expressed as a fractional number such as 0.2, now has better floating point precision.
-
-The [Plot.indexOf](https://observablehq.com/plot/features/transforms#indexOf) channel transform, used internally by some mark shorthand, is now exported.
-
-Plot has a few improvements for server-side rendering. Plot now assumes a high pixel density display when headless. The default class name for plots is now deterministically generated (`plot-d6a7b5`) rather than randomly generated; this makes it easier to apply overrides to Plot’s default styles with an external stylesheet. (The default class name will change if Plot’s default styles change in a future release.) The **className** plot option is now inherited by a plot’s legends, if any. The density mark now respects the Plot’s **document** option, and the **caption** option now uses a duck test instead of testing against the global Node.
-
-## 0.6.5
-
-[Released April 1, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.5)
-
-**TypeScript!** The feature you asked for most is here. 🎉 Plot now exports TypeScript type declarations (.d.ts) with precise types and documentation for all of Plot’s public API. This allows code editors such as VS Code to offer better suggestions, complete with inline documentation. For example, if you type “Plot.g”, Plot.geo is now suggested:
-
-
-
-Function calls also have hints, showing their expected arguments.
-
-
-
-Types are especially useful for suggesting well-known names, such as projections, scale types, and color schemes.
-
-
-
-TypeScript also helps for reading code; hover any symbol and immediately see the relevant documentation.
-
-
-
-Lastly, if you’re using Plot in a TypeScript project, you can now benefit from additional static type checking to find errors.
-
-
-
-If you have suggestions or clarifications on how to improve the documentation, please open an issue or discussion, or send us a pull request.
-
-The **strokeOpacity**, **fillOpacity**, and **opacity** channels now opt-out of the *opacity* scale by default when all values are numbers in the range [0, 1]; to opt-in to the *opacity* scale, specify the channel as {value, scale: true}. The raster mark now also supports scale overrides.
-
-The scale **nice** option now accepts named time intervals such as *day* and *week*. The new *quarter* and *half* time intervals represent quarters (three months) and half-years (six months). The quantile scale now supports the **unknown** option. BigInt channels are now supported; values are coerced to numbers. Color legends now support the **opacity** option.
-
-The text mark no longer crashes when a NaN tick value is specified for an axis. The **className** option now allows mixed-case class names. Channel-derived scale domains now correctly handle non-orderable values, duplicate values, and mark transforms. Plot.auto now renders correctly when a non-zero **reduce** is used in conjunction with the *bar* **mark**, and in some cases when the **zero** option is specified. The **zero** option no longer changes the default mark type, and Plot.autoSpec now materializes the **zero** option.
-
-The interfaces for reduce and map implementations have changed. To better disambiguate from arrays, reduce implementations now have a *reduceIndex* method instead of a *reduce* method, and map implementations now have a *mapIndex* method instead of a *map* method. The old interfaces are still supported, but now deprecated.
-
-## 0.6.4
-
-[Released February 28, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.4)
-
-The new top-level [**aspectRatio** option](https://observablehq.com/plot/features/plots#aspectRatio) changes the default plot **height** such that, assuming both *x* and *y* are *linear* scales, a scaled unit distance in *x* divided by a scaled unit distance in *y* is the given aspect ratio. For example, if *x* and *y* represent the same units (say, degrees Fahrenheit), and if the **aspectRatio** is one, then scaled distances in *x* and *y* will be equivalent.
-
-
-
-```js
-Plot.plot({
- aspectRatio: 1,
- grid: true,
- x: {label: "Daily low temperature (°F) →", tickSpacing: 40},
- y: {label: "↑ Daily temperature variation (Δ°F)", tickSpacing: 40},
- color: {scheme: "rainbow", nice: true, legend: true, tickFormat: "%b"},
- marks: [
- Plot.ruleY([0]),
- Plot.dot(weather, {
- x: (d) => d.temp_min,
- y: (d) => d.temp_max - d.temp_min,
- fill: (d) => (d = new Date(d.date), d.setUTCFullYear(2020), d)
- })
- ]
-})
-```
-
-The new **textOverflow** option for the [text mark](https://observablehq.com/plot/marks/text) allows text to be truncated when a line of text is longer than the specified **lineWidth**. Overflowing characters can either be clipped (*clip*) or replaced with an ellipsis (*ellipsis*), either at the start, middle, or end of each line.
-
-
-
-When wrapping or truncating, the text mark now more accurately estimates the width of ellipses and emojis, and no longer separates combining marks or emoji character sequences such as 👨👩👧👦.
-
-The [link mark](https://observablehq.com/plot/marks/link) now respects the current [**projection**](https://observablehq.com/plot/features/projections), if any, given the default [**curve**](https://observablehq.com/plot/features/curves) of *auto*. This matches the behavior of the line mark. To opt-out of the projection and draw a straight line, set the **curve** to *linear*.
-
-The [image mark](https://observablehq.com/plot/marks/image) now supports the **imageRendering** option. (Note: Safari currently ignores the SVG image-rendering attribute.)
-
-You can now override the scale for a given [mark channel](https://observablehq.com/plot/features/marks#marks-have-channels) by specifying the corresponding option as a {value, scale} object. For example, to force the **stroke** channel to be unscaled, interpreting the associated values as literal color strings:
-
-```js
-Plot.dot(data, {stroke: {value: "foo", scale: null}})
-```
-
-To instead force the **stroke** channel to be bound to the *color* scale regardless of the provided values, say:
-
-```js
-Plot.dot(data, {stroke: {value: "foo", scale: "color"}})
-```
-
-Color channels (**fill** and **stroke**) are bound to the *color* scale by default, unless the provided values are all valid CSS color strings or nullish, in which case the values are interpreted literally and unscaled. Likewise, if the dot mark’s **symbol** channel values are all symbols, symbol names, or nullish, values are interpreted literally and unscaled; otherwise, the channel is bound to the *symbol* scale. (If some color channels are literal values while other color channels are not, the channels with literal values will now automatically opt-out of the color scale; the same goes for symbol channels. This deviates from the previous behavior, where *all* channels associated with a scale were required to be literal values in order to have the scale default to an *identity* scale.)
-
-The mark [**facetAnchor** option](https://observablehq.com/plot/features/facets#facetAnchor) can now be set to *empty* such that a mark is only rendered on empty facets. This is typically used for annotation.
-
-The new Plot.autoSpec method takes *data* and *options* suitable for [Plot.auto](https://observablehq.com/plot/marks/auto) and returns a corresponding *options* object with default options realized. While intended primarily as an internal helper, Plot.autoSpec may be useful for debugging by letting you inspect which mark and reducers are chosen by Plot.auto.
-
-Fix Plot.auto to only default to the *bar* mark if *x* or *y* is zeroed. Fix Plot.auto’s support for the *area* mark. Fix Plot.auto’s use of the *bar* mark with possibly ordinal reducers. Fix a bug where arrays of values could be erroneously interpreted as reducers. Fix a crash when the mark **facet** option is set to *exclude*, but the mark is not faceted; the option is now ignored. Fix a crash coercing BigInt values to numbers.
-
-## 0.6.3
-
-[Released February 6, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.3)
-
-The new [auto mark](https://observablehq.com/plot/marks/auto) ([Plot.auto](https://observablehq.com/plot/marks/auto#auto)) automatically selects a mark type that best represents the given dimensions of data according to some simple heuristics. For example,
-
-[](https://observablehq.com/plot/marks/auto)
-
-```js
-Plot.auto(olympians, {x: "height", y: "weight"}).plot()
-```
-
-makes a scatterplot (equivalent to [dot](https://observablehq.com/plot/marks/dot)); adding **color** as
-
-[](https://observablehq.com/plot/marks/auto)
-
-```js
-Plot.auto(olympians, {x: "height", y: "weight", color: "count"}).plot()
-```
-
-makes a heatmap (equivalent to [rect](https://observablehq.com/plot/marks/rect) and [bin](https://observablehq.com/plot/transforms/bin); chosen since _height_ and _weight_ are quantitative); switching to
-
-[](https://observablehq.com/plot/marks/auto)
-
-```js
-Plot.auto(aapl, {x: "Date", y: "Close"}).plot()
-```
-
-makes a line chart (equivalent to [lineY](https://observablehq.com/plot/marks/line#lineY); chosen because the selected *x* dimension *Date* is temporal and monotonic, _i.e._, the data is in chronological order);
-
-[](https://observablehq.com/plot/marks/auto)
-
-```js
-Plot.auto(penguins, {x: "body_mass_g"}).plot()
-```
-
-makes a histogram (equivalent to [rectY](https://observablehq.com/plot/marks/rect#rectY) and [binX](https://observablehq.com/plot/transforms/bin#binX); chosen because the _body_mass_g_ column is quantitative); and
-
-[](https://observablehq.com/plot/marks/auto)
-
-```js
-Plot.auto(penguins, {x: "island"}).plot()
-```
-
-makes a bar chart (equivalent to [barY](https://observablehq.com/plot/marks/bar#barY) and [groupX](https://observablehq.com/plot/transforms/group#groupX); chosen because the _island_ column is categorical). The auto mark is intended to support fast exploratory analysis where the goal is to get a useful plot as quickly as possible. It’s also great if you’re new to Plot, since you can get started with a minimal API.
-
-Plot’s new [axis](https://observablehq.com/plot/marks/axis) and [grid](https://observablehq.com/plot/marks/grid) marks allow customization and styling of axes. This has been one of our most asked-for features, closing more than a dozen feature requests (see [#1197](https://github.com/observablehq/plot/pull/1197))! The new axis mark composes a [vector](https://observablehq.com/plot/marks/vector) for tick marks and a [text](https://observablehq.com/plot/marks/text) for tick and axis labels. As such, you can use the rich capabilities of these marks, such the **lineWidth** option to wrap long text labels.
-
-[](https://observablehq.com/plot/marks/auto)
-
-```js
-Plot.plot({
- y: {percent: true},
- marks: [
- Plot.axisX({label: null, lineWidth: 8, marginBottom: 40}),
- Plot.axisY({label: "↑ Responses (%)"}),
- Plot.barY(responses, {x: "name", y: "value"}),
- Plot.ruleY([0])
- ]
-})
-```
-
-And since axes and grids are now proper marks, you can interleave them with other marks, for example to produce ggplot2-style axes with a gray background and white grid lines.
-
-[](https://observablehq.com/plot/marks/auto)
-
-```js
-Plot.plot({
- inset: 10,
- marks: [
- Plot.frame({fill: "#eaeaea"}),
- Plot.gridY({stroke: "#fff", strokeOpacity: 1}),
- Plot.gridX({stroke: "#fff", strokeOpacity: 1}),
- Plot.line(aapl, {x: "Date", y: "Close"})
- ]
-})
-```
-
-The *x* and *y* axes are now automatically repeated in empty facets, improving readability by reducing eye travel to read tick values. Below, note that the *x* axis for culmen depth (with ticks at 15 and 20 mm) is rendered below the Adelie/null-sex facet in the top-right.
-
-[](ttps://observablehq.com/plot/marks/axis)
-
-```js
-Plot.plot({
- facet: {marginRight: 80},
- marks: [
- Plot.dot(penguins, {x: "culmen_depth_mm", y: "culmen_length_mm", stroke: "#ddd"}),
- Plot.frame(),
- Plot.gridX(),
- Plot.gridY(),
- Plot.dot(penguins, {x: "culmen_depth_mm", y: "culmen_length_mm", fx: "sex", fy: "species"})
- ]
-})
-```
-
-See [Plot: Axes](https://observablehq.com/plot/marks/axis) for more examples, including the new _both_ **axis** option to repeat axes on both sides of the plot, dashed grid lines via the **strokeDasharray** option, data-driven tick placement, and layering axes to show hierarchical time intervals (years, months, weeks).
-
-Marks can now declare default margins via the **marginTop**, **marginRight**, **marginBottom**, and **marginLeft** options, and the **margin** shorthand. For each side, the maximum corresponding margin across marks becomes the plot’s default. While most marks default to zero margins (because they are drawn inside the chart area), Plot‘s axis mark provides default margins depending on their anchor. The facet margin options (*e.g.*, facet.**marginRight**) now correctly affect the positioning of the *x* and *y* axis labels.
-
-The new [*mark*.**facetAnchor**](https://observablehq.com/plot/features/facets#facetAnchor) mark option controls the facets in which the mark will appear when faceting. It defaults to null for all marks except for axis marks, where it defaults to *top-empty* if the axis anchor is *top*, *right-empty* if anchor is *right*, *bottom-empty* if anchor is *bottom*, and *left-empty* if anchor is *left*. This ensures the proper positioning of the axes with respect to empty facets.
-
-The [frame mark](https://observablehq.com/plot/marks/frame)’s new **anchor** option allows you to draw a line on one side of the frame (as opposed to the default behavior where a rect is drawn around all four sides); this feature is now used by the *scale*.**line** option for *x* and *y* scales. The [text mark](https://observablehq.com/plot/marks/text) now supports soft hyphens (`\xad`); lines are now eligible to break at soft hyphens, in which case a hyphen (-) will appear at the end of the line before the break. The [raster mark](https://observablehq.com/plot/marks/raster) no longer crashes with an _identity_ color scale. The [voronoi mark](https://observablehq.com/plot/marks/delaunay#voronoi) now correctly respects the **target**, **mixBlendMode**, and **opacity** options.
-
-## 0.6.2
-
-[Released January 18, 2023.](https://github.com/observablehq/plot/releases/tag/v0.6.2)
-
-The new [raster mark](https://observablehq.com/plot/marks/raster) and [contour mark](https://observablehq.com/plot/marks/contour) generate a raster image and smooth contours, respectively, from spatial samples. For example, the plot below shows a gridded digital elevation model of Maungawhau (R’s [`volcano` dataset](./test/data/volcano.json)) with contours every 10 meters:
-
-[](https://observablehq.com/plot/marks/raster)
-
-```js
-Plot.plot({
- color: {legend: true, label: "Height (m)"},
- marks: [
- Plot.raster(volcano.values, {width: volcano.width, height: volcano.height}),
- Plot.contour(volcano.values, {width: volcano.width, height: volcano.height, interval: 10})
- ]
-})
-```
-
-For non-gridded or sparse data, the raster and contour marks implement a variety of [spatial interpolation methods](https://observablehq.com/plot/marks/raster#spatial-interpolators) to populate the raster grid. The *barycentric* interpolation method, shown below with data from the [Great Britain aeromagnetic survey](https://www.bgs.ac.uk/datasets/gb-aeromagnetic-survey/), uses barycentric coordinates from a Delaunay triangulation of the samples (small black dots).
-
-[](https://observablehq.com/plot/marks/raster)
-
-```js
-Plot.plot({
- width: 640,
- height: 484,
- inset: 4,
- x: {tickFormat: "s"},
- y: {tickFormat: "s", ticks: 5},
- color: {type: "diverging", legend: true},
- marks: [
- Plot.raster(ca55, {x: "LONGITUDE", y: "LATITUDE", fill: "MAG_IGRF90", interpolate: "barycentric"}),
- Plot.dot(ca55, {x: "LONGITUDE", y: "LATITUDE", r: 0.75, fill: "currentColor"})
- ]
-})
-```
-
-The same data, with a smidge of blur, as filled contours in projected coordinates:
-
-[](https://observablehq.com/plot/marks/contour)
-
-```js
-Plot.plot({
- width: 640,
- height: 484,
- color: {type: "diverging", legend: true},
- projection: {type: "reflect-y", domain: {type: "MultiPoint", coordinates: ca55.map((d) => [d.GRID_EAST, d.GRID_NORTH])}},
- marks: [Plot.contour(ca55, {x: "GRID_EAST", y: "GRID_NORTH", fill: "MAG_IGRF90", stroke: "currentColor", blur: 2})]
-})
-```
-
-Naturally, the raster and contour mark are compatible with Plot’s [projection system](https://observablehq.com/plot/features/projections), allowing spatial samples to be shown in any geographic projection and in conjunction with other geographic data. The *equirectangular* projection is the natural choice for this gridded global water vapor dataset from [NASA Earth Observations](https://neo.gsfc.nasa.gov/view.php?datasetId=MYDAL2_M_SKY_WV&date=2022-11-01).
-
-[](https://observablehq.com/plot/marks/raster)
-
-```js
-Plot.plot({
- projection: "equirectangular",
- color: {
- scheme: "ylgnbu",
- unknown: "#ccc",
- legend: true,
- label: "Water vapor (cm)"
- },
- marks: [
- Plot.raster(vapor, {
- x1: -180,
- y1: 90,
- x2: 180,
- y2: -90,
- width: 720,
- height: 360
- }),
- Plot.graticule(),
- Plot.frame()
- ]
-})
-```
-
-The raster and contour mark also support sampling continuous spatial functions *f*(*x*, *y*). For example, here is the famous Mandelbrot set, with color encoding the number of iterations before the point escapes:
-
-[](https://observablehq.com/plot/marks/raster)
-
-```js
-Plot.plot({
- height: 500,
- marks: [
- Plot.raster({
- fill: (x, y) => {
- for (let n = 0, zr = 0, zi = 0; n < 80; ++n) {
- [zr, zi] = [zr * zr - zi * zi + x, 2 * zr * zi + y];
- if (zr * zr + zi * zi > 4) return n;
- }
- },
- x1: -2,
- y1: -1.164,
- x2: 1,
- y2: 1.164,
- pixelSize: 0.5
- })
- ]
-})
-```
-
-The [vector mark](https://observablehq.com/plot/marks/vector) now supports the **shape** constant option; the built-in shapes are *arrow* (default) and *spike*. A custom shape can also be implemented, returning the corresponding SVG path data for the desired shape. The new [spike convenience constructor](https://observablehq.com/plot/marks/vector#spike) creates a vector suitable for spike maps. The vector mark also now supports an **r** constant option to set the shape radius.
-
-[](https://observablehq.com/@observablehq/plot-spike)
-
-```js
-Plot.plot({
- width: 960,
- height: 600,
- projection: "albers-usa",
- length: {
- range: [0, 200]
- },
- marks: [
- Plot.geo(nation, {fill: "#e0e0e0"}),
- Plot.geo(statemesh, {stroke: "white"}),
- Plot.spike(counties.features, Plot.geoCentroid({stroke: "red", length: (d) => population.get(d.id)}))
- ]
-});
-```
-
-The new [geoCentroid transform](https://observablehq.com/plot/transforms/centroid#geoCentroid) and [centroid initializer](https://observablehq.com/plot/transforms/centroid#centroid) compute the spherical and projected planar centroids of geometry, respectively. The new [identity](https://observablehq.com/plot/features/transforms#identity) channel helper returns a source array as-is, avoiding an extra copy.
-
-The **interval** option now supports named time intervals such as “sunday” and “hour”, equivalent to the corresponding d3-time interval (_e.g._, d3.utcSunday and d3.utcHour). The [bin transform](https://observablehq.com/plot/transforms/bin) is now many times faster, especially when there are many bins and when binning temporal data.
-
-Diverging scales now correctly handle descending domains. When the stack **order** option is used without a *z* channel, a helpful error message is now thrown. The **clip** option *frame* now correctly handles band scales. Using D3 7.8, generated SVG path data is now rounded to three decimal points to reduce output size. Fix a crash when a facet scale’s domain includes a value for which there is no corresponding facet data. The bin, group, and hexbin transforms now correctly ignore undefined outputs. Upgrade D3 to 7.8.2.
+Upgrades D3 to 7.9.0.
---
-For earlier changes, continue to the [2022 CHANGELOG](./CHANGELOG-2022.md).
+For earlier changes, continue to the [2023 CHANGELOG](./CHANGELOG-2023.md).
diff --git a/package.json b/package.json
index bceac5688c..59da041c0d 100644
--- a/package.json
+++ b/package.json
@@ -1,7 +1,7 @@
{
"name": "@observablehq/plot",
"description": "A JavaScript library for exploratory data visualization.",
- "version": "0.6.13",
+ "version": "0.6.14",
"author": {
"name": "Observable, Inc.",
"url": "https://observablehq.com"