Files
docmost/.claude/superpowers/specs/2026-04-20-base-view-draft-design.md
T

33 KiB
Raw Blame History

Base View Draft (Local-First Filter & Sort) — Design Spec

Date: 2026-04-20 Status: Draft Feature area: apps/client/src/features/base (client-only)

Goal

Make filter and sort changes on a base view local-first: they apply instantly for the editing user, are scoped to their own browser/profile, and never touch the server baseline until the user explicitly clicks "Save for everyone". A banner at the top of the table surfaces the draft state and lets the user either promote the draft to the shared baseline or discard it.

This removes the current Notion-unlike behavior where every filter/sort tweak is auto-persisted and immediately inflicted on every teammate viewing the same view.

Non-goals (v1)

  • Column layout in draft mode. Column visibility, order, and widths continue to flow through the existing debounced persistViewConfig path in use-base-table.ts:371-396. No draft behavior for them. (Listed as a future extension.)
  • Server-side per-user drafts. localStorage only. A user clearing their browser storage, switching devices, or using a different browser profile loses drafts — by design.
  • "Save as new view". The screenshot hints at a dropdown caret next to the Save button for a "save as new view" split-action. Not in v1.
  • Kanban / calendar. Only the table view type exists today; spec scopes to it but the hook is type-agnostic and will apply trivially when other view types land.
  • Automatic garbage collection of stale drafts. Drafts persist indefinitely until the user resets or saves. No TTL, no eager cleanup when baseline values match the draft.
  • Conflict UI. If another user writes a new baseline while I have local drafts, my draft silently wins on my client. No "baseline changed" warning.

UX overview

Draft banner

Placement: between the page title and BaseToolbar, inside base-table.tsx above the <BaseToolbar /> node (around base-table.tsx:192). The banner is part of the table's own layout, not a workspace-level chrome element, because it's tied to a specific view.

Render condition: isDirty === true (see "Dirty check").

Layout (match the reference screenshot):

  • Mantine <Paper withBorder radius="sm" px="md" py="xs"> with a soft background (bg="yellow.0" or bg="orange.0" depending on theme palette — pick whichever tolerates dark mode) and a small info icon on the left.
  • Left region: short message — t("Filter and sort changes are visible only to you.").
  • Right region (a <Group gap="sm">):
    • <Button variant="subtle" color="gray" size="xs">{t("Reset")}</Button> — underline-on-hover "text link" feel; wipes the draft.
    • <Button variant="filled" size="xs">{t("Save for everyone")}</Button> — primary accent (project's default theme color — orange in the screenshot maps to Mantine's configured primaryColor, so color is omitted and the theme default is used).
  • The "Save for everyone" button is omitted entirely for users without edit permission (see "Permission gating"). "Reset" always shows.
  • The banner never animates in/out on every keystroke — it only appears/disappears when isDirty flips. Add a Mantine <Transition mounted={isDirty} transition="slide-down" duration={120}> wrap if the flip is jarring; otherwise mount unconditionally with a {isDirty && ...} guard.

Filter/sort editors in draft mode

No UI affordance changes inside the filter or sort popovers themselves. They keep the same open-on-click, add/remove/edit flow. The only behavioral change is that their onChange callback writes to the draft store rather than firing updateView — completely transparent to the editor components.

Reset behavior

Click Reset → the draft hook removes its localStorage entry → the table re-renders reading filter/sorts from activeView.config (the server baseline). Any currently-open filter/sort popover closes on outside click as usual; if it's open when the user clicks Reset, the next render shows the baseline values. No notification — the banner disappearing is sufficient feedback.

Save for everyone

Click Save → call the existing useUpdateViewMutation from base-view-query.ts:43-112 with { viewId, baseId, config: { ...serverBaseline, filter: draft.filter, sorts: draft.sorts } }. On success, clear the localStorage key and show a Mantine notification t("View updated for everyone"). On error, keep the draft; the mutation already wires the error toast.

Permission gating

A user can edit this base iff their space membership grants SpaceCaslAction.Edit, SpaceCaslSubject.Base — the same check the server enforces in base-view.controller.ts:68. Viewers still get local drafts (the entire point is that local changes don't require edit permission), but their "Save for everyone" button is hidden.

Client caveat: permissions.type.ts currently only exports Settings, Member, and Page subjects. The server enum has Base but the client enum doesn't. The spec adds Base = "base" to SpaceCaslSubject and widens the SpaceAbility union — that's a one-line change plus import fix.

Data model

localStorage key

docmost:base-view-draft:v1:{userId}:{baseId}:{viewId}
  • Namespace prefix docmost:base-view-draft: keeps us from colliding with other consumers.
  • v1 is the schema version so a future breaking change can shed old entries by skipping.
  • {userId} scopes drafts so a shared-device login-swap doesn't leak drafts across accounts. userId comes from the existing useCurrentUser() hook (returns { data: ICurrentUser } — read user?.user.id), the same helper used by other authenticated client code.
  • {baseId} and {viewId} together uniquely identify which table state the draft applies to.

Value shape

// apps/client/src/features/base/types/base.types.ts (additive)
export type BaseViewDraft = {
  filter?: FilterGroup;
  sorts?: ViewSortConfig[];
  updatedAt: string; // ISO timestamp, written on each put — used only for diagnostics
};

Both filter and sorts are optional, independently. An absent field means "inherit baseline for that axis". That matters because a user who's only dirtied sorts but not filters should see the baseline filter unchanged if the baseline's filter later shifts.

Serialized as JSON by Jotai's atomWithStorage (which JSON-stringifies on write and parses on read). No schema validation on read — if the parse fails or the shape looks wrong, Jotai yields null and the hook falls back to baseline.

Client architecture

Storage atom family

File: apps/client/src/features/base/atoms/view-draft-atom.ts

Follow the existing Jotai storage pattern in home-tab-atom.ts and auth-tokens-atom.tsatomWithStorage is the codebase convention for localStorage-backed state. Since our key is dynamic per (user, base, view), pair it with atomFamily from jotai/utils:

import { atomFamily, atomWithStorage } from "jotai/utils";
import { BaseViewDraft } from "@/features/base/types/base.types";

export type ViewDraftKey = {
  userId: string;
  baseId: string;
  viewId: string;
};

const keyFor = (k: ViewDraftKey) =>
  `docmost:base-view-draft:v1:${k.userId}:${k.baseId}:${k.viewId}`;

export const viewDraftAtomFamily = atomFamily(
  (k: ViewDraftKey) =>
    atomWithStorage<BaseViewDraft | null>(keyFor(k), null),
  (a, b) =>
    a.userId === b.userId && a.baseId === b.baseId && a.viewId === b.viewId,
);

atomWithStorage handles JSON serialization, cross-tab sync via the storage event, and SSR-safe lazy reads out of the box — no hand-rolled localStorage.getItem/setItem or window.addEventListener("storage", ...) needed. The comparator passed as atomFamily's second argument ensures the same (user, base, view) triple always resolves to the same atom instance, so React Query-style object identity issues don't cause atoms to be recreated per render.

Hook: useViewDraft

File: apps/client/src/features/base/hooks/use-view-draft.ts

Thin wrapper that binds the atom family to the rendering layer, adds the passthrough-when-undefined guard, and derives effectiveFilter / effectiveSorts / isDirty / buildPromotedConfig from the atom's value:

export type ViewDraftState = {
  draft: BaseViewDraft | null;
  effectiveFilter: FilterGroup | undefined;
  effectiveSorts: ViewSortConfig[] | undefined;
  isDirty: boolean;
  setFilter: (filter: FilterGroup | undefined) => void;
  setSorts: (sorts: ViewSortConfig[] | undefined) => void;
  reset: () => void;
  buildPromotedConfig: (baseline: ViewConfig) => ViewConfig;
};

export function useViewDraft(args: {
  userId: string | undefined;
  baseId: string | undefined;
  viewId: string | undefined;
  baselineFilter: FilterGroup | undefined;
  baselineSorts: ViewSortConfig[] | undefined;
}): ViewDraftState;

Behavior:

  1. If any of userId / baseId / viewId is undefined → return a passthrough state (draft=null, isDirty=false, setters no-op, effective* fall through to baseline). Guards the initial-load window where auth / activeView hasn't resolved yet.
  2. Otherwise, useAtom(viewDraftAtomFamily({ userId, baseId, viewId })) gives [draft, setDraft]. Jotai reads from localStorage on first access and writes on every set.
  3. setFilter(next) and setSorts(next) compute merged = { ...(draft ?? {}), [axis]: next, updatedAt: new Date().toISOString() }. If the result has both filter and sorts back to undefined (the user cleared all local divergence), call setDraft(RESET) instead of writing an empty object. (RESET is jotai/utils' sentinel — it removes the key from localStorage.) This keeps "orphan" drafts from lingering.
  4. reset() is setDraft(RESET).
  5. isDirty is draft !== null && (!shallowEqualFilter(draft.filter, baselineFilter) || !shallowEqualSorts(draft.sorts, baselineSorts)). Note the per-axis ?? fallback doesn't appear here because null/undefined is the "no local divergence" signal for that axis; only a defined-and-different value counts as dirty.
  6. buildPromotedConfig(baseline) returns { ...baseline, filter: draft?.filter ?? baseline.filter, sorts: draft?.sorts ?? baseline.sorts }. Preserves all non-draft config fields (widths, order, visibility) and only overwrites the two axes that may have diverged.

Return composition:

  • effectiveFilter = draft?.filter ?? baselineFilter
  • effectiveSorts = draft?.sorts ?? baselineSorts

Cross-tab sync is free. atomWithStorage subscribes to the storage event internally — a filter change in tab A triggers a re-render in tab B with no extra code. No manual listener required.

Integration into useBaseTable and base-table.tsx

useBaseTable at use-base-table.ts:224 currently derives the table's initial sort from activeView.config.sorts. In the new world the table's sort/filter state must come from the effective values (draft-or-baseline), not the raw activeView.config.

Two cut options were considered:

Option A (chosen): drive from effective values via props. useBaseTable takes an additional effectiveConfig?: ViewConfig parameter (or, cleaner, the caller passes a shallow-merged activeView whose config is { ...activeView.config, filter: effective.filter, sorts: effective.sorts }). buildSortingState and the row query already read from activeView.config, so the cleanest shape is to mutate the config the hook receives, not to introduce a new parameter.

Option B (rejected): thread draft deep into useBaseTable. Adds the concept of drafts to a hook that only cares about the rendered state. Muddies responsibilities.

Going with A. In base-table.tsx:

// NEW: wire the draft hook
const { data: user } = useCurrentUser();
const { draft, effectiveFilter, effectiveSorts, isDirty, setFilter, setSorts, reset, buildPromotedConfig } =
  useViewDraft({
    userId: user?.user.id,
    baseId,
    viewId: activeView?.id,
    baselineFilter: activeView?.config?.filter,
    baselineSorts: activeView?.config?.sorts,
  });

// Swap the raw `activeView` for a view with effective config so the table and row query see drafts.
const effectiveView = useMemo(
  () =>
    activeView
      ? { ...activeView, config: { ...activeView.config, filter: effectiveFilter, sorts: effectiveSorts } }
      : undefined,
  [activeView, effectiveFilter, effectiveSorts],
);

// Row query reads effective filter/sorts.
const { data: rowsData, ... } = useBaseRowsQuery(
  base ? baseId : undefined,
  effectiveFilter,
  effectiveSorts,
);

// Table is seeded from effectiveView for rendering, but the auto-persist
// write-path uses the real `activeView.config` as the baseline so draft
// filter/sort values can never leak into a column-layout save.
// See "Filter & sort write-path changes" below for the exact mechanism.
const { table, persistViewConfig } = useBaseTable(base, rows, effectiveView, {
  baselineConfig: activeView?.config,
});

The server-roundtrip persistViewConfig keeps being called for column layout changes. It reads from baselineConfig — never from the effective/draft state — so a pending layout write cannot bake draft filter/sort values into the server baseline. See the next subsection for the exact implementation.

Filter & sort write-path changes

Today, filter/sort editors feed BaseToolbar's handlers:

  • base-toolbar.tsx:135-148 handleSortsChange → builds config via buildViewConfigFromTable(table, activeView.config, { sorts: newSorts })updateViewMutation.mutate(...).
  • base-toolbar.tsx:150-169 handleFiltersChange → same pattern with { filter }.

Both write directly to the server. That's the exact site to branch.

New base-toolbar.tsx: accept two new callbacks from base-table.tsx:

onDraftSortsChange: (sorts: ViewSortConfig[]) => void;
onDraftFiltersChange: (filter: FilterGroup | undefined) => void;

The toolbar drops its internal updateViewMutation.mutate calls for sort/filter (retains them for view tabs / view type flip if any exists elsewhere). handleSortsChange becomes:

const handleSortsChange = useCallback(
  (newSorts: ViewSortConfig[]) => {
    onDraftSortsChange(newSorts); // writes to useViewDraft via base-table
  },
  [onDraftSortsChange],
);

Same for filters — the FilterCondition[]→FilterGroup wrapping logic at base-toolbar.tsx:152-157 stays; only the final dispatch target changes.

base-table.tsx wires those callbacks to the draft hook:

const handleDraftSortsChange = useCallback(
  (sorts: ViewSortConfig[]) => setSorts(sorts.length ? sorts : undefined),
  [setSorts],
);
const handleDraftFiltersChange = useCallback(
  (filter: FilterGroup | undefined) => setFilter(filter),
  [setFilter],
);

The "normalize empty to undefined" rule is how we let the draft go clean after the user deletes every filter — the draft hook's "remove key if both axes are undefined" rule then kicks in.

Toolbar badge counts: base-toolbar.tsx:118-128 currently derives sorts and conditions from activeView.config. Switch these to read from the effective config (effectiveView.config) so the toolbar badges reflect the draft's count, not the baseline. The toolbar already accepts activeView — pass it effectiveView instead, since everything the toolbar reads from activeView (name, sorts, filter) should be in the effective form.

The buildViewConfigFromTable call site in handleColumnReorder / handleResizeEnd / field-visibility: these continue reading from activeView.config (the real baseline) and going through updateViewMutation. They do not read from the draft. This is deliberate — column layout stays auto-persisted.

However: buildViewConfigFromTable currently spreads its base argument and emits sorts from the live table state. For the debounced persistViewConfig call at use-base-table.ts:382, the base arg is the effective config (because we pass effectiveView into useBaseTable), but the emitted sorts comes from the table's live state — which was seeded from effective. That means if the user drafts a sort and then reorders a column, the debounced persist would write { ...effectiveConfig, sorts: draftSorts } back to the server. Bug.

Fix: when building the config for the auto-persist path in persistViewConfig, override the emitted sorts and filter with the baseline values, not the effective ones. Concretely, change use-base-table.ts:382 to

const config = buildViewConfigFromTable(table, activeView.config, {
  sorts: activeView.config?.sorts,
  filter: activeView.config?.filter,
});

where activeView in that callsite is the real activeView (not the effective one). So useBaseTable needs both: the effective view for seeding and rendering, and the real baseline for the persist path.

Simplest refactor: give useBaseTable an optional baselineConfig?: ViewConfig argument. If omitted (existing callers), behave as today. If provided, persistViewConfig uses baselineConfig for sort/filter overrides. base-table.tsx passes activeView.config as the baseline and the effective-wrapped view as the active.

This keeps useBaseTable's own responsibilities tidy and makes the "drafts don't leak into the layout write-path" rule explicit.

Note on useBaseTable's re-seed effect: A draft edit changes effectiveView.config.filter/sorts, which propagates through the derivedColumnOrder / derivedColumnVisibility memos and re-fires the sync effect at use-base-table.ts:280. This is harmless because (a) activeView.id is unchanged, so the full re-seed branch doesn't trigger, and (b) the hasPendingEdit branch preserves live column state when no layout mutation is pending, and adopts derived values otherwise — those derived values are still driven by the same properties, so they're content-equal. No action required, but worth naming so the implementer doesn't chase a non-issue.

Banner component

File: apps/client/src/features/base/components/base-view-draft-banner.tsx

type BaseViewDraftBannerProps = {
  isDirty: boolean;
  canSave: boolean;
  onReset: () => void;
  onSave: () => void;
  saving: boolean;
};

export function BaseViewDraftBanner({ isDirty, canSave, onReset, onSave, saving }: BaseViewDraftBannerProps) {
  const { t } = useTranslation();
  if (!isDirty) return null;
  return (
    <Paper withBorder radius="sm" px="md" py="xs" /* soft bg per theme */>
      <Group justify="space-between" wrap="nowrap">
        <Group gap="xs" wrap="nowrap">
          <IconInfoCircle size={16} />
          <Text size="sm">{t("Filter and sort changes are visible only to you.")}</Text>
        </Group>
        <Group gap="sm" wrap="nowrap">
          <Button variant="subtle" color="gray" size="xs" onClick={onReset}>{t("Reset")}</Button>
          {canSave && (
            <Button size="xs" onClick={onSave} loading={saving}>{t("Save for everyone")}</Button>
          )}
        </Group>
      </Group>
    </Paper>
  );
}

Wiring in base-table.tsx, inserted between the existing page chrome and <BaseToolbar />:

const { data: space } = useSpaceQuery(base?.spaceId ?? "");
const spaceAbility = useSpaceAbility(space?.membership?.permissions);
const canSave = spaceAbility.can(SpaceCaslAction.Edit, SpaceCaslSubject.Base);
const updateViewMutation = useUpdateViewMutation();
const handleSaveDraft = useCallback(async () => {
  if (!activeView || !base) return;
  const config = buildPromotedConfig(activeView.config);
  await updateViewMutation.mutateAsync({ viewId: activeView.id, baseId: base.id, config });
  reset();
  notifications.show({ message: t("View updated for everyone") });
}, [activeView, base, buildPromotedConfig, reset, updateViewMutation, t]);

return (
  <div style={{...}}>
    <BaseViewDraftBanner
      isDirty={isDirty}
      canSave={canSave}
      onReset={reset}
      onSave={handleSaveDraft}
      saving={updateViewMutation.isPending}
    />
    <BaseToolbar ... />
    <GridContainer ... />
  </div>
);

The useSpaceQuery/useSpaceAbility pair follows the same pattern as use-history-restore.tsx:35-41.

Cross-tab sync

Inherited from atomWithStorage. Its internal subscription to the storage event re-notifies any Jotai-connected component on other tabs when the matching localStorage key changes, triggering a re-render with the new draft value. No hand-rolled listener in useViewDraft.

React Query's row cache is keyed by (baseId, filter, sorts, search) — when the updated draft flows through effectiveFilter / effectiveSorts on the other tab, the row query refetches as a fresh infinite query via the normal path.

Edge case: two tabs editing simultaneously — both writes land in localStorage, last-write-wins (same-user scope, acceptable).

Save flow (pseudocode)

async function onSaveForEveryone() {
  if (!activeView || !base) return;
  // 1. Compose the promoted config from the server baseline + draft values.
  //    baseline is activeView.config (NOT effectiveView.config) because the
  //    baseline might include layout fields (propertyWidths, propertyOrder,
  //    hiddenPropertyIds, visiblePropertyIds) that we must preserve verbatim.
  const config: ViewConfig = {
    ...activeView.config,
    filter: draft.filter ?? activeView.config.filter,
    sorts: draft.sorts ?? activeView.config.sorts,
  };
  // 2. Fire the existing mutation. `updateViewMutation` already:
  //    - optimistically updates the ["bases", baseId] query cache
  //    - rolls back on error
  //    - writes the server response back on success
  await updateViewMutation.mutateAsync({ viewId: activeView.id, baseId: base.id, config });
  // 3. Clear the draft. Because the baseline has now caught up to what the
  //    draft said, isDirty flips to false and the banner unmounts.
  reset();
  notifications.show({ message: t("View updated for everyone") });
}

Error handling: useUpdateViewMutation already shows a red toast and rolls back the optimistic cache update on failure. We do not call reset() in that case — the draft stays, the banner stays, the user can retry.

Dirty check

isDirty lives inside useViewDraft. Returns true iff the draft file exists AND at least one of these is true:

  • draft.filter !== undefined AND !deepEqualFilter(draft.filter, baselineFilter)
  • draft.sorts !== undefined AND !deepEqualSorts(draft.sorts, baselineSorts)

Deep equality: the codebase has no lodash or fast-deep-equal in client package.json. Options:

  1. JSON.stringify both sides and compare strings. Trivially correct for FilterGroup (a pure data tree) and ViewSortConfig[]. Key ordering inside objects is deterministic in V8+ for non-numeric keys, which is the case here. Pick this — it's 4 lines and good enough for this shape.
  2. Hand-written structural compare — overkill for two types with known finite shapes.

Go with option 1. Helpers live in use-view-draft.ts:

function filterEq(a: FilterGroup | undefined, b: FilterGroup | undefined) {
  return JSON.stringify(a ?? null) === JSON.stringify(b ?? null);
}
function sortsEq(a: ViewSortConfig[] | undefined, b: ViewSortConfig[] | undefined) {
  return JSON.stringify(a ?? null) === JSON.stringify(b ?? null);
}

Orphan suppression. The agreed rule: when the draft's values equal the baseline, the banner hides. The dirty check above already does that — a draft with filter: X where baseline is also X yields filterEq === true for that axis, and if the sorts axis is also equal (or absent), isDirty === false. The key stays in localStorage (no eager GC), but the banner is invisible until the user next diverges or another tab updates the baseline.

Testing

Per CLAUDE.md, the client has no test infrastructure (no vitest in the workspace). This spec does not block on adding one. Testing is primarily manual QA + optional unit tests if Vitest is introduced alongside this feature.

Unit tests (proposed, Vitest — gated on harness being added)

use-view-draft.test.ts:

  • Initialize with no stored value. Hook returns draft=null, isDirty=false, effective values fall through to baseline.
  • setFilter writes to localStorage and updates state. After setFilter(X), localStorage.getItem(key) parses back to { filter: X, updatedAt: ... }, draft.filter === X, isDirty === true.
  • setSorts writes independently. draft.filter stays undefined even after setSorts(...), and vice versa.
  • setFilter(undefined) then setSorts(undefined) removes the key. After both axes are cleared, localStorage.getItem(key) is null.
  • reset clears both state and storage.
  • Draft values equal to baseline → isDirty === false without clearing storage. Set baseline to B, set draft filter to B, assert isDirty === false and localStorage.getItem(key) is still non-null (no eager GC).
  • Baseline change while draft exists. Baseline shifts from B1 to B2, draft filter is X. Effective filter stays X, isDirty stays true. Then baseline shifts again to XisDirty flips to false without draft being cleared.
  • Cross-tab propagation (integration-level, not strictly a unit test). atomWithStorage handles the storage event internally; the only thing our hook contributes is the derivation of effectiveFilter / effectiveSorts / isDirty from the atom value. A single assertion that writing to the atom value in one Provider context reflects in another suffices.
  • Malformed storage value. Seed localStorage with garbage under the computed key → atomWithStorage yields null, hook reports draft=null, isDirty=false, table receives baseline.
  • userId missing → passthrough. All setters are no-ops, isDirty=false, effective = baseline.

Manual QA checklist

Single user, single tab.

  • Apply a filter. Banner appears. Row list updates locally.
  • Click Reset. Banner disappears. Filter in the popover reverts to baseline. Row list reverts.
  • Apply a filter and a sort. Click Save for everyone. Banner disappears. Refresh the page — the filter/sort is now the new baseline (i.e. came back from the server).
  • Apply a filter, then manually delete it via the filter popover. Banner disappears. Subsequent refresh does not restore the deleted filter (baseline untouched).

Single user, multiple tabs.

  • Open base in tab A and tab B. In tab A, add a sort. Tab B re-renders with the same sort applied (verified by checking the sort popover badge and the row order). Tab B shows the banner.
  • In tab B, click Reset. Tab A's banner disappears and sort reverts.

Multi-user baseline race.

  • User X (editor) opens base. Applies a filter (draft). User Y (editor) in another session saves a brand-new baseline via their own Save flow. User X's client receives the websocket base:schema:bumped["bases", baseId] invalidates → activeView.config updates. User X's effectiveFilter still shows X's draft filter (draft wins). Banner stays. No UI prompt. If X now clicks Reset, they see Y's new baseline.

Permission gating.

  • As a space Viewer (who has Read but not Edit on Base): open base, apply a filter. Banner appears but shows only "Reset" — no "Save for everyone" button.
  • Server check: attempting Save as a viewer would have been blocked by base-view.controller.ts:68 anyway; the UI gate is belt-and-suspenders.

Reset with popover open.

  • Open the filter popover and add conditions. Without closing the popover, click Reset (the banner is visible behind the popover dropdown — it's positioned above). Popover closes on outside-click, baseline conditions show next open.

Save clears draft + updates server.

  • Save. Banner vanishes. localStorage key for {user,base,view} is absent. Re-open the base in an incognito/second-account browser — the filter/sort shows too (from the server).

Browser storage cleared.

  • In DevTools, wipe localStorage. Base re-renders with baseline. Banner gone. Expected.

Rollout

  • No DB migration. No server change.
  • No feature flag. Behavior change ships as-is.
  • No data migration. Existing users have no drafts; the system starts empty.
  • Behavioral change vs. today. Existing users' muscle memory is "touch a filter → auto-saves for everyone". After this ships, that becomes "touch a filter → only I see it until I hit Save for everyone". This is the entire point of the feature but will surprise power users on day one.
    • Mitigation: none in v1. A one-time popover/tooltip pointing at the banner ("New: filter and sort changes are now a draft until you save") is worth doing, but falls squarely in YAGNI territory for the first ship.
    • Followup: consider a dismissible one-time in-product hint the first time a user diverges from baseline after the deploy. Flag this as a follow-up task; do not ship with v1.

Risks & open questions

  • localStorage quota. FilterGroup + ViewSortConfig[] is tiny — a realistic draft is under 2KB. A worst-case malicious user with thousands of views could hit the 510MB per-origin cap, but practically negligible. No cleanup logic needed.
  • Users losing drafts via browser data clear. Expected. The banner is a live indicator, not a durable source of truth. Flagged in non-goals.
  • Multi-device divergence. Same user on laptop and phone: drafts don't sync. Expected and flagged.
  • Dropdown caret ("Save as new view") in the screenshot. Explicitly out of scope for v1. If we add it, the caret menu would include:
    1. "Save for everyone" (current behavior)
    2. "Save as new view" (creates a new IBaseView with draft values baked into config)
  • Baseline layout fields overriding draft. Save flow does { ...activeView.config, filter: X, sorts: Y }. If another user changed column widths right before Save, those widths land in the Save's payload (we already read the latest optimistic cache). Acceptable — the alternative (send a sparse patch with only {filter, sorts}) would require a server-side partial-update endpoint we don't have.
  • Invalid draft for stale schema. If a property is deleted while a user's draft references it by id, the predicate/sort engine on the server silently drops unknown property ids. Client-side, the sort/filter popover shows the condition with a missing-property label (existing behavior — the toolbar already does properties.find((p) => p.id === …) and tolerates the undefined case). No special handling needed here; the draft just falls away when the user next edits and doesn't re-add the dead condition.
  • SpaceCaslSubject.Base missing from client enum. Single-line fix at permissions.type.ts:12. Flagged so reviewers notice.

Future extension

  1. Draft column layout. Extend the draft shape to carry propertyWidths, propertyOrder, hiddenPropertyIds, visiblePropertyIds. Column reorder / hide / resize call the draft hook instead of persistViewConfig. useBaseTable then seeds column state from effective values. Mechanically identical to filter/sort — the hook already takes arbitrary ViewConfig fragments. The only reason this isn't in v1 is to minimize behavioral change surface and keep the spec scope narrow.
  2. Server-side per-user drafts. For cross-device sync, add a base_view_drafts table keyed by (userId, viewId) storing the same shape. The client hook swaps localStorage for a paired mutation + query. The banner UX stays identical.
  3. Split-button save. Dropdown caret next to "Save for everyone" offering "Save as new view" — creates an IBaseView via createView with the effective config. Deepens the Notion parallel.
  4. Draft conflict hint. When baseline changes while I have drafts, show a subtle "Baseline has changed since your last edit" line inside the banner with a "Discard draft and load latest" affordance. Expected to be low value in practice — flag once real users report it.