@actuate-media/cms-admin 0.6.0 → 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 (215) hide show
  1. package/dist/AdminRoot.d.ts.map +1 -1
  2. package/dist/AdminRoot.js +17 -0
  3. package/dist/AdminRoot.js.map +1 -1
  4. package/dist/actuate-admin.css +1 -1
  5. package/dist/components/ErrorBoundary.js +1 -1
  6. package/dist/components/ErrorBoundary.js.map +1 -1
  7. package/dist/hooks/useBuilderState.d.ts +49 -0
  8. package/dist/hooks/useBuilderState.d.ts.map +1 -0
  9. package/dist/hooks/useBuilderState.js +238 -0
  10. package/dist/hooks/useBuilderState.js.map +1 -0
  11. package/dist/index.d.ts +3 -0
  12. package/dist/index.d.ts.map +1 -1
  13. package/dist/index.js +2 -0
  14. package/dist/index.js.map +1 -1
  15. package/dist/layout/Sidebar.d.ts.map +1 -1
  16. package/dist/layout/Sidebar.js +2 -2
  17. package/dist/layout/Sidebar.js.map +1 -1
  18. package/dist/views/FormSubmissions.js +11 -11
  19. package/dist/views/FormSubmissions.js.map +1 -1
  20. package/dist/views/Forms.js +1 -1
  21. package/dist/views/Forms.js.map +1 -1
  22. package/dist/views/MediaBrowser.d.ts.map +1 -1
  23. package/dist/views/MediaBrowser.js +28 -8
  24. package/dist/views/MediaBrowser.js.map +1 -1
  25. package/dist/views/Posts.js +1 -1
  26. package/dist/views/Posts.js.map +1 -1
  27. package/dist/views/Redirects.js +2 -2
  28. package/dist/views/Redirects.js.map +1 -1
  29. package/dist/views/SEO.js +3 -3
  30. package/dist/views/SEO.js.map +1 -1
  31. package/dist/views/Users.js +3 -3
  32. package/dist/views/Users.js.map +1 -1
  33. package/dist/views/page-builder/AIBlockAssist.d.ts +9 -0
  34. package/dist/views/page-builder/AIBlockAssist.d.ts.map +1 -0
  35. package/dist/views/page-builder/AIBlockAssist.js +40 -0
  36. package/dist/views/page-builder/AIBlockAssist.js.map +1 -0
  37. package/dist/views/page-builder/AIGenerateDialog.d.ts +8 -0
  38. package/dist/views/page-builder/AIGenerateDialog.d.ts.map +1 -0
  39. package/dist/views/page-builder/AIGenerateDialog.js +170 -0
  40. package/dist/views/page-builder/AIGenerateDialog.js.map +1 -0
  41. package/dist/views/page-builder/BlockEditor.d.ts +11 -0
  42. package/dist/views/page-builder/BlockEditor.d.ts.map +1 -0
  43. package/dist/views/page-builder/BlockEditor.js +67 -0
  44. package/dist/views/page-builder/BlockEditor.js.map +1 -0
  45. package/dist/views/page-builder/BlockPicker.d.ts +7 -0
  46. package/dist/views/page-builder/BlockPicker.d.ts.map +1 -0
  47. package/dist/views/page-builder/BlockPicker.js +102 -0
  48. package/dist/views/page-builder/BlockPicker.js.map +1 -0
  49. package/dist/views/page-builder/BottomBar.d.ts +9 -0
  50. package/dist/views/page-builder/BottomBar.d.ts.map +1 -0
  51. package/dist/views/page-builder/BottomBar.js +13 -0
  52. package/dist/views/page-builder/BottomBar.js.map +1 -0
  53. package/dist/views/page-builder/BuilderToolbar.d.ts +21 -0
  54. package/dist/views/page-builder/BuilderToolbar.d.ts.map +1 -0
  55. package/dist/views/page-builder/BuilderToolbar.js +18 -0
  56. package/dist/views/page-builder/BuilderToolbar.js.map +1 -0
  57. package/dist/views/page-builder/ContextPanel.d.ts +20 -0
  58. package/dist/views/page-builder/ContextPanel.d.ts.map +1 -0
  59. package/dist/views/page-builder/ContextPanel.js +40 -0
  60. package/dist/views/page-builder/ContextPanel.js.map +1 -0
  61. package/dist/views/page-builder/DesignScore.d.ts +6 -0
  62. package/dist/views/page-builder/DesignScore.d.ts.map +1 -0
  63. package/dist/views/page-builder/DesignScore.js +93 -0
  64. package/dist/views/page-builder/DesignScore.js.map +1 -0
  65. package/dist/views/page-builder/NodeSettings.d.ts +12 -0
  66. package/dist/views/page-builder/NodeSettings.d.ts.map +1 -0
  67. package/dist/views/page-builder/NodeSettings.js +80 -0
  68. package/dist/views/page-builder/NodeSettings.js.map +1 -0
  69. package/dist/views/page-builder/PageBuilder.d.ts +8 -0
  70. package/dist/views/page-builder/PageBuilder.d.ts.map +1 -0
  71. package/dist/views/page-builder/PageBuilder.js +126 -0
  72. package/dist/views/page-builder/PageBuilder.js.map +1 -0
  73. package/dist/views/page-builder/PageSettings.d.ts +7 -0
  74. package/dist/views/page-builder/PageSettings.d.ts.map +1 -0
  75. package/dist/views/page-builder/PageSettings.js +27 -0
  76. package/dist/views/page-builder/PageSettings.js.map +1 -0
  77. package/dist/views/page-builder/PageTemplates.d.ts +5 -0
  78. package/dist/views/page-builder/PageTemplates.d.ts.map +1 -0
  79. package/dist/views/page-builder/PageTemplates.js +13 -0
  80. package/dist/views/page-builder/PageTemplates.js.map +1 -0
  81. package/dist/views/page-builder/SEOPanel.d.ts +10 -0
  82. package/dist/views/page-builder/SEOPanel.d.ts.map +1 -0
  83. package/dist/views/page-builder/SEOPanel.js +105 -0
  84. package/dist/views/page-builder/SEOPanel.js.map +1 -0
  85. package/dist/views/page-builder/SavedSections.d.ts +6 -0
  86. package/dist/views/page-builder/SavedSections.d.ts.map +1 -0
  87. package/dist/views/page-builder/SavedSections.js +145 -0
  88. package/dist/views/page-builder/SavedSections.js.map +1 -0
  89. package/dist/views/page-builder/TemplatePicker.d.ts +7 -0
  90. package/dist/views/page-builder/TemplatePicker.d.ts.map +1 -0
  91. package/dist/views/page-builder/TemplatePicker.js +68 -0
  92. package/dist/views/page-builder/TemplatePicker.js.map +1 -0
  93. package/dist/views/page-builder/block-renderers/CTAPreview.d.ts +3 -0
  94. package/dist/views/page-builder/block-renderers/CTAPreview.d.ts.map +1 -0
  95. package/dist/views/page-builder/block-renderers/CTAPreview.js +19 -0
  96. package/dist/views/page-builder/block-renderers/CTAPreview.js.map +1 -0
  97. package/dist/views/page-builder/block-renderers/CardsPreview.d.ts +3 -0
  98. package/dist/views/page-builder/block-renderers/CardsPreview.d.ts.map +1 -0
  99. package/dist/views/page-builder/block-renderers/CardsPreview.js +22 -0
  100. package/dist/views/page-builder/block-renderers/CardsPreview.js.map +1 -0
  101. package/dist/views/page-builder/block-renderers/CodePreview.d.ts +3 -0
  102. package/dist/views/page-builder/block-renderers/CodePreview.d.ts.map +1 -0
  103. package/dist/views/page-builder/block-renderers/CodePreview.js +16 -0
  104. package/dist/views/page-builder/block-renderers/CodePreview.js.map +1 -0
  105. package/dist/views/page-builder/block-renderers/FAQPreview.d.ts +3 -0
  106. package/dist/views/page-builder/block-renderers/FAQPreview.d.ts.map +1 -0
  107. package/dist/views/page-builder/block-renderers/FAQPreview.js +24 -0
  108. package/dist/views/page-builder/block-renderers/FAQPreview.js.map +1 -0
  109. package/dist/views/page-builder/block-renderers/FallbackPreview.d.ts +6 -0
  110. package/dist/views/page-builder/block-renderers/FallbackPreview.d.ts.map +1 -0
  111. package/dist/views/page-builder/block-renderers/FallbackPreview.js +7 -0
  112. package/dist/views/page-builder/block-renderers/FallbackPreview.js.map +1 -0
  113. package/dist/views/page-builder/block-renderers/FormPreview.d.ts +3 -0
  114. package/dist/views/page-builder/block-renderers/FormPreview.d.ts.map +1 -0
  115. package/dist/views/page-builder/block-renderers/FormPreview.js +14 -0
  116. package/dist/views/page-builder/block-renderers/FormPreview.js.map +1 -0
  117. package/dist/views/page-builder/block-renderers/GalleryPreview.d.ts +3 -0
  118. package/dist/views/page-builder/block-renderers/GalleryPreview.d.ts.map +1 -0
  119. package/dist/views/page-builder/block-renderers/GalleryPreview.js +21 -0
  120. package/dist/views/page-builder/block-renderers/GalleryPreview.js.map +1 -0
  121. package/dist/views/page-builder/block-renderers/HeroPreview.d.ts +3 -0
  122. package/dist/views/page-builder/block-renderers/HeroPreview.d.ts.map +1 -0
  123. package/dist/views/page-builder/block-renderers/HeroPreview.js +19 -0
  124. package/dist/views/page-builder/block-renderers/HeroPreview.js.map +1 -0
  125. package/dist/views/page-builder/block-renderers/ImagePreview.d.ts +3 -0
  126. package/dist/views/page-builder/block-renderers/ImagePreview.d.ts.map +1 -0
  127. package/dist/views/page-builder/block-renderers/ImagePreview.js +17 -0
  128. package/dist/views/page-builder/block-renderers/ImagePreview.js.map +1 -0
  129. package/dist/views/page-builder/block-renderers/TextPreview.d.ts +3 -0
  130. package/dist/views/page-builder/block-renderers/TextPreview.d.ts.map +1 -0
  131. package/dist/views/page-builder/block-renderers/TextPreview.js +26 -0
  132. package/dist/views/page-builder/block-renderers/TextPreview.js.map +1 -0
  133. package/dist/views/page-builder/block-renderers/VideoPreview.d.ts +3 -0
  134. package/dist/views/page-builder/block-renderers/VideoPreview.d.ts.map +1 -0
  135. package/dist/views/page-builder/block-renderers/VideoPreview.js +21 -0
  136. package/dist/views/page-builder/block-renderers/VideoPreview.js.map +1 -0
  137. package/dist/views/page-builder/block-renderers/index.d.ts +9 -0
  138. package/dist/views/page-builder/block-renderers/index.d.ts.map +1 -0
  139. package/dist/views/page-builder/block-renderers/index.js +25 -0
  140. package/dist/views/page-builder/block-renderers/index.js.map +1 -0
  141. package/dist/views/page-builder/canvas/BlockRenderer.d.ts +8 -0
  142. package/dist/views/page-builder/canvas/BlockRenderer.d.ts.map +1 -0
  143. package/dist/views/page-builder/canvas/BlockRenderer.js +30 -0
  144. package/dist/views/page-builder/canvas/BlockRenderer.js.map +1 -0
  145. package/dist/views/page-builder/canvas/BuilderCanvas.d.ts +10 -0
  146. package/dist/views/page-builder/canvas/BuilderCanvas.d.ts.map +1 -0
  147. package/dist/views/page-builder/canvas/BuilderCanvas.js +26 -0
  148. package/dist/views/page-builder/canvas/BuilderCanvas.js.map +1 -0
  149. package/dist/views/page-builder/canvas/ColumnRenderer.d.ts +8 -0
  150. package/dist/views/page-builder/canvas/ColumnRenderer.d.ts.map +1 -0
  151. package/dist/views/page-builder/canvas/ColumnRenderer.js +36 -0
  152. package/dist/views/page-builder/canvas/ColumnRenderer.js.map +1 -0
  153. package/dist/views/page-builder/canvas/ContainerRenderer.d.ts +8 -0
  154. package/dist/views/page-builder/canvas/ContainerRenderer.d.ts.map +1 -0
  155. package/dist/views/page-builder/canvas/ContainerRenderer.js +33 -0
  156. package/dist/views/page-builder/canvas/ContainerRenderer.js.map +1 -0
  157. package/dist/views/page-builder/canvas/RowRenderer.d.ts +8 -0
  158. package/dist/views/page-builder/canvas/RowRenderer.d.ts.map +1 -0
  159. package/dist/views/page-builder/canvas/RowRenderer.js +32 -0
  160. package/dist/views/page-builder/canvas/RowRenderer.js.map +1 -0
  161. package/dist/views/page-builder/canvas/SectionRenderer.d.ts +8 -0
  162. package/dist/views/page-builder/canvas/SectionRenderer.d.ts.map +1 -0
  163. package/dist/views/page-builder/canvas/SectionRenderer.js +54 -0
  164. package/dist/views/page-builder/canvas/SectionRenderer.js.map +1 -0
  165. package/dist/views/page-builder/canvas/index.d.ts +3 -0
  166. package/dist/views/page-builder/canvas/index.d.ts.map +1 -0
  167. package/dist/views/page-builder/canvas/index.js +2 -0
  168. package/dist/views/page-builder/canvas/index.js.map +1 -0
  169. package/package.json +3 -2
  170. package/src/AdminRoot.tsx +21 -0
  171. package/src/components/ErrorBoundary.tsx +3 -3
  172. package/src/hooks/useBuilderState.ts +328 -0
  173. package/src/index.ts +4 -0
  174. package/src/layout/Sidebar.tsx +5 -0
  175. package/src/views/FormSubmissions.tsx +12 -12
  176. package/src/views/Forms.tsx +1 -1
  177. package/src/views/MediaBrowser.tsx +46 -15
  178. package/src/views/Posts.tsx +1 -1
  179. package/src/views/Redirects.tsx +2 -2
  180. package/src/views/SEO.tsx +3 -3
  181. package/src/views/Users.tsx +3 -3
  182. package/src/views/page-builder/AIBlockAssist.tsx +68 -0
  183. package/src/views/page-builder/AIGenerateDialog.tsx +574 -0
  184. package/src/views/page-builder/BlockEditor.tsx +352 -0
  185. package/src/views/page-builder/BlockPicker.tsx +338 -0
  186. package/src/views/page-builder/BottomBar.tsx +64 -0
  187. package/src/views/page-builder/BuilderToolbar.tsx +218 -0
  188. package/src/views/page-builder/ContextPanel.tsx +145 -0
  189. package/src/views/page-builder/DesignScore.tsx +258 -0
  190. package/src/views/page-builder/NodeSettings.tsx +515 -0
  191. package/src/views/page-builder/PageBuilder.tsx +288 -0
  192. package/src/views/page-builder/PageSettings.tsx +161 -0
  193. package/src/views/page-builder/PageTemplates.tsx +105 -0
  194. package/src/views/page-builder/SEOPanel.tsx +485 -0
  195. package/src/views/page-builder/SavedSections.tsx +486 -0
  196. package/src/views/page-builder/TemplatePicker.tsx +201 -0
  197. package/src/views/page-builder/block-renderers/CTAPreview.tsx +81 -0
  198. package/src/views/page-builder/block-renderers/CardsPreview.tsx +71 -0
  199. package/src/views/page-builder/block-renderers/CodePreview.tsx +46 -0
  200. package/src/views/page-builder/block-renderers/FAQPreview.tsx +90 -0
  201. package/src/views/page-builder/block-renderers/FallbackPreview.tsx +18 -0
  202. package/src/views/page-builder/block-renderers/FormPreview.tsx +69 -0
  203. package/src/views/page-builder/block-renderers/GalleryPreview.tsx +93 -0
  204. package/src/views/page-builder/block-renderers/HeroPreview.tsx +103 -0
  205. package/src/views/page-builder/block-renderers/ImagePreview.tsx +54 -0
  206. package/src/views/page-builder/block-renderers/TextPreview.tsx +81 -0
  207. package/src/views/page-builder/block-renderers/VideoPreview.tsx +78 -0
  208. package/src/views/page-builder/block-renderers/index.ts +34 -0
  209. package/src/views/page-builder/canvas/BlockRenderer.tsx +62 -0
  210. package/src/views/page-builder/canvas/BuilderCanvas.tsx +90 -0
  211. package/src/views/page-builder/canvas/ColumnRenderer.tsx +86 -0
  212. package/src/views/page-builder/canvas/ContainerRenderer.tsx +71 -0
  213. package/src/views/page-builder/canvas/RowRenderer.tsx +72 -0
  214. package/src/views/page-builder/canvas/SectionRenderer.tsx +97 -0
  215. package/src/views/page-builder/canvas/index.ts +2 -0
@@ -0,0 +1,33 @@
1
+ 'use client';
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import { useState } from 'react';
4
+ import { RowRenderer } from './RowRenderer.js';
5
+ export function ContainerRenderer({ node, selectedNodeId, onSelectNode }) {
6
+ const [hovered, setHovered] = useState(false);
7
+ const isSelected = selectedNodeId === node.id;
8
+ const maxWidth = node.settings.maxWidth ?? '1200px';
9
+ const alignment = node.settings.alignment ?? 'center';
10
+ const padding = node.settings.padding;
11
+ const marginMap = {
12
+ left: '0 auto 0 0',
13
+ center: '0 auto',
14
+ right: '0 0 0 auto',
15
+ };
16
+ const handleClick = (e) => {
17
+ e.stopPropagation();
18
+ onSelectNode(node.id);
19
+ };
20
+ return (_jsxs("div", { "data-node-id": node.id, className: `relative transition-shadow ${isSelected
21
+ ? 'ring-2 ring-primary ring-offset-2'
22
+ : hovered
23
+ ? 'ring-1 ring-primary/50'
24
+ : ''}`, style: {
25
+ maxWidth,
26
+ margin: marginMap[alignment] ?? '0 auto',
27
+ padding,
28
+ }, onClick: handleClick, onMouseEnter: (e) => {
29
+ e.stopPropagation();
30
+ setHovered(true);
31
+ }, onMouseLeave: () => setHovered(false), children: [(hovered || isSelected) && (_jsx("span", { className: "absolute -top-2 -left-1 text-xs px-1.5 py-0.5 bg-primary text-primary-foreground rounded font-medium z-10", children: "Container" })), _jsx("div", { className: "flex flex-col gap-4", children: node.children.map((row) => (_jsx(RowRenderer, { node: row, selectedNodeId: selectedNodeId, onSelectNode: onSelectNode }, row.id))) })] }));
32
+ }
33
+ //# sourceMappingURL=ContainerRenderer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ContainerRenderer.js","sourceRoot":"","sources":["../../../../src/views/page-builder/canvas/ContainerRenderer.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAEb,OAAO,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAEjC,OAAO,EAAE,WAAW,EAAE,MAAM,kBAAkB,CAAC;AAQ/C,MAAM,UAAU,iBAAiB,CAAC,EAAE,IAAI,EAAE,cAAc,EAAE,YAAY,EAA0B;IAC9F,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC9C,MAAM,UAAU,GAAG,cAAc,KAAK,IAAI,CAAC,EAAE,CAAC;IAC9C,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,IAAI,QAAQ,CAAC;IACpD,MAAM,SAAS,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,IAAI,QAAQ,CAAC;IACtD,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC;IAEtC,MAAM,SAAS,GAA2B;QACxC,IAAI,EAAE,YAAY;QAClB,MAAM,EAAE,QAAQ;QAChB,KAAK,EAAE,YAAY;KACpB,CAAC;IAEF,MAAM,WAAW,GAAG,CAAC,CAAmB,EAAE,EAAE;QAC1C,CAAC,CAAC,eAAe,EAAE,CAAC;QACpB,YAAY,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACxB,CAAC,CAAC;IAEF,OAAO,CACL,+BACgB,IAAI,CAAC,EAAE,EACrB,SAAS,EAAE,8BACT,UAAU;YACR,CAAC,CAAC,mCAAmC;YACrC,CAAC,CAAC,OAAO;gBACP,CAAC,CAAC,wBAAwB;gBAC1B,CAAC,CAAC,EACR,EAAE,EACF,KAAK,EAAE;YACL,QAAQ;YACR,MAAM,EAAE,SAAS,CAAC,SAAS,CAAC,IAAI,QAAQ;YACxC,OAAO;SACR,EACD,OAAO,EAAE,WAAW,EACpB,YAAY,EAAE,CAAC,CAAC,EAAE,EAAE;YAClB,CAAC,CAAC,eAAe,EAAE,CAAC;YACpB,UAAU,CAAC,IAAI,CAAC,CAAC;QACnB,CAAC,EACD,YAAY,EAAE,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC,aAEpC,CAAC,OAAO,IAAI,UAAU,CAAC,IAAI,CAC1B,eAAM,SAAS,EAAC,2GAA2G,0BAEpH,CACR,EAED,cAAK,SAAS,EAAC,qBAAqB,YACjC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAC1B,KAAC,WAAW,IAEV,IAAI,EAAE,GAAG,EACT,cAAc,EAAE,cAAc,EAC9B,YAAY,EAAE,YAAY,IAHrB,GAAG,CAAC,EAAE,CAIX,CACH,CAAC,GACE,IACF,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,8 @@
1
+ import type { RowNode } from '@actuate-media/cms-core';
2
+ export interface RowRendererProps {
3
+ node: RowNode;
4
+ selectedNodeId: string | null;
5
+ onSelectNode: (id: string | null) => void;
6
+ }
7
+ export declare function RowRenderer({ node, selectedNodeId, onSelectNode }: RowRendererProps): import("react/jsx-runtime").JSX.Element;
8
+ //# sourceMappingURL=RowRenderer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"RowRenderer.d.ts","sourceRoot":"","sources":["../../../../src/views/page-builder/canvas/RowRenderer.tsx"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,yBAAyB,CAAC;AAGvD,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,OAAO,CAAC;IACd,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,YAAY,EAAE,CAAC,EAAE,EAAE,MAAM,GAAG,IAAI,KAAK,IAAI,CAAC;CAC3C;AAED,wBAAgB,WAAW,CAAC,EAAE,IAAI,EAAE,cAAc,EAAE,YAAY,EAAE,EAAE,gBAAgB,2CA2DnF"}
@@ -0,0 +1,32 @@
1
+ 'use client';
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import { useState } from 'react';
4
+ import { ColumnRenderer } from './ColumnRenderer.js';
5
+ export function RowRenderer({ node, selectedNodeId, onSelectNode }) {
6
+ const [hovered, setHovered] = useState(false);
7
+ const isSelected = selectedNodeId === node.id;
8
+ const gap = node.settings.gap ?? '16px';
9
+ const verticalAlign = node.settings.verticalAlign ?? 'stretch';
10
+ const alignMap = {
11
+ top: 'start',
12
+ center: 'center',
13
+ bottom: 'end',
14
+ stretch: 'stretch',
15
+ };
16
+ const handleClick = (e) => {
17
+ e.stopPropagation();
18
+ onSelectNode(node.id);
19
+ };
20
+ return (_jsxs("div", { "data-node-id": node.id, className: `relative transition-shadow ${isSelected
21
+ ? 'ring-2 ring-primary ring-offset-2'
22
+ : hovered
23
+ ? 'ring-1 ring-primary/50'
24
+ : ''}`, onClick: handleClick, onMouseEnter: (e) => {
25
+ e.stopPropagation();
26
+ setHovered(true);
27
+ }, onMouseLeave: () => setHovered(false), children: [(hovered || isSelected) && (_jsx("span", { className: "absolute -top-2 -left-1 text-xs px-1.5 py-0.5 bg-primary text-primary-foreground rounded font-medium z-10", children: "Row" })), _jsx("div", { className: "grid grid-cols-12", style: {
28
+ gap,
29
+ alignItems: alignMap[verticalAlign] ?? 'stretch',
30
+ }, children: node.children.map((col) => (_jsx(ColumnRenderer, { node: col, selectedNodeId: selectedNodeId, onSelectNode: onSelectNode }, col.id))) })] }));
31
+ }
32
+ //# sourceMappingURL=RowRenderer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"RowRenderer.js","sourceRoot":"","sources":["../../../../src/views/page-builder/canvas/RowRenderer.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAEb,OAAO,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAEjC,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AAQrD,MAAM,UAAU,WAAW,CAAC,EAAE,IAAI,EAAE,cAAc,EAAE,YAAY,EAAoB;IAClF,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC9C,MAAM,UAAU,GAAG,cAAc,KAAK,IAAI,CAAC,EAAE,CAAC;IAC9C,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,IAAI,MAAM,CAAC;IACxC,MAAM,aAAa,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,IAAI,SAAS,CAAC;IAE/D,MAAM,QAAQ,GAA2B;QACvC,GAAG,EAAE,OAAO;QACZ,MAAM,EAAE,QAAQ;QAChB,MAAM,EAAE,KAAK;QACb,OAAO,EAAE,SAAS;KACnB,CAAC;IAEF,MAAM,WAAW,GAAG,CAAC,CAAmB,EAAE,EAAE;QAC1C,CAAC,CAAC,eAAe,EAAE,CAAC;QACpB,YAAY,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACxB,CAAC,CAAC;IAEF,OAAO,CACL,+BACgB,IAAI,CAAC,EAAE,EACrB,SAAS,EAAE,8BACT,UAAU;YACR,CAAC,CAAC,mCAAmC;YACrC,CAAC,CAAC,OAAO;gBACP,CAAC,CAAC,wBAAwB;gBAC1B,CAAC,CAAC,EACR,EAAE,EACF,OAAO,EAAE,WAAW,EACpB,YAAY,EAAE,CAAC,CAAC,EAAE,EAAE;YAClB,CAAC,CAAC,eAAe,EAAE,CAAC;YACpB,UAAU,CAAC,IAAI,CAAC,CAAC;QACnB,CAAC,EACD,YAAY,EAAE,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC,aAEpC,CAAC,OAAO,IAAI,UAAU,CAAC,IAAI,CAC1B,eAAM,SAAS,EAAC,2GAA2G,oBAEpH,CACR,EAED,cACE,SAAS,EAAC,mBAAmB,EAC7B,KAAK,EAAE;oBACL,GAAG;oBACH,UAAU,EAAE,QAAQ,CAAC,aAAa,CAAC,IAAI,SAAS;iBACjD,YAEA,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAC1B,KAAC,cAAc,IAEb,IAAI,EAAE,GAAG,EACT,cAAc,EAAE,cAAc,EAC9B,YAAY,EAAE,YAAY,IAHrB,GAAG,CAAC,EAAE,CAIX,CACH,CAAC,GACE,IACF,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,8 @@
1
+ import type { SectionNode } from '@actuate-media/cms-core';
2
+ export interface SectionRendererProps {
3
+ node: SectionNode;
4
+ selectedNodeId: string | null;
5
+ onSelectNode: (id: string | null) => void;
6
+ }
7
+ export declare function SectionRenderer({ node, selectedNodeId, onSelectNode }: SectionRendererProps): import("react/jsx-runtime").JSX.Element;
8
+ //# sourceMappingURL=SectionRenderer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SectionRenderer.d.ts","sourceRoot":"","sources":["../../../../src/views/page-builder/canvas/SectionRenderer.tsx"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,yBAAyB,CAAC;AAI3D,MAAM,WAAW,oBAAoB;IACnC,IAAI,EAAE,WAAW,CAAC;IAClB,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,YAAY,EAAE,CAAC,EAAE,EAAE,MAAM,GAAG,IAAI,KAAK,IAAI,CAAC;CAC3C;AAED,wBAAgB,eAAe,CAAC,EAAE,IAAI,EAAE,cAAc,EAAE,YAAY,EAAE,EAAE,oBAAoB,2CAmF3F"}
@@ -0,0 +1,54 @@
1
+ 'use client';
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import { useState } from 'react';
4
+ import { ContainerRenderer } from './ContainerRenderer.js';
5
+ import { RowRenderer } from './RowRenderer.js';
6
+ export function SectionRenderer({ node, selectedNodeId, onSelectNode }) {
7
+ const [hovered, setHovered] = useState(false);
8
+ const isSelected = selectedNodeId === node.id;
9
+ const handleClick = (e) => {
10
+ e.stopPropagation();
11
+ onSelectNode(node.id);
12
+ };
13
+ const backgroundStyle = {};
14
+ if (node.settings.background) {
15
+ backgroundStyle.backgroundColor = node.settings.background;
16
+ }
17
+ if (node.settings.backgroundImage) {
18
+ backgroundStyle.backgroundImage = `url(${node.settings.backgroundImage})`;
19
+ backgroundStyle.backgroundSize = 'cover';
20
+ backgroundStyle.backgroundPosition = 'center';
21
+ }
22
+ if (node.settings.backgroundGradient) {
23
+ backgroundStyle.backgroundImage = node.settings.backgroundGradient;
24
+ }
25
+ if (node.settings.paddingTop) {
26
+ backgroundStyle.paddingTop = node.settings.paddingTop;
27
+ }
28
+ if (node.settings.paddingBottom) {
29
+ backgroundStyle.paddingBottom = node.settings.paddingBottom;
30
+ }
31
+ if (node.settings.marginTop) {
32
+ backgroundStyle.marginTop = node.settings.marginTop;
33
+ }
34
+ if (node.settings.marginBottom) {
35
+ backgroundStyle.marginBottom = node.settings.marginBottom;
36
+ }
37
+ return (_jsxs("div", { "data-node-id": node.id, className: `relative w-full transition-shadow ${isSelected
38
+ ? 'ring-2 ring-primary ring-offset-2'
39
+ : hovered
40
+ ? 'ring-1 ring-primary/50'
41
+ : ''}`, style: backgroundStyle, onClick: handleClick, onMouseEnter: (e) => {
42
+ e.stopPropagation();
43
+ setHovered(true);
44
+ }, onMouseLeave: () => setHovered(false), children: [(hovered || isSelected) && (_jsx("span", { className: "absolute -top-2 -left-1 text-xs px-1.5 py-0.5 bg-primary text-primary-foreground rounded font-medium z-10", children: "Section" })), node.children.map((child) => {
45
+ if (child.type === 'container') {
46
+ return (_jsx(ContainerRenderer, { node: child, selectedNodeId: selectedNodeId, onSelectNode: onSelectNode }, child.id));
47
+ }
48
+ if (child.type === 'row') {
49
+ return (_jsx(RowRenderer, { node: child, selectedNodeId: selectedNodeId, onSelectNode: onSelectNode }, child.id));
50
+ }
51
+ return null;
52
+ })] }));
53
+ }
54
+ //# sourceMappingURL=SectionRenderer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SectionRenderer.js","sourceRoot":"","sources":["../../../../src/views/page-builder/canvas/SectionRenderer.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ export { BuilderCanvas } from './BuilderCanvas.js';
2
+ export type { BuilderCanvasProps } from './BuilderCanvas.js';
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/views/page-builder/canvas/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AACnD,YAAY,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { BuilderCanvas } from './BuilderCanvas.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/views/page-builder/canvas/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@actuate-media/cms-admin",
3
- "version": "0.6.0",
3
+ "version": "0.7.1",
4
4
  "repository": {
5
5
  "type": "git",
6
6
  "url": "https://github.com/actuate-media/actuatecms.git",
@@ -69,7 +69,8 @@
69
69
  "@types/react-dom": "^19.0.0",
70
70
  "tailwindcss": "^4.0.0",
71
71
  "typescript": "^5.7.0",
72
- "vitest": "^3.0.0"
72
+ "vitest": "^3.0.0",
73
+ "@actuate-media/cms-core": "0.10.2"
73
74
  },
74
75
  "scripts": {
75
76
  "build": "tsc --project tsconfig.json && npx @tailwindcss/cli -i src/styles/build-input.css -o dist/actuate-admin.css --minify",
package/src/AdminRoot.tsx CHANGED
@@ -23,6 +23,9 @@ import { ErrorBoundary } from './components/ErrorBoundary.js';
23
23
  import { ThemeProvider } from './components/ThemeProvider.js';
24
24
  import { LocaleProvider } from './components/LocaleProvider.js';
25
25
  import { useKeyboardShortcuts } from './hooks/useKeyboardShortcuts.js';
26
+ import { PageBuilder } from './views/page-builder/PageBuilder.js';
27
+ import { SavedSections } from './views/page-builder/SavedSections.js';
28
+ import { PageTemplates } from './views/page-builder/PageTemplates.js';
26
29
 
27
30
  export interface AdminRootProps {
28
31
  config: any;
@@ -99,6 +102,16 @@ function AdminShell({ config, session, basePath = '/admin', initialPath = '/', s
99
102
  return <Dashboard config={config} session={session} onNavigate={navigate} />;
100
103
  }
101
104
 
105
+ const pageBuilderNew = matchRoute('/page-builder/new');
106
+ if (pageBuilderNew) {
107
+ return <PageBuilder collectionSlug="pages" config={config} onNavigate={navigate} />;
108
+ }
109
+
110
+ const pageBuilderEdit = matchRoute('/page-builder/:id');
111
+ if (pageBuilderEdit?.id) {
112
+ return <PageBuilder documentId={pageBuilderEdit.id} collectionSlug="pages" config={config} onNavigate={navigate} />;
113
+ }
114
+
102
115
  for (const slug of collectionSlugs) {
103
116
  const newMatch = matchRoute(`/${slug}/new`);
104
117
  if (newMatch) {
@@ -171,6 +184,14 @@ function AdminShell({ config, session, basePath = '/admin', initialPath = '/', s
171
184
  return <ScriptTags onNavigate={navigate} />;
172
185
  }
173
186
 
187
+ if (matchRoute('/saved-sections')) {
188
+ return <SavedSections onNavigate={navigate} config={config} />;
189
+ }
190
+
191
+ if (matchRoute('/page-templates')) {
192
+ return <PageTemplates onNavigate={navigate} />;
193
+ }
194
+
174
195
  if (matchRoute('/users')) {
175
196
  return <Users onNavigate={navigate} />;
176
197
  }
@@ -32,13 +32,13 @@ export class ErrorBoundary extends Component<Props, State> {
32
32
  return (
33
33
  <div className="flex min-h-[200px] items-center justify-center p-6">
34
34
  <div className="text-center">
35
- <h2 className="text-lg font-semibold text-gray-900 mb-2">Something went wrong</h2>
36
- <p className="text-sm text-gray-600 mb-4">
35
+ <h2 className="text-lg font-medium text-foreground mb-2">Something went wrong</h2>
36
+ <p className="text-sm text-muted-foreground mb-4">
37
37
  {this.state.error?.message ?? 'An unexpected error occurred'}
38
38
  </p>
39
39
  <button
40
40
  onClick={() => this.setState({ hasError: false, error: null })}
41
- className="px-4 py-2 text-sm font-medium text-white bg-blue-600 rounded-md hover:bg-blue-700"
41
+ className="px-4 py-2 text-sm font-medium text-primary-foreground bg-primary rounded-md hover:bg-primary/90 transition-colors"
42
42
  >
43
43
  Try Again
44
44
  </button>
@@ -0,0 +1,328 @@
1
+ 'use client';
2
+
3
+ import { useState, useCallback, useRef, useMemo } from 'react';
4
+ import type {
5
+ PageNode,
6
+ BuilderNode,
7
+ BlockNode,
8
+ SectionNode,
9
+ ColumnNode,
10
+ } from '@actuate-media/cms-core';
11
+ import {
12
+ addNode,
13
+ removeNode,
14
+ moveNode,
15
+ updateNodeSettings,
16
+ updateBlockData,
17
+ findNode,
18
+ findParent,
19
+ createEmptyPage,
20
+ createSection,
21
+ createContainer,
22
+ createRow,
23
+ createColumn,
24
+ createBlock,
25
+ hasChildren,
26
+ } from '@actuate-media/cms-core';
27
+
28
+ export type DeviceMode = 'desktop' | 'tablet' | 'mobile';
29
+ export type PanelTab = 'block' | 'node' | 'page' | 'seo' | 'design';
30
+
31
+ export interface PageSettings {
32
+ title: string;
33
+ slug: string;
34
+ template?: string;
35
+ metaTitle?: string;
36
+ metaDescription?: string;
37
+ ogImage?: string;
38
+ focusKeyphrase?: string;
39
+ schemaType?: string;
40
+ }
41
+
42
+ export interface BuilderState {
43
+ tree: PageNode;
44
+ selectedNodeId: string | null;
45
+ selectedNode: BuilderNode | null;
46
+ deviceMode: DeviceMode;
47
+ activeTab: PanelTab;
48
+ pageSettings: PageSettings;
49
+ dirty: boolean;
50
+ canUndo: boolean;
51
+ canRedo: boolean;
52
+ showGridOverlay: boolean;
53
+ }
54
+
55
+ export interface BuilderActions {
56
+ selectNode: (id: string | null) => void;
57
+ setDeviceMode: (mode: DeviceMode) => void;
58
+ setActiveTab: (tab: PanelTab) => void;
59
+ setPageSettings: (settings: Partial<PageSettings>) => void;
60
+ setShowGridOverlay: (show: boolean) => void;
61
+
62
+ addSection: () => void;
63
+ addRowToSection: (sectionId: string) => void;
64
+ addBlockToColumn: (columnId: string, blockType: string, variant?: string) => void;
65
+ addNodeAtId: (parentId: string, node: BuilderNode, index?: number) => void;
66
+ removeNodeById: (id: string) => void;
67
+ moveNodeById: (id: string, newParentId: string, index?: number) => void;
68
+ updateSettings: (id: string, settings: Record<string, unknown>) => void;
69
+ updateBlock: (id: string, data: Record<string, unknown>) => void;
70
+ duplicateNode: (id: string) => void;
71
+ moveNodeUp: (id: string) => void;
72
+ moveNodeDown: (id: string) => void;
73
+
74
+ undo: () => void;
75
+ redo: () => void;
76
+ markClean: () => void;
77
+ replaceTree: (tree: PageNode) => void;
78
+ }
79
+
80
+ const MAX_HISTORY = 50;
81
+
82
+ function deepClone<T>(obj: T): T {
83
+ return JSON.parse(JSON.stringify(obj));
84
+ }
85
+
86
+ function regenerateIds(node: BuilderNode): BuilderNode {
87
+ const cloned = deepClone(node);
88
+ let counter = 0;
89
+ function walk(n: any) {
90
+ n.id = `node_${Date.now().toString(36)}_${(counter++).toString(36)}_${Math.random().toString(36).slice(2, 8)}`;
91
+ if (Array.isArray(n.children)) {
92
+ for (const child of n.children) walk(child);
93
+ }
94
+ }
95
+ walk(cloned);
96
+ return cloned;
97
+ }
98
+
99
+ export function useBuilderState(
100
+ initialTree?: PageNode,
101
+ initialPageSettings?: Partial<PageSettings>,
102
+ ): BuilderState & BuilderActions {
103
+ const [tree, setTree] = useState<PageNode>(() => initialTree ?? createEmptyPage());
104
+ const [selectedNodeId, setSelectedNodeId] = useState<string | null>(null);
105
+ const [deviceMode, setDeviceMode] = useState<DeviceMode>('desktop');
106
+ const [activeTab, setActiveTab] = useState<PanelTab>('page');
107
+ const [showGridOverlay, setShowGridOverlay] = useState(false);
108
+ const [pageSettings, setPageSettingsState] = useState<PageSettings>({
109
+ title: '',
110
+ slug: '',
111
+ ...initialPageSettings,
112
+ });
113
+ const [dirty, setDirty] = useState(false);
114
+
115
+ const undoStack = useRef<PageNode[]>([]);
116
+ const redoStack = useRef<PageNode[]>([]);
117
+
118
+ const pushHistory = useCallback((currentTree: PageNode) => {
119
+ undoStack.current = [...undoStack.current.slice(-MAX_HISTORY + 1), currentTree];
120
+ redoStack.current = [];
121
+ }, []);
122
+
123
+ const [canUndo, setCanUndo] = useState(false);
124
+ const [canRedo, setCanRedo] = useState(false);
125
+
126
+ const applyTreeChange = useCallback((fn: (prev: PageNode) => PageNode) => {
127
+ setTree((prev) => {
128
+ pushHistory(prev);
129
+ const next = fn(prev);
130
+ setDirty(true);
131
+ setCanUndo(true);
132
+ setCanRedo(false);
133
+ return next;
134
+ });
135
+ }, [pushHistory]);
136
+
137
+ const selectedNode = useMemo(() => {
138
+ if (!selectedNodeId) return null;
139
+ return findNode(tree, selectedNodeId) ?? null;
140
+ }, [tree, selectedNodeId]);
141
+
142
+ const selectNode = useCallback((id: string | null) => {
143
+ setSelectedNodeId(id);
144
+ if (id) {
145
+ const node = findNode(tree, id);
146
+ if (node) {
147
+ if (node.type === 'block') setActiveTab('block');
148
+ else if (node.type === 'page') setActiveTab('page');
149
+ else setActiveTab('node');
150
+ }
151
+ }
152
+ }, [tree]);
153
+
154
+ const setPageSettings = useCallback((settings: Partial<PageSettings>) => {
155
+ setPageSettingsState((prev) => ({ ...prev, ...settings }));
156
+ setDirty(true);
157
+ }, []);
158
+
159
+ const addSection = useCallback(() => {
160
+ applyTreeChange((prev) => {
161
+ const section = createSection({ paddingTop: '64px', paddingBottom: '64px' });
162
+ const container = createContainer();
163
+ const col = createColumn(12);
164
+ const row = createRow([col]);
165
+ (container as any).children = [row];
166
+ (section as any).children = [container];
167
+ return addNode(prev, prev.id, section as BuilderNode);
168
+ });
169
+ }, [applyTreeChange]);
170
+
171
+ const addRowToSection = useCallback((sectionId: string) => {
172
+ applyTreeChange((prev) => {
173
+ const section = findNode(prev, sectionId);
174
+ if (!section || !hasChildren(section)) return prev;
175
+ const containers = (section as SectionNode).children.filter((c) => c.type === 'container');
176
+ const targetId = containers.length > 0 ? containers[containers.length - 1]!.id : sectionId;
177
+ const col = createColumn(12);
178
+ const row = createRow([col]);
179
+ return addNode(prev, targetId, row as BuilderNode);
180
+ });
181
+ }, [applyTreeChange]);
182
+
183
+ const addBlockToColumn = useCallback((columnId: string, blockType: string, variant?: string) => {
184
+ applyTreeChange((prev) => {
185
+ const block = createBlock(blockType, variant);
186
+ return addNode(prev, columnId, block as BuilderNode);
187
+ });
188
+ }, [applyTreeChange]);
189
+
190
+ const addNodeAtId = useCallback((parentId: string, node: BuilderNode, index?: number) => {
191
+ applyTreeChange((prev) => addNode(prev, parentId, node, index));
192
+ }, [applyTreeChange]);
193
+
194
+ const removeNodeById = useCallback((id: string) => {
195
+ applyTreeChange((prev) => {
196
+ const result = removeNode(prev, id);
197
+ if (selectedNodeId === id) setSelectedNodeId(null);
198
+ return result;
199
+ });
200
+ }, [applyTreeChange, selectedNodeId]);
201
+
202
+ const moveNodeById = useCallback((id: string, newParentId: string, index?: number) => {
203
+ applyTreeChange((prev) => moveNode(prev, id, newParentId, index));
204
+ }, [applyTreeChange]);
205
+
206
+ const updateSettings = useCallback((id: string, settings: Record<string, unknown>) => {
207
+ applyTreeChange((prev) => updateNodeSettings(prev, id, settings));
208
+ }, [applyTreeChange]);
209
+
210
+ const updateBlock = useCallback((id: string, data: Record<string, unknown>) => {
211
+ applyTreeChange((prev) => updateBlockData(prev, id, data));
212
+ }, [applyTreeChange]);
213
+
214
+ const duplicateNode = useCallback((id: string) => {
215
+ applyTreeChange((prev) => {
216
+ const node = findNode(prev, id);
217
+ if (!node) return prev;
218
+ const parent = findParent(prev, id);
219
+ if (!parent) return prev;
220
+ const siblings = (parent as any).children as BuilderNode[];
221
+ const idx = siblings.findIndex((c) => c.id === id);
222
+ const cloned = regenerateIds(node);
223
+ return addNode(prev, parent.id, cloned, idx + 1);
224
+ });
225
+ }, [applyTreeChange]);
226
+
227
+ const moveNodeUp = useCallback((id: string) => {
228
+ applyTreeChange((prev) => {
229
+ const parent = findParent(prev, id);
230
+ if (!parent) return prev;
231
+ const siblings = (parent as any).children as BuilderNode[];
232
+ const idx = siblings.findIndex((c) => c.id === id);
233
+ if (idx <= 0) return prev;
234
+ const withoutNode = removeNode(prev, id);
235
+ const node = findNode(prev, id);
236
+ if (!node) return prev;
237
+ return addNode(withoutNode, parent.id, deepClone(node), idx - 1);
238
+ });
239
+ }, [applyTreeChange]);
240
+
241
+ const moveNodeDown = useCallback((id: string) => {
242
+ applyTreeChange((prev) => {
243
+ const parent = findParent(prev, id);
244
+ if (!parent) return prev;
245
+ const siblings = (parent as any).children as BuilderNode[];
246
+ const idx = siblings.findIndex((c) => c.id === id);
247
+ if (idx >= siblings.length - 1) return prev;
248
+ const withoutNode = removeNode(prev, id);
249
+ const node = findNode(prev, id);
250
+ if (!node) return prev;
251
+ return addNode(withoutNode, parent.id, deepClone(node), idx + 1);
252
+ });
253
+ }, [applyTreeChange]);
254
+
255
+ const undo = useCallback(() => {
256
+ if (undoStack.current.length === 0) return;
257
+ const prev = undoStack.current[undoStack.current.length - 1]!;
258
+ undoStack.current = undoStack.current.slice(0, -1);
259
+ setTree((current) => {
260
+ redoStack.current = [...redoStack.current, current];
261
+ setCanUndo(undoStack.current.length > 0);
262
+ setCanRedo(true);
263
+ return prev;
264
+ });
265
+ }, []);
266
+
267
+ const redo = useCallback(() => {
268
+ if (redoStack.current.length === 0) return;
269
+ const next = redoStack.current[redoStack.current.length - 1]!;
270
+ redoStack.current = redoStack.current.slice(0, -1);
271
+ setTree((current) => {
272
+ undoStack.current = [...undoStack.current, current];
273
+ setCanRedo(redoStack.current.length > 0);
274
+ setCanUndo(true);
275
+ return next;
276
+ });
277
+ }, []);
278
+
279
+ const markClean = useCallback(() => {
280
+ setDirty(false);
281
+ }, []);
282
+
283
+ const replaceTree = useCallback((newTree: PageNode) => {
284
+ setTree((prev) => {
285
+ pushHistory(prev);
286
+ setDirty(true);
287
+ setCanUndo(true);
288
+ setCanRedo(false);
289
+ return newTree;
290
+ });
291
+ }, [pushHistory]);
292
+
293
+ return {
294
+ tree,
295
+ selectedNodeId,
296
+ selectedNode,
297
+ deviceMode,
298
+ activeTab,
299
+ pageSettings,
300
+ dirty,
301
+ canUndo,
302
+ canRedo,
303
+ showGridOverlay,
304
+
305
+ selectNode,
306
+ setDeviceMode,
307
+ setActiveTab,
308
+ setPageSettings,
309
+ setShowGridOverlay,
310
+
311
+ addSection,
312
+ addRowToSection,
313
+ addBlockToColumn,
314
+ addNodeAtId,
315
+ removeNodeById,
316
+ moveNodeById,
317
+ updateSettings,
318
+ updateBlock,
319
+ duplicateNode,
320
+ moveNodeUp,
321
+ moveNodeDown,
322
+
323
+ undo,
324
+ redo,
325
+ markClean,
326
+ replaceTree,
327
+ };
328
+ }
package/src/index.ts CHANGED
@@ -33,6 +33,10 @@ export type { ResetPasswordProps } from './views/ResetPassword.js';
33
33
  export { CollectionList } from './views/CollectionList.js';
34
34
  export { DocumentEdit } from './views/DocumentEdit.js';
35
35
 
36
+ export { PageBuilder } from './views/page-builder/PageBuilder.js';
37
+ export { useBuilderState } from './hooks/useBuilderState.js';
38
+ export type { BuilderState, BuilderActions, DeviceMode, PanelTab, PageSettings as BuilderPageSettings } from './hooks/useBuilderState.js';
39
+
36
40
  export { Breadcrumbs } from './components/Breadcrumbs.js';
37
41
  export { CommandPalette } from './components/CommandPalette.js';
38
42
  export { ErrorBoundary } from './components/ErrorBoundary.js';
@@ -20,6 +20,8 @@ import {
20
20
  PanelBottom,
21
21
  Layers,
22
22
  Code2,
23
+ LayoutTemplate,
24
+ Library,
23
25
  } from 'lucide-react';
24
26
  import type { LucideIcon } from 'lucide-react';
25
27
 
@@ -254,6 +256,9 @@ function buildNavItems(config: any): NavItem[] {
254
256
  }
255
257
 
256
258
  items.push(
259
+ { path: '/page-builder/new', label: 'Page Builder', icon: LayoutTemplate, group: 'Pages' },
260
+ { path: '/saved-sections', label: 'Saved Sections', icon: Library, group: 'Pages' },
261
+ { path: '/page-templates', label: 'Templates', icon: Layers, group: 'Pages' },
257
262
  { path: '/media', label: 'Media', icon: Image },
258
263
  { path: '/forms', label: 'Forms', icon: ClipboardList },
259
264
  { path: '/seo', label: 'SEO', icon: SearchIcon },