@actuate-media/cms-admin 0.6.0 → 0.7.0
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.
- package/dist/AdminRoot.d.ts.map +1 -1
- package/dist/AdminRoot.js +13 -0
- package/dist/AdminRoot.js.map +1 -1
- package/dist/actuate-admin.css +1 -1
- package/dist/components/ErrorBoundary.js +1 -1
- package/dist/components/ErrorBoundary.js.map +1 -1
- package/dist/hooks/useBuilderState.d.ts +49 -0
- package/dist/hooks/useBuilderState.d.ts.map +1 -0
- package/dist/hooks/useBuilderState.js +238 -0
- package/dist/hooks/useBuilderState.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +2 -0
- package/dist/index.js.map +1 -1
- package/dist/layout/Sidebar.d.ts.map +1 -1
- package/dist/layout/Sidebar.js +2 -2
- package/dist/layout/Sidebar.js.map +1 -1
- package/dist/views/page-builder/AIBlockAssist.d.ts +9 -0
- package/dist/views/page-builder/AIBlockAssist.d.ts.map +1 -0
- package/dist/views/page-builder/AIBlockAssist.js +40 -0
- package/dist/views/page-builder/AIBlockAssist.js.map +1 -0
- package/dist/views/page-builder/AIGenerateDialog.d.ts +8 -0
- package/dist/views/page-builder/AIGenerateDialog.d.ts.map +1 -0
- package/dist/views/page-builder/AIGenerateDialog.js +170 -0
- package/dist/views/page-builder/AIGenerateDialog.js.map +1 -0
- package/dist/views/page-builder/BlockEditor.d.ts +11 -0
- package/dist/views/page-builder/BlockEditor.d.ts.map +1 -0
- package/dist/views/page-builder/BlockEditor.js +67 -0
- package/dist/views/page-builder/BlockEditor.js.map +1 -0
- package/dist/views/page-builder/BlockPicker.d.ts +7 -0
- package/dist/views/page-builder/BlockPicker.d.ts.map +1 -0
- package/dist/views/page-builder/BlockPicker.js +102 -0
- package/dist/views/page-builder/BlockPicker.js.map +1 -0
- package/dist/views/page-builder/BottomBar.d.ts +9 -0
- package/dist/views/page-builder/BottomBar.d.ts.map +1 -0
- package/dist/views/page-builder/BottomBar.js +13 -0
- package/dist/views/page-builder/BottomBar.js.map +1 -0
- package/dist/views/page-builder/BuilderToolbar.d.ts +21 -0
- package/dist/views/page-builder/BuilderToolbar.d.ts.map +1 -0
- package/dist/views/page-builder/BuilderToolbar.js +18 -0
- package/dist/views/page-builder/BuilderToolbar.js.map +1 -0
- package/dist/views/page-builder/ContextPanel.d.ts +20 -0
- package/dist/views/page-builder/ContextPanel.d.ts.map +1 -0
- package/dist/views/page-builder/ContextPanel.js +40 -0
- package/dist/views/page-builder/ContextPanel.js.map +1 -0
- package/dist/views/page-builder/DesignScore.d.ts +6 -0
- package/dist/views/page-builder/DesignScore.d.ts.map +1 -0
- package/dist/views/page-builder/DesignScore.js +93 -0
- package/dist/views/page-builder/DesignScore.js.map +1 -0
- package/dist/views/page-builder/NodeSettings.d.ts +12 -0
- package/dist/views/page-builder/NodeSettings.d.ts.map +1 -0
- package/dist/views/page-builder/NodeSettings.js +80 -0
- package/dist/views/page-builder/NodeSettings.js.map +1 -0
- package/dist/views/page-builder/PageBuilder.d.ts +8 -0
- package/dist/views/page-builder/PageBuilder.d.ts.map +1 -0
- package/dist/views/page-builder/PageBuilder.js +126 -0
- package/dist/views/page-builder/PageBuilder.js.map +1 -0
- package/dist/views/page-builder/PageSettings.d.ts +7 -0
- package/dist/views/page-builder/PageSettings.d.ts.map +1 -0
- package/dist/views/page-builder/PageSettings.js +27 -0
- package/dist/views/page-builder/PageSettings.js.map +1 -0
- package/dist/views/page-builder/SEOPanel.d.ts +10 -0
- package/dist/views/page-builder/SEOPanel.d.ts.map +1 -0
- package/dist/views/page-builder/SEOPanel.js +105 -0
- package/dist/views/page-builder/SEOPanel.js.map +1 -0
- package/dist/views/page-builder/SavedSections.d.ts +6 -0
- package/dist/views/page-builder/SavedSections.d.ts.map +1 -0
- package/dist/views/page-builder/SavedSections.js +145 -0
- package/dist/views/page-builder/SavedSections.js.map +1 -0
- package/dist/views/page-builder/TemplatePicker.d.ts +7 -0
- package/dist/views/page-builder/TemplatePicker.d.ts.map +1 -0
- package/dist/views/page-builder/TemplatePicker.js +68 -0
- package/dist/views/page-builder/TemplatePicker.js.map +1 -0
- package/dist/views/page-builder/block-renderers/CTAPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/CTAPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/CTAPreview.js +19 -0
- package/dist/views/page-builder/block-renderers/CTAPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/CardsPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/CardsPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/CardsPreview.js +22 -0
- package/dist/views/page-builder/block-renderers/CardsPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/CodePreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/CodePreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/CodePreview.js +16 -0
- package/dist/views/page-builder/block-renderers/CodePreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/FAQPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/FAQPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/FAQPreview.js +24 -0
- package/dist/views/page-builder/block-renderers/FAQPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/FallbackPreview.d.ts +6 -0
- package/dist/views/page-builder/block-renderers/FallbackPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/FallbackPreview.js +7 -0
- package/dist/views/page-builder/block-renderers/FallbackPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/FormPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/FormPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/FormPreview.js +14 -0
- package/dist/views/page-builder/block-renderers/FormPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/GalleryPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/GalleryPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/GalleryPreview.js +21 -0
- package/dist/views/page-builder/block-renderers/GalleryPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/HeroPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/HeroPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/HeroPreview.js +19 -0
- package/dist/views/page-builder/block-renderers/HeroPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/ImagePreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/ImagePreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/ImagePreview.js +17 -0
- package/dist/views/page-builder/block-renderers/ImagePreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/TextPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/TextPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/TextPreview.js +26 -0
- package/dist/views/page-builder/block-renderers/TextPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/VideoPreview.d.ts +3 -0
- package/dist/views/page-builder/block-renderers/VideoPreview.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/VideoPreview.js +21 -0
- package/dist/views/page-builder/block-renderers/VideoPreview.js.map +1 -0
- package/dist/views/page-builder/block-renderers/index.d.ts +9 -0
- package/dist/views/page-builder/block-renderers/index.d.ts.map +1 -0
- package/dist/views/page-builder/block-renderers/index.js +25 -0
- package/dist/views/page-builder/block-renderers/index.js.map +1 -0
- package/dist/views/page-builder/canvas/BlockRenderer.d.ts +8 -0
- package/dist/views/page-builder/canvas/BlockRenderer.d.ts.map +1 -0
- package/dist/views/page-builder/canvas/BlockRenderer.js +30 -0
- package/dist/views/page-builder/canvas/BlockRenderer.js.map +1 -0
- package/dist/views/page-builder/canvas/BuilderCanvas.d.ts +10 -0
- package/dist/views/page-builder/canvas/BuilderCanvas.d.ts.map +1 -0
- package/dist/views/page-builder/canvas/BuilderCanvas.js +26 -0
- package/dist/views/page-builder/canvas/BuilderCanvas.js.map +1 -0
- package/dist/views/page-builder/canvas/ColumnRenderer.d.ts +8 -0
- package/dist/views/page-builder/canvas/ColumnRenderer.d.ts.map +1 -0
- package/dist/views/page-builder/canvas/ColumnRenderer.js +36 -0
- package/dist/views/page-builder/canvas/ColumnRenderer.js.map +1 -0
- package/dist/views/page-builder/canvas/ContainerRenderer.d.ts +8 -0
- package/dist/views/page-builder/canvas/ContainerRenderer.d.ts.map +1 -0
- package/dist/views/page-builder/canvas/ContainerRenderer.js +33 -0
- package/dist/views/page-builder/canvas/ContainerRenderer.js.map +1 -0
- package/dist/views/page-builder/canvas/RowRenderer.d.ts +8 -0
- package/dist/views/page-builder/canvas/RowRenderer.d.ts.map +1 -0
- package/dist/views/page-builder/canvas/RowRenderer.js +32 -0
- package/dist/views/page-builder/canvas/RowRenderer.js.map +1 -0
- package/dist/views/page-builder/canvas/SectionRenderer.d.ts +8 -0
- package/dist/views/page-builder/canvas/SectionRenderer.d.ts.map +1 -0
- package/dist/views/page-builder/canvas/SectionRenderer.js +54 -0
- package/dist/views/page-builder/canvas/SectionRenderer.js.map +1 -0
- package/dist/views/page-builder/canvas/index.d.ts +3 -0
- package/dist/views/page-builder/canvas/index.d.ts.map +1 -0
- package/dist/views/page-builder/canvas/index.js +2 -0
- package/dist/views/page-builder/canvas/index.js.map +1 -0
- package/package.json +3 -2
- package/src/AdminRoot.tsx +16 -0
- package/src/components/ErrorBoundary.tsx +3 -3
- package/src/hooks/useBuilderState.ts +328 -0
- package/src/index.ts +4 -0
- package/src/layout/Sidebar.tsx +5 -0
- package/src/views/page-builder/AIBlockAssist.tsx +68 -0
- package/src/views/page-builder/AIGenerateDialog.tsx +574 -0
- package/src/views/page-builder/BlockEditor.tsx +352 -0
- package/src/views/page-builder/BlockPicker.tsx +338 -0
- package/src/views/page-builder/BottomBar.tsx +64 -0
- package/src/views/page-builder/BuilderToolbar.tsx +218 -0
- package/src/views/page-builder/ContextPanel.tsx +145 -0
- package/src/views/page-builder/DesignScore.tsx +258 -0
- package/src/views/page-builder/NodeSettings.tsx +515 -0
- package/src/views/page-builder/PageBuilder.tsx +288 -0
- package/src/views/page-builder/PageSettings.tsx +161 -0
- package/src/views/page-builder/SEOPanel.tsx +485 -0
- package/src/views/page-builder/SavedSections.tsx +486 -0
- package/src/views/page-builder/TemplatePicker.tsx +201 -0
- package/src/views/page-builder/block-renderers/CTAPreview.tsx +81 -0
- package/src/views/page-builder/block-renderers/CardsPreview.tsx +71 -0
- package/src/views/page-builder/block-renderers/CodePreview.tsx +46 -0
- package/src/views/page-builder/block-renderers/FAQPreview.tsx +90 -0
- package/src/views/page-builder/block-renderers/FallbackPreview.tsx +18 -0
- package/src/views/page-builder/block-renderers/FormPreview.tsx +69 -0
- package/src/views/page-builder/block-renderers/GalleryPreview.tsx +93 -0
- package/src/views/page-builder/block-renderers/HeroPreview.tsx +103 -0
- package/src/views/page-builder/block-renderers/ImagePreview.tsx +54 -0
- package/src/views/page-builder/block-renderers/TextPreview.tsx +81 -0
- package/src/views/page-builder/block-renderers/VideoPreview.tsx +78 -0
- package/src/views/page-builder/block-renderers/index.ts +34 -0
- package/src/views/page-builder/canvas/BlockRenderer.tsx +62 -0
- package/src/views/page-builder/canvas/BuilderCanvas.tsx +90 -0
- package/src/views/page-builder/canvas/ColumnRenderer.tsx +86 -0
- package/src/views/page-builder/canvas/ContainerRenderer.tsx +71 -0
- package/src/views/page-builder/canvas/RowRenderer.tsx +72 -0
- package/src/views/page-builder/canvas/SectionRenderer.tsx +97 -0
- package/src/views/page-builder/canvas/index.ts +2 -0
|
@@ -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 @@
|
|
|
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 @@
|
|
|
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.
|
|
3
|
+
"version": "0.7.0",
|
|
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.9.0"
|
|
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,8 @@ 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';
|
|
26
28
|
|
|
27
29
|
export interface AdminRootProps {
|
|
28
30
|
config: any;
|
|
@@ -99,6 +101,16 @@ function AdminShell({ config, session, basePath = '/admin', initialPath = '/', s
|
|
|
99
101
|
return <Dashboard config={config} session={session} onNavigate={navigate} />;
|
|
100
102
|
}
|
|
101
103
|
|
|
104
|
+
const pageBuilderEdit = matchRoute('/page-builder/:id');
|
|
105
|
+
if (pageBuilderEdit?.id) {
|
|
106
|
+
return <PageBuilder documentId={pageBuilderEdit.id} collectionSlug="pages" config={config} onNavigate={navigate} />;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
const pageBuilderNew = matchRoute('/page-builder/new');
|
|
110
|
+
if (pageBuilderNew) {
|
|
111
|
+
return <PageBuilder collectionSlug="pages" config={config} onNavigate={navigate} />;
|
|
112
|
+
}
|
|
113
|
+
|
|
102
114
|
for (const slug of collectionSlugs) {
|
|
103
115
|
const newMatch = matchRoute(`/${slug}/new`);
|
|
104
116
|
if (newMatch) {
|
|
@@ -171,6 +183,10 @@ function AdminShell({ config, session, basePath = '/admin', initialPath = '/', s
|
|
|
171
183
|
return <ScriptTags onNavigate={navigate} />;
|
|
172
184
|
}
|
|
173
185
|
|
|
186
|
+
if (matchRoute('/saved-sections')) {
|
|
187
|
+
return <SavedSections onNavigate={navigate} config={config} />;
|
|
188
|
+
}
|
|
189
|
+
|
|
174
190
|
if (matchRoute('/users')) {
|
|
175
191
|
return <Users onNavigate={navigate} />;
|
|
176
192
|
}
|
|
@@ -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-
|
|
36
|
-
<p className="text-sm text-
|
|
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-
|
|
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';
|
package/src/layout/Sidebar.tsx
CHANGED
|
@@ -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 },
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import { useState, useCallback } from 'react';
|
|
4
|
+
import { Sparkles, Loader2 } from 'lucide-react';
|
|
5
|
+
import { toast } from 'sonner';
|
|
6
|
+
import { cmsApi } from '../../lib/api.js';
|
|
7
|
+
import type { BlockNode } from '@actuate-media/cms-core';
|
|
8
|
+
|
|
9
|
+
export interface AIBlockAssistProps {
|
|
10
|
+
block: BlockNode;
|
|
11
|
+
pageTitle?: string;
|
|
12
|
+
businessName?: string;
|
|
13
|
+
onUpdateData: (data: Record<string, unknown>) => void;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export function AIBlockAssist({
|
|
17
|
+
block,
|
|
18
|
+
pageTitle,
|
|
19
|
+
businessName,
|
|
20
|
+
onUpdateData,
|
|
21
|
+
}: AIBlockAssistProps) {
|
|
22
|
+
const [generating, setGenerating] = useState(false);
|
|
23
|
+
|
|
24
|
+
const handleGenerate = useCallback(async () => {
|
|
25
|
+
setGenerating(true);
|
|
26
|
+
|
|
27
|
+
try {
|
|
28
|
+
const res = await cmsApi<Record<string, unknown>>('/page-builder/generate-block', {
|
|
29
|
+
method: 'POST',
|
|
30
|
+
body: JSON.stringify({
|
|
31
|
+
blockType: block.settings.blockType,
|
|
32
|
+
variant: block.settings.variant,
|
|
33
|
+
pageContext: {
|
|
34
|
+
pageTitle,
|
|
35
|
+
businessName,
|
|
36
|
+
},
|
|
37
|
+
}),
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
if (res.error) {
|
|
41
|
+
toast.error(res.error);
|
|
42
|
+
} else if (res.data) {
|
|
43
|
+
onUpdateData(res.data);
|
|
44
|
+
toast.success('Content generated');
|
|
45
|
+
}
|
|
46
|
+
} catch {
|
|
47
|
+
toast.error('Failed to generate content');
|
|
48
|
+
} finally {
|
|
49
|
+
setGenerating(false);
|
|
50
|
+
}
|
|
51
|
+
}, [block, pageTitle, businessName, onUpdateData]);
|
|
52
|
+
|
|
53
|
+
return (
|
|
54
|
+
<button
|
|
55
|
+
onClick={handleGenerate}
|
|
56
|
+
disabled={generating}
|
|
57
|
+
className="flex items-center gap-1.5 px-2.5 py-1 text-xs text-primary hover:bg-primary/10 rounded-md transition-colors disabled:opacity-50"
|
|
58
|
+
title="Generate content with AI"
|
|
59
|
+
>
|
|
60
|
+
{generating ? (
|
|
61
|
+
<Loader2 size={12} className="animate-spin" />
|
|
62
|
+
) : (
|
|
63
|
+
<Sparkles size={12} />
|
|
64
|
+
)}
|
|
65
|
+
{generating ? 'Generating...' : 'AI Fill'}
|
|
66
|
+
</button>
|
|
67
|
+
);
|
|
68
|
+
}
|