Changelog
v17.0.0-beta.1 (2026-06-29)
Section titled “v17.0.0-beta.1 (2026-06-29)”The second 17.0 beta, continuing to iron out the pivot-table kinks from beta.0. Still subject to further breaking changes before the stable release.
Features
Section titled “Features”Workbook.deleteCells(range, shift)andWorkbook.insertCells(range, shift)implement Excel’s “Delete cells → shift up/left” and “Insert cells → shift down/right” for an arbitrary bounded rectangle (e.g.B3:C5), the bounded-rectangle counterparts to the existing full-row/columndeleteRows/deleteColumns/insertRows/insertColumns. Surviving cells slide in (delete) or displaced cells push out (insert) to fill or open the gap; this workbook’s own formulas are rewritten in place, with references to removed or pushed-off cells becoming#REF!, and aRewriteFormulacallback is returned to adjust other workbooks’ references. Overlapping tables, pivot tables, and merges move whole, grow, or dissolve per Excel’s rules; an edit that would tear a table or pivot table, or push non-empty content off the grid, is refused.- New
PivotTable.pageAreareports the geometry of the page-field (report filter) area that sits above a pivot’s output range, so a renderer can locate and style those cells. - New
PivotTable.rowRoles/PivotTable.colRoleslist which output rows and columns are subheadings or subtotals (as 0-based offsets from the output’s top-left, each with its kind and 1-based nesting level), so a renderer can apply a declared pivot style’s subheading and subtotal styling.
Bug fixes
Section titled “Bug fixes”- What-if data tables now evaluate each scenario with a real model recalculation (substitute the input cells, recalculate, read the source cell), matching Excel’s documented behaviour. Previously only the source formula’s own AST was re-evaluated against the substituted input, which silently produced wrong results whenever the source reached the input indirectly: through intermediate formula cells,
INDIRECT/OFFSET, defined names, cross-sheet references, volatile functions, array spills, or iterative-calculation cycles. - A custom workbook theme now survives model serialization.
deserializeModelpreviously dropped the theme and reverted every workbook to the default palette (withtheme.isDefaultset true); it now restores the serialized theme, so theme-resolved colours and fonts round-trip. Already-serialized models recover their theme too, since the data was present all along and only the read path was at fault. - A pivot table’s saved output cells are now kept as-is on load instead of being recomputed, matching Excel — computed output fills only the cells the file left empty. A cache flagged
refreshOnLoad="1"is instead rebuilt from its worksheet source and re-rendered once at load time. - Refreshing a pivot cache that holds a derived field (a calculated field or a derived grouped field) no longer throws when the source columns are reordered, renamed, added, or deleted. The derived fields are kept and reconciled against the new column set, matching Excel; previously the presence of any derived field, even one unrelated to the change, aborted the refresh.
- The single-data-field pivot header (“corner”) cell now matches modern Excel: the data field’s display name (e.g.
Sum of Sales) in every layout, the bare field name only on pivots carrying a legacy AutoFormat, and an empty cell when the data field has no name. Previously it wrote the bare cache-field name unconditionally. - “Show items with no data” (
showAll) densification on load now matches Excel: it defaults to dense for ordinary discrete fields (the spec default), lists only the items surviving the pivot’s active filters, evaluates a label/caption filter against the field’s full item set (so an item the caption filter keeps still shows even with no data), and includes orphan cache items as empty “no data” rows. - Pivots mixing compact and tabular row-field layouts now match Excel: an outer-compact mix collapses to one compact “Row Labels” column, an outer-tabular mix captions its first column “Row Labels”, and a tabular outer field writes its group label on the group’s first output row. Field-name row captions appear only on row-only pivots whose row fields are all tabular with
compactDataoff. - A pivot that places the Values pseudo-field at a mid position among its column fields now lays the data fields out as interleaved sub-blocks at that depth, with the field-button row in raw field order, the way Excel does. Previously any non-first Values position was treated as innermost.
- A Values-last multi-data pivot now serializes its
colItemswith the data field varying fastest (df-innermost interleaving), matching Excel. Previously even a dormant load→save rewrote them into a df-major order Excel does not use. - A Values-outer pivot’s column axis-label row no longer drops the
Valuesbutton: it now readsValuesahead of the real field buttons, matching Excel. - A multi-data pivot whose Values pseudo-field sits on the row axis now lays the data fields out as row blocks — a synthetic “Values” row-label column, with per-data-field subtotal and grand-total captions — matching Excel. Previously the row-axis Values position was ignored and the data fields were always emitted on the column axis.
- A multi-data pivot with no column field now omits the synthetic “Values” caption row — the data-field names form the header row — matching Excel, instead of emitting an extra caption row that pushed the body down.
- On refresh, a pivot’s header-row offsets (
firstHeaderRow/firstDataRow) are now re-derived from its definition so the output lands where Excel’s refresh would, regardless of the offsets the file stored: an over-reserved blank row above the headers is removed, and a multi-data pivot missing its reserved Values header row has that row rebuilt (instead of collapsing to a single data field and inventinghideValuesRow="1"on export). - Pivot column widths are sized to the formatted value text (currency, percent, and date number formats) instead of the raw number’s digits, so formatted value columns are no longer too narrow.
- Date fields on a pivot’s row or column axis now keep their number format on refresh, so header cells show formatted dates instead of raw serials. Previously only data, subtotal, and grand-total cells kept the format through re-materialization.
- Pivots with report filters now emit their page-field cells — a caption/selected-item cell pair per page field, in Excel’s single-column stack or wrapped multi-column grid — in the rows above the output range. Previously the report-filter area was omitted entirely.
- A pivot page-field (report filter) selection cell now carries the field’s own number format, so a date filter shows a formatted date instead of a raw serial.
- Pivot structural cell styles now match Excel: the column-caption dropdown-button flag is stamped on the axis-label row and only when the pivot has a real column field, and row-label left alignment is written only when the row axis has at least one compact field (not on fully-tabular layouts).
- Classic (
gridDropZones) pivots now show field-name captions on their row and column axes instead of the modern literalRow Labels/Column Labels, matching Excel. - Pivot calculated-field formulas now evaluate through the engine’s normal formula path, removing divergences from Excel’s coercion semantics that the old per-record closure tree had drifted into (e.g. unary-plus coercion on text, and empty-string arithmetic).
- Exported pivots now stamp
showAll="0"on axis fields that were materialized sparse (“show items with no data” off). Excel reads an absentshowAllas dense, so refreshing a previously-exported pivot re-densified the layout to the full item set and overflowed the saved footprint. - A fully non-compact (tabular/outline) pivot now writes the table-level
compact="0" compactData="0"attributes on export. Excel derives the row-label caption from these, so without them an Apiary-saved tabular pivot came back retitled (field names →Row Labels) on its first refresh. - A grouped (date/range/discrete) pivot field now exports its full item universe in
<items>(backfilled from the field group’s items), not just the materialized subset. Excel rejects a grouped field whose items list omits any bucket. - A pivot field carrying an explicit
defaultSubtotal: falsewithoutsubtotalFunctionsno longer gets a contradictory trailing default-subtotal item on export — a combination Excel offers to “repair” once the field also carriesshowAll="0". - Exporting a page-fields-only single-data pivot no longer produces a file Excel offers to repair: its
<location>geometry (anchor, header/data offsets, and the blank separator below the page area) now matches what Excel writes. - Exported pivots now compress members repeated from the previous item of
<rowItems>/<colItems>into ther(“repeated item count”) attribute, the way Excel writes them. Without it, Excel drew phantom expand/collapse buttons on the blank continuation label cells of saved multi-row-field pivots until the table was refreshed. Workbook.deleteRows/deleteColumnsnow match Excel in guarding a pivot table’s output range. A deletion that partially overlaps a pivot table is refused (throwsPivotEditBlocked, like the analogousinsertRows/insertColumnsguard); a deletion that fully encompasses a pivot table removes it (and its cache if not also referenced by another pivot table).
Performance
Section titled “Performance”- Data tables now recompute only when something they depend on changed (their source, parameters, input cells, or any cell those transitively read), and at most once per recalculation, instead of unconditionally on every recalculation. Tables whose source closure contains a dynamic reference (
INDIRECT/OFFSET) still recompute on every recalculation, conservatively. - Per-call function and operator timing now reads
performance.now()only when ametricsCallbackis set, removing a small per-call cost from the common no-metrics path (including hot inner loops). Behaviour is unchanged when metrics are enabled.
dist/now ships code-split output alongside the existingindex.{js,cjs,d.ts,d.cts}entry bundles (e.g.parser-*,toXlsx-*,fsHelpers-*). The package’s public entry points (exports,main,module,types) and what consumers see when theyimport '@grid-is/apiary'are unchanged. Consumers reaching pastexportsto load files insidedist/directly would need to update: chunk filenames are content-hashed and the chunking strategy is an internal implementation detail.
v17.0.0-beta.0 (2026-06-11)
Section titled “v17.0.0-beta.0 (2026-06-11)”First beta of the 17.0 major line. Pivot tables — the headline 17.0 feature — land here, in beta: some known kinks are still being ironed out. Subject to further breaking changes before the stable release.
Breaking changes
Section titled “Breaking changes”(Note, these are breaking change since v17.0.0-alpha.0; note also the breaking changes reported for that version.)
Operations on Workbook may now throw new errors when they conflict with pivot-table output:
PivotEditBlocked— thrown byinsertRows/insertColumnswhen the insert lands strictly inside a pivot’s output area, bymoveRows/moveColumnswhen the move’s source or destination crosses a pivot’s boundary, and bymoveCells/clearCellswhen the affected range overlaps pivot output. Excel’s GUI refuses the same operations (usually silently); Apiary throws so the host can surface the refusal.PivotCellWriteBlocked— thrown byWorkbook.editCell(and thewritewrappers around it) when the target cell falls inside a pivot’s output region.PivotOutputOverlapsData— thrown by a pivot cache refresh when the refreshed pivot’s output would grow onto populated cells previously outside it.
StyleManager now reserves styles index 0 as the workbook default style (XLSX cellXf 0). A workbook built from scratch seeds slot 0 with the empty style {}, so workbook.styles.length starts at 1 and the first style created via findOrCreate gets index 1 (previously 0). Exported JSF for styled built workbooks carries the {} default at styles[0]; unstyled built workbooks still export with no styles array. Consumers should resolve styles through cell s indices rather than assuming the first created style sits at index 0.
Features
Section titled “Features”-
Pivot table support (beta — some known kinks are still being ironed out):
- Evaluation matches Excel: all OOXML aggregations and
showDataAsmodes, calculated fields, grouped fields, label/value/top-N filters, custom-list andautoSortScopesorting. - The
GETPIVOTDATAspreadsheet function is now supported. PivotManager.refreshCacherefreshes a pivot cache from its source; structural edits adjust pivot output and cache source ranges; pivots round-trip through JSF and XLSX.- Look up pivots via
workbook.pivots.get(name, sheetName?)andworkbook.pivots.findAt(sheetName, row, col). - Visual pivot styling is not baked into cells: as in Excel, only the declared pivot style name is stored, and resolving it to an appearance is the renderer’s job.
- Evaluation matches Excel: all OOXML aggregations and
-
Formula parsing accepts more of Excel’s grammar (via
@grid-is/waspiary1.13.0): the@implicit-intersection prefix operator and the structured-reference@this-row shorthand, reference operators and calls after parenthesised references ((A1):B2,(A1)#,(A1)(B2), …), and workbook/sheet prefixes on structured references ([0]Sheet1!Table1[Col]). Calling a parenthesised non-lambda reference, e.g.(A1)(B2), evaluates to#REF!as in Excel.
API changes
Section titled “API changes”- New
StyleManager.defaultStylegetter andsetDefaultStyle(style)method read and replace the workbook default style (slot 0). The default cannot carryextendsStyle. - New
recalcAll?: booleanonAddWorkbookOptionsforces (or, whenfalse, suppresses) a full recalculation when the workbook is attached. The default honours the file’sfullCalcOnLoadsignal (see Bug fixes).
Bug fixes
Section titled “Bug fixes”<calcPr fullCalcOnLoad="1"/>(ECMA-376 §18.2.2) is now honoured: attaching a workbook that carries the flag (and is not in manual calculation mode) performs a full recalculation instead of trusting cached values as-is, matching Excel. Previously the flag was ignored, which could leave formula cells without a usable value when the writer emitted empty cached values. The flag is cleared after a successful full recalculation, as the spec requires.ValueSnapshot.applyTono longer leaves a stale cell node in the R-tree when it rebuilds a spill anchor that had been collapsed to a value cell beforehand; previously a later value write over the restored anchor tripped an internal invariant.- A formula written via
editCellto a blank-but-formatted cell that a pre-existing formula depends on now evaluates. Previously its dependency-graph vertex kept pointing at a throwaway cell object created at load, so reads returnednulland recalculation recorded an “Invariant violation” model error, with no recovery even via a full recalc.
v17.0.0-alpha.0 (2026-05-30)
Section titled “v17.0.0-alpha.0 (2026-05-30)”First alpha of the 17.0 major line. Subject to further breaking changes before the stable release; notably, pivot-table support is still to come.
Breaking changes
Section titled “Breaking changes”Structural edits and writes that would change part of a what-if data table now throw DataTableEditBlocked (exported from the package index) instead of silently corrupting the table. Excel refuses each of these edits in its UI (via a modal alert or a silent no-op); Apiary surfaces the refusal so the host application can decide how to react. The refused edits, matching Excel’s empirically verified rules, are:
- row/column inserts strictly inside a data table’s block (the range spanning its source cell, parameter cells, and result range);
- row/column deletes overlapping the block without covering the whole result range (covering it removes the table, like Excel);
moveCells/moveRows/moveColumnstaking part of the block (a move must contain all of it --- the whole table then moves) or landing on part of the result range (landing on all of it removes the table);- writes (
editCelland thewritewrappers) targeting a result cell, andclearCellscovering part but not all of the result range.
The error carries dataTableRef (the A1 range of the table’s result cells) and operation (a stable DataTableBlockedOperation identifier of the refused edit kind, e.g. 'insert-rows' or 'move-cells-onto') for programmatic handling, e.g. a localized host-application message.
The source and parameter cells remain freely editable, and the table recomputes from the new values. Allowed structural edits now also maintain the table’s metadata --- previously any insert/delete/move silently desynchronized it.
Transient writes and “reset” are gone, replaced by value snapshots (see below). Model.write, Model.writeMultiple, and Workbook.write still exist but are now thin wrappers around Workbook.editCell. They are persistent edits, not transient overlays. Notable differences in behaviour:
-
One state, no
reset. Each write mutates the workbook in place; there is no separate transient state to enumerate or roll back. To undo a value edit later, capture aValueSnapshotbeforehand and apply it to revert. -
No formula neutralization option. Writing a value to a formula cell now unconditionally replaces the formula with that value. The
neutralizeFormulaOnSingleCellWriteflag is removed. -
No range writes.
Model.write/Workbook.writeno longer accept a multi-cell range reference. To assign values to many cells, useModel.writeMultiple([[refA, val], [refB, val], …]); for a real spill, useWorkbook.editCellwith an array formula. -
No writes inside merges.
Model.writeetc. now wrapWorkbook.editCell, which throws when the target is a non-anchor cell of a merge range. Write to the merge anchor instead, or unmerge first. -
No writes to defined names.
Model.write/Model.writeMultiple/Workbook.writeacceptstring | A1Referenceand throw on a defined-name reference (including a string that parses to one):- The “named values” feature (writing a literal value as a defined name’s formula) is gone.
Workbook.editCellon a name reference now requires an explicitfformula - To write through a chained defined-name reference, resolve first:
const ref = Reference.from(name, { ctx: model }).resolveToNonName()and thenmodel.write(ref, v) - Existing workbooks with literal-formula defined names still load and evaluate read-only
- The “named values” feature (writing a literal value as a defined name’s formula) is gone.
The following public APIs are removed:
Model.reset,Workbook.reset,WorkSheet.reset.- The
'reset'model event. Model.writes,Workbook.writes,Model.writeState,Workbook.writeState, theModelStateTreetype, and theGRID.SCENARIO()spreadsheet function. No replacement: these all reported transient write state, which no longer exists.=GRID.SCENARIO()now resolves to#NAME?.- The
neutralizeFormulaOnSingleCellWriteandresetproperties ofWriteOptions. The surviving properties areforceRecalc,skipRecalc, andskipVolatiles. Cell._vandCell.resetValueBoxed(the per-cell reset value).
Serialization: the _rangeWrites and _spillResetState fields and the per-cell reset value are no longer written. v17.x still loads models serialized by older Apiary versions; the dropped fields ride along in the payload and are silently ignored. Older Apiary versions cannot load v17.x serializations.
The RecalcUreachedEvent type is renamed to RecalcUnreachedEvent (the old name was a typo). The event payload is unchanged.
Features
Section titled “Features”- Value snapshots:
ValueSnapshot.capture(model)returns a reusable snapshot of cell and defined-name values (boxed, so number formats are preserved) and spill matrices across every workbook; formulas are captured as fingerprints only.snapshot.applyTo(model, { skipRecalc?, includeDeferred? })restores values and spills, then triggers a recalc. Any cell or name whose formula changed (added, edited, or removed) since capture is skipped, and one WARNING-levelModelErrorlisting every affected cell/name is recorded per workbook. The check is per-position and best-effort: a structural edit that rearranges cells may produce a partially-warned, partially-corrupted revert.- Styles and structural edits (row/column inserts/deletes, sheet adds/removes/renames,
moveCells) are out of scope and invalidate the snapshot; applying after such edits may leave the model in an incorrect state with no per-cell WARNING. For guaranteed-correct reverts across formula or structural edits, serialise the model (binary, JSF, or XLSX) and restore from that.
API changes
Section titled “API changes”Workbook.editCellis the canonical cell-mutation entry point: it keeps recalc bookkeeping in sync automatically.Model.write,Model.writeMultiple, andWorkbook.writeare now thin wrappers around it.Cell.userZis now marked@internaland excluded from the public.d.tstype. To read the effective number format useCell.z, orCell.stylefor the fullStyleRelaxed(which includes the user-assigned format undernumberFormat). To set a number format useWorkbook.editCell({ z })orcell.style = { ... }. No runtime change.- The
RewriteFormulacallback returned by structural-edit methods (deleteRows/deleteColumns/insertRows/insertColumns/moveRows/moveColumns/moveCells) gains two context parameters:(formula, formulaSheet, formulaWorkbook) => string. Callers that apply the returned rewriter to formulas in another sheet or workbook must now pass that context. Previously the rewriter assumed every unprefixed reference lived in the mutation’s source sheet/workbook, so it corrupted formulas applied elsewhere;moveCellsin particular returned a rewriter that was a silent no-op across workbooks.
Bug fixes
Section titled “Bug fixes”- XLSX export font fix (via
@grid-is/jsf2xlsx2.2.0): the cell style cascade did not resolve fonts correctly, so cells could export with the wrong typeface.
Packaging
Section titled “Packaging”- Bumped Excel import/export dependencies:
@grid-is/jsf2xlsx^2.1.0→^2.2.0,@borgar/xlsx-convert^5.2.0→^5.3.0, and@jsfkit/types^2.2.0→^2.4.0.
v16.1.0
Section titled “v16.1.0”Features
Section titled “Features”Workbook.calcModeis now a public getter/setter. Reading it returns the workbook’s current calculation mode ('auto','autoNoTable', or'manual'). Setting it to'autoNoTable'also enables data-table deferral (equivalent toModel.deferDataTables = true).
Bug fixes
Section titled “Bug fixes”deleteRowsanddeleteColumnsno longer rewrite formula references on sheets other than the one being modified. Previously, unqualified references on every sheet were treated as if they lived on the target sheet.- Loading JSF containing external workbooks whose defined names carry no value no longer crashes with “Cannot set defined name with invalid name or empty formula”.
Model.writeMultiplenow correctly honours theskipVolatilesoption. Previously the option was silently ignored, causing volatile functions such asRAND()andNOW()to be recalculated even when the caller asked to skip them.
v16.0.2
Section titled “v16.0.2”API changes
Section titled “API changes”Cell.editandDefinedName.editare now@deprecatedaliases that forward to internal_editmethods. Consumers should useWorkbook.editCellinstead —editCellkeeps the dependency graph and recalc bookkeeping in sync, which the raweditpaths do not. No runtime behaviour change.
Bug fixes
Section titled “Bug fixes”Workbook.editCell({ v: '#N/A' })(andCell.edit) no longer coerces error-looking strings to error values —vis preserved as a string, matching theCellconstructor. Coercion still happens when the input includest: 'e'.Workbook.editCellno longer clears existingModelErrors on a cell before its per-branch validation guards run, so a formula edit rejected by the merge-follower guard now leaves prior errors intact.toJSFtyping fix:External.namesentries are typed asExternalDefinedName(wasJSFDefinedName, structurally compatible but wrong). Type-only fix, no behaviour change.
Performance
Section titled “Performance”Model.fromJSFexternal-load loop is no longer O(externals × main-cells)._addExternalsdefers per-externalrecalculate()andoptimize()to a single trailing pass; on a real 125-external, 18k-cell workbook,Model.fromJSFdrops from ~16s to ~2s. Callers passingrecalcVolatiles: false, recalcErrors: false(the round-trip harnesses) still see no recalc.
Packaging
Section titled “Packaging”numfmtmoved frompeerDependenciestodependencies(range unchanged,~3.2.2). Apiary usesnumfmtinternally and consumers never pass instances in or out of the API; the peer-dependency placement was incidental, and demoting it meansnpm install @grid-is/apiaryworks without consumers having to add a matchingnumfmtentry.@types/nodemoved fromdependenciesto optionalpeerDependencies(range:>=20.19.23 <21 || >=22.19.0 <23 || >=24.10.0). The publisheddist/index.d.tsstill references Node’sBuffer, so a TypeScript Node consumer needs@types/noderesolvable — but the consumer’s own pin now wins. Pure-JS consumers see no missing-peer warning thanks tooptional: true. The per-major minimum reflects the late-2025 DefinitelyTyped round that narrowedBuffer.bufferfromArrayBufferLiketoArrayBuffer; below that, our publicWorkbook.fromXLSX/Model.fromXlsxtypes are unsound.@borgar/xlsx-convertbumped to 5.2.0, picking up an XLSX color/theme parsing fix that surfaced throughModel.fromXlsxin an add-ins consumer (undefined is not an object (evaluating 'readColor(elm.children[0], …)')).
v16.0.1
Section titled “v16.0.1”API changes
Section titled “API changes”- Removed the
NamedStyleandStylere-exports from the package entry point. These were unintentionally exposed during the 16.0 cycle and were never meant to be part of Apiary’s public API; import them from@jsfkit/typesdirectly. Type-only change with no runtime impact.
Bug fixes
Section titled “Bug fixes”Modelnow emitsaddsheetwhen a sheet is added (previously dropped silently), and no longer emits a spuriousrecalcfor that operation.- Three independent deserialization-fidelity bugs in the binary cache, jointly affecting
autoNoTable-mode roundtrips, are fixed:Workbook._calcModeis restored from the payload (was silently reverting to'auto').Cells._dataTableCells(the data-table-anchor index) is repopulated, soiterDataTableCells()works on deserialized workbooks — unblocking theModel.deferDataTables = truesetter andModel.attachWorkbook’sautoNoTablehandling.Model._deferDataTablesand the deferral cell Sets (cellsToDefer,staleDeferredCells) are now serialized and restored, so a model in deferral mode no longer loses it through a roundtrip.
toJSFcompacts full-row/full-column spill ranges incell.F(A3:XFD3→3:3,B1:B1048576→B:B), eliminating spurious roundtrip diffs when a code path holds an expandedF.- Error subclasses now set their own
namefield, so stack traces,console.log,inspect(), and Sentry display the actual class name (e.g.FormulaSyntaxError) instead of the inheritedError. @jsfkit/typesmoved fromdevDependenciestodependenciesso it is installed transitively when consumersnpm install @grid-is/apiary. The publisheddist/index.d.{ts,cts}references@jsfkit/typesdirectly because those types appear in Apiary’s public API; consumers’ type-checking previously failed unless they happened to depend on@jsfkit/typesfor other reasons.typescriptmoved fromdependenciestodevDependencies. It is only used here as a CLI build tool (tsc --noEmit,tsup); nothing inlib/imports it and the published types do not reference it. There is no public-API reason for consumers to receive Apiary’s pinnedtypescriptversion transitively. Originally added todependenciesduring the apiary monorepo extraction without a deliberate placement choice.
Serialization
Section titled “Serialization”CURRENT_SERIALIZATION_VERSIONbumped 5 → 6 to carry the previously-omittedModel._deferDataTablesboolean and the two deferral cell Sets.MINIMUM_ACCEPTABLE_SERIALIZATION_VERSIONis unchanged — older payloads still load, with the new fields defaulted to off / empty.
Documentation
Section titled “Documentation”- JSF types referenced from the generated API reference (
NamedStyle,Note,Style,Theme,ThemeColorScheme,ThemeCustomColor,ThemeFontCollection,ThreadedComment) now link tojsfkit.github.ioinstead of being mirrored as local Apiary type-alias pages. EvaluateExpressionOptions.cropTois now documented.- Cleaned up a confusing JSF-normalizer docstring.
v16.0.0 (2026-04-30)
Section titled “v16.0.0 (2026-04-30)”Breaking changes
Section titled “Breaking changes”Workbook.viewsis now aViewManager(wasWorkbookView[] | undefined).Workbook.getViews()andWorkSheet.getViews()are removed;WorkSheet.viewsis now@internal. Per-sheet view state is now read/written viaworkbook.views.get(index).getSheetView(sheetName).Workbook.themeis now aThemeManager(always present), notTheme | undefined.WorkSheet.showGridLinesproperty removed; gridline visibility now lives onWorkSheet.views[].showGridLines(per-view).@jsfkit/types1.x → 2.x: colours are now structuredColorobjects instead of hex strings, themes are restructured, and several types are reshaped.Workbook.rowHeight(rowIndex, sheetName)now throwsSheetNotFoundErrorwhen the sheet does not exist (previously returned the default16silently), and consultssheet.defaults.rowHeightbefore falling back to the Excel default. The newWorkbook.columnWidth()follows the same precedence on its axis.WorkSheet.rowHeightsandWorkSheet.colWidthsfields removed. Per-row height and per-column width are now read fromWorkSheet.rowsandWorkSheet.columns(theGridSize[]span arrays).
Removed @deprecated symbols (each deprecation note already pointed at the replacement shown here):
Cell.hasValue()/DefinedName.hasValue()andCell.isBlank()/DefinedName.isBlank()removed. UsehasValueOrFormula().Workbook.getTables()andWorkbook.getTable(name)removed. Useworkbook.tables.getAll()andworkbook.tables.get(name).WorkSheet.getComments()andWorkSheet.getNotes()removed. Useworkbook.comments.get(sheetName)andworkbook.notes.get(sheetName).Model.fromDataremoved. UseModel.fromCsf.Workbook.lazyImportPromise,Model.lazyImportPromise,loadLazy(), and theloadLazyre-export removed. All four were already no-ops; just delete the calls.- A batch of
@deprecated @internalre-exports removed from the package entry point: formula-evaluation helpers (run,errorForCode,errorTable, …) and dependency graph helpers (cellToVertexId,wbNameToKey, …). Usecell.vertexId/definedName.vertexIdinstead ofcellToVertexId(cell).
(Several dependency-graph items previously marked @deprecated @internal (or not exported at all) are now part of the public API: VertexId, CellVertexId, NameVertexId, RangeVertexId, KnownVertexId, VertexIdSet. The helpers referenceToVertexId, vertexIdToCell, and vertexIdToReference are re-exported with @internal markers — available, but excluded from documentation.)
Features
Section titled “Features”- Views CRUD via
Workbook.views, with two new public typesViewManagerandWorkbookView. - Theme CRUD via
Workbook.theme(newThemeManager): read/write the theme as a whole, the 12-colour scheme, major/minor fonts, and custom colours. - Named-style CRUD via
styleManager.named(newNamedStylescollection), with case-insensitive lookup and original casing preserved onNamedStyle.name. Workbook.reorderSheet(name, newIndex)andWorkbook.copySheet(name, newName?).Tableis now a public type export with new methods (appendRow,insertRow,rename,rowRef) and a newdataRowCountgetter.TableManageris also exported as a type. (The runtime classes are not re-exported: instances come fromworkbook.tables.get(name)/workbook.tables.add().)- Public-API types now exported from the package entry point (previously reachable only via
Parameters<typeof model.write>[2]-style tricks):WriteOptions,AddWorkbookOptions,WorkbookOptions,ReplaceTableReferencesFn,ModelEventArgs,ModelEventType,ModelEventListener,EvaluateExpressionOptions,ModelMeta. NotesManagerandCommentsManagerare now exported so they show up in the generated public docs.Workbook.toXLSX(outputType?, options?)gains both arguments.outputTypeis'nodebuffer'(default) or'arraybuffer'for browser-compatible XLSX export.options(typed asToXLSXOptions) carries acompressionLevelsetting (DEFLATE 0–9), now defaulting to6, down from9— measurably faster export with negligible size impact on typical XLSX payloads.Workbook.columnWidth(columnIndex, sheetName)— analogous toWorkbook.rowHeight(). Returns the column width in pixels.Workbook.moveCellsandWorkbook.rewriteFormulasaccept an optionalrecalcNow(defaulttrue); passfalseto defer the trailingrecalculate().
API changes
Section titled “API changes”Workbook.editCell()added as the preferred name forwriteCellData()(which stays as a@deprecatedalias). Its parameter type simplifies to plainCellData, which now includes the optionalftfield.- Identifier casing standardized on uppercase
JSF/XLSX:Model.fromJSF,toJSF,csfToJSF,normalizeJSFForComparison,toXLSX,toXLSXFile,validateXLSXFilePath, plus typeXLSXOutputType. The old title-case spellings remain as@deprecatedaliases with identical signatures — no breaking change.
Bug fixes
Section titled “Bug fixes”moveCellsnow handles merged cells. Destination merges are dissolved silently (matching Excel Scripts and Office.js behavior), and cells moved out of a source merge no longer carry stale.Mmerge pointers.- Loading a workbook with a table whose name is not valid as a defined name (e.g.
T1,A1,R1C1) now sanitizes and uniquifies the name instead of dropping the table with aModelError. - Iterative calculation now converges correctly when the iterative set contains a spill; previously the calc ran to
maxIterationsevery time and dependents of unchanged spills could be flagged as false-positive circular references. getEntitiesno longer crashes on workbooks containing sheet-scoped defined names.SORTNsilently ignores out-of-bounds sort-column indices instead of returning#VALUE!, matching verified Google Sheets behavior.toJSFpreserves values in merged-cell “ghosts” (non-anchor cells);mergeCellskeeps styles on follower cells and only clears values, matching Excel.- Deserializing models serialized by older Apiary versions no longer crashes on missing
views/comments/notes/people/externals/rowsproperties or on thestable.csf->stable.jsfTable key rename. replaceTableReferencesnow correctly rewrites bare-name references to tables (not just bracketed structured references).- Dynamic dependencies of LAMBDA closures passed to
BYROW/BYCOL/MAP/REDUCE/SCANwithINDIRECT/OFFSET/ANCHORARRAYin the body are now tracked, so edits to referenced cells correctly trigger recalc. - Lambda parameter resolution during reference analysis no longer crashes on
INDIRECT/OFFSETinside directly-called lambdas or nested lambda scopes. toJSFno longer crashes on full-row or full-column spill ranges.rewriteFormulasskips unparseable formulas instead of throwing.- Loading CSF workbooks: the literal string
"#SPILL!"in a cell is no longer coerced to an error value, fixing an R-tree invariant violation that crasheddescribeWorkbookin some cases. - Cells in JSF with error-looking string values (
"#VALUE!","#N/A", …) but no explicitt: 'e'marker are preserved as strings, not coerced to errors.
Performance
Section titled “Performance”- Faster label detection in
describeWorkbookvia cheaper subordinate-style checks (weighted geomean ~23.5% faster, worst-case workbooks up to 43% faster).
Serialization
Section titled “Serialization”CURRENT_SERIALIZATION_VERSIONbumped 3 → 5 over the v16 cycle. The 3 → 4 step was a marker for shape changes that had already landed without a version bump (Tablecsf→jsfkey,cellsWithDynamicDependencies, notes/comments/views/tables), giving consumers that cached v3 payloads a proper signal to invalidate. The 4 → 5 step adds thenamedStylesfield and folds the legacy parallelrowHeights/colWidthsmaps into theWorkSheet.rows/WorkSheet.columnsspan arrays.MINIMUM_ACCEPTABLE_SERIALIZATION_VERSIONstays at 1 — older cached payloads still load.
Documentation
Section titled “Documentation”ModelError.toJSONAPI documentation cleaned up (the raw inferred-return-type property list is gone); the MDN link gets a readable link text.
Dependencies
Section titled “Dependencies”@jsfkit/types^1.4.1→^2.2.0(see Breaking changes for the structural shifts).@borgar/xlsx-convertand@grid-is/jsf2xlsxupgraded for XLSX import/export of named styles and the CSE array-formula flag, plus round-trip and repair-dialog fixes.@grid-is/waspiary→ 1.11.0 for structured-reference bare-name rewrite.
v16.0.0-rc.0 (2026-04-16)
Section titled “v16.0.0-rc.0 (2026-04-16)”First release candidate for the 16.0 major line. Covers the deprecation cleanup, API additions, and fixes landed since v16.0.0-alpha.0.
Breaking changes
Section titled “Breaking changes”All of the following remove symbols that were already @deprecated, with their deprecation notes pointing at the replacement shown here.
Cell.hasValue()/DefinedName.hasValue()andCell.isBlank()/DefinedName.isBlank()removed. UsehasValueOrFormula().Workbook.getTables()andWorkbook.getTable(name)removed. Useworkbook.tables.getAll()andworkbook.tables.get(name).WorkSheet.getComments()andWorkSheet.getNotes()removed. Useworkbook.comments.get(sheetName)andworkbook.notes.get(sheetName).Model.fromDataremoved. UseModel.fromCsf.Workbook.lazyImportPromise,Model.lazyImportPromise,loadLazy(), and theloadLazyre-export removed. All four were already no-ops; just delete the calls.- A batch of
@deprecated @internalre-exports removed from the package entry point: formula-evaluation helpers (run,errorForCode,errorTable, …) and dependency graph helpers (cellToVertexId,wbNameToKey, …). Usecell.vertexId/definedName.vertexIdinstead ofcellToVertexId(cell).
(Some dependency graph classes and utilities were previously marked @deprecated @internal but are now instead made public: VertexId, CellVertexId, NameVertexId, RangeVertexId, KnownVertexId, VertexIdSet, referenceToVertexId, vertexIdToCell, vertexIdToReference.)
Features
Section titled “Features”- Public-API types now exported from the package entry point (previously reachable only via
Parameters<typeof model.write>[2]-style tricks):WriteOptions,AddWorkbookOptions,WorkbookOptions,ReplaceTableReferencesFn,ModelEventArgs,ModelEventType,ModelEventListener,EvaluateExpressionOptions,ModelMeta. - Named-style CRUD via
styleManager.named(newNamedStylescollection), with case-insensitive lookup and original casing preserved onNamedStyle.name.
API changes
Section titled “API changes”Workbook.editCell()added as the preferred name forwriteCellData()(which stays as a@deprecatedalias). Its parameter type simplifies to plainCellData, which now includes the optionalftfield.- Identifier casing standardized on uppercase
JSF/XLSX:Model.fromJSF,toJSF,csfToJSF,normalizeJSFForComparison,toXLSX,toXLSXFile,validateXLSXFilePath, plus typesXLSXOutputMapandXLSXOutputType. The old title-case spellings remain as@deprecatedaliases with identical signatures — no breaking change.
Bug fixes
Section titled “Bug fixes”moveCellsnow handles merged cells. Destination merges are dissolved silently (matching Excel Scripts and Office.js behavior), and cells moved out of a source merge no longer carry stale.Mmerge pointers.- Loading a workbook with a table whose name is not valid as a defined name (e.g.
T1,A1,R1C1) now sanitizes and uniquifies the name instead of dropping the table with aModelError.
Serialization
Section titled “Serialization”CURRENT_SERIALIZATION_VERSIONbumped 3 → 4 as a marker for shape changes that had landed without a version bump (Tablecsf→jsfkey,cellsWithDynamicDependencies, notes/comments/views/tables), giving consumers that cached v3 payloads a proper signal to invalidate.CURRENT_SERIALIZATION_VERSIONbumped 4 → 5 for the named-styles addition.MINIMUM_ACCEPTABLE_SERIALIZATION_VERSIONstays at 1 — older cached payloads still load.
Documentation
Section titled “Documentation”ModelError.toJSONAPI documentation cleaned up (the raw inferred-return-type property list is gone); the MDN link gets a readable link text.
v16.0.0-alpha.0 (2026-04-08)
Section titled “v16.0.0-alpha.0 (2026-04-08)”First alpha of the 16.0 major line. Subject to further breaking changes before the stable release.
Breaking changes
Section titled “Breaking changes”Workbook.viewsis now aViewManager(wasWorkbookView[] | undefined).Workbook.getViews()andWorkSheet.getViews()are removed;WorkSheet.viewsis now@internal. Per-sheet view state is now read/written viaworkbook.views.get(index).getSheetView(sheetName).Workbook.themeis now aThemeManager(always present), notTheme | undefined.WorkSheet.showGridLinesproperty removed; gridline visibility now lives onWorkSheet.views[].showGridLines(per-view).@jsfkit/types1.x → 2.0.0: colours are now structuredColorobjects instead of hex strings, themes are restructured, andGridSizegains an optionalsize.
Features
Section titled “Features”- Views CRUD via
Workbook.views, with two new public typesViewManagerandWorkbookView. - Theme CRUD via
Workbook.theme(newThemeManager): read/write the theme as a whole, the 12-colour scheme, major/minor fonts, and custom colours. Workbook.reorderSheet(name, newIndex)andWorkbook.copySheet(name, newName?).Tableclass is now a public export with new methods (appendRow,insertRow,rename,rowRef) and a newdataRowCountgetter.TableManageris also exported.toXlsxaccepts an optionaloutputTypeargument ('nodebuffer'(default) or'arraybuffer') for browser-compatible XLSX export. New exportsXlsxOutputTypeandXlsxOutputMap.NotesManagerandCommentsManagerare now exported so they show up in the generated public docs.Workbook.moveCellsandWorkbook.rewriteFormulasaccept an optionalrecalcNow(defaulttrue); passfalseto defer the trailingrecalculate().
Bug fixes
Section titled “Bug fixes”getEntitiesno longer crashes on workbooks containing sheet-scoped defined names.SORTNsilently ignores out-of-bounds sort-column indices instead of returning#VALUE!, matching verified Google Sheets behavior.toJsfpreserves values in merged-cell “ghosts” (non-anchor cells);mergeCellskeeps styles on follower cells and only clears values, matching Excel.- Deserializing models serialized by older Apiary versions no longer crashes on missing
views/comments/notes/people/externals/rowsproperties or on thestable.csf->stable.jsfTable key rename. replaceTableReferencesnow correctly rewrites bare-name references to tables (not just bracketed structured references).- Dynamic dependencies of LAMBDA closures passed to
BYROW/BYCOL/MAP/REDUCE/SCANwithINDIRECT/OFFSET/ANCHORARRAYin the body are now tracked, so edits to referenced cells correctly trigger recalc. - Lambda parameter resolution during reference analysis no longer crashes on
INDIRECT/OFFSETinside directly-called lambdas or nested lambda scopes. toJsfno longer crashes on full-row or full-column spill ranges.rewriteFormulasskips unparseable formulas instead of throwing.- Loading CSF workbooks: the literal string
"#SPILL!"in a cell is no longer coerced to an error value, fixing an R-tree invariant violation that crasheddescribeWorkbookin some cases. - Cells in JSF with error-looking string values (
"#VALUE!","#N/A", …) but no explicitt: 'e'marker are preserved as strings, not coerced to errors.
Performance
Section titled “Performance”- Faster label detection in
describeWorkbookvia cheaper subordinate-style checks (weighted geomean ~23.5% faster, worst-case workbooks up to 43% faster).
Dependencies
Section titled “Dependencies”@jsfkit/types^1.4.1->^2.0.0(see Breaking changes).@grid-is/waspiary->1.11.0for structured-reference bare-name rewrite.
v15.5.0 (2026-03-05)
Section titled “v15.5.0 (2026-03-05)”Features
Section titled “Features”- Cell notes CRUD via
NotesManager(workbook.notes):add,get,getByCell,update,delete
Bug fixes
Section titled “Bug fixes”AND/OR/XOR/NOTevaluation with strings and other edge-case arguments, matching Excel and Google Sheets behaviorTEXTJOINandJOINnow apply cell display formats (e.g. date formats) in Google mode
Interface changes
Section titled “Interface changes”These are technically breaking, but very minor and in very recent features, so did not demand a major-version bump.
CommentsManagernow throwsSheetNotFoundError(instead of plainError) for missing sheets, with a differently worded message.WorkSheet.notestype changed toNote[](no longer possibly undefined)
Dependencies
Section titled “Dependencies”- Update
@grid-is/jsf2xlsxto ^1.3.2, updating default column width and fixing extraneousactiveTab="0"attributes inworkbookViewelements.
v15.4.0 (2026-03-02)
Section titled “v15.4.0 (2026-03-02)”Features
Section titled “Features”- Respect ‘autoNoTables’ from XLSX file, plus
deferDataTablesoption for manual control- And more generally, mark any formula cells to be deferred in recalculation
- Threaded comments CRUD via
CommentsManager(workbook.comments)
Bug fixes
Section titled “Bug fixes”MATCHreturned#N/Ainstead of#REF!in Excel missing-sheet edge casetoJsfexport expanded external workbooks to nonexistent cells if referenced in ranges- Error-like strings (e.g.
#REF!) not treated as error values when loading styled cells from CSF - Default (named) styles were not preserved in
toJsfexport
Performance
Section titled “Performance”- Much faster worst-case
describeWorkbook(label-detection) performance
Documentation
Section titled “Documentation”- Update quickstart to use
Model.fromXlsxFileinstead of manualxlsx-convert
v15.3.1 (2026-02-23)
Section titled “v15.3.1 (2026-02-23)”Bug fixes
Section titled “Bug fixes”- Preserve formula type (array or not) when result is 1x1, and classify correctly on edit
- Serialization did not preserve views, annotations, rows, and externals
- Remove Node.js util imports so consumers need not polyfill for browser compatibility
Documentation
Section titled “Documentation”- Add changelog to public docs
v15.3.0 (2026-02-19)
Section titled “v15.3.0 (2026-02-19)”Features
Section titled “Features”- Excel data tables
- Function
SORTN getViewsmethod inWorkSheetandWorkbook- Google Sheets
FLATTENfunction - Merge/unmerge cells
Bug fixes
Section titled “Bug fixes”- Make
HYPERLINK()-derived hyperlinks ephemeral - Preserve external workbooks in JSF roundtrip
- Update comment and note cell refs on move, insert, and delete
- Wrap
fs/promisesimports in try-catch to avoid bundler errors
Performance
Section titled “Performance”- Speed up
describeWorkbookby pre-filteringparseDatecalls
Documentation
Section titled “Documentation”- Pare down public API, marking stuff
@deprecatedand/or@internal - Use code blocks in Typedoc-generated docs
v15.2.0 (2026-02-10)
Section titled “v15.2.0 (2026-02-10)”Features
Section titled “Features”- Support JSF cell notes and threaded comments
Bug fixes
Section titled “Bug fixes”describeWorkbookcrash on cross-workbook ref to formula cell- Support
QUERYwithCOUNT(DISTINCT value)
v15.1.1 (2026-02-10)
Section titled “v15.1.1 (2026-02-10)”Bug fixes
Section titled “Bug fixes”- Missed dynamic dependencies in lambda-calling functions at init time
- Preserve cells with explicit empty styles through round-trip
- Strip
numberFormatFromFormulafrom exported styles intoJSF
v15.1.0 (2026-02-04)
Section titled “v15.1.0 (2026-02-04)”Features
Section titled “Features”toJsftoXlsx- Add
fromXlsxmethods for direct XLSX loading - Native
QUERYwithoutpg-mem(calling it “feat” because it supports a lot that the old one did not) - Output style indexes on rows and columns
Bug fixes
Section titled “Bug fixes”- Real workbooks override external workbooks with same name
- Fix invariant error on insert after row is deleted
- Export
ftproperty intoCSFfor array formulas - Init recalc was skipped if
!recalcVolatiles && !recalcErrors - Load XLSX files in Excel mode by default
- Respect
recalcVolatilesoption during initial recalculation - Preserve text strings that look like errors in JSF loading
ERROR.TYPEdid not support modern error codesdescribeWorkbookthrows on JSF workbook with externals- Upgrade
xlsx-convertto 3.6.4 for bugfixes - Fix
handlerRelsto work with absolute paths - Fix external refs order
- The
rproperty on cells is optional
Breaking changes
Section titled “Breaking changes”- Remove
Flagsmodule andflagsexport
v15.0.0 (2026-01-14)
Section titled “v15.0.0 (2026-01-14)”Features
Section titled “Features”- Public
Model.attachWorkbook, and documentaddWorkbookas just a wrapper - Export
StyleManager - Export
CellDatatype - Add support for JSF workbook/worksheet view props
- Specific
Errorsubclass for invalid reference strings - Add missing detail messages on
#CALC!errors
Bug fixes
Section titled “Bug fixes”- Google-mode early-date date function discrepancies
Matrix.expandmakes fully-populated matrices, causing performance disasters- Nonresolving name in
LOOKUPvector arg borks reference analysis - Propagate
markDirtyto all dependents of spill ranges - Blank spill anchor missing in
Cells.getCells - Don’t propagate recalc from unchanged spill matrix
numberFormatFromFormulapropagationSUMIFSetc. wrong forMatrixwith defaulted regions- Incorrect swap in Brent root-finding setup
QUERYwithnow()not treated as volatile if no dependencies- GSDV blocker in extended range should not block other columns
- Evaluate array expressions with mismatched row widths as
#VALUE! - Fix default column-width units discrepancy
- Fix
Tableserialization property name mismatch - Fix serialization invariant violation
- Fix errors in
toCSFexport structures - Fix GSDV spill range inference: use offset not absolute row
- Fix regression in
evaluateExpression - Preserve CSF style array when loading workbooks
Performance
Section titled “Performance”- Add
vertexIdtoCellandDefinedName - Non-volatile
OFFSETandINDIRECTandANCHORARRAYwith dynamic dependencies
Breaking changes
Section titled “Breaking changes”- CSF type should have
filename, notname - Change
NameReference.widthand.heightto beundefined, not throw
Dependencies
Section titled “Dependencies”- Update
fxto 5.0 - Upgrade Waspiary to 1.10.0 to permit
RandCnames
v15.0.0-rc3 (2026-01-08)
Section titled “v15.0.0-rc3 (2026-01-08)”Features
Section titled “Features”- Add missing detail messages on
#CALC!errors - Export
CellDatatype - Specific
Errorsubclass for invalid reference strings - Add support for JSF workbook/worksheet view props
- Preserve CSF style array when loading workbooks
Bug fixes
Section titled “Bug fixes”- GSDV blocker in extended range should not block other columns
numberFormatFromFormulapropagationSUMIFSetc. wrong forMatrixwith defaulted regions- Incorrect swap in Brent root-finding setup
QUERYwithnow()not treated as volatile if no dependencies- Evaluate array expressions with mismatched row widths as
#VALUE! - Fix GSDV spill range inference: use offset not absolute row
- Fix
Tableserialization property name mismatch - Fix serialization invariant violation
- Fix errors in
toCSFexport structures
v15.0.0-rc2 (2025-12-10)
Section titled “v15.0.0-rc2 (2025-12-10)”Bug fixes
Section titled “Bug fixes”- Blank spill anchor missing in
Cells.getCells - Don’t propagate recalc from unchanged spill matrix
- Fix regression in
evaluateExpression - CSF type should have
filename, notname
Performance
Section titled “Performance”- Non-volatile
OFFSETandINDIRECTandANCHORARRAYwith dynamic dependencies
v15.0.0-rc1 (2025-12-08)
Section titled “v15.0.0-rc1 (2025-12-08)”Features
Section titled “Features”- Public
Model.attachWorkbook, and documentaddWorkbookas just a wrapper - Export
StyleManager
Bug fixes
Section titled “Bug fixes”- Google-mode early-date date function discrepancies
Matrix.expandmakes fully-populated matrices, causing performance disasters- Non-resolving name in
LOOKUPvector arg borks reference analysis - Propagate
markDirtyto all dependents of spill ranges - Fix default column-width units discrepancy
Performance
Section titled “Performance”- Add
vertexIdtoCellandDefinedName
Breaking changes
Section titled “Breaking changes”- Change
NameReference.widthand.heightto beundefined, not throw
Dependencies
Section titled “Dependencies”- Update
fxto 5.0
v14.13.0 (2025-11-19)
Section titled “v14.13.0 (2025-11-19)”Features
Section titled “Features”- Read JSF directly
Bug fixes
Section titled “Bug fixes”- Bugs in row and column deletion and moves
- Support R1C1 syntax in
INDIRECT - Handle changed serialization
- Replace
ERROR_SPILL'foo'placeholder with proper detail message - Serialize model with lambdas in arrays
VLOOKUP/HLOOKUPcan return non-up-to-date value- Direct call to
UPLUSfails with internal error - Recalculate affected formulas if
clearGsdvclears cells
Breaking changes
Section titled “Breaking changes”- Remove
resetvalue fromDefinedNameandCellInterface
Dependencies
Section titled “Dependencies”- Switch to
@jsfkit/typesfor JSF type definitions
- Install from, and publish to, npmjs.com registry
v14.12.0 (2025-10-17)
Section titled “v14.12.0 (2025-10-17)”Breaking changes
Section titled “Breaking changes”- Make
FormulaErrorandReferencefully immutable
Features
Section titled “Features”- Store origin cell in
FormulaError - Add and improve error-value detail messages
- Add detail messages to
#SPILL!errors
Bug fixes
Section titled “Bug fixes”- Make
Cell.isBlank()closer to what its name implies - Minor bugs in reference intersection and add a test
- Rename and deprecate
Cellmethods
Documentation
Section titled “Documentation”- Deploy public API docs to GitHub Pages site
- Remove internal stuff from docs
- Mention case-insensitivity in name parameters
v14.11.0 (2025-10-15)
Section titled “v14.11.0 (2025-10-15)”Features
Section titled “Features”- Recalculate error-valued cells at init time
- Remove defined name
Bug fixes
Section titled “Bug fixes”- Wrong spill-operator result for blocked spill
QUERYcoerces dates to string by local timezone- Support structured references in
INDIRECT IFresult should not be marked dynamic- Make apiary bin work again
v14.10.1 (2025-10-01)
Section titled “v14.10.1 (2025-10-01)”Features
Section titled “Features”- Allow disabling common-subexpression extraction
Bug fixes
Section titled “Bug fixes”- Fix 1904 epoch bug when converting JSF to CSF
v14.10.0 (2025-09-25)
Section titled “v14.10.0 (2025-09-25)”Features
Section titled “Features”- Export
FormulaSyntaxErrorandEvaluationError - Add
Model.empty()convenience method - Export recalculation option symbols
Bug fixes
Section titled “Bug fixes”ADDRESSwith invalid coordinates yields wrong results or throws exception- Use source number format in
TO_TEXT - Set number format in
TO_*functions XLOOKUPthird argument wrongly forced up-to-dateROWetc. fail to update when name ref updates- False-pos circdep in
SUMIF/AVERAGEIFon target range extension Workbook.iterFormulaCellshad a bug and a redundant sibling- Update JSF support
Documentation
Section titled “Documentation”- Add a quick start guide
- Add Starlight-based documentation system
- Add tutorial for creating a spreadsheet from scratch
- Document some methods on the
Modelclass
v14.9.5 (2025-09-01)
Section titled “v14.9.5 (2025-09-01)”Features
Section titled “Features”- Add support for loading and converting JSF format
v14.9.3 (2025-08-12)
Section titled “v14.9.3 (2025-08-12)”Bug fixes
Section titled “Bug fixes”CONVERTfailed to handle reference value argument
Dependencies
Section titled “Dependencies”- Bump
fxfrom 4.11.2 to 4.12.0 [bot] - Bump
msgpackrfrom 1.11.4 to 1.11.5 [bot]
v14.9.2 (2025-07-23)
Section titled “v14.9.2 (2025-07-23)”Bug fixes
Section titled “Bug fixes”- Fix defined names causing slow iterative calculation
v14.9.1 (2025-07-22)
Section titled “v14.9.1 (2025-07-22)”Version revoked due to incorrect fix.
v14.9.0 (2025-06-27)
Section titled “v14.9.0 (2025-06-27)”- Detect parent labels
v14.8.7 (2025-06-26)
Section titled “v14.8.7 (2025-06-26)”Bug fixes
Section titled “Bug fixes”- Label sequence confusion from sparse values at island top/left
Dependencies
Section titled “Dependencies”- Bump
component-emitterfrom 1.3.0 to 2.0.0 - Upgrade
pg-memto 2.6.15
v14.8.6 (2025-06-12)
Section titled “v14.8.6 (2025-06-12)”Bug fixes
Section titled “Bug fixes”- Leftmost-column data mistaken for a label sequence
v14.8.5 (2025-06-08)
Section titled “v14.8.5 (2025-06-08)”Bug fixes
Section titled “Bug fixes”- Skip over gaps in mixed-type label subsequences
v14.8.4 (2025-06-08)
Section titled “v14.8.4 (2025-06-08)”- Improve island-based label detection