@motion-proto/live-tokens 0.6.2 → 0.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (212) hide show
  1. package/README.md +14 -13
  2. package/dist-plugin/index.cjs +147 -136
  3. package/dist-plugin/index.d.cts +1 -1
  4. package/dist-plugin/index.d.ts +1 -1
  5. package/dist-plugin/index.js +145 -135
  6. package/package.json +25 -40
  7. package/src/{component-editor → editor/component-editor}/BadgeEditor.svelte +8 -82
  8. package/src/{component-editor → editor/component-editor}/CalloutEditor.svelte +4 -4
  9. package/src/{component-editor → editor/component-editor}/CardEditor.svelte +28 -76
  10. package/src/{component-editor → editor/component-editor}/CollapsibleSectionEditor.svelte +3 -3
  11. package/src/{component-editor → editor/component-editor}/CornerBadgeEditor.svelte +31 -93
  12. package/src/{component-editor → editor/component-editor}/DialogEditor.svelte +60 -57
  13. package/src/editor/component-editor/ImageEditor.svelte +30 -0
  14. package/src/{component-editor → editor/component-editor}/InlineEditActionsEditor.svelte +6 -4
  15. package/src/editor/component-editor/MenuSelectEditor.svelte +160 -0
  16. package/src/{component-editor → editor/component-editor}/NotificationEditor.svelte +64 -37
  17. package/src/{component-editor → editor/component-editor}/ProgressBarEditor.svelte +5 -4
  18. package/src/{component-editor → editor/component-editor}/RadioButtonEditor.svelte +3 -3
  19. package/src/{component-editor → editor/component-editor}/SectionDividerEditor.svelte +57 -84
  20. package/src/{component-editor → editor/component-editor}/SegmentedControlEditor.svelte +2 -2
  21. package/src/{component-editor → editor/component-editor}/StandardButtonsEditor.svelte +16 -20
  22. package/src/{component-editor → editor/component-editor}/TabBarEditor.svelte +9 -14
  23. package/src/{component-editor → editor/component-editor}/TableEditor.svelte +9 -18
  24. package/src/{component-editor → editor/component-editor}/TooltipEditor.svelte +11 -47
  25. package/src/{component-editor → editor/component-editor}/registry.ts +28 -18
  26. package/src/{component-editor → editor/component-editor}/scaffolding/AngleDial.svelte +2 -2
  27. package/src/{component-editor → editor/component-editor}/scaffolding/ComponentEditorBase.svelte +3 -51
  28. package/src/{component-editor → editor/component-editor}/scaffolding/ComponentFileManager.svelte +144 -416
  29. package/src/{component-editor → editor/component-editor}/scaffolding/ComponentFileMenu.svelte +18 -170
  30. package/src/{component-editor → editor/component-editor}/scaffolding/ComponentsTab.svelte +2 -2
  31. package/src/{component-editor → editor/component-editor}/scaffolding/CopyFromMenu.svelte +44 -4
  32. package/src/{component-editor → editor/component-editor}/scaffolding/DividerEditor.svelte +1 -1
  33. package/src/{component-editor → editor/component-editor}/scaffolding/FieldsetWrapper.svelte +1 -1
  34. package/src/{component-editor → editor/component-editor}/scaffolding/GradientCard.svelte +6 -6
  35. package/src/{component-editor → editor/component-editor}/scaffolding/LinkageChart.svelte +6 -6
  36. package/src/{component-editor → editor/component-editor}/scaffolding/LinkedBlock.svelte +6 -11
  37. package/src/editor/component-editor/scaffolding/NonStylableConfig.svelte +38 -0
  38. package/src/{component-editor → editor/component-editor}/scaffolding/SaveAsDialog.svelte +66 -12
  39. package/src/editor/component-editor/scaffolding/ShadowBackdrop.svelte +72 -0
  40. package/src/editor/component-editor/scaffolding/ShadowBackdropControls.svelte +132 -0
  41. package/src/editor/component-editor/scaffolding/StateBlock.svelte +257 -0
  42. package/src/{component-editor → editor/component-editor}/scaffolding/TokenLayout.svelte +9 -7
  43. package/src/editor/component-editor/scaffolding/VariantGroup.svelte +644 -0
  44. package/src/{component-editor → editor/component-editor}/scaffolding/editorContext.ts +19 -9
  45. package/src/{component-editor → editor/component-editor}/scaffolding/linkedBlock.ts +2 -2
  46. package/src/{component-editor → editor/component-editor}/scaffolding/types.ts +14 -0
  47. package/src/{lib → editor/core/components}/componentConfigService.ts +2 -2
  48. package/src/{lib → editor/core/components}/componentPersist.ts +5 -5
  49. package/src/editor/core/flashStatus.ts +30 -0
  50. package/src/{lib → editor/core/fonts}/fontLoader.ts +2 -2
  51. package/src/{lib → editor/core/fonts}/fontMigration.ts +4 -4
  52. package/src/{lib → editor/core/fonts}/fontParse.ts +1 -1
  53. package/src/editor/core/manifests/manifestService.ts +116 -0
  54. package/src/{lib → editor/core/palettes}/paletteDerivation.ts +2 -2
  55. package/src/{lib → editor/core/palettes}/tokenRegistry.ts +5 -5
  56. package/src/editor/core/productionPulse.ts +37 -0
  57. package/src/{lib → editor/core/routing}/router.ts +1 -1
  58. package/src/{lib/files/versionedFileResource.ts → editor/core/storage/files/versionedFileResourceClient.ts} +8 -1
  59. package/src/{lib → editor/core/store}/editorCore.ts +24 -8
  60. package/src/{lib → editor/core/store}/editorPersistence.ts +3 -3
  61. package/src/{lib → editor/core/store}/editorRenderer.ts +2 -2
  62. package/src/{lib → editor/core/store}/editorStore.ts +17 -17
  63. package/src/{lib → editor/core/store}/editorTypes.ts +1 -1
  64. package/src/{lib → editor/core/themes}/slices/columns.ts +2 -2
  65. package/src/{lib → editor/core/themes}/slices/components.ts +2 -2
  66. package/src/{lib → editor/core/themes}/slices/fonts.ts +1 -1
  67. package/src/{lib → editor/core/themes}/slices/gradients.ts +2 -2
  68. package/src/{lib → editor/core/themes}/slices/overlays.ts +1 -1
  69. package/src/{lib → editor/core/themes}/slices/palettes.ts +1 -1
  70. package/src/{lib → editor/core/themes}/slices/shadows.ts +3 -3
  71. package/src/{lib → editor/core/themes}/themeInit.ts +6 -6
  72. package/src/{lib → editor/core/themes}/themeService.ts +6 -6
  73. package/src/{lib → editor/core/themes}/themeTypes.ts +11 -7
  74. package/src/editor/index.ts +69 -0
  75. package/src/{lib → editor/overlay}/LiveEditorOverlay.svelte +79 -125
  76. package/src/{lib → editor/overlay}/columnsOverlay.ts +2 -2
  77. package/src/{pages → editor/pages}/ComponentEditorPage.svelte +12 -12
  78. package/src/{pages → editor/pages}/Editor.svelte +4 -4
  79. package/src/{pages → editor/pages}/EditorShell.svelte +18 -36
  80. package/src/{styles → editor/styles}/ui-editor.css +41 -21
  81. package/src/{styles → editor/styles}/ui-form-controls.css +8 -8
  82. package/src/{ui → editor/ui}/BezierCurveEditor.svelte +8 -8
  83. package/src/{ui → editor/ui}/ColorEditPanel.svelte +13 -13
  84. package/src/{ui → editor/ui}/EditorViewSwitcher.svelte +8 -6
  85. package/src/editor/ui/FileLoadList.svelte +350 -0
  86. package/src/editor/ui/FilePill.svelte +80 -0
  87. package/src/{ui → editor/ui}/FontStackEditor.svelte +7 -7
  88. package/src/{ui → editor/ui}/GradientEditor.svelte +11 -11
  89. package/src/{ui → editor/ui}/GradientStopPicker.svelte +1 -1
  90. package/src/editor/ui/ManifestFileManager.svelte +371 -0
  91. package/src/{ui → editor/ui}/PaletteEditor.svelte +132 -598
  92. package/src/{ui → editor/ui}/ProjectFontsSection.svelte +102 -144
  93. package/src/{ui → editor/ui}/SurfacesTab.svelte +3 -3
  94. package/src/{ui → editor/ui}/TextTab.svelte +3 -3
  95. package/src/{ui → editor/ui}/ThemeFileManager.svelte +286 -519
  96. package/src/{ui → editor/ui}/UICopyPopover.svelte +4 -4
  97. package/src/{ui → editor/ui}/UIFontFamilySelector.svelte +6 -6
  98. package/src/{ui → editor/ui}/UIFontSizeSelector.svelte +1 -1
  99. package/src/editor/ui/UIInfoPopover.svelte +244 -0
  100. package/src/{ui → editor/ui}/UILineHeightSelector.svelte +5 -5
  101. package/src/{ui → editor/ui}/UILinkToggle.svelte +2 -2
  102. package/src/{ui → editor/ui}/UIPaddingSelector.svelte +6 -6
  103. package/src/{ui → editor/ui}/UIPaletteSelector.svelte +26 -26
  104. package/src/editor/ui/UIPillButton.svelte +138 -0
  105. package/src/{ui → editor/ui}/UIRadio.svelte +2 -2
  106. package/src/{ui → editor/ui}/UIRelinkConfirmPopover.svelte +4 -4
  107. package/src/editor/ui/UISquareButton.svelte +172 -0
  108. package/src/{ui → editor/ui}/UITokenSelector.svelte +10 -10
  109. package/src/{ui → editor/ui}/UIVariantSelector.svelte +1 -1
  110. package/src/{ui → editor/ui}/VariablesTab.svelte +31 -8
  111. package/src/{ui → editor/ui}/palette/GradientStopEditor.svelte +13 -13
  112. package/src/{ui → editor/ui}/palette/OverridesPanel.svelte +13 -13
  113. package/src/{ui → editor/ui}/palette/PaletteBase.svelte +8 -5
  114. package/src/{ui → editor/ui}/palette/paletteEditorState.ts +1 -1
  115. package/src/editor/ui/palette/paletteMath.ts +275 -0
  116. package/src/{ui → editor/ui}/sections/ColumnsSection.svelte +137 -17
  117. package/src/{ui → editor/ui}/sections/GradientsSection.svelte +7 -7
  118. package/src/{ui → editor/ui}/sections/OverlaysSection.svelte +17 -17
  119. package/src/{ui → editor/ui}/sections/ShadowsSection.svelte +22 -22
  120. package/src/{ui → editor/ui}/sections/TokenScaleTable.svelte +3 -3
  121. package/src/{components → system/components}/Badge.svelte +0 -36
  122. package/src/{components → system/components}/Card.svelte +8 -62
  123. package/src/{components → system/components}/CornerBadge.svelte +8 -24
  124. package/src/{components → system/components}/Dialog.svelte +1 -1
  125. package/src/system/components/FloatingTokenTags.css +256 -0
  126. package/src/system/components/FloatingTokenTags.svelte +592 -0
  127. package/src/{components → system/components}/InlineEditActions.svelte +6 -4
  128. package/src/system/components/MenuSelect.svelte +229 -0
  129. package/src/{components → system/components}/ProgressBar.svelte +29 -11
  130. package/src/{components → system/components}/SegmentedControl.svelte +49 -43
  131. package/src/{components → system/components}/TabBar.svelte +81 -65
  132. package/src/{components → system/components}/Table.svelte +17 -3
  133. package/src/{components → system/components}/Tooltip.svelte +6 -4
  134. package/src/system/styles/CONVENTIONS.md +178 -0
  135. package/src/{styles → system/styles}/fonts.css +6 -3
  136. package/src/{styles → system/styles}/tokens.css +149 -29
  137. package/src/component-editor/ImageEditor.svelte +0 -74
  138. package/src/component-editor/scaffolding/NonStylableConfig.svelte +0 -62
  139. package/src/component-editor/scaffolding/ShadowBackdrop.svelte +0 -37
  140. package/src/component-editor/scaffolding/ShadowBackdropControls.svelte +0 -61
  141. package/src/component-editor/scaffolding/StateBlock.svelte +0 -132
  142. package/src/component-editor/scaffolding/VariantGroup.svelte +0 -310
  143. package/src/data/google-fonts.json +0 -75
  144. package/src/lib/index.ts +0 -68
  145. package/src/lib/presetService.ts +0 -214
  146. package/src/lib/productionPulse.ts +0 -32
  147. package/src/ui/PresetFileManager.svelte +0 -1116
  148. package/src/ui/UnsavedComponentsDialog.svelte +0 -315
  149. /package/src/{styles → app}/site.css +0 -0
  150. /package/src/{component-editor → editor/component-editor}/index.ts +0 -0
  151. /package/src/{component-editor → editor/component-editor}/scaffolding/DemoHeader.svelte +0 -0
  152. /package/src/{component-editor → editor/component-editor}/scaffolding/TypeEditor.svelte +0 -0
  153. /package/src/{component-editor → editor/component-editor}/scaffolding/buildTypeGroupTokens.ts +0 -0
  154. /package/src/{component-editor → editor/component-editor}/scaffolding/componentSectionType.ts +0 -0
  155. /package/src/{component-editor → editor/component-editor}/scaffolding/componentSources.ts +0 -0
  156. /package/src/{component-editor → editor/component-editor}/scaffolding/defaultSections.ts +0 -0
  157. /package/src/{component-editor → editor/component-editor}/scaffolding/siblings.ts +0 -0
  158. /package/src/{lib → editor/core/components}/componentConfigKeys.ts +0 -0
  159. /package/src/{lib → editor/core}/cssVarSync.ts +0 -0
  160. /package/src/{lib → editor/core/palettes}/oklch.ts +0 -0
  161. /package/src/{lib → editor/core/routing}/navLinkTypes.ts +0 -0
  162. /package/src/{lib → editor/core/routing}/parentRouteStore.ts +0 -0
  163. /package/src/{lib → editor/core/storage}/storage.ts +0 -0
  164. /package/src/{lib → editor/core/store}/editorConfig.ts +0 -0
  165. /package/src/{lib → editor/core/store}/editorConfigStore.ts +0 -0
  166. /package/src/{lib → editor/core/store}/editorKeybindings.ts +0 -0
  167. /package/src/{lib → editor/core/store}/editorViewStore.ts +0 -0
  168. /package/src/{lib → editor/core/themes}/migrations/2026-04-24-component-prefix-and-suffix-renames.ts +0 -0
  169. /package/src/{lib → editor/core/themes}/migrations/2026-04-24-legacy-keys-and-bg-to-canvas.ts +0 -0
  170. /package/src/{lib → editor/core/themes}/migrations/2026-04-27-segmentedcontrol-disabled-flatten.ts +0 -0
  171. /package/src/{lib → editor/core/themes}/migrations/2026-05-08-collapsiblesection-frame-and-cleanup.ts +0 -0
  172. /package/src/{lib → editor/core/themes}/migrations/2026-05-08-collapsiblesection-variant-namespace.ts +0 -0
  173. /package/src/{lib → editor/core/themes}/migrations/2026-05-10-sectiondivider-gradient-stops.ts +0 -0
  174. /package/src/{lib → editor/core/themes}/migrations/2026-05-13-primary-to-brand.ts +0 -0
  175. /package/src/{lib → editor/core/themes}/migrations/index.ts +0 -0
  176. /package/src/{lib → editor/core/themes}/parsers/globalRootBlock.ts +0 -0
  177. /package/src/{lib → editor/core/themes}/slices/domainVars.ts +0 -0
  178. /package/src/{lib → editor/overlay}/ColumnsOverlay.svelte +0 -0
  179. /package/src/{lib → editor/overlay}/overlayState.ts +0 -0
  180. /package/src/{pages → editor/pages}/ComponentEditorPage.svelte.d.ts +0 -0
  181. /package/src/{pages → editor/pages}/Editor.svelte.d.ts +0 -0
  182. /package/src/{ui → editor/ui}/Toggle.svelte +0 -0
  183. /package/src/{ui → editor/ui}/UIDialog.svelte +0 -0
  184. /package/src/{ui → editor/ui}/UIFontWeightSelector.svelte +0 -0
  185. /package/src/{ui → editor/ui}/UIOptionItem.svelte +0 -0
  186. /package/src/{ui → editor/ui}/UIOptionList.svelte +0 -0
  187. /package/src/{ui → editor/ui}/UIRadioGroup.svelte +0 -0
  188. /package/src/{lib → editor/ui}/copyPopover.ts +0 -0
  189. /package/src/{ui → editor/ui}/curveEngine.ts +0 -0
  190. /package/src/{ui → editor/ui}/index.ts +0 -0
  191. /package/src/{ui → editor/ui}/keepInViewport.ts +0 -0
  192. /package/src/{ui → editor/ui}/palette/ScaleCurveEditor.svelte +0 -0
  193. /package/src/{lib → editor/ui}/scrollSection.ts +0 -0
  194. /package/src/{ui → editor/ui}/sections/tokenScales.ts +0 -0
  195. /package/src/{ui → editor/ui}/variantScales.ts +0 -0
  196. /package/src/{assets → system/assets}/newspaper.webp +0 -0
  197. /package/src/{assets → system/assets}/offering.webp +0 -0
  198. /package/src/{components → system/components}/Button.svelte +0 -0
  199. /package/src/{components → system/components}/Callout.svelte +0 -0
  200. /package/src/{components → system/components}/CollapsibleSection.svelte +0 -0
  201. /package/src/{components → system/components}/Image.svelte +0 -0
  202. /package/src/{components → system/components}/Notification.svelte +0 -0
  203. /package/src/{components → system/components}/RadioButton.svelte +0 -0
  204. /package/src/{components → system/components}/SectionDivider.svelte +0 -0
  205. /package/src/{components → system/components}/types.ts +0 -0
  206. /package/src/{styles → system/styles}/_padding.scss +0 -0
  207. /package/src/{styles → system/styles}/fonts/Fraunces/Fraunces-italic-latin-ext.woff2 +0 -0
  208. /package/src/{styles → system/styles}/fonts/Fraunces/Fraunces-italic-latin.woff2 +0 -0
  209. /package/src/{styles → system/styles}/fonts/Fraunces/Fraunces-roman-latin-ext.woff2 +0 -0
  210. /package/src/{styles → system/styles}/fonts/Fraunces/Fraunces-roman-latin.woff2 +0 -0
  211. /package/src/{styles → system/styles}/fonts/Manrope/Manrope-latin-ext.woff2 +0 -0
  212. /package/src/{styles → system/styles}/fonts/Manrope/Manrope-latin.woff2 +0 -0
@@ -0,0 +1,371 @@
1
+ <script lang="ts">
2
+ import { onMount } from 'svelte';
3
+ import type { ManifestMeta } from '../core/themes/themeTypes';
4
+ import {
5
+ listManifests,
6
+ deleteManifest,
7
+ getActiveManifest,
8
+ applyManifest,
9
+ saveAsManifest,
10
+ saveActiveManifest,
11
+ } from '../core/manifests/manifestService';
12
+ import { dirty, componentDirty } from '../core/store/editorStore';
13
+ import { productionRevision, activeManifest } from '../core/productionPulse';
14
+ import { flashStatus } from '../core/flashStatus';
15
+ import UIInfoPopover from './UIInfoPopover.svelte';
16
+ import FileLoadList from './FileLoadList.svelte';
17
+ import FilePill from './FilePill.svelte';
18
+ import SaveAsDialog from '../component-editor/scaffolding/SaveAsDialog.svelte';
19
+
20
+ let files: ManifestMeta[] = $state([]);
21
+ let showFileList = $state(false);
22
+ let saveAsDialog = $state(false);
23
+ let saveStatus: 'idle' | 'saving' | 'saved' | 'error' = $state('idle');
24
+
25
+ let activeFileName = $state('default');
26
+ let currentDisplayName = $state('Default');
27
+ let activeIsProtected = $derived(activeFileName === 'default');
28
+
29
+ type SaveState = 'idle' | 'saving' | 'saved' | 'error';
30
+ const setSaveStatus = (s: SaveState) => (saveStatus = s);
31
+
32
+ let dirtyComponentCount = $derived(
33
+ Object.values($componentDirty).filter(Boolean).length,
34
+ );
35
+ let editorDirty = $derived($dirty || dirtyComponentCount > 0);
36
+
37
+ async function refreshFiles() {
38
+ try {
39
+ files = await listManifests();
40
+ } catch {
41
+ // silent — empty list
42
+ }
43
+ }
44
+
45
+ async function refreshActive() {
46
+ try {
47
+ const active = await getActiveManifest();
48
+ if (active) {
49
+ activeFileName = active._fileName ?? 'default';
50
+ // Default is always labeled "Default" regardless of what the on-disk
51
+ // name field says (older default.json files may have "Default Preset").
52
+ currentDisplayName = activeFileName === 'default'
53
+ ? 'Default'
54
+ : (active.name ?? activeFileName);
55
+ const meta = (await listManifests()).find((f) => f.fileName === activeFileName) ?? null;
56
+ activeManifest.set(meta);
57
+ }
58
+ } catch {
59
+ // silent
60
+ }
61
+ }
62
+
63
+ onMount(async () => {
64
+ await refreshFiles();
65
+ await refreshActive();
66
+ });
67
+
68
+ // Re-read active manifest whenever a sibling Adopt fires — the server
69
+ // patches our active file in those moments, so the timestamp + refs
70
+ // displayed here need to track. Skip the first tick (refreshActive ran
71
+ // already on mount).
72
+ let pulseInitialised = false;
73
+ $effect(() => {
74
+ void $productionRevision;
75
+ if (!pulseInitialised) {
76
+ pulseInitialised = true;
77
+ return;
78
+ }
79
+ refreshActive();
80
+ });
81
+
82
+ async function handleSave() {
83
+ if (activeIsProtected) return;
84
+ saveStatus = 'saving';
85
+ try {
86
+ await saveActiveManifest(currentDisplayName);
87
+ await refreshActive();
88
+ flashStatus(setSaveStatus, 'saved');
89
+ } catch {
90
+ flashStatus(setSaveStatus, 'error');
91
+ }
92
+ }
93
+
94
+ function openSaveAs() {
95
+ showFileList = false;
96
+ saveAsDialog = true;
97
+ }
98
+
99
+ async function confirmSaveAs(detail: { displayName: string; fileName: string }) {
100
+ saveStatus = 'saving';
101
+ try {
102
+ await saveAsManifest(detail.fileName, detail.displayName);
103
+ await refreshFiles();
104
+ await refreshActive();
105
+ flashStatus(setSaveStatus, 'saved');
106
+ } catch {
107
+ flashStatus(setSaveStatus, 'error');
108
+ }
109
+ }
110
+
111
+ async function handleApply(file: ManifestMeta) {
112
+ if (editorDirty) {
113
+ const ok = window.confirm(
114
+ 'Loading a manifest will reload the editor and discard unsaved changes. Continue?',
115
+ );
116
+ if (!ok) return;
117
+ }
118
+ showFileList = false;
119
+ try {
120
+ await applyManifest(file.fileName);
121
+ // applyManifest atomically flips active pointers; reload to rehydrate
122
+ // the editor from the now-active theme + component configs.
123
+ window.location.reload();
124
+ } catch (err) {
125
+ window.alert(`Failed to apply manifest: ${(err as Error).message}`);
126
+ }
127
+ }
128
+
129
+ async function handleDelete(file: ManifestMeta) {
130
+ if (file.isProtected) return;
131
+ if (file.fileName === activeFileName) {
132
+ window.alert('Cannot delete the active manifest. Load another manifest first.');
133
+ return;
134
+ }
135
+ const ok = window.confirm(`Delete manifest "${file.name}"?`);
136
+ if (!ok) return;
137
+ try {
138
+ await deleteManifest(file.fileName);
139
+ await refreshFiles();
140
+ } catch (err) {
141
+ window.alert(`Failed to delete: ${(err as Error).message}`);
142
+ }
143
+ }
144
+
145
+ function toggleFileList() {
146
+ showFileList = !showFileList;
147
+ if (showFileList) refreshFiles();
148
+ }
149
+ </script>
150
+
151
+ <div class="manifest-file-manager">
152
+ <div class="mfm-header">
153
+ <span class="mfm-header-label">Manifest</span>
154
+ <UIInfoPopover title="Manifests" ariaLabel="About manifests">
155
+ <p>
156
+ A <strong>manifest</strong> pins one theme plus one config file per component.
157
+ </p>
158
+ <p>
159
+ The <strong>active</strong> manifest is what the editor reads and what production runs. Theme and component <strong>Adopt</strong> actions auto-update its file.
160
+ </p>
161
+ <p>
162
+ <strong>Default</strong> is protected. To start customizing, <strong>Save As</strong> a new manifest first.
163
+ </p>
164
+ </UIInfoPopover>
165
+ </div>
166
+
167
+ <div class="mfm-card" class:protected={activeIsProtected}>
168
+ <span class="mfm-rail" aria-hidden="true"></span>
169
+ <div class="mfm-card-head">
170
+ <span class="mfm-card-label">Active</span>
171
+ {#if activeIsProtected}
172
+ <span class="mfm-badge protected" title="The default manifest is read-only">
173
+ <i class="fas fa-lock" aria-hidden="true"></i>
174
+ <span>protected</span>
175
+ </span>
176
+ {/if}
177
+ </div>
178
+ <FilePill
179
+ name={currentDisplayName}
180
+ isProtected={activeIsProtected}
181
+ protectedTitle="Protected default manifest"
182
+ title={currentDisplayName}
183
+ style="display: flex;"
184
+ />
185
+ <div class="mfm-card-actions">
186
+ <button
187
+ class="mfm-btn mfm-btn-row save-btn"
188
+ class:saving={saveStatus === 'saving'}
189
+ class:saved={saveStatus === 'saved'}
190
+ class:error={saveStatus === 'error'}
191
+ onclick={handleSave}
192
+ disabled={activeIsProtected || saveStatus === 'saving'}
193
+ title={activeIsProtected
194
+ ? 'Default is read-only — use Save As to capture under a new name'
195
+ : 'Re-stamp the active manifest with the current editor state'}
196
+ >
197
+ <i
198
+ class="fas"
199
+ class:fa-save={saveStatus === 'idle'}
200
+ class:fa-spinner={saveStatus === 'saving'}
201
+ class:fa-check={saveStatus === 'saved'}
202
+ class:fa-times={saveStatus === 'error'}
203
+ ></i>
204
+ <span>
205
+ {#if saveStatus === 'idle'}Save{:else if saveStatus === 'saving'}Saving{:else if saveStatus === 'saved'}Saved{:else}Error{/if}
206
+ </span>
207
+ </button>
208
+ <button class="mfm-btn mfm-btn-row" onclick={openSaveAs} title="Save current state as a new manifest">
209
+ <i class="fas fa-copy"></i>
210
+ <span>Save As…</span>
211
+ </button>
212
+ <button
213
+ class="mfm-btn mfm-btn-row"
214
+ class:active={showFileList}
215
+ onclick={toggleFileList}
216
+ title="Load a manifest"
217
+ >
218
+ <i class="fas fa-folder-open"></i>
219
+ <span>Load…</span>
220
+ </button>
221
+ </div>
222
+ </div>
223
+ </div>
224
+
225
+ <FileLoadList
226
+ bind:show={showFileList}
227
+ title="Load Manifest"
228
+ {files}
229
+ {activeFileName}
230
+ onload={handleApply}
231
+ ondelete={handleDelete}
232
+ />
233
+
234
+ <SaveAsDialog
235
+ bind:show={saveAsDialog}
236
+ {currentDisplayName}
237
+ {files}
238
+ title="Save Manifest As"
239
+ placeholder="Manifest name…"
240
+ reservedNameMessage='The name "default" is reserved for the protected baseline.'
241
+ branchFromDefaultName="my-manifest"
242
+ onsave={confirmSaveAs}
243
+ />
244
+
245
+ <style>
246
+ .manifest-file-manager {
247
+ --mfm-active: #5aa85e;
248
+ --mfm-rail-neutral: var(--ui-border);
249
+ --mfm-rail-active: var(--mfm-active);
250
+
251
+ display: flex;
252
+ flex-direction: column;
253
+ gap: var(--ui-space-8);
254
+ }
255
+
256
+ .mfm-header {
257
+ display: flex;
258
+ align-items: center;
259
+ justify-content: space-between;
260
+ gap: var(--ui-space-4);
261
+ padding: 0 var(--ui-space-4);
262
+ }
263
+
264
+ .mfm-header-label {
265
+ font-size: var(--ui-font-size-xs);
266
+ color: var(--ui-text-secondary);
267
+ text-transform: uppercase;
268
+ letter-spacing: 0.05em;
269
+ }
270
+
271
+ .mfm-card {
272
+ position: relative;
273
+ display: flex;
274
+ flex-direction: column;
275
+ gap: var(--ui-space-6);
276
+ padding: var(--ui-space-8) var(--ui-space-10) var(--ui-space-10) var(--ui-space-16);
277
+ background: var(--ui-surface-lower);
278
+ border: 1px solid var(--ui-border-low);
279
+ border-radius: var(--ui-radius-md);
280
+ }
281
+
282
+ .mfm-rail {
283
+ position: absolute;
284
+ left: 0;
285
+ top: 0;
286
+ bottom: 0;
287
+ width: 3px;
288
+ border-radius: var(--ui-radius-md) 0 0 var(--ui-radius-md);
289
+ background: var(--mfm-rail-active);
290
+ transition: background var(--ui-transition-base);
291
+ }
292
+
293
+ .mfm-card.protected .mfm-rail {
294
+ background: var(--mfm-rail-neutral);
295
+ }
296
+
297
+ .mfm-card-head {
298
+ display: flex;
299
+ align-items: baseline;
300
+ justify-content: space-between;
301
+ gap: var(--ui-space-8);
302
+ }
303
+
304
+ .mfm-card-label {
305
+ font-size: 10px;
306
+ font-weight: var(--ui-font-weight-semibold);
307
+ letter-spacing: 0.08em;
308
+ text-transform: uppercase;
309
+ color: var(--ui-text-tertiary);
310
+ }
311
+
312
+ .mfm-badge {
313
+ display: inline-flex;
314
+ align-items: center;
315
+ gap: var(--ui-space-4);
316
+ font-size: var(--ui-font-size-xs);
317
+ color: var(--ui-text-tertiary);
318
+ }
319
+
320
+ .mfm-badge.protected i {
321
+ font-size: 0.8em;
322
+ }
323
+
324
+ .mfm-card-actions {
325
+ display: flex;
326
+ flex-direction: column;
327
+ gap: var(--ui-space-4);
328
+ }
329
+
330
+ .mfm-btn {
331
+ display: inline-flex;
332
+ align-items: center;
333
+ justify-content: flex-start;
334
+ gap: var(--ui-space-8);
335
+ padding: var(--ui-space-6) var(--ui-space-10);
336
+ background: var(--ui-surface-high);
337
+ border: 1px solid var(--ui-border-low);
338
+ border-radius: var(--ui-radius-sm);
339
+ color: var(--ui-text-primary);
340
+ font-family: inherit;
341
+ font-size: var(--ui-font-size-sm);
342
+ line-height: 1;
343
+ cursor: pointer;
344
+ transition: background var(--ui-transition-fast), color var(--ui-transition-fast);
345
+ }
346
+
347
+ .mfm-btn:hover:not(:disabled) {
348
+ background: var(--ui-hover);
349
+ }
350
+
351
+ .mfm-btn:disabled {
352
+ cursor: not-allowed;
353
+ opacity: 0.5;
354
+ }
355
+
356
+ .mfm-btn.active {
357
+ background: var(--ui-active);
358
+ }
359
+
360
+ .mfm-btn.saved { color: var(--mfm-active); }
361
+ .mfm-btn.error { color: var(--ui-error, #c0392b); }
362
+
363
+ .mfm-btn .fa-spinner {
364
+ animation: mfm-spin 0.8s linear infinite;
365
+ }
366
+
367
+ @keyframes mfm-spin {
368
+ from { transform: rotate(0deg); }
369
+ to { transform: rotate(360deg); }
370
+ }
371
+ </style>