Commit 53bc0ad
authored
ENG-1454: Base getter() task (#828)
* ENG-1465: Add "Use new settings store" feature flag (#811)
* ENG-1454: Enable dual read feature flag
* add caller
* resolve merge conflicts
* review
* example fix
* review
* ENG-1455: Dual-read from old-system settings and from blockprops (#812)
* ENG-1455: Dual-read from old-system settings and from blockprops
* remove console.logs
* address review
* prettier
* add comment describing why we do it like this
* reviewed
* rebase, review
* address review
* explicit throw, now zodschema default reads
* replace throw with warn, throw was too strong, with warn we test if legacy and block props are same
* ENG-1472: Refactor BlockPropSettingPanels to add accessor-backed default reads (#813)
* ENG-1472: Refactor BlockPropSettingPanels to add accessor-backed default reads (with initialValue fallback)
* review, fix
* Eng-1473 port non boolean personal setting consumers (#823)
* ENG-1473: Rebase onto updated eng-1472, resolve conflicts and fix missing import
* ENG-1473: Review fixes — remove dead extensionAPI params, fix type casts
* ENG-1473: Fix restrict-template-expressions warnings in query utils
* rename persistQueryPages → setQueryPages, setQueryPages → setInitialQueryPages
* restore legacy type coercion for query-pages
* ENG-1467: Port global setting consumer reads (→ getGlobalSetting) (#824)
* ENG-1467: Port global setting consumer reads (→ getGlobalSetting)
* prettier
* rebase onto updated eng-1472, resolve conflicts and fix tsc
* ENG-1456: Migrate personal boolean flag consumers (getSetting → getPersonalSetting) (#843)
* ENG-1479: Port suggestive mode settings to dual-read (#845)
* ENG-1479: Port suggestive mode settings to dual-read
Add dual-read routing to getFeatureFlag for flags with legacy counterparts.
FEATURE_FLAG_LEGACY_MAP maps "Suggestive mode enabled" and "Enable left
sidebar" to their config tree reads, gated by isNewSettingsStoreEnabled().
Flags without legacy entries go straight to block props (no behavior change).
Migrate value reads from getFormattedConfigTree() / extensionAPI.settings
to accessors:
- index.ts: getUidAndBooleanSetting → getFeatureFlag (accessor handles
legacy fallback now)
- AdminPanel: suggestiveModeEnabled.value → getFeatureFlag in both
FeatureFlagsTab and main component. Removed unused useMemo.
- SuggestiveModeSettings: includePageRelations state init →
getGlobalSetting. Dropped initialValue from both GlobalFlagPanels.
- hyde.ts: orphan extensionAPI.settings.get() reads →
getGlobalSetting. Fixes bug where sync config toggles had no
runtime effect (keys were never written by any code).
Structural UIDs remain with getFormattedConfigTree().
pageGroups.groups NOT migrated — type mismatch (legacy PageGroup has
UIDs, Zod PageGroup does not). Deferred to ENG-1470.
* Align hyde.ts fallback defaults with Zod schema
* Fix eslint naming-convention warnings in FEATURE_FLAG_LEGACY_MAP
* Add TODO(ENG-1484) for suggestive mode reactivity
* ENG-1468: Port node and relations tree based consumers (#825)
* Rebase ENG-1468 onto eng-1472 (fresh redo)
* Remove unnecessary lazy initializer from complement useState
* Break circular dep: inline DISCOURSE_CONFIG_PAGE_TITLE
* Address review: case-insensitive attribute lookup, empty-string fallback, move DISCOURSE_CONFIG_PAGE_TITLE to data/constants
* ENG-1478: Port left sidebar to dual-readers + reactivity
* ENG-1478: DRY merge helpers, real dual-read in settings panels, emitter-based reactivity
- Extract mergeGlobalSectionWithAccessor/mergePersonalSectionsWithAccessor to
getLeftSidebarSettings.ts (used by LeftSidebarView, GlobalSettings, PersonalSettings)
- Settings panels now actually use accessor values (not fire-and-forget)
- Replace setLeftSidebarFlagHandler with settingsEmitter pub/sub
- useConfig subscribes to emitter instead of old subscribe()
- Wire up global/personal/flag handlers in pullWatchers to emit
- Remove dead newValue===oldValue guard (hasPropChanged already checks)
- Remove dead ?./?? in personal merge (Zod defaults guarantee values)
* ENG-1478: refresh config tree before emitter-driven rebuild
toggleFoldedState mutates folded.uid in-place then dual-writes to
block props, triggering pull watch → emitter → buildConfig(). Without
refreshConfigTree(), buildConfig() reads stale UIDs from the cached
tree, overwriting the in-place mutation and orphaning Roam blocks.
* ENG-1478: extract emitter key constants, fix stale TODO
* ENG-1478: fix legacy adapter child.uid → child.text, fix settingKeys on personal panels
* ENG-1478: skip reload prompt when new settings store is active
* ENG-1478: guard getGlobalSetting/getPersonalSetting against empty keys
* ENG-1469: Refactor getDiscourseRelations and getDiscourseNodes to read from block props
* Fix: use backedBy 'user' for initial discourse nodes in block props
initSingleDiscourseNode was writing backedBy: "default" which caused
nodes to be filtered out of the settings panel when the flag is ON.
Also fixes up existing graphs that already stored the wrong value.
* Hardcode backedBy 'user' in toDiscourseNode to match legacy behavior
* ENG-1484: reactive settings for triggers and suggestive mode overlay
* ENG-1484: use nullish coalescing for trigger fallbacks to match init semantics
* ENG-1484: reactive suggestive mode flag, gate reload prompts on new settings store
* ENG-1484: broad selector for suggestive overlay cleanup to match legacy
* ENG-1484: fix prettier formatting (root config)
* ENG-1484: drop reload prompts — reactivity is unconditional via dual-write
* ENG-1499: Replace raw string accessor paths with shared constants from settingKeys.ts
Replaces all raw string literals used as block-prop accessor paths in consumer files
with typed constants from settingKeys.ts. Adds TEMPLATE_SETTING_KEYS constant and
tightens KeyboardShortcutInput blockPropKey prop from string to keyof PersonalSettings.
* Fix missed keyImageOption getter to use shared constants
* ENG-1519: Add legacy-to-blockprops migration, remove backedBy from schema
* Fix shouldWrite skipping after reconciliation, warn on missing personal block
* Add migration telemetry for legacy parse failures
* Use backend query for node migration
* ENG-735: Remove inline settings UI from config pages
* Use Object.values instead of unused destructured key
* ENG-1503: Replace getFormattedConfigTree consumers with direct helper calls
* ENG-1484: extract constants, remove dead defaults per review
* Fix useConfig to call buildConfig() for dual-read merge
* prettier
* Fix import path for DISCOURSE_CONFIG_PAGE_TITLE
* fix review
* ENG-1470: Fix dual-read gaps found during flag-ON validation (#896)
* ENG-1470: Fix dual-read gaps found during flag-ON validation
Bootstrap legacy config blocks in initSchema so the plugin works on
fresh graphs without createConfigObserver/configPageTabs:
- trigger, grammar/relations, export, Suggestive Mode, Left Sidebar
- Reuses existing ensureBlocksExist/buildBlockMap helpers + DEFAULT_RELATION_VALUES
Fix duplicate block accumulation bugs:
- Page Groups: getSubTree auto-create race (ensureLegacyConfigBlocks pre-creates)
- Folded: lookup-based delete via getBasicTreeByParentUid instead of stale uid
- scratch/enabled: switched getSubTree({ parentUid }) to tree-based reads
- Folded in convertToComplexSection: removed erroneous block creation
Fix dual-read comparison:
- Replace JSON.stringify with deepEqual (handles key order, undefined/empty/false)
- Deterministic async ordering: await legacy write → refreshConfigTree → blockProp write
(BlockPropSettingPanels, LeftSidebarView toggleFoldedState, AdminPanel suggestive mode)
- Use getPersonalSettings() (raw read) in toggleFoldedState to avoid mid-write comparison
Fix storedRelations import path (renderNodeConfigPage → data/constants)
* Fix dual-read mismatches and ZodError on discourse node parse
* Fix dual-read mismatches: alias timing, key-image re-render, deepEqual null
* Fix prettier formatting
* ENG-1574: Add dual-read console logs to setting setters (#914)
* ENG-1574: Add dual-read console logs to setting setters
Log legacy and block prop values with match/mismatch status
when a setting is changed. Fix broken import in storedRelations.
* ENG-1574: Add init-time dual-read log and window.dgDualReadLog()
Log all legacy vs block prop settings on init. Remove setter
logging. Expose dgDualReadLog() on window for on-demand use.
* ENG-1574: Fix eslint naming-convention warnings in init.ts
* ENG-1574: Use deepEqual instead of JSON.stringify for comparison
JSON.stringify is key-order dependent, causing false mismatches
when legacy and block props return keys in different order.
* ENG-1574: Remove dead code, use deepEqual for comparison
* ENG-1574: Fix review feedback — try-catch, flag exclusion, type guard
* Eng 1616 add getconfigtree equivalent for block pros on init (#944)
* ENG-1616: Bulk-read settings + thread snapshot (with timing logs)
Cut plugin load from ~20925ms to ~1327ms (94%) on a real graph by collapsing
per-call settings accessors into a single bulk read at init and threading
that snapshot through the init chain + observer callbacks.
Key changes:
- accessors.ts: bulkReadSettings() runs ONE pull query against the settings
page's direct children and returns { featureFlags, globalSettings,
personalSettings } parsed via Zod. readPathValue exported.
- getDiscourseNodes / getDiscourseRelations / getAllRelations: optional
snapshot param threaded through, no breaking changes to existing callers.
- initializeDiscourseNodes + refreshConfigTree (+ registerDiscourseDatalog-
Translators, getDiscourseRelationLabels): accept and forward snapshot.
- index.ts: bulkReadSettings() at the top of init; snapshot threaded into
initializeDiscourseNodes, refreshConfigTree, initObservers,
installDiscourseFloatingMenu, setInitialQueryPages, and the 3 sync sites
inside index.ts itself.
- initializeObserversAndListeners.ts: snapshot threaded into the sync-init
body; pageTitleObserver + leftSidebarObserver callbacks call
bulkReadSettings() per fire (fresh, not stale); nodeTagPopupButtonObserver
uses per-sync-batch memoization via queueMicrotask; hashChangeListener
and nodeCreationPopoverListener use bulkReadSettings() per fire.
- findDiscourseNode: snapshot param added; getDiscourseNodes() default-arg
moved inside the cache-miss branch so cache hits don't waste the call.
- isQueryPage / isCanvasPage / QueryPagesPanel.getQueryPages: optional
snapshot param.
- LeftSidebarView.buildConfig / useConfig / mountLeftSidebar: optional
initialSnapshot threaded for the first render; emitter-driven updates
keep using live reads for post-mount reactivity.
- DiscourseFloatingMenu.installDiscourseFloatingMenu: optional snapshot.
- posthog.initPostHog: removed redundant internal getPersonalSetting check
(caller already guards from the snapshot).
- migrateLegacyToBlockProps.hasGraphMigrationMarker: accepts the existing
blockMap and does an O(1) lookup instead of a getBlockUidByTextOnPage scan.
Includes per-phase timing console.logs across index.ts, refreshConfigTree,
init.ts, initSettingsPageBlocks, and initObservers. Committed as a
checkpoint so we can reference measurements later; will be removed in the
next commit.
* ENG-1616: Remove plugin-load timing logs
Removes the per-phase console.log instrumentation added in the previous
commit. All the [DG Plugin] / [DG Nav] logs and their `mark()` / `markPhase()`
helpers are gone. Code behavior unchanged.
Dropped in this commit:
- index.ts: mark() closure, load start/done logs, and all phase marks.
- initializeObserversAndListeners.ts: markPhase() closure, per-observer marks,
pageTitleObserver fire log, hashChangeListener [DG Nav] logs.
- LeftSidebarView.tsx: openTarget [DG Nav] click/resolve logs.
- refreshConfigTree.ts: mark() closure and all phase marks.
- init.ts: mark() closures in initSchema and initSettingsPageBlocks.
- accessors.ts: bulkReadSettings internal timing log.
- index.ts: unused getPluginElapsedTime import.
Previous commit (343dc11) kept as a checkpoint for future drill-downs.
* ENG-1616: Address review — typed indexing, restore dgDualReadLog, optional snapshot
- index.ts: move initPluginTimer() back to its original position (after
early-return checks) so timing isn't started for graphs that bail out.
- Replace readPathValue + `as T | undefined` casts with direct typed
indexing on the Zod-derived snapshot types across:
- index.ts (disallowDiagnostics, isStreamlineStylingEnabled)
- initializeObserversAndListeners.ts (suggestiveModeOverlay,
pagePreview, discourseContextOverlay, globalTrigger,
personalTriggerCombo, customTrigger) — also drops dead `?? "\\"`
and `?? "@"` fallbacks since Zod defaults already populate them.
- isCanvasPage.ts (canvasPageFormat)
- setQueryPages.ts + QueryPagesPanel.tsx (nested [Query][Query pages])
- setQueryPages.setInitialQueryPages: snapshot is now optional with a
getPersonalSetting fallback, matching the pattern used elsewhere
(getQueryPages, isCanvasPage, etc.).
- init.ts: restore logDualReadComparison + window.dgDualReadLog so the
on-demand console helper is available again. NOT auto-called on init —
invoke window.dgDualReadLog() manually to dump the comparison.
* ENG-1616: Log total plugin load time
Capture performance.now() at the top of runExtension and log the
elapsed milliseconds just before the unload handler is wired, so we
have a single broad measurement of plugin init cost on each load.
* ENG-1616: Tighten init-only leaves to required snapshot, AGENTS.md compliance
Make snapshot required at six init-only leaves where caller audit
showed every site already passed one: installDiscourseFloatingMenu,
initializeDiscourseNodes, setInitialQueryPages, isQueryPage,
isCurrentPageCanvas, isSidebarCanvas. No cascade — only at the leaves.
Drop dead fallback code that was reachable only via the optional path:
- setQueryPages: legacy string|Record coercion ladder (snapshot is Zod-typed string[])
- DiscourseFloatingMenu: getPersonalSetting<boolean> cast site
- DiscourseFloatingMenu: unused props parameter (no caller ever overrode default)
- initializeObserversAndListeners: !== false dead pattern (Zod boolean default)
- initializeObserversAndListeners: as IKeyCombo cast (schema is structurally compatible)
AGENTS.md compliance for >2-arg functions:
- mountLeftSidebar: object-destructured params, both call sites updated
- installDiscourseFloatingMenu: kept at 2 positional via dead-props removal
posthog: collapse doInitPostHog wrapper into initPostHog (caller-side gating).
accessors: revert speculative readPathValue export (no consumer).
LeftSidebarView/DiscourseFloatingMenu: eslint-disable react/no-deprecated on
ReactDOM.render rewritten lines, matching existing codebase convention.
* ENG-1616: Address review — rename snapshot vars, flag-gate bulkRead, move PostHog guards
- Rename settingsSnapshot/callbackSnapshot/snap/navSnapshot → settings
- bulkReadSettings now checks "Use new settings store" flag and falls
back to legacy reads when off, matching individual getter behavior
- Move encryption/offline guards into initPostHog (diagnostics check
stays at call site to avoid race with async setSetting in enablePostHog)
* Fix legacy bulk settings fallback
* ENG-1617: se existing 'getConfigTree equivalent functions' for specific setting groups (#946)
* ENG-1616: Bulk-read settings + thread snapshot (with timing logs)
Cut plugin load from ~20925ms to ~1327ms (94%) on a real graph by collapsing
per-call settings accessors into a single bulk read at init and threading
that snapshot through the init chain + observer callbacks.
Key changes:
- accessors.ts: bulkReadSettings() runs ONE pull query against the settings
page's direct children and returns { featureFlags, globalSettings,
personalSettings } parsed via Zod. readPathValue exported.
- getDiscourseNodes / getDiscourseRelations / getAllRelations: optional
snapshot param threaded through, no breaking changes to existing callers.
- initializeDiscourseNodes + refreshConfigTree (+ registerDiscourseDatalog-
Translators, getDiscourseRelationLabels): accept and forward snapshot.
- index.ts: bulkReadSettings() at the top of init; snapshot threaded into
initializeDiscourseNodes, refreshConfigTree, initObservers,
installDiscourseFloatingMenu, setInitialQueryPages, and the 3 sync sites
inside index.ts itself.
- initializeObserversAndListeners.ts: snapshot threaded into the sync-init
body; pageTitleObserver + leftSidebarObserver callbacks call
bulkReadSettings() per fire (fresh, not stale); nodeTagPopupButtonObserver
uses per-sync-batch memoization via queueMicrotask; hashChangeListener
and nodeCreationPopoverListener use bulkReadSettings() per fire.
- findDiscourseNode: snapshot param added; getDiscourseNodes() default-arg
moved inside the cache-miss branch so cache hits don't waste the call.
- isQueryPage / isCanvasPage / QueryPagesPanel.getQueryPages: optional
snapshot param.
- LeftSidebarView.buildConfig / useConfig / mountLeftSidebar: optional
initialSnapshot threaded for the first render; emitter-driven updates
keep using live reads for post-mount reactivity.
- DiscourseFloatingMenu.installDiscourseFloatingMenu: optional snapshot.
- posthog.initPostHog: removed redundant internal getPersonalSetting check
(caller already guards from the snapshot).
- migrateLegacyToBlockProps.hasGraphMigrationMarker: accepts the existing
blockMap and does an O(1) lookup instead of a getBlockUidByTextOnPage scan.
Includes per-phase timing console.logs across index.ts, refreshConfigTree,
init.ts, initSettingsPageBlocks, and initObservers. Committed as a
checkpoint so we can reference measurements later; will be removed in the
next commit.
* ENG-1616: Remove plugin-load timing logs
Removes the per-phase console.log instrumentation added in the previous
commit. All the [DG Plugin] / [DG Nav] logs and their `mark()` / `markPhase()`
helpers are gone. Code behavior unchanged.
Dropped in this commit:
- index.ts: mark() closure, load start/done logs, and all phase marks.
- initializeObserversAndListeners.ts: markPhase() closure, per-observer marks,
pageTitleObserver fire log, hashChangeListener [DG Nav] logs.
- LeftSidebarView.tsx: openTarget [DG Nav] click/resolve logs.
- refreshConfigTree.ts: mark() closure and all phase marks.
- init.ts: mark() closures in initSchema and initSettingsPageBlocks.
- accessors.ts: bulkReadSettings internal timing log.
- index.ts: unused getPluginElapsedTime import.
Previous commit (343dc11) kept as a checkpoint for future drill-downs.
* ENG-1616: Address review — typed indexing, restore dgDualReadLog, optional snapshot
- index.ts: move initPluginTimer() back to its original position (after
early-return checks) so timing isn't started for graphs that bail out.
- Replace readPathValue + `as T | undefined` casts with direct typed
indexing on the Zod-derived snapshot types across:
- index.ts (disallowDiagnostics, isStreamlineStylingEnabled)
- initializeObserversAndListeners.ts (suggestiveModeOverlay,
pagePreview, discourseContextOverlay, globalTrigger,
personalTriggerCombo, customTrigger) — also drops dead `?? "\\"`
and `?? "@"` fallbacks since Zod defaults already populate them.
- isCanvasPage.ts (canvasPageFormat)
- setQueryPages.ts + QueryPagesPanel.tsx (nested [Query][Query pages])
- setQueryPages.setInitialQueryPages: snapshot is now optional with a
getPersonalSetting fallback, matching the pattern used elsewhere
(getQueryPages, isCanvasPage, etc.).
- init.ts: restore logDualReadComparison + window.dgDualReadLog so the
on-demand console helper is available again. NOT auto-called on init —
invoke window.dgDualReadLog() manually to dump the comparison.
* ENG-1616: Log total plugin load time
Capture performance.now() at the top of runExtension and log the
elapsed milliseconds just before the unload handler is wired, so we
have a single broad measurement of plugin init cost on each load.
* ENG-1616: Tighten init-only leaves to required snapshot, AGENTS.md compliance
Make snapshot required at six init-only leaves where caller audit
showed every site already passed one: installDiscourseFloatingMenu,
initializeDiscourseNodes, setInitialQueryPages, isQueryPage,
isCurrentPageCanvas, isSidebarCanvas. No cascade — only at the leaves.
Drop dead fallback code that was reachable only via the optional path:
- setQueryPages: legacy string|Record coercion ladder (snapshot is Zod-typed string[])
- DiscourseFloatingMenu: getPersonalSetting<boolean> cast site
- DiscourseFloatingMenu: unused props parameter (no caller ever overrode default)
- initializeObserversAndListeners: !== false dead pattern (Zod boolean default)
- initializeObserversAndListeners: as IKeyCombo cast (schema is structurally compatible)
AGENTS.md compliance for >2-arg functions:
- mountLeftSidebar: object-destructured params, both call sites updated
- installDiscourseFloatingMenu: kept at 2 positional via dead-props removal
posthog: collapse doInitPostHog wrapper into initPostHog (caller-side gating).
accessors: revert speculative readPathValue export (no consumer).
LeftSidebarView/DiscourseFloatingMenu: eslint-disable react/no-deprecated on
ReactDOM.render rewritten lines, matching existing codebase convention.
* ENG-1617: Single-pull settings reads + dialog snapshot threading
`getBlockPropBasedSettings` now does one Roam `pull` that returns the
settings page's children with their string + uid + props in one shot,
replacing the `q`-based `getBlockUidByTextOnPage` (~290ms per call) plus
a second `pull` for props. `setBlockPropBasedSettings` reuses the same
helper for the uid lookup so we have a single pull-and-walk pattern.
`SettingsDialog` captures a full settings snapshot once at mount via
`useState(() => bulkReadSettings())` and threads `featureFlags` and
`personalSettings` down to `HomePersonalSettings`. Each child component
(`PersonalFlagPanel`, `NodeMenuTriggerComponent`,
`NodeSearchMenuTriggerSetting`, `KeyboardShortcutInput`) accepts an
`initialValue` prop and seeds its local state from the snapshot instead
of calling `getPersonalSetting` on mount. `PersonalFlagPanel`'s
`initialValue` precedence flips so the prop wins when provided;
`QuerySettings` callers without a prop still hit the existing fallback.
`getDiscourseNodes`, `getDiscourseRelations`, and `getAllRelations`
narrow their snapshot parameter to `Pick<SettingsSnapshot, ...>` to
declare which fields each function actually reads.
Adds a one-line `console.log` in `SettingsDialog` reporting the dialog
open time, kept as an ongoing perf monitor.
* ENG-1617: Refresh snapshot on Home tab nav + reuse readPathValue
CodeRabbit catch: with `renderActiveTabPanelOnly={true}`, the Home tab's
panel unmounts/remounts when the user navigates away and back. Each
re-mount re-runs `useState(() => initialValue ?? false)` in
`BaseFlagPanel`, re-seeding from whatever `initialValue` is currently
passed. Because the dialog held the snapshot in a non-updating
`useState`, that path served stale values, so toggles made earlier in
the same dialog session would visually revert after a tab round-trip.
Fix: hold the snapshot in a stateful slot and refresh it via
`bulkReadSettings()` from the Tabs `onChange` handler when the user
navigates back to Home. The setState batches with `setActiveTabId`, so
the new mount sees the fresh snapshot in the same render.
Also replace the inline reducer in `getBlockPropBasedSettings` with the
existing `readPathValue` util — same traversal but consistent with the
rest of the file and adds array-index handling for free.
* ENG-1617: Per-tab snapshot threading via bulkReadSettings
Replaces the dialog-level snapshot from earlier commits with a per-tab
snapshot model that scales to every tab without per-tab plumbing in the
dialog itself.
In accessors.ts, the three plural getters (getFeatureFlags,
getGlobalSettings, getPersonalSettings) now delegate to the existing
bulkReadSettings, which does one Roam pull on the settings page and
parses all three schemas in a single pass. The slow q-based
getBlockPropBasedSettings is deleted (it was only used by the plural
getters and the set path); setBlockPropBasedSettings goes back to
calling getBlockUidByTextOnPage directly. Writes are infrequent enough
that the q cost is acceptable on the set path.
Each tab container that renders panels at mount captures one snapshot
via useState(() => bulkReadSettings()) and threads the relevant slice as
initialValue down to its panels: HomePersonalSettings, QuerySettings,
GeneralSettings, ExportSettings. The Personal and Global panels in
BlockPropSettingPanels.tsx flip their initialValue precedence to prefer
the prop and fall back to the live read only when the prop is missing,
so callers that don't pass initialValue (e.g. LeftSidebarGlobalSettings,
which already passes its own value) continue to behave the same way.
NodeMenuTriggerComponent, NodeSearchMenuTriggerSetting, and
KeyboardShortcutInput accept an initialValue prop and seed local state
from it instead of calling getPersonalSetting in their useState
initializer.
Settings.tsx wraps getDiscourseNodes() in useState so it doesn't re-run
on every dialog re-render.
The dialog-level snapshot, the snapshot-refresh-on-Home-tab-nav
workaround, and the Pick<SettingsSnapshot, ...> type narrowings are all
gone.
* ENG-1617: Lift settings snapshot to SettingsDialog, thread to all tabs
Move bulkReadSettings() from per-tab useState into a single call at
SettingsDialog mount. Each tab receives its snapshot slice (globalSettings,
personalSettings, featureFlags) as props. Remove dual-read mismatch
console.warn logic from accessors. Make initialValue caller-provided in
BlockPropSettingPanel wrappers. Add TabTiming wrapper for per-tab
commit/paint perf measurement.
* ENG-1617: Remove timing instrumentation, per-call dual-read, flag-aware bulkReadSettings
- Remove TabTiming component and all console.log timing from Settings dialog
- Remove per-call dual-read comparison from getGlobalSetting, getPersonalSetting,
getDiscourseNodeSetting (keep logDualReadComparison for manual use)
- Make bulkReadSettings flag-aware: reads from legacy when flag is OFF,
block props when ON
- Remove accessor fallbacks from Global/Personal wrapper panels (initialValue
now always passed from snapshot)
- Remove getGlobalSetting/getPersonalSetting imports from BlockPropSettingPanels
* ENG-1617: Eliminate double bulkReadSettings calls in accessor functions
getGlobalSetting, getPersonalSetting, getFeatureFlag, getDiscourseNodeSetting
now each do a single bulkReadSettings() call instead of calling
isNewSettingsStoreEnabled() (which triggered a separate bulkReadSettings)
followed by another bulkReadSettings via the getter. bulkReadSettings already
handles the flag check and legacy/block-props routing internally.
* ENG-1617: Re-read snapshot on tab change to prevent stale initialValues
Replace useState with useMemo keyed on activeTabId so bulkReadSettings()
runs fresh each time the user switches tabs. Fixes stale snapshot when
renderActiveTabPanelOnly unmounts/remounts panels.
* ENG-1616: Address review — rename snapshot vars, flag-gate bulkRead, move PostHog guards
- Rename settingsSnapshot/callbackSnapshot/snap/navSnapshot → settings
- bulkReadSettings now checks "Use new settings store" flag and falls
back to legacy reads when off, matching individual getter behavior
- Move encryption/offline guards into initPostHog (diagnostics check
stays at call site to avoid race with async setSetting in enablePostHog)
* ENG-1617: Fix DiscourseNodeSelectPanel fallback to use first option instead of empty string
* ENG-1617: Rename snapshot variables to settings for clarity
* Fix legacy bulk settings fallback
* fix bug similar code
* Fix stale startup snapshot: populate config tree before bulkReadSettings
Without a prior refreshConfigTree() call, bulkReadSettings() reads an empty
discourseConfigRef.tree when the flag is OFF, returning legacy defaults
instead of user data. Populate the tree first so the initial snapshot
reflects actual settings.
* Fix toggleFoldedState race on rapid chevron clicks
Move setIsOpen to the top of the function so the optimistic UI update
lands synchronously. Add an isTogglingRef guard in PersonalSectionItem
and GlobalSection so concurrent clicks while Roam writes are in flight
are dropped instead of firing duplicate createBlock/deleteBlock on the
same Folded node.
* Keep Query pages schema default at discourse-graph/queries/*
Previous commit accidentally included an older change that set
"Query pages" default to []. Pre-migration, setQueryPages auto-seeds
this path on every plugin load, so the effective user-facing default
has always been ["discourse-graph/queries/*"]. Restore that value.
* Include Reified relation triples in legacy personal settings
Add reifiedRelationTriples to PERSONAL_KEYS and map
"Reified relation triples" -> "use-reified-relations" in
PERSONAL_SCHEMA_PATH_TO_LEGACY_KEY. Without these entries,
readAllLegacyPersonalSettings iterates over PERSONAL_KEYS values
and omits this field, so the migration to block props fills in the
Zod default (false) instead of the user's stored preference.
* Close panel races between syncToBlock and refreshConfigTree
BaseTextPanel: track the inner setTimeout via debounceRef so unmount
cleanup cancels it instead of letting it fire on an unmounted component.
BaseNumberPanel and BaseSelectPanel: add the same 100ms ordering gap
between the non-awaited syncToBlock write and refreshConfigTree that
BaseFlagPanel and BaseTextPanel already have, tracked via
refreshTimeoutRef for cleanup. Without the gap, refreshConfigTree can
read the tree before the async Roam write settles.
* Report per-key mismatches in dgDualReadLog; fix legacy returnNode read
- Iterate top-level keys per group (Personal, Global, Feature Flags, each
node) and log which specific keys mismatch, with legacy vs block values
side-by-side. Previously only group-level match/mismatch was reported.
- Skip template, specification, and index in node comparisons — their
legacy and block-prop shapes diverge structurally (uid + empty children
on templates; condition uids stripped on save) not semantically. Inline
comment cites the responsible readers/writers.
- Fix getLegacyQuerySettingByParentUid returning a hardcoded "node" for
returnNode instead of reading scratch > return > [value] from the tree.
* ENG-1664: Replace getBlockUidByTextOnPage scan with pull in settings writer (#975)
The Datalog .q scan that resolved the top-level settings block on every
write took ~318 ms per call on real graphs. Replaced with a hash-indexed
roamAlphaAPI.pull by page title that walks :block/children — same pattern
bulkReadSettings already uses.
Behavior equivalence: the old query matched any descendant via :block/page;
the new pull traverses direct :block/children only. Safe here because
setBlockPropBasedSettings is only reached with keys[0] set to one of the
three top-level settings keys (feature flags, global, personal user-id),
all guaranteed to be direct children of the settings page.
Measured on a real graph: BaseFlagPanel click handler 420 ms → 97 ms
(−77%). The remaining 95 ms is refreshConfigTree > register translators,
the inherent dual-read cost ENG-1616 flagged for post-ENG-1470 cleanup.
* ENG-1659: Use typed canvasSettings.color keys on color picker writes (#980)
* ENG-1663 Await scratch/conditions/return creation before writing children (#982)
Legacy tree init was calling getSubTree with parentUid, which fires
createBlock without awaiting. Subsequent createBlock calls under those
uids raced Roam's async write and failed with 'Parent entity doesn't
exist'. Replace the three fire-and-forget lookups with tree-mode
getSubTree + awaited createBlock fallback.
* ENG-1671 Default Reified relation triples to true for block props (#1004)
* ENG-1668: Perf: slow load time (#984)
* ENG-1668: Remove dead initDiscourseNodePages + skip redundant refreshConfigTree
Remove initDiscourseNodePages (and helpers hasNonDefaultNodes,
initSingleDiscourseNode) from initSchema — migrateDiscourseNodes already
handles block prop writes, making this redundant on every load.
Make the second refreshConfigTree(settings) conditional — only runs when
initializeDiscourseNodes actually creates pages (first install). On
existing graphs the tree is identical to what the first call already read.
* Add per-step load timing logs to index.ts
* Add per-step load timing logs to initObservers
* Split listeners + queryBuilder timing logs
* Add detailed timing logs to nodeTagPopupButtonObserver callback
* Add detailed timing logs to all observer callbacks
* Thread config tree through legacy getters to avoid redundant fetches
* Remove timing instrumentation from index.ts and initObservers
* Add total plugin load time log
* Eng 1672: Settings not reading from block props (#983)
* ENG-1672: Flip left sidebar merge helpers to iterate snapshot
* ENG-1672: Route getExportSettings runtime reads through bulkReadSettings
* ENG-1672: Restore defaultValue hydration for Template and Attributes panels
PR #784 dropped the defaultValue prop when porting BlocksPanel to
DualWriteBlocksPanel, breaking block-props reads when legacy is empty.
Restore the hydration path and add flag-gated dev nuke command.
* ENG-1672: Fix review findings — children:[] not undefined, merge attributes with tree UIDs
* ENG-1672: Route createDiscourseNode template reads through getDiscourseNodeSetting
* fix: render left sidebar settings from selected store
* ENG-1672: Preserve legacy sidebar config when accessor snapshot is undefined
* Revert "ENG-1672: Preserve legacy sidebar config when accessor snapshot is undefined"
This reverts commit 3b3f197.
* ENG-1713: Relations do not show up in settings panel (#1010)
* ENG-1713: Read Relations panel from globalSettings, fix migrate clobbering reconcile
DiscourseRelationConfigPanel.refreshRelations now derives the list from
globalSettings.Relations (flag-aware via bulkReadSettings) instead of
querying the legacy tree directly. This removes the panel's dependency
on the cached relations parent uid, which is "" on first open of a fresh
graph until the dialog is closed and reopened.
initSchema now refreshes discourseConfigRef.tree between
initSettingsPageBlocks and migrateGraphLevel when the cache hasn't
caught up to the seeded grammar block. Without it, getLegacyGlobalSetting
reads the stale cache, returns schema-default placeholder-keyed Relations
(_INFO-rel, etc.), and migrateSection overwrites the real-uid keys
reconcileRelationKeys just wrote — leaving block props keyed by
placeholders that the legacy edit panel cannot resolve.
Cold path adds ~6ms once per graph; hot path skips via .some() check.
* ENG-1713: Defer back() 50ms after relation save so list reflects write
setGlobalSetting fires window.roamAlphaAPI.data.block.update without
awaiting the returned promise. The previous code called back() in the
same tick, so handleBack's refreshRelations ran a sync pull before
Roam's index reflected the write — the list rendered pre-edit data
on add / edit / delete from inside RelationEditPanel.
setTimeout(back, 50) gives Roam a tick to commit before the read.
This is a band-aid: the proper fix is either to await the write chain
(plumb async through setBlockProps -> setGlobalSetting) or refresh via
a pull watcher / saved-data callback. Tracked separately for cleanup.
Also drops the explanatory comment above the conditional refreshConfigTree
in initSchema; the why is captured on ENG-1471 instead.
* ENG-1717: Default max filename length on invalid stored value (#1008)
* Merge origin/main into migration-block-init-staging-branch
Single conflict: apps/roam/src/components/LeftSidebarView.tsx — HEAD's
dual-read plumbing (initialSnapshot, sectionIndex, handleToggle) unioned
with main's drag-and-drop (SortableList, dragHandle, reorderGlobalChildren).
SectionChildren removed (dead after main's ChildRow/SortableList path).
Known followup: DnD reorder + fold toggle can race on stale section index
when toggle fires before reorder write settles. To be tracked separately.
* ENG-1716: Bug fix: Template does not render from block props (#1016)
* ENG-1716: Render template settings from block props
When the new settings store flag is on, materialize the block-props
template into an ephemeral Template-Block-props block as the last
child of node.type, render that, and delete it on unmount. Flag-off
behavior is unchanged.
* ENG-1716: Dual-write template edits back to legacy Template block
When the new settings store flag is on, edits in the buffer
Template-Block-props block are mirrored to the legacy Template block
via position-walked block.update calls. On length mismatch (only
expected during testing/manual divergence) the mirror logs a warning
and skips that subtree.
* ENG-1716: Extend buffer->legacy mirror to handle add/delete
Mirror now creates legacy children when buffer has extras and deletes
legacy children when buffer has fewer, in addition to in-place
block.update for matching positions. Replaces the prior length-match
guard which caused dual-write to silently skip whenever the user
added or removed template lines.
* ENG-1716: Guard render effect against stale async continuation
The ensureChildren promise resolves asynchronously, so its .then
callback could run after the effect's cleanup fired - re-registering
a pull watch on a stale renderUid and stomping pullWatchArgsRef.
Add a cancelled flag (same pattern as the buffer-lifecycle effect)
so the continuation aborts after teardown.
* ENG-1716: Mirror heading and open alongside string
The mirror only pushed block string to legacy, so a heading-level
or collapse-state change in the buffer never propagated. Update
the same block.update call to include heading and open whenever
any of the three differ between buffer and legacy.
Also drops uid from the buffer-creation effect's dependency array
since the body no longer references it.
* ENG-1716: Rename useNewStore -> isNewStore
useNewStore reads like a React hook (use* convention) but is a
boolean. Rename to isNewStore so the name matches the type.
* ENG-1755: Normalize legacy query-pages snapshot before array operations (#1034)
setInitialQueryPages reads the personal query-pages setting from the
snapshot and immediately calls .includes / spreads it. With the new
settings store flag OFF, bulkReadSettings returns the raw legacy value
cast as PersonalSettings, but the legacy extensionAPI shape is
string | string[] | Record<string, string>. That throws on objects and
spreads strings per-character.
Route through the existing getQueryPages helper (already used by
isQueryPage, listActiveQueries, resolveQueryBuilderRef), which coerces
the legacy shapes to string[] before the append logic runs.
* ENG-1757: Read fresh relation settings on edit to prevent duplicate data loss (#1035)
* ENG-1757: Read fresh relation settings in editor to prevent duplicate-edit data loss
RelationEditPanel was reading the relation from the globalSettings prop,
which is a snapshot taken when the Relations tab was opened. After
handleDuplicate writes a new relation via setGlobalSetting, the parent's
snapshot is not refreshed. Opening the editor on the duplicated row found
no entry, initialized source/destination/complement to blanks, and saving
overwrote the relation with empty endpoints.
Switch the lookup to getGlobalSettings().Relations[uid] so the editor
always reads current state. The prop is now unused in both panels and is
removed along with the Settings.tsx pass-through.
* ENG-1757 Refresh relation state after relation mutations
* ENG-1756: Preserve existing feature flag values during legacy migration (#1036)
* ENG-1756: Preserve existing feature flag block-prop values during legacy migration
readAllLegacyFeatureFlags only sources Enable left sidebar from legacy
config. The migration wrote the full FeatureFlags object (with Zod defaults
filling the unsourced keys as false) via setBlockProps, which clobbered any
pre-existing true values for Duplicate node alert enabled and Suggestive
mode overlay enabled.
Expose LEGACY_SOURCED_FEATURE_FLAG_KEYS to identify which keys the legacy
reader actually sources. The migration now overlays only those keys onto the
existing block-prop values, leaving everything else untouched.
* ENG-1756: Remove type assertion on LEGACY_SOURCED_FEATURE_FLAG_KEYS
Declare the keys as a const tuple and derive the map's key type from it,
so the array's type accurately describes its contents without an assertion.
* remove nuke command (#1039)
* remove unused1 parent d99f23f commit 53bc0ad
75 files changed
Lines changed: 3113 additions & 1379 deletions
File tree
- apps/roam/src
- components
- canvas
- settings
- components
- utils
- data
- utils
Some content is hidden
Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
12 | 12 | | |
13 | 13 | | |
14 | 14 | | |
15 | | - | |
16 | | - | |
17 | 15 | | |
18 | | - | |
19 | | - | |
20 | | - | |
| 16 | + | |
| 17 | + | |
| 18 | + | |
21 | 19 | | |
22 | 20 | | |
23 | 21 | | |
| |||
191 | 189 | | |
192 | 190 | | |
193 | 191 | | |
194 | | - | |
195 | | - | |
196 | | - | |
197 | | - | |
198 | | - | |
| 192 | + | |
| 193 | + | |
| 194 | + | |
| 195 | + | |
199 | 196 | | |
200 | 197 | | |
201 | 198 | | |
| |||
204 | 201 | | |
205 | 202 | | |
206 | 203 | | |
207 | | - | |
208 | | - | |
209 | | - | |
210 | | - | |
211 | | - | |
212 | | - | |
213 | | - | |
214 | | - | |
215 | | - | |
216 | | - | |
217 | | - | |
| 204 | + | |
| 205 | + | |
| 206 | + | |
| 207 | + | |
| 208 | + | |
218 | 209 | | |
219 | 210 | | |
220 | 211 | | |
| |||
223 | 214 | | |
224 | 215 | | |
225 | 216 | | |
226 | | - | |
| 217 | + | |
227 | 218 | | |
228 | 219 | | |
229 | 220 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
13 | 13 | | |
14 | 14 | | |
15 | 15 | | |
| 16 | + | |
| 17 | + | |
16 | 18 | | |
17 | 19 | | |
18 | 20 | | |
| |||
116 | 118 | | |
117 | 119 | | |
118 | 120 | | |
119 | | - | |
120 | | - | |
121 | | - | |
122 | | - | |
123 | | - | |
| 121 | + | |
124 | 122 | | |
125 | 123 | | |
126 | 124 | | |
127 | 125 | | |
128 | 126 | | |
129 | 127 | | |
130 | 128 | | |
131 | | - | |
| 129 | + | |
132 | 130 | | |
133 | 131 | | |
| 132 | + | |
134 | 133 | | |
135 | 134 | | |
136 | | - | |
137 | | - | |
138 | | - | |
| 135 | + | |
| 136 | + | |
| 137 | + | |
139 | 138 | | |
140 | 139 | | |
141 | 140 | | |
| |||
146 | 145 | | |
147 | 146 | | |
148 | 147 | | |
| 148 | + | |
149 | 149 | | |
150 | 150 | | |
151 | 151 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
28 | 28 | | |
29 | 29 | | |
30 | 30 | | |
| 31 | + | |
| 32 | + | |
31 | 33 | | |
32 | 34 | | |
33 | 35 | | |
| |||
455 | 457 | | |
456 | 458 | | |
457 | 459 | | |
| 460 | + | |
458 | 461 | | |
459 | 462 | | |
| 463 | + | |
460 | 464 | | |
461 | 465 | | |
462 | 466 | | |
463 | | - | |
464 | | - | |
465 | | - | |
466 | | - | |
467 | | - | |
| 467 | + | |
| 468 | + | |
468 | 469 | | |
469 | 470 | | |
470 | 471 | | |
| |||
477 | 478 | | |
478 | 479 | | |
479 | 480 | | |
480 | | - | |
| 481 | + | |
481 | 482 | | |
482 | 483 | | |
483 | 484 | | |
| |||
499 | 500 | | |
500 | 501 | | |
501 | 502 | | |
502 | | - | |
| 503 | + | |
503 | 504 | | |
504 | 505 | | |
505 | 506 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
24 | 24 | | |
25 | 25 | | |
26 | 26 | | |
27 | | - | |
28 | 27 | | |
29 | 28 | | |
| 29 | + | |
30 | 30 | | |
31 | 31 | | |
32 | 32 | | |
| |||
706 | 706 | | |
707 | 707 | | |
708 | 708 | | |
| 709 | + | |
709 | 710 | | |
710 | 711 | | |
| 712 | + | |
711 | 713 | | |
712 | 714 | | |
713 | 715 | | |
714 | | - | |
| 716 | + | |
715 | 717 | | |
716 | 718 | | |
717 | 719 | | |
| |||
726 | 728 | | |
727 | 729 | | |
728 | 730 | | |
729 | | - | |
| 731 | + | |
730 | 732 | | |
731 | 733 | | |
732 | 734 | | |
| |||
0 commit comments