site-builder: dynamic CTAs, section anchors, edit-with-Sitesmith

Three related features:

1. Dynamic CTA buttons on HeroSimple, CTASection, CallToAction.
   New shared ctas[] array (text + href + variant + target) replaces the
   primary/secondary pair. Settings panel gets add/remove/reorder controls.
   Legacy fields stay readable for backwards compat — first user edit
   migrates the section onto the new array.

2. Anchor IDs on all layout/section components (Container, Section,
   BackgroundSection, ColumnLayout, plus 6 section blocks done by parallel
   subagent, plus Hero/CTA/CallToAction). Anchor input lives in the
   settings panel with an "auto from heading" button that walks the
   subtree for the first Heading.text. Renders as id="..." on the
   outermost element so #anchor URLs resolve.

3. Edit-with-Sitesmith targeted invocation. Right-click → "Ask Sitesmith"
   and a button at the top of the right-side settings panel both open the
   modal pre-targeted at the selected node. The node's serialized subtree
   is sent to the server; system prompt is augmented to require a patch
   with replace_node. Editor lifts modal state into a new SitesmithContext.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-05-25 12:43:28 -07:00
parent 7b747f775f
commit d0925d9e2d
24 changed files with 723 additions and 237 deletions

View File

@@ -6,18 +6,29 @@ import { WhpConfig } from './types';
import { EditorConfigProvider } from './state/EditorConfigContext'; import { EditorConfigProvider } from './state/EditorConfigContext';
import { SiteDesignProvider } from './state/SiteDesignContext'; import { SiteDesignProvider } from './state/SiteDesignContext';
import { PageProvider } from './state/PageContext'; import { PageProvider } from './state/PageContext';
import { SitesmithProvider, useSitesmithModal } from './state/SitesmithContext';
import { SitesmithModal } from './panels/sitesmith/SitesmithModal';
interface AppProps { interface AppProps {
whpConfig: WhpConfig | null; whpConfig: WhpConfig | null;
} }
const SitesmithModalMount: React.FC = () => {
const { isOpen, target, close } = useSitesmithModal();
if (!isOpen) return null;
return <SitesmithModal target={target} onClose={close} />;
};
export const App: React.FC<AppProps> = ({ whpConfig }) => { export const App: React.FC<AppProps> = ({ whpConfig }) => {
return ( return (
<EditorConfigProvider config={whpConfig}> <EditorConfigProvider config={whpConfig}>
<SiteDesignProvider> <SiteDesignProvider>
<Editor resolver={componentResolver} enabled={true}> <Editor resolver={componentResolver} enabled={true}>
<PageProvider> <PageProvider>
<SitesmithProvider>
<EditorShell /> <EditorShell />
<SitesmithModalMount />
</SitesmithProvider>
</PageProvider> </PageProvider>
</Editor> </Editor>
</SiteDesignProvider> </SiteDesignProvider>

View File

@@ -2,6 +2,7 @@ import React, { CSSProperties } from 'react';
import { useNode, Element, UserComponent } from '@craftjs/core'; import { useNode, Element, UserComponent } from '@craftjs/core';
import { Container } from './Container'; import { Container } from './Container';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface BackgroundSectionProps { interface BackgroundSectionProps {
bgImage?: string; bgImage?: string;
@@ -11,6 +12,7 @@ interface BackgroundSectionProps {
innerMaxWidth?: string; innerMaxWidth?: string;
style?: CSSProperties; style?: CSSProperties;
children?: React.ReactNode; children?: React.ReactNode;
anchorId?: string;
} }
export const BackgroundSection: UserComponent<BackgroundSectionProps> = ({ export const BackgroundSection: UserComponent<BackgroundSectionProps> = ({
@@ -20,12 +22,14 @@ export const BackgroundSection: UserComponent<BackgroundSectionProps> = ({
overlayOpacity = 0.4, overlayOpacity = 0.4,
innerMaxWidth = '1200px', innerMaxWidth = '1200px',
style = {}, style = {},
anchorId,
}) => { }) => {
const { connectors: { connect, drag } } = useNode(); const { connectors: { connect, drag } } = useNode();
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
position: 'relative', position: 'relative',
width: '100%', width: '100%',
@@ -77,6 +81,7 @@ const BackgroundSectionSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Background Image URL</label> <label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Background Image URL</label>
<input <input
@@ -162,6 +167,7 @@ BackgroundSection.craft = {
overlayOpacity: 0.4, overlayOpacity: 0.4,
innerMaxWidth: '1200px', innerMaxWidth: '1200px',
style: { padding: '0' }, style: { padding: '0' },
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -176,6 +182,7 @@ BackgroundSection.craft = {
/* ---------- HTML export ---------- */ /* ---------- HTML export ---------- */
(BackgroundSection as any).toHtml = (props: BackgroundSectionProps, childrenHtml: string) => { (BackgroundSection as any).toHtml = (props: BackgroundSectionProps, childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const outerStyle = cssPropsToString({ const outerStyle = cssPropsToString({
position: 'relative', position: 'relative',
width: '100%', width: '100%',
@@ -200,7 +207,8 @@ BackgroundSection.craft = {
margin: '0 auto', margin: '0 auto',
padding: '60px 20px', padding: '60px 20px',
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
return { return {
html: `<section${outerStyle ? ` style="${outerStyle}"` : ''}><div${overlayStyle ? ` style="${overlayStyle}"` : ''}></div><div${innerStyle ? ` style="${innerStyle}"` : ''}>${childrenHtml}</div></section>`, html: `<section${idAttr}${outerStyle ? ` style="${outerStyle}"` : ''}><div${overlayStyle ? ` style="${overlayStyle}"` : ''}></div><div${innerStyle ? ` style="${innerStyle}"` : ''}>${childrenHtml}</div></section>`,
}; };
}; };

View File

@@ -2,6 +2,7 @@ import React, { CSSProperties, useState } from 'react';
import { useNode, Element, UserComponent } from '@craftjs/core'; import { useNode, Element, UserComponent } from '@craftjs/core';
import { Container } from './Container'; import { Container } from './Container';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
type SplitOption = type SplitOption =
| '100' | '100'
@@ -18,6 +19,7 @@ interface ColumnLayoutProps {
gap?: string; gap?: string;
style?: CSSProperties; style?: CSSProperties;
children?: React.ReactNode; children?: React.ReactNode;
anchorId?: string;
} }
const splitToWidths: Record<string, string[]> = { const splitToWidths: Record<string, string[]> = {
@@ -59,6 +61,7 @@ export const ColumnLayout: UserComponent<ColumnLayoutProps> = ({
split = '50-50', split = '50-50',
gap = '16px', gap = '16px',
style = {}, style = {},
anchorId,
}) => { }) => {
const { connectors: { connect, drag } } = useNode(); const { connectors: { connect, drag } } = useNode();
const widths = getWidths(split, columns); const widths = getWidths(split, columns);
@@ -66,6 +69,7 @@ export const ColumnLayout: UserComponent<ColumnLayoutProps> = ({
return ( return (
<div <div
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
display: 'flex', display: 'flex',
flexWrap: 'wrap', flexWrap: 'wrap',
@@ -124,6 +128,7 @@ const ColumnLayoutSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
{/* Preset layouts */} {/* Preset layouts */}
<div> <div>
<label style={labelStyle}>Column Layout</label> <label style={labelStyle}>Column Layout</label>
@@ -270,6 +275,7 @@ ColumnLayout.craft = {
split: '50-50', split: '50-50',
gap: '16px', gap: '16px',
style: {}, style: {},
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -284,6 +290,7 @@ ColumnLayout.craft = {
/* ---------- HTML export ---------- */ /* ---------- HTML export ---------- */
(ColumnLayout as any).toHtml = (props: ColumnLayoutProps, childrenHtml: string) => { (ColumnLayout as any).toHtml = (props: ColumnLayoutProps, childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const gap = props.gap || '16px'; const gap = props.gap || '16px';
const outerStyle = cssPropsToString({ const outerStyle = cssPropsToString({
display: 'flex', display: 'flex',
@@ -292,7 +299,8 @@ ColumnLayout.craft = {
width: '100%', width: '100%',
...props.style, ...props.style,
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
return { return {
html: `<div${outerStyle ? ` style="${outerStyle}"` : ''}>${childrenHtml}</div>`, html: `<div${idAttr}${outerStyle ? ` style="${outerStyle}"` : ''}>${childrenHtml}</div>`,
}; };
}; };

View File

@@ -4,6 +4,7 @@ import { cssPropsToString } from '../../utils/style-helpers';
import { SettingsTabs } from '../../ui/SettingsTabs'; import { SettingsTabs } from '../../ui/SettingsTabs';
import { BorderControl } from '../../ui/BorderControl'; import { BorderControl } from '../../ui/BorderControl';
import { AdvancedTab } from '../../ui/AdvancedTab'; import { AdvancedTab } from '../../ui/AdvancedTab';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface ContainerProps { interface ContainerProps {
style?: CSSProperties; style?: CSSProperties;
@@ -11,6 +12,7 @@ interface ContainerProps {
children?: React.ReactNode; children?: React.ReactNode;
cssId?: string; cssId?: string;
cssClass?: string; cssClass?: string;
anchorId?: string;
hideOnDesktop?: boolean; hideOnDesktop?: boolean;
hideOnTablet?: boolean; hideOnTablet?: boolean;
hideOnMobile?: boolean; hideOnMobile?: boolean;
@@ -37,6 +39,7 @@ export const Container: UserComponent<ContainerProps> = ({
children, children,
fullWidth = false, fullWidth = false,
contentWidth = 'full', contentWidth = 'full',
anchorId,
}) => { }) => {
const { connectors: { connect, drag } } = useNode(); const { connectors: { connect, drag } } = useNode();
@@ -56,6 +59,7 @@ export const Container: UserComponent<ContainerProps> = ({
ref: (ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }, ref: (ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); },
style: outerStyle, style: outerStyle,
'data-craft-container': 'true', 'data-craft-container': 'true',
id: anchorId || undefined,
}, },
needsBoxedWrapper needsBoxedWrapper
? React.createElement('div', { style: { maxWidth: '1200px', margin: '0 auto', ...flexStyles } }, children) ? React.createElement('div', { style: { maxWidth: '1200px', margin: '0 auto', ...flexStyles } }, children)
@@ -120,6 +124,7 @@ const ContainerSettings: React.FC = () => {
<SettingsTabs <SettingsTabs
general={ general={
<div style={{ display: 'flex', flexDirection: 'column', gap: 14 }}> <div style={{ display: 'flex', flexDirection: 'column', gap: 14 }}>
<AnchorIdField />
{/* Tag */} {/* Tag */}
<div> <div>
<label style={cLabelStyle}>HTML Element</label> <label style={cLabelStyle}>HTML Element</label>
@@ -304,6 +309,7 @@ Container.craft = {
tag: 'div', tag: 'div',
fullWidth: false, fullWidth: false,
contentWidth: 'full', contentWidth: 'full',
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -318,6 +324,7 @@ Container.craft = {
/* ---------- HTML export ---------- */ /* ---------- HTML export ---------- */
(Container as any).toHtml = (props: ContainerProps, childrenHtml: string) => { (Container as any).toHtml = (props: ContainerProps, childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const tag = props.tag || 'div'; const tag = props.tag || 'div';
const isBoxed = props.contentWidth === 'boxed'; const isBoxed = props.contentWidth === 'boxed';
const flexStyles = flexAlignFromTextAlign(props.style?.textAlign); const flexStyles = flexAlignFromTextAlign(props.style?.textAlign);
@@ -333,11 +340,12 @@ Container.craft = {
} }
const styleStr = cssPropsToString(outerCss); const styleStr = cssPropsToString(outerCss);
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
if (isBoxed) { if (isBoxed) {
const innerStyle = cssPropsToString({ maxWidth: '1200px', margin: '0 auto', ...flexStyles }); const innerStyle = cssPropsToString({ maxWidth: '1200px', margin: '0 auto', ...flexStyles });
return { html: `<${tag}${styleStr ? ` style="${styleStr}"` : ''}><div${innerStyle ? ` style="${innerStyle}"` : ''}>${childrenHtml}</div></${tag}>` }; return { html: `<${tag}${idAttr}${styleStr ? ` style="${styleStr}"` : ''}><div${innerStyle ? ` style="${innerStyle}"` : ''}>${childrenHtml}</div></${tag}>` };
} }
return { html: `<${tag}${styleStr ? ` style="${styleStr}"` : ''}>${childrenHtml}</${tag}>` }; return { html: `<${tag}${idAttr}${styleStr ? ` style="${styleStr}"` : ''}>${childrenHtml}</${tag}>` };
}; };

View File

@@ -2,6 +2,7 @@ import React, { CSSProperties } from 'react';
import { useNode, Element, UserComponent } from '@craftjs/core'; import { useNode, Element, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { Container } from './Container'; import { Container } from './Container';
import { AnchorIdField } from '../../ui/AnchorIdField';
/* ---------- Shape Divider SVG Paths ---------- */ /* ---------- Shape Divider SVG Paths ---------- */
@@ -27,6 +28,7 @@ interface SectionProps {
bottomDivider?: DividerShape; bottomDivider?: DividerShape;
bottomDividerColor?: string; bottomDividerColor?: string;
bottomDividerHeight?: string; bottomDividerHeight?: string;
anchorId?: string;
} }
/* ---------- Divider renderer ---------- */ /* ---------- Divider renderer ---------- */
@@ -85,6 +87,7 @@ export const Section: UserComponent<SectionProps> = ({
bottomDivider = 'none', bottomDivider = 'none',
bottomDividerColor = '#ffffff', bottomDividerColor = '#ffffff',
bottomDividerHeight = '50px', bottomDividerHeight = '50px',
anchorId,
}) => { }) => {
const { connectors: { connect, drag } } = useNode(); const { connectors: { connect, drag } } = useNode();
@@ -94,6 +97,7 @@ export const Section: UserComponent<SectionProps> = ({
return ( return (
<section <section
ref={(ref: HTMLElement | null) => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null) => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
width: '100%', width: '100%',
position: (hasTopDivider || hasBottomDivider) ? 'relative' : undefined, position: (hasTopDivider || hasBottomDivider) ? 'relative' : undefined,
@@ -229,6 +233,7 @@ const SectionSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Background Color</label> <label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Background Color</label>
<div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}> <div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}>
@@ -333,6 +338,7 @@ Section.craft = {
bottomDivider: 'none', bottomDivider: 'none',
bottomDividerColor: '#ffffff', bottomDividerColor: '#ffffff',
bottomDividerHeight: '50px', bottomDividerHeight: '50px',
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -377,6 +383,7 @@ function buildDividerHtml(
} }
(Section as any).toHtml = (props: SectionProps, childrenHtml: string) => { (Section as any).toHtml = (props: SectionProps, childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const hasTopDivider = props.topDivider && props.topDivider !== 'none'; const hasTopDivider = props.topDivider && props.topDivider !== 'none';
const hasBottomDivider = props.bottomDivider && props.bottomDivider !== 'none'; const hasBottomDivider = props.bottomDivider && props.bottomDivider !== 'none';
@@ -394,8 +401,9 @@ function buildDividerHtml(
const topHtml = buildDividerHtml(props.topDivider, props.topDividerColor, props.topDividerHeight, 'top'); const topHtml = buildDividerHtml(props.topDivider, props.topDividerColor, props.topDividerHeight, 'top');
const bottomHtml = buildDividerHtml(props.bottomDivider, props.bottomDividerColor, props.bottomDividerHeight, 'bottom'); const bottomHtml = buildDividerHtml(props.bottomDivider, props.bottomDividerColor, props.bottomDividerHeight, 'bottom');
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
return { return {
html: `<section${outerStyle ? ` style="${outerStyle}"` : ''}>${topHtml}<div${innerStyle ? ` style="${innerStyle}"` : ''}>${childrenHtml}</div>${bottomHtml}</section>`, html: `<section${idAttr}${outerStyle ? ` style="${outerStyle}"` : ''}>${topHtml}<div${innerStyle ? ` style="${innerStyle}"` : ''}>${childrenHtml}</div>${bottomHtml}</section>`,
}; };
}; };

View File

@@ -1,6 +1,7 @@
import React, { CSSProperties, useState } from 'react'; import React, { CSSProperties, useState } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface AccordionItem { interface AccordionItem {
title: string; title: string;
@@ -15,6 +16,7 @@ interface AccordionProps {
headerColor?: string; headerColor?: string;
contentBg?: string; contentBg?: string;
borderColor?: string; borderColor?: string;
anchorId?: string;
} }
const defaultItems: AccordionItem[] = [ const defaultItems: AccordionItem[] = [
@@ -30,6 +32,7 @@ export const Accordion: UserComponent<AccordionProps> = ({
headerColor = '#18181b', headerColor = '#18181b',
contentBg = '#ffffff', contentBg = '#ffffff',
borderColor = '#e2e8f0', borderColor = '#e2e8f0',
anchorId,
}) => { }) => {
const { const {
connectors: { connect, drag }, connectors: { connect, drag },
@@ -56,6 +59,7 @@ export const Accordion: UserComponent<AccordionProps> = ({
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
padding: '60px 20px', padding: '60px 20px',
backgroundColor: '#ffffff', backgroundColor: '#ffffff',
@@ -161,6 +165,7 @@ const AccordionSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={labelStyle}>Header Background</label> <label style={labelStyle}>Header Background</label>
<div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}> <div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}>
@@ -279,6 +284,7 @@ Accordion.craft = {
headerColor: '#18181b', headerColor: '#18181b',
contentBg: '#ffffff', contentBg: '#ffffff',
borderColor: '#e2e8f0', borderColor: '#e2e8f0',
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -293,11 +299,12 @@ Accordion.craft = {
/* ---------- HTML export ---------- */ /* ---------- HTML export ---------- */
(Accordion as any).toHtml = (props: AccordionProps, _childrenHtml: string) => { (Accordion as any).toHtml = (props: AccordionProps, _childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;'); const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const sectionStyle = cssPropsToString({ const sectionStyle = cssPropsToString({
padding: '60px 20px', padding: '60px 20px',
...props.style, ...props.style,
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
const headerBg = props.headerBg || '#f8fafc'; const headerBg = props.headerBg || '#f8fafc';
const headerColor = props.headerColor || '#18181b'; const headerColor = props.headerColor || '#18181b';
const contentBg = props.contentBg || '#ffffff'; const contentBg = props.contentBg || '#ffffff';
@@ -320,7 +327,7 @@ Accordion.craft = {
}).join('\n '); }).join('\n ');
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
<div style="max-width:800px;margin:0 auto;display:flex;flex-direction:column"> <div style="max-width:800px;margin:0 auto;display:flex;flex-direction:column">
${panels} ${panels}
</div> </div>

View File

@@ -1,13 +1,18 @@
import React, { CSSProperties } from 'react'; import React, { CSSProperties } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { CtaButton, CtasEditor, normalizeCtas, ctaInlineStyle, ctasToHtml } from './_cta-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface CTASectionProps { interface CTASectionProps {
heading?: string; heading?: string;
description?: string; description?: string;
ctas?: CtaButton[];
/** Legacy props kept for backward compat with saved projects. */
buttonText?: string; buttonText?: string;
buttonHref?: string; buttonHref?: string;
gradient?: string; gradient?: string;
anchorId?: string;
style?: CSSProperties; style?: CSSProperties;
} }
@@ -16,9 +21,11 @@ const defaultGradient = 'linear-gradient(135deg, #2563eb 0%, #7c3aed 100%)';
export const CTASection: UserComponent<CTASectionProps> = ({ export const CTASection: UserComponent<CTASectionProps> = ({
heading = 'Ready to Get Started?', heading = 'Ready to Get Started?',
description = 'Join thousands of satisfied users and start building your dream website today.', description = 'Join thousands of satisfied users and start building your dream website today.',
buttonText = 'Start Free Trial', ctas,
buttonHref = '#', buttonText,
buttonHref,
gradient = defaultGradient, gradient = defaultGradient,
anchorId,
style = {}, style = {},
}) => { }) => {
const { const {
@@ -28,9 +35,13 @@ export const CTASection: UserComponent<CTASectionProps> = ({
selected: node.events.selected, selected: node.events.selected,
})); }));
const effectiveCtas = normalizeCtas({ ctas, buttonText, buttonHref });
const ctaDefaults = { primaryBg: '#ffffff', primaryText: '#18181b', outlineText: '#ffffff' };
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
background: gradient, background: gradient,
padding: '80px 20px', padding: '80px 20px',
@@ -46,22 +57,14 @@ export const CTASection: UserComponent<CTASectionProps> = ({
<p style={{ fontSize: '18px', color: 'rgba(255,255,255,0.85)', marginBottom: '28px', lineHeight: '1.6' }}> <p style={{ fontSize: '18px', color: 'rgba(255,255,255,0.85)', marginBottom: '28px', lineHeight: '1.6' }}>
{description} {description}
</p> </p>
<a <div style={{ display: 'flex', gap: '12px', justifyContent: 'center', flexWrap: 'wrap' }}>
href={buttonHref} {effectiveCtas.map((cta, i) => (
onClick={(e) => e.preventDefault()} <a key={i} href={cta.href || '#'} onClick={(e) => e.preventDefault()}
style={{ style={ctaInlineStyle(cta, ctaDefaults)}>
display: 'inline-block', {cta.text}
padding: '14px 36px',
backgroundColor: '#ffffff',
color: '#18181b',
textDecoration: 'none',
borderRadius: '8px',
fontWeight: '600',
fontSize: '16px',
}}
>
{buttonText}
</a> </a>
))}
</div>
</div> </div>
</section> </section>
); );
@@ -83,8 +86,11 @@ const CTASectionSettings: React.FC = () => {
{ label: 'Ocean', value: 'linear-gradient(135deg, #0ea5e9 0%, #6366f1 100%)' }, { label: 'Ocean', value: 'linear-gradient(135deg, #0ea5e9 0%, #6366f1 100%)' },
]; ];
const effectiveCtas = normalizeCtas(props);
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Heading</label> <label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Heading</label>
<input <input
@@ -105,26 +111,14 @@ const CTASectionSettings: React.FC = () => {
/> />
</div> </div>
<div> <CtasEditor
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Button Text</label> ctas={effectiveCtas}
<input onChange={(next) => setProp((p: CTASectionProps) => {
type="text" p.ctas = next;
value={props.buttonText || ''} p.buttonText = undefined;
onChange={(e) => setProp((p: CTASectionProps) => { p.buttonText = e.target.value; })} p.buttonHref = undefined;
style={{ width: '100%', padding: '4px 8px', background: '#27272a', color: '#e4e4e7', border: '1px solid #3f3f46', borderRadius: 4, fontSize: 12 }} })}
/> />
</div>
<div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Button URL</label>
<input
type="text"
value={props.buttonHref || ''}
onChange={(e) => setProp((p: CTASectionProps) => { p.buttonHref = e.target.value; })}
placeholder="https://..."
style={{ width: '100%', padding: '4px 8px', background: '#27272a', color: '#e4e4e7', border: '1px solid #3f3f46', borderRadius: 4, fontSize: 12 }}
/>
</div>
<div> <div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Gradient</label> <label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Gradient</label>
@@ -155,9 +149,11 @@ CTASection.craft = {
props: { props: {
heading: 'Ready to Get Started?', heading: 'Ready to Get Started?',
description: 'Join thousands of satisfied users and start building your dream website today.', description: 'Join thousands of satisfied users and start building your dream website today.',
buttonText: 'Start Free Trial', ctas: [
buttonHref: '#', { text: 'Start Free Trial', href: '#', variant: 'primary' },
] as CtaButton[],
gradient: defaultGradient, gradient: defaultGradient,
anchorId: '',
style: {}, style: {},
}, },
rules: { rules: {
@@ -180,12 +176,15 @@ CTASection.craft = {
textAlign: 'center', textAlign: 'center',
...props.style, ...props.style,
}); });
const ctas = normalizeCtas(props);
const buttonsHtml = ctasToHtml(ctas, { primaryBg: '#ffffff', primaryText: '#18181b', outlineText: '#ffffff' });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
<div style="max-width:700px;margin:0 auto"> <div style="max-width:700px;margin:0 auto">
<h2 style="font-size:36px;font-weight:700;color:#ffffff;margin-bottom:12px">${esc(props.heading || '')}</h2> <h2 style="font-size:36px;font-weight:700;color:#ffffff;margin-bottom:12px">${esc(props.heading || '')}</h2>
<p style="font-size:18px;color:rgba(255,255,255,0.85);margin-bottom:28px;line-height:1.6">${esc(props.description || '')}</p> <p style="font-size:18px;color:rgba(255,255,255,0.85);margin-bottom:28px;line-height:1.6">${esc(props.description || '')}</p>
<a href="${props.buttonHref || '#'}" style="display:inline-block;padding:14px 36px;background-color:#ffffff;color:#18181b;text-decoration:none;border-radius:8px;font-weight:600;font-size:16px">${esc(props.buttonText || '')}</a> <div style="display:flex;gap:12px;justify-content:center;flex-wrap:wrap">${buttonsHtml}</div>
</div> </div>
</section>`, </section>`,
}; };

View File

@@ -1,10 +1,14 @@
import React, { CSSProperties } from 'react'; import React, { CSSProperties } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { CtaButton, CtasEditor, normalizeCtas, ctaInlineStyle, ctasToHtml } from './_cta-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface CallToActionProps { interface CallToActionProps {
heading?: string; heading?: string;
description?: string; description?: string;
ctas?: CtaButton[];
/** Legacy props kept for backward compat with saved projects. */
buttonText?: string; buttonText?: string;
buttonHref?: string; buttonHref?: string;
secondaryButtonText?: string; secondaryButtonText?: string;
@@ -15,6 +19,7 @@ interface CallToActionProps {
overlayOpacity?: number; overlayOpacity?: number;
textColor?: string; textColor?: string;
buttonColor?: string; buttonColor?: string;
anchorId?: string;
style?: CSSProperties; style?: CSSProperties;
} }
@@ -23,16 +28,18 @@ const defaultGradient = 'linear-gradient(135deg, #2563eb 0%, #7c3aed 100%)';
export const CallToAction: UserComponent<CallToActionProps> = ({ export const CallToAction: UserComponent<CallToActionProps> = ({
heading = 'Ready to Get Started?', heading = 'Ready to Get Started?',
description = 'Join thousands of satisfied users and start building your dream website today.', description = 'Join thousands of satisfied users and start building your dream website today.',
buttonText = 'Get Started', ctas,
buttonHref = '#', buttonText,
secondaryButtonText = '', buttonHref,
secondaryButtonHref = '#', secondaryButtonText,
secondaryButtonHref,
bgType = 'gradient', bgType = 'gradient',
bgValue = defaultGradient, bgValue = defaultGradient,
overlayColor = '#000000', overlayColor = '#000000',
overlayOpacity = 0, overlayOpacity = 0,
textColor = '#ffffff', textColor = '#ffffff',
buttonColor = '#ffffff', buttonColor = '#ffffff',
anchorId,
style = {}, style = {},
}) => { }) => {
const { const {
@@ -56,9 +63,13 @@ export const CallToAction: UserComponent<CallToActionProps> = ({
const isButtonDark = buttonColor === '#ffffff' || buttonColor === '#f8fafc'; const isButtonDark = buttonColor === '#ffffff' || buttonColor === '#f8fafc';
const buttonTextColor = isButtonDark ? '#18181b' : '#ffffff'; const buttonTextColor = isButtonDark ? '#18181b' : '#ffffff';
const effectiveCtas = normalizeCtas({ ctas, buttonText, buttonHref, secondaryButtonText, secondaryButtonHref });
const ctaDefaults = { primaryBg: buttonColor, primaryText: buttonTextColor, outlineText: textColor };
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
position: 'relative', position: 'relative',
padding: '80px 20px', padding: '80px 20px',
@@ -89,41 +100,12 @@ export const CallToAction: UserComponent<CallToActionProps> = ({
{description} {description}
</p> </p>
<div style={{ display: 'flex', gap: '12px', justifyContent: 'center', flexWrap: 'wrap' }}> <div style={{ display: 'flex', gap: '12px', justifyContent: 'center', flexWrap: 'wrap' }}>
<a {effectiveCtas.map((cta, i) => (
href={buttonHref} <a key={i} href={cta.href || '#'} onClick={(e) => e.preventDefault()}
onClick={(e) => e.preventDefault()} style={ctaInlineStyle(cta, ctaDefaults)}>
style={{ {cta.text}
display: 'inline-block',
padding: '14px 36px',
backgroundColor: buttonColor,
color: buttonTextColor,
textDecoration: 'none',
borderRadius: '8px',
fontWeight: '600',
fontSize: '16px',
}}
>
{buttonText}
</a> </a>
{secondaryButtonText && ( ))}
<a
href={secondaryButtonHref}
onClick={(e) => e.preventDefault()}
style={{
display: 'inline-block',
padding: '14px 36px',
backgroundColor: 'transparent',
color: textColor,
textDecoration: 'none',
borderRadius: '8px',
fontWeight: '600',
fontSize: '16px',
border: `2px solid ${textColor}`,
}}
>
{secondaryButtonText}
</a>
)}
</div> </div>
</div> </div>
</section> </section>
@@ -155,8 +137,11 @@ const CallToActionSettings: React.FC = () => {
border: '1px solid #3f3f46', borderRadius: 4, fontSize: 12, border: '1px solid #3f3f46', borderRadius: 4, fontSize: 12,
}; };
const effectiveCtas = normalizeCtas(props);
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Heading</label> <label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Heading</label>
<input <input
@@ -177,52 +162,16 @@ const CallToActionSettings: React.FC = () => {
/> />
</div> </div>
{/* Primary Button */} <CtasEditor
<div> ctas={effectiveCtas}
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Primary Button Text</label> onChange={(next) => setProp((p: CallToActionProps) => {
<input p.ctas = next;
type="text" p.buttonText = undefined;
value={props.buttonText || ''} p.buttonHref = undefined;
onChange={(e) => setProp((p: CallToActionProps) => { p.buttonText = e.target.value; })} p.secondaryButtonText = undefined;
style={inputStyle} p.secondaryButtonHref = undefined;
})}
/> />
</div>
<div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Primary Button URL</label>
<input
type="text"
value={props.buttonHref || ''}
onChange={(e) => setProp((p: CallToActionProps) => { p.buttonHref = e.target.value; })}
placeholder="https://..."
style={inputStyle}
/>
</div>
{/* Secondary Button */}
<div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Secondary Button Text <span style={{ opacity: 0.5 }}>(leave empty to hide)</span></label>
<input
type="text"
value={props.secondaryButtonText || ''}
onChange={(e) => setProp((p: CallToActionProps) => { p.secondaryButtonText = e.target.value; })}
placeholder="e.g. Learn More"
style={inputStyle}
/>
</div>
{props.secondaryButtonText && (
<div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Secondary Button URL</label>
<input
type="text"
value={props.secondaryButtonHref || ''}
onChange={(e) => setProp((p: CallToActionProps) => { p.secondaryButtonHref = e.target.value; })}
placeholder="https://..."
style={inputStyle}
/>
</div>
)}
{/* Background Type */} {/* Background Type */}
<div> <div>
@@ -380,10 +329,11 @@ CallToAction.craft = {
props: { props: {
heading: 'Ready to Get Started?', heading: 'Ready to Get Started?',
description: 'Join thousands of satisfied users and start building your dream website today.', description: 'Join thousands of satisfied users and start building your dream website today.',
buttonText: 'Get Started', ctas: [
buttonHref: '#', { text: 'Get Started', href: '#', variant: 'primary' },
secondaryButtonText: 'Learn More', { text: 'Learn More', href: '#', variant: 'outline' },
secondaryButtonHref: '#', ] as CtaButton[],
anchorId: '',
bgType: 'gradient', bgType: 'gradient',
bgValue: defaultGradient, bgValue: defaultGradient,
overlayColor: '#000000', overlayColor: '#000000',
@@ -445,41 +395,16 @@ CallToAction.craft = {
overlayHtml = `<div${overlayStyle ? ` style="${overlayStyle}"` : ''}></div>`; overlayHtml = `<div${overlayStyle ? ` style="${overlayStyle}"` : ''}></div>`;
} }
let secondaryBtnHtml = ''; const ctas = normalizeCtas(props);
if (props.secondaryButtonText) { const buttonsHtml = ctasToHtml(ctas, { primaryBg: buttonColor, primaryText: buttonTextColor, outlineText: textColor });
const secStyle = cssPropsToString({ const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
display: 'inline-block',
padding: '14px 36px',
backgroundColor: 'transparent',
color: textColor,
textDecoration: 'none',
borderRadius: '8px',
fontWeight: '600',
fontSize: '16px',
border: `2px solid ${textColor}`,
});
secondaryBtnHtml = `\n <a href="${props.secondaryButtonHref || '#'}"${secStyle ? ` style="${secStyle}"` : ''}>${esc(props.secondaryButtonText)}</a>`;
}
const btnStyle = cssPropsToString({
display: 'inline-block',
padding: '14px 36px',
backgroundColor: buttonColor,
color: buttonTextColor,
textDecoration: 'none',
borderRadius: '8px',
fontWeight: '600',
fontSize: '16px',
});
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
${overlayHtml}<div style="max-width:700px;margin:0 auto;position:relative;z-index:1"> ${overlayHtml}<div style="max-width:700px;margin:0 auto;position:relative;z-index:1">
<h2 style="font-size:36px;font-weight:700;color:${textColor};margin-bottom:12px">${esc(props.heading || '')}</h2> <h2 style="font-size:36px;font-weight:700;color:${textColor};margin-bottom:12px">${esc(props.heading || '')}</h2>
<p style="font-size:18px;color:${textColor};opacity:0.85;margin-bottom:28px;line-height:1.6">${esc(props.description || '')}</p> <p style="font-size:18px;color:${textColor};opacity:0.85;margin-bottom:28px;line-height:1.6">${esc(props.description || '')}</p>
<div style="display:flex;gap:12px;justify-content:center;flex-wrap:wrap"> <div style="display:flex;gap:12px;justify-content:center;flex-wrap:wrap">${buttonsHtml}</div>
<a href="${props.buttonHref || '#'}"${btnStyle ? ` style="${btnStyle}"` : ''}>${esc(props.buttonText || '')}</a>${secondaryBtnHtml}
</div>
</div> </div>
</section>`, </section>`,
}; };

View File

@@ -1,6 +1,7 @@
import React, { CSSProperties, useEffect, useState, useCallback } from 'react'; import React, { CSSProperties, useEffect, useState, useCallback } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface CountdownProps { interface CountdownProps {
targetDate?: string; targetDate?: string;
@@ -9,6 +10,7 @@ interface CountdownProps {
digitColor?: string; digitColor?: string;
labelColor?: string; labelColor?: string;
bgColor?: string; bgColor?: string;
anchorId?: string;
} }
interface TimeLeft { interface TimeLeft {
@@ -44,6 +46,7 @@ export const Countdown: UserComponent<CountdownProps> = ({
digitColor = '#ffffff', digitColor = '#ffffff',
labelColor = 'rgba(255,255,255,0.7)', labelColor = 'rgba(255,255,255,0.7)',
bgColor = '#18181b', bgColor = '#18181b',
anchorId,
}) => { }) => {
const { const {
connectors: { connect, drag }, connectors: { connect, drag },
@@ -98,6 +101,7 @@ export const Countdown: UserComponent<CountdownProps> = ({
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
padding: '60px 20px', padding: '60px 20px',
textAlign: 'center', textAlign: 'center',
@@ -141,6 +145,7 @@ const CountdownSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
{/* Target date */} {/* Target date */}
<div> <div>
<label style={labelStyle}>Target Date</label> <label style={labelStyle}>Target Date</label>
@@ -235,6 +240,7 @@ Countdown.craft = {
digitColor: '#ffffff', digitColor: '#ffffff',
labelColor: 'rgba(255,255,255,0.7)', labelColor: 'rgba(255,255,255,0.7)',
bgColor: '#18181b', bgColor: '#18181b',
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -265,6 +271,7 @@ Countdown.craft = {
backgroundColor: bgColor, backgroundColor: bgColor,
...style, ...style,
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
const headingHtml = heading const headingHtml = heading
? `<h2 style="font-size:32px;font-weight:700;color:${digitColor};margin-bottom:32px;font-family:Inter,sans-serif">${esc(heading)}</h2>` ? `<h2 style="font-size:32px;font-weight:700;color:${digitColor};margin-bottom:32px;font-family:Inter,sans-serif">${esc(heading)}</h2>`
@@ -278,7 +285,7 @@ Countdown.craft = {
const uid = 'cd_' + Math.random().toString(36).slice(2, 8); const uid = 'cd_' + Math.random().toString(36).slice(2, 8);
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
${headingHtml} ${headingHtml}
<div style="display:flex;justify-content:center;gap:24px;flex-wrap:wrap"> <div style="display:flex;justify-content:center;gap:24px;flex-wrap:wrap">
<div style="${boxStyle}"><span id="${uid}_d" style="${dStyle}">00</span><span style="${lStyle}">Days</span></div> <div style="${boxStyle}"><span id="${uid}_d" style="${dStyle}">00</span><span style="${lStyle}">Days</span></div>

View File

@@ -1,6 +1,7 @@
import React, { CSSProperties } from 'react'; import React, { CSSProperties } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface FeatureItem { interface FeatureItem {
title: string; title: string;
@@ -11,6 +12,7 @@ interface FeatureItem {
interface FeaturesGridProps { interface FeaturesGridProps {
features?: FeatureItem[]; features?: FeatureItem[];
style?: CSSProperties; style?: CSSProperties;
anchorId?: string;
} }
const defaultFeatures: FeatureItem[] = [ const defaultFeatures: FeatureItem[] = [
@@ -22,6 +24,7 @@ const defaultFeatures: FeatureItem[] = [
export const FeaturesGrid: UserComponent<FeaturesGridProps> = ({ export const FeaturesGrid: UserComponent<FeaturesGridProps> = ({
features = defaultFeatures, features = defaultFeatures,
style = {}, style = {},
anchorId,
}) => { }) => {
const { const {
connectors: { connect, drag }, connectors: { connect, drag },
@@ -33,6 +36,7 @@ export const FeaturesGrid: UserComponent<FeaturesGridProps> = ({
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
padding: '80px 20px', padding: '80px 20px',
backgroundColor: '#ffffff', backgroundColor: '#ffffff',
@@ -102,6 +106,7 @@ const FeaturesGridSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Background</label> <label style={{ fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 6 }}>Background</label>
<div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}> <div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}>
@@ -163,6 +168,7 @@ FeaturesGrid.craft = {
props: { props: {
features: defaultFeatures, features: defaultFeatures,
style: { backgroundColor: '#ffffff' }, style: { backgroundColor: '#ffffff' },
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -177,11 +183,12 @@ FeaturesGrid.craft = {
/* ---------- HTML export ---------- */ /* ---------- HTML export ---------- */
(FeaturesGrid as any).toHtml = (props: FeaturesGridProps, _childrenHtml: string) => { (FeaturesGrid as any).toHtml = (props: FeaturesGridProps, _childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;'); const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const sectionStyle = cssPropsToString({ const sectionStyle = cssPropsToString({
padding: '80px 20px', padding: '80px 20px',
...props.style, ...props.style,
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
const cards = (props.features || defaultFeatures).map((feat) => { const cards = (props.features || defaultFeatures).map((feat) => {
return `<div style="text-align:center;padding:32px 24px;border-radius:12px;background-color:#f8fafc;border:1px solid #e2e8f0"> return `<div style="text-align:center;padding:32px 24px;border-radius:12px;background-color:#f8fafc;border:1px solid #e2e8f0">
<div style="font-size:36px;margin-bottom:16px">${esc(feat.icon)}</div> <div style="font-size:36px;margin-bottom:16px">${esc(feat.icon)}</div>
@@ -191,7 +198,7 @@ FeaturesGrid.craft = {
}).join('\n '); }).join('\n ');
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
<div style="max-width:1100px;margin:0 auto;display:grid;grid-template-columns:repeat(3,1fr);gap:32px"> <div style="max-width:1100px;margin:0 auto;display:grid;grid-template-columns:repeat(3,1fr);gap:32px">
${cards} ${cards}
</div> </div>

View File

@@ -1,10 +1,15 @@
import React, { CSSProperties } from 'react'; import React, { CSSProperties } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { CtaButton, CtasEditor, normalizeCtas, ctaInlineStyle, ctasToHtml } from './_cta-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface HeroProps { interface HeroProps {
heading?: string; heading?: string;
subtitle?: string; subtitle?: string;
/** New dynamic CTAs. When set (length > 0), legacy primary/secondary fields are ignored. */
ctas?: CtaButton[];
/** Legacy — kept for backwards compatibility with saved projects. */
buttonText?: string; buttonText?: string;
buttonHref?: string; buttonHref?: string;
secondaryButtonText?: string; secondaryButtonText?: string;
@@ -24,6 +29,7 @@ interface HeroProps {
minHeight?: string; minHeight?: string;
verticalAlign?: 'top' | 'center' | 'bottom'; verticalAlign?: 'top' | 'center' | 'bottom';
textAlign?: 'left' | 'center' | 'right'; textAlign?: 'left' | 'center' | 'right';
anchorId?: string;
style?: CSSProperties; style?: CSSProperties;
} }
@@ -43,10 +49,11 @@ function buildBackground(props: HeroProps): string {
export const HeroSimple: UserComponent<HeroProps> = ({ export const HeroSimple: UserComponent<HeroProps> = ({
heading = 'Build Something Amazing', heading = 'Build Something Amazing',
subtitle = 'Create beautiful websites without writing a single line of code.', subtitle = 'Create beautiful websites without writing a single line of code.',
buttonText = 'Get Started', ctas,
buttonHref = '#', buttonText,
secondaryButtonText = '', buttonHref,
secondaryButtonHref = '#', secondaryButtonText,
secondaryButtonHref,
bgType = 'color', bgType = 'color',
bgColor = '#1e293b', bgColor = '#1e293b',
bgGradientFrom = '#667eea', bgGradientFrom = '#667eea',
@@ -62,6 +69,7 @@ export const HeroSimple: UserComponent<HeroProps> = ({
minHeight = '500px', minHeight = '500px',
verticalAlign = 'center', verticalAlign = 'center',
textAlign = 'center', textAlign = 'center',
anchorId,
style = {}, style = {},
}) => { }) => {
const { connectors: { connect, drag } } = useNode(); const { connectors: { connect, drag } } = useNode();
@@ -72,9 +80,17 @@ export const HeroSimple: UserComponent<HeroProps> = ({
const justifyMap = { top: 'flex-start', center: 'center', bottom: 'flex-end' }; const justifyMap = { top: 'flex-start', center: 'center', bottom: 'flex-end' };
const effectiveCtas = normalizeCtas({ ctas, buttonText, buttonHref, secondaryButtonText, secondaryButtonHref });
const ctaDefaults = {
primaryBg: buttonBgColor,
primaryText: buttonTextColor,
outlineText: textColor,
};
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
...style, ...style,
background: bgType !== 'image' ? bg : undefined, background: bgType !== 'image' ? bg : undefined,
@@ -134,25 +150,12 @@ export const HeroSimple: UserComponent<HeroProps> = ({
{subtitle} {subtitle}
</p> </p>
<div style={{ display: 'flex', gap: '12px', justifyContent: textAlign === 'center' ? 'center' : textAlign === 'right' ? 'flex-end' : 'flex-start', flexWrap: 'wrap' }}> <div style={{ display: 'flex', gap: '12px', justifyContent: textAlign === 'center' ? 'center' : textAlign === 'right' ? 'flex-end' : 'flex-start', flexWrap: 'wrap' }}>
{buttonText && ( {effectiveCtas.map((cta, i) => (
<a href={buttonHref} onClick={(e) => e.preventDefault()} style={{ <a key={i} href={cta.href || '#'} onClick={(e) => e.preventDefault()}
display: 'inline-block', padding: '14px 36px', backgroundColor: buttonBgColor, style={ctaInlineStyle(cta, ctaDefaults)}>
color: buttonTextColor, textDecoration: 'none', borderRadius: '8px', {cta.text}
fontWeight: '600', fontSize: '16px',
}}>
{buttonText}
</a> </a>
)} ))}
{secondaryButtonText && (
<a href={secondaryButtonHref} onClick={(e) => e.preventDefault()} style={{
display: 'inline-block', padding: '14px 36px',
backgroundColor: 'transparent', color: textColor,
textDecoration: 'none', borderRadius: '8px', fontWeight: '600',
fontSize: '16px', border: `2px solid ${textColor}`,
}}>
{secondaryButtonText}
</a>
)}
</div> </div>
</div> </div>
</section> </section>
@@ -181,8 +184,11 @@ const HeroSettings: React.FC = () => {
props: node.data.props as HeroProps, props: node.data.props as HeroProps,
})); }));
const effectiveCtas = normalizeCtas(props);
return ( return (
<div style={{ padding: 12, display: 'flex', flexDirection: 'column', gap: 12 }}> <div style={{ padding: 12, display: 'flex', flexDirection: 'column', gap: 12 }}>
<AnchorIdField />
{/* Content */} {/* Content */}
<div> <div>
<label style={labelStyle}>Heading</label> <label style={labelStyle}>Heading</label>
@@ -192,18 +198,20 @@ const HeroSettings: React.FC = () => {
<label style={labelStyle}>Subtitle</label> <label style={labelStyle}>Subtitle</label>
<textarea value={props.subtitle || ''} onChange={(e) => setProp((p: HeroProps) => { p.subtitle = e.target.value; })} rows={3} style={{ ...inputStyle, resize: 'vertical' as const }} /> <textarea value={props.subtitle || ''} onChange={(e) => setProp((p: HeroProps) => { p.subtitle = e.target.value; })} rows={3} style={{ ...inputStyle, resize: 'vertical' as const }} />
</div> </div>
<div>
<label style={labelStyle}>Button Text</label> {/* Dynamic CTAs */}
<input type="text" value={props.buttonText || ''} onChange={(e) => setProp((p: HeroProps) => { p.buttonText = e.target.value; })} style={inputStyle} /> <CtasEditor
</div> ctas={effectiveCtas}
<div> onChange={(next) => setProp((p: HeroProps) => {
<label style={labelStyle}>Button URL</label> p.ctas = next;
<input type="text" value={props.buttonHref || ''} onChange={(e) => setProp((p: HeroProps) => { p.buttonHref = e.target.value; })} placeholder="#" style={inputStyle} /> // Once the user touches CTAs, the legacy fields are no longer
</div> // authoritative — clear them so the array is the only source.
<div> p.buttonText = undefined;
<label style={labelStyle}>Secondary Button Text</label> p.buttonHref = undefined;
<input type="text" value={props.secondaryButtonText || ''} onChange={(e) => setProp((p: HeroProps) => { p.secondaryButtonText = e.target.value; })} placeholder="Leave blank to hide" style={inputStyle} /> p.secondaryButtonText = undefined;
</div> p.secondaryButtonHref = undefined;
})}
/>
{/* Background Type */} {/* Background Type */}
<div> <div>
@@ -370,10 +378,9 @@ HeroSimple.craft = {
props: { props: {
heading: 'Build Something Amazing', heading: 'Build Something Amazing',
subtitle: 'Create beautiful websites without writing a single line of code.', subtitle: 'Create beautiful websites without writing a single line of code.',
buttonText: 'Get Started', ctas: [
buttonHref: '#', { text: 'Get Started', href: '#', variant: 'primary' },
secondaryButtonText: '', ] as CtaButton[],
secondaryButtonHref: '#',
bgType: 'color', bgType: 'color',
bgColor: '#1e293b', bgColor: '#1e293b',
bgGradientFrom: '#667eea', bgGradientFrom: '#667eea',
@@ -389,6 +396,7 @@ HeroSimple.craft = {
minHeight: '500px', minHeight: '500px',
verticalAlign: 'center', verticalAlign: 'center',
textAlign: 'center', textAlign: 'center',
anchorId: '',
style: {}, style: {},
}, },
rules: { rules: {
@@ -436,16 +444,16 @@ HeroSimple.craft = {
const textAlign = props.textAlign || 'center'; const textAlign = props.textAlign || 'center';
const justifyBtn = textAlign === 'center' ? 'center' : textAlign === 'right' ? 'flex-end' : 'flex-start'; const justifyBtn = textAlign === 'center' ? 'center' : textAlign === 'right' ? 'flex-end' : 'flex-start';
let buttonsHtml = ''; const ctas = normalizeCtas(props);
if (props.buttonText) { const buttonsHtml = ctasToHtml(ctas, {
buttonsHtml += `<a href="${props.buttonHref || '#'}" style="display:inline-block;padding:14px 36px;background-color:${props.buttonBgColor || '#3b82f6'};color:${props.buttonTextColor || '#fff'};text-decoration:none;border-radius:8px;font-weight:600;font-size:16px">${esc(props.buttonText)}</a>`; primaryBg: props.buttonBgColor || '#3b82f6',
} primaryText: props.buttonTextColor || '#fff',
if (props.secondaryButtonText) { outlineText: props.textColor || '#fff',
buttonsHtml += `<a href="${props.secondaryButtonHref || '#'}" style="display:inline-block;padding:14px 36px;background:transparent;color:${props.textColor || '#fff'};text-decoration:none;border-radius:8px;font-weight:600;font-size:16px;border:2px solid ${props.textColor || '#fff'}">${esc(props.secondaryButtonText)}</a>`; });
}
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
return { return {
html: `<section style="${sectionStyle}"> html: `<section${idAttr} style="${sectionStyle}">
${videoHtml}${overlayHtml} ${videoHtml}${overlayHtml}
<div style="max-width:800px;width:100%;position:relative;z-index:2;text-align:${textAlign}"> <div style="max-width:800px;width:100%;position:relative;z-index:2;text-align:${textAlign}">
<h1 style="font-size:48px;font-weight:700;color:${props.textColor || '#fff'};margin-bottom:16px;line-height:1.2">${esc(props.heading || '')}</h1> <h1 style="font-size:48px;font-weight:700;color:${props.textColor || '#fff'};margin-bottom:16px;line-height:1.2">${esc(props.heading || '')}</h1>

View File

@@ -1,6 +1,7 @@
import React, { CSSProperties } from 'react'; import React, { CSSProperties } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface PricingPlan { interface PricingPlan {
name: string; name: string;
@@ -17,6 +18,7 @@ interface PricingTableProps {
style?: CSSProperties; style?: CSSProperties;
featuredBg?: string; featuredBg?: string;
bulletType?: string; bulletType?: string;
anchorId?: string;
} }
const bulletChars: Record<string, string> = { const bulletChars: Record<string, string> = {
@@ -58,6 +60,7 @@ export const PricingTable: UserComponent<PricingTableProps> = ({
style = {}, style = {},
featuredBg = '#3b82f6', featuredBg = '#3b82f6',
bulletType = 'check', bulletType = 'check',
anchorId,
}) => { }) => {
const { const {
connectors: { connect, drag }, connectors: { connect, drag },
@@ -69,6 +72,7 @@ export const PricingTable: UserComponent<PricingTableProps> = ({
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
padding: '80px 20px', padding: '80px 20px',
backgroundColor: '#ffffff', backgroundColor: '#ffffff',
@@ -271,6 +275,7 @@ const PricingTableSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={labelStyle}>Background</label> <label style={labelStyle}>Background</label>
<div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}> <div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}>
@@ -384,6 +389,7 @@ PricingTable.craft = {
style: { backgroundColor: '#ffffff' }, style: { backgroundColor: '#ffffff' },
featuredBg: '#3b82f6', featuredBg: '#3b82f6',
bulletType: 'check', bulletType: 'check',
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -398,12 +404,13 @@ PricingTable.craft = {
/* ---------- HTML export ---------- */ /* ---------- HTML export ---------- */
(PricingTable as any).toHtml = (props: PricingTableProps, _childrenHtml: string) => { (PricingTable as any).toHtml = (props: PricingTableProps, _childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;'); const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const bulletType = props.bulletType || 'check'; const bulletType = props.bulletType || 'check';
const sectionStyle = cssPropsToString({ const sectionStyle = cssPropsToString({
padding: '80px 20px', padding: '80px 20px',
...props.style, ...props.style,
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
const plans = props.plans || defaultPlans; const plans = props.plans || defaultPlans;
const featuredBg = props.featuredBg || '#3b82f6'; const featuredBg = props.featuredBg || '#3b82f6';
@@ -442,7 +449,7 @@ PricingTable.craft = {
}).join('\n '); }).join('\n ');
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
<div style="max-width:1100px;margin:0 auto;display:flex;gap:24px;justify-content:center;align-items:stretch;flex-wrap:wrap"> <div style="max-width:1100px;margin:0 auto;display:flex;gap:24px;justify-content:center;align-items:stretch;flex-wrap:wrap">
${cards} ${cards}
</div> </div>

View File

@@ -1,6 +1,7 @@
import React, { CSSProperties, useState } from 'react'; import React, { CSSProperties, useState } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface TabItem { interface TabItem {
label: string; label: string;
@@ -15,6 +16,7 @@ interface TabsProps {
inactiveTabBg?: string; inactiveTabBg?: string;
inactiveTabColor?: string; inactiveTabColor?: string;
contentBg?: string; contentBg?: string;
anchorId?: string;
} }
const defaultTabs: TabItem[] = [ const defaultTabs: TabItem[] = [
@@ -31,6 +33,7 @@ export const Tabs: UserComponent<TabsProps> = ({
inactiveTabBg = '#f1f5f9', inactiveTabBg = '#f1f5f9',
inactiveTabColor = '#64748b', inactiveTabColor = '#64748b',
contentBg = '#ffffff', contentBg = '#ffffff',
anchorId,
}) => { }) => {
const { const {
connectors: { connect, drag }, connectors: { connect, drag },
@@ -44,6 +47,7 @@ export const Tabs: UserComponent<TabsProps> = ({
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
padding: '60px 20px', padding: '60px 20px',
backgroundColor: '#ffffff', backgroundColor: '#ffffff',
@@ -139,6 +143,7 @@ const TabsSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
<div> <div>
<label style={labelStyle}>Active Tab Background</label> <label style={labelStyle}>Active Tab Background</label>
<div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}> <div style={{ display: 'flex', gap: 4, flexWrap: 'wrap' }}>
@@ -276,6 +281,7 @@ Tabs.craft = {
inactiveTabBg: '#f1f5f9', inactiveTabBg: '#f1f5f9',
inactiveTabColor: '#64748b', inactiveTabColor: '#64748b',
contentBg: '#ffffff', contentBg: '#ffffff',
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -290,11 +296,12 @@ Tabs.craft = {
/* ---------- HTML export ---------- */ /* ---------- HTML export ---------- */
(Tabs as any).toHtml = (props: TabsProps, _childrenHtml: string) => { (Tabs as any).toHtml = (props: TabsProps, _childrenHtml: string) => {
const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;'); const esc = (s: any) => String(s ?? "").replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
const sectionStyle = cssPropsToString({ const sectionStyle = cssPropsToString({
padding: '60px 20px', padding: '60px 20px',
...props.style, ...props.style,
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
const tabs = props.tabs || defaultTabs; const tabs = props.tabs || defaultTabs;
const activeTabBg = props.activeTabBg || '#3b82f6'; const activeTabBg = props.activeTabBg || '#3b82f6';
const activeTabColor = props.activeTabColor || '#ffffff'; const activeTabColor = props.activeTabColor || '#ffffff';
@@ -326,7 +333,7 @@ function ${tabId}_switch(idx){
</script>`; </script>`;
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
<div style="max-width:800px;margin:0 auto"> <div style="max-width:800px;margin:0 auto">
<div style="display:flex;gap:2px;border-bottom:2px solid #e2e8f0"> <div style="display:flex;gap:2px;border-bottom:2px solid #e2e8f0">
${tabButtons} ${tabButtons}

View File

@@ -1,6 +1,7 @@
import React, { CSSProperties, useState } from 'react'; import React, { CSSProperties, useState } from 'react';
import { useNode, UserComponent } from '@craftjs/core'; import { useNode, UserComponent } from '@craftjs/core';
import { cssPropsToString } from '../../utils/style-helpers'; import { cssPropsToString } from '../../utils/style-helpers';
import { AnchorIdField } from '../../ui/AnchorIdField';
interface Testimonial { interface Testimonial {
quote: string; quote: string;
@@ -16,6 +17,7 @@ interface TestimonialsProps {
style?: CSSProperties; style?: CSSProperties;
cardBg?: string; cardBg?: string;
starColor?: string; starColor?: string;
anchorId?: string;
} }
const defaultTestimonials: Testimonial[] = [ const defaultTestimonials: Testimonial[] = [
@@ -52,6 +54,7 @@ export const Testimonials: UserComponent<TestimonialsProps> = ({
style = {}, style = {},
cardBg = '#f8fafc', cardBg = '#f8fafc',
starColor = '#f59e0b', starColor = '#f59e0b',
anchorId,
}) => { }) => {
const { const {
connectors: { connect, drag }, connectors: { connect, drag },
@@ -86,6 +89,7 @@ export const Testimonials: UserComponent<TestimonialsProps> = ({
return ( return (
<section <section
ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }} ref={(ref: HTMLElement | null): void => { if (ref) connect(drag(ref)); }}
id={anchorId || undefined}
style={{ style={{
padding: '80px 20px', padding: '80px 20px',
backgroundColor: '#ffffff', backgroundColor: '#ffffff',
@@ -187,6 +191,7 @@ const TestimonialsSettings: React.FC = () => {
return ( return (
<div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}> <div style={{ padding: '12px', display: 'flex', flexDirection: 'column', gap: '14px' }}>
<AnchorIdField />
{/* Layout */} {/* Layout */}
<div> <div>
<label style={labelStyle}>Layout</label> <label style={labelStyle}>Layout</label>
@@ -357,6 +362,7 @@ Testimonials.craft = {
style: { backgroundColor: '#ffffff' }, style: { backgroundColor: '#ffffff' },
cardBg: '#f8fafc', cardBg: '#f8fafc',
starColor: '#f59e0b', starColor: '#f59e0b',
anchorId: '',
}, },
rules: { rules: {
canDrag: () => true, canDrag: () => true,
@@ -388,6 +394,7 @@ Testimonials.craft = {
backgroundColor: '#ffffff', backgroundColor: '#ffffff',
...style, ...style,
}); });
const idAttr = props.anchorId ? ` id="${esc(props.anchorId)}"` : '';
const cardCss = `background-color:${cardBg};border-radius:12px;padding:32px 24px;text-align:center;border:1px solid #e2e8f0`; const cardCss = `background-color:${cardBg};border-radius:12px;padding:32px 24px;text-align:center;border:1px solid #e2e8f0`;
@@ -403,7 +410,7 @@ Testimonials.craft = {
if (layout === 'single') { if (layout === 'single') {
// For single layout, export as grid with 1 column (simpler static export) // For single layout, export as grid with 1 column (simpler static export)
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
<div style="max-width:600px;margin:0 auto;display:grid;grid-template-columns:1fr;gap:24px"> <div style="max-width:600px;margin:0 auto;display:grid;grid-template-columns:1fr;gap:24px">
${cards} ${cards}
</div> </div>
@@ -412,7 +419,7 @@ Testimonials.craft = {
} }
return { return {
html: `<section${sectionStyle ? ` style="${sectionStyle}"` : ''}> html: `<section${idAttr}${sectionStyle ? ` style="${sectionStyle}"` : ''}>
<div style="max-width:1100px;margin:0 auto;display:grid;grid-template-columns:repeat(${columns},1fr);gap:24px"> <div style="max-width:1100px;margin:0 auto;display:grid;grid-template-columns:repeat(${columns},1fr);gap:24px">
${cards} ${cards}
</div> </div>

View File

@@ -0,0 +1,201 @@
import React, { CSSProperties } from 'react';
export type CtaVariant = 'primary' | 'outline' | 'ghost';
export interface CtaButton {
text: string;
href: string;
variant?: CtaVariant;
target?: '_blank';
}
export interface CtaStyleDefaults {
primaryBg: string;
primaryText: string;
outlineText: string;
}
/**
* Read the effective list of CTAs for a section, falling back to legacy
* primary/secondary props when ctas[] is absent. New sections write ctas[]
* directly; old sections keep rendering until the user touches the settings.
*/
export function normalizeCtas(props: {
ctas?: CtaButton[];
buttonText?: string;
buttonHref?: string;
secondaryButtonText?: string;
secondaryButtonHref?: string;
}): CtaButton[] {
if (Array.isArray(props.ctas) && props.ctas.length > 0) {
return props.ctas.filter((c) => c && (c.text || c.href));
}
const legacy: CtaButton[] = [];
if (props.buttonText) legacy.push({ text: props.buttonText, href: props.buttonHref || '#', variant: 'primary' });
if (props.secondaryButtonText) legacy.push({ text: props.secondaryButtonText, href: props.secondaryButtonHref || '#', variant: 'outline' });
return legacy;
}
export function ctaInlineStyle(cta: CtaButton, defaults: CtaStyleDefaults): CSSProperties {
const variant = cta.variant || 'primary';
switch (variant) {
case 'outline':
return {
display: 'inline-block', padding: '14px 36px',
backgroundColor: 'transparent', color: defaults.outlineText,
textDecoration: 'none', borderRadius: '8px',
fontWeight: 600, fontSize: '16px',
border: `2px solid ${defaults.outlineText}`,
};
case 'ghost':
return {
display: 'inline-block', padding: '14px 24px',
backgroundColor: 'transparent', color: defaults.outlineText,
textDecoration: 'underline', borderRadius: '8px',
fontWeight: 600, fontSize: '16px',
};
case 'primary':
default:
return {
display: 'inline-block', padding: '14px 36px',
backgroundColor: defaults.primaryBg, color: defaults.primaryText,
textDecoration: 'none', borderRadius: '8px',
fontWeight: 600, fontSize: '16px',
};
}
}
export function ctaCssString(cta: CtaButton, defaults: CtaStyleDefaults): string {
const variant = cta.variant || 'primary';
switch (variant) {
case 'outline':
return `display:inline-block;padding:14px 36px;background-color:transparent;color:${defaults.outlineText};text-decoration:none;border-radius:8px;font-weight:600;font-size:16px;border:2px solid ${defaults.outlineText}`;
case 'ghost':
return `display:inline-block;padding:14px 24px;background-color:transparent;color:${defaults.outlineText};text-decoration:underline;border-radius:8px;font-weight:600;font-size:16px`;
case 'primary':
default:
return `display:inline-block;padding:14px 36px;background-color:${defaults.primaryBg};color:${defaults.primaryText};text-decoration:none;border-radius:8px;font-weight:600;font-size:16px`;
}
}
const esc = (s: any) => String(s ?? '').replace(/</g, '&lt;').replace(/>/g, '&gt;');
export function ctasToHtml(ctas: CtaButton[], defaults: CtaStyleDefaults): string {
return ctas.map((c) => {
const target = c.target === '_blank' ? ' target="_blank" rel="noopener noreferrer"' : '';
return `<a href="${esc(c.href || '#')}"${target} style="${ctaCssString(c, defaults)}">${esc(c.text || '')}</a>`;
}).join('');
}
/* ---------- CTAs editor (settings UI) ---------- */
interface CtasEditorProps {
ctas: CtaButton[];
/** Called whenever the user mutates the array. Sections wire this via setProp. */
onChange: (next: CtaButton[]) => void;
/** Max items the user can add. Default 4. */
max?: number;
}
const inputStyle: CSSProperties = {
width: '100%', padding: '6px 8px', background: '#27272a',
color: '#e4e4e7', border: '1px solid #3f3f46', borderRadius: 4, fontSize: 12,
};
const labelStyle: CSSProperties = {
fontSize: 11, color: '#a1a1aa', display: 'block', marginBottom: 4,
};
export const CtasEditor: React.FC<CtasEditorProps> = ({ ctas, onChange, max = 4 }) => {
const update = (i: number, patch: Partial<CtaButton>) => {
const next = ctas.slice();
next[i] = { ...next[i], ...patch };
onChange(next);
};
const remove = (i: number) => onChange(ctas.filter((_, j) => j !== i));
const add = () => {
if (ctas.length >= max) return;
onChange([...ctas, { text: 'New button', href: '#', variant: ctas.length === 0 ? 'primary' : 'outline' }]);
};
const move = (i: number, dir: -1 | 1) => {
const j = i + dir;
if (j < 0 || j >= ctas.length) return;
const next = ctas.slice();
[next[i], next[j]] = [next[j], next[i]];
onChange(next);
};
return (
<div style={{ display: 'flex', flexDirection: 'column', gap: 10 }}>
<div style={{ fontSize: 11, color: '#a1a1aa', fontWeight: 600, textTransform: 'uppercase', letterSpacing: '0.5px' }}>
Buttons ({ctas.length})
</div>
{ctas.length === 0 && (
<div style={{ fontSize: 11, color: '#71717a', fontStyle: 'italic', padding: '8px 0' }}>
No buttons. Click "Add button" to insert one.
</div>
)}
{ctas.map((cta, i) => (
<div key={i} style={{
background: '#18181b', border: '1px solid #3f3f46', borderRadius: 6,
padding: 10, display: 'flex', flexDirection: 'column', gap: 6,
}}>
<div style={{ display: 'flex', alignItems: 'center', gap: 4 }}>
<span style={{ fontSize: 10, color: '#a1a1aa', fontWeight: 600, flex: 1 }}>Button {i + 1}</span>
<button onClick={() => move(i, -1)} disabled={i === 0} title="Move up"
style={iconBtn(i === 0)}></button>
<button onClick={() => move(i, 1)} disabled={i === ctas.length - 1} title="Move down"
style={iconBtn(i === ctas.length - 1)}></button>
<button onClick={() => remove(i)} title="Remove"
style={{ ...iconBtn(false), color: '#fca5a5' }}></button>
</div>
<div>
<label style={labelStyle}>Text</label>
<input type="text" value={cta.text} onChange={(e) => update(i, { text: e.target.value })} style={inputStyle} />
</div>
<div>
<label style={labelStyle}>URL</label>
<input type="text" value={cta.href} onChange={(e) => update(i, { href: e.target.value })}
placeholder="https://… or #anchor" style={inputStyle} />
</div>
<div style={{ display: 'flex', gap: 6 }}>
<div style={{ flex: 1 }}>
<label style={labelStyle}>Style</label>
<select value={cta.variant || 'primary'}
onChange={(e) => update(i, { variant: e.target.value as CtaVariant })}
style={{ ...inputStyle, padding: '5px 6px' }}>
<option value="primary">Primary (filled)</option>
<option value="outline">Outline</option>
<option value="ghost">Ghost (text)</option>
</select>
</div>
<div style={{ flex: '0 0 auto', display: 'flex', alignItems: 'flex-end' }}>
<label style={{ fontSize: 11, color: '#a1a1aa', display: 'inline-flex', alignItems: 'center', gap: 4, cursor: 'pointer', whiteSpace: 'nowrap' }}>
<input type="checkbox" checked={cta.target === '_blank'}
onChange={(e) => update(i, { target: e.target.checked ? '_blank' : undefined })} />
New tab
</label>
</div>
</div>
</div>
))}
{ctas.length < max && (
<button onClick={add} style={{
padding: '8px 12px', fontSize: 12, fontWeight: 600,
color: '#3b82f6', background: 'rgba(59,130,246,0.1)',
border: '1px dashed #3b82f6', borderRadius: 4, cursor: 'pointer',
}}>
+ Add button{ctas.length === 0 ? '' : ` (${max - ctas.length} more)`}
</button>
)}
</div>
);
};
function iconBtn(disabled: boolean): CSSProperties {
return {
width: 22, height: 22, fontSize: 11,
background: '#27272a', color: disabled ? '#52525b' : '#a1a1aa',
border: '1px solid #3f3f46', borderRadius: 4,
cursor: disabled ? 'not-allowed' : 'pointer',
};
}

View File

@@ -34,14 +34,20 @@ export function useSitesmith(siteId: number) {
useEffect(() => { void refreshEntitlement(); void fetchHistory(); }, [refreshEntitlement, fetchHistory]); useEffect(() => { void refreshEntitlement(); void fetchHistory(); }, [refreshEntitlement, fetchHistory]);
const send = useCallback(async (userText: string, canvasSummary: string): Promise<SendResult> => { const send = useCallback(async (
userText: string,
canvasSummary: string,
target?: { node_id: string; display_name: string; tree_json: string },
): Promise<SendResult> => {
if (!whpConfig) return { ok: false, status: 'BLOCKED', message: 'No WHP config' }; if (!whpConfig) return { ok: false, status: 'BLOCKED', message: 'No WHP config' };
setMessages((m) => [...m, { role: 'user', content: userText, response_type: null, created_at: new Date().toISOString() }]); setMessages((m) => [...m, { role: 'user', content: userText, response_type: null, created_at: new Date().toISOString() }]);
const body: Record<string, unknown> = { site_id: siteId, message: userText, canvas_summary: canvasSummary };
if (target) body.target = target;
const r = await fetch(`${apiBase(whpConfig.apiUrl)}?action=send`, { const r = await fetch(`${apiBase(whpConfig.apiUrl)}?action=send`, {
method: 'POST', method: 'POST',
credentials: 'include', credentials: 'include',
headers: { 'Content-Type': 'application/json', 'X-CSRF-Token': whpConfig.csrfToken }, headers: { 'Content-Type': 'application/json', 'X-CSRF-Token': whpConfig.csrfToken },
body: JSON.stringify({ site_id: siteId, message: userText, canvas_summary: canvasSummary }), body: JSON.stringify(body),
}); });
const j: SendResult = await r.json(); const j: SendResult = await r.json();
void fetchHistory(); void fetchHistory();

View File

@@ -1,6 +1,8 @@
import React, { useEffect, useCallback, useRef } from 'react'; import React, { useEffect, useCallback, useRef } from 'react';
import { useEditor } from '@craftjs/core'; import { useEditor } from '@craftjs/core';
import { findDeletableTarget } from '../../utils/craft-helpers'; import { findDeletableTarget } from '../../utils/craft-helpers';
import { useSitesmithModal } from '../../state/SitesmithContext';
import { buildSitesmithTarget } from '../../utils/sitesmith-target';
interface ContextMenuProps { interface ContextMenuProps {
visible: boolean; visible: boolean;
@@ -27,6 +29,7 @@ export const ContextMenu: React.FC<ContextMenuProps> = ({
onClose, onClose,
}) => { }) => {
const { actions, query } = useEditor(); const { actions, query } = useEditor();
const { open: openSitesmith } = useSitesmithModal();
const menuRef = useRef<HTMLDivElement>(null); const menuRef = useRef<HTMLDivElement>(null);
const clipboardRef = useRef<string | null>(null); const clipboardRef = useRef<string | null>(null);
@@ -143,6 +146,17 @@ export const ContextMenu: React.FC<ContextMenuProps> = ({
onClose(); onClose();
}, [nodeId, actions, getParentId, onClose]); }, [nodeId, actions, getParentId, onClose]);
const askSitesmith = useCallback(() => {
if (!nodeId || nodeId === 'ROOT') return;
try {
const target = buildSitesmithTarget(query, nodeId);
if (target) openSitesmith(target);
} catch (e) {
console.error('Ask Sitesmith failed:', e);
}
onClose();
}, [nodeId, query, openSitesmith, onClose]);
const deleteNode = useCallback(() => { const deleteNode = useCallback(() => {
const target = findDeletableTarget(query, nodeId); const target = findDeletableTarget(query, nodeId);
if (!target) { if (!target) {
@@ -162,6 +176,12 @@ export const ContextMenu: React.FC<ContextMenuProps> = ({
const isRoot = nodeId === 'ROOT' || !nodeId; const isRoot = nodeId === 'ROOT' || !nodeId;
const items: MenuItem[] = [ const items: MenuItem[] = [
{
label: '✨ Ask Sitesmith',
action: askSitesmith,
disabled: isRoot,
dividerAfter: true,
},
{ {
label: 'Duplicate', label: 'Duplicate',
shortcut: 'Ctrl+D', shortcut: 'Ctrl+D',

View File

@@ -2,6 +2,8 @@ import React from 'react';
import { useEditor } from '@craftjs/core'; import { useEditor } from '@craftjs/core';
import { componentResolver } from '../../components/resolver'; import { componentResolver } from '../../components/resolver';
import { SiteDesignPanel } from './SiteDesignPanel'; import { SiteDesignPanel } from './SiteDesignPanel';
import { useSitesmithModal } from '../../state/SitesmithContext';
import { buildSitesmithTarget } from '../../utils/sitesmith-target';
import { import {
TextStylePanel, TextStylePanel,
ButtonStylePanel, ButtonStylePanel,
@@ -30,6 +32,8 @@ import {
export const GuidedStyles: React.FC = () => { export const GuidedStyles: React.FC = () => {
const resolverMap = componentResolver as Record<string, any>; const resolverMap = componentResolver as Record<string, any>;
const { open: openSitesmith } = useSitesmithModal();
const { query } = useEditor();
const { selected, selectedType, nodeProps, resolvedName } = useEditor((state) => { const { selected, selectedType, nodeProps, resolvedName } = useEditor((state) => {
const currentNodeId = state.events.selected const currentNodeId = state.events.selected
@@ -97,14 +101,33 @@ export const GuidedStyles: React.FC = () => {
: isUtility ? 'fa-ellipsis-h' : isUtility ? 'fa-ellipsis-h'
: 'fa-cube'; : 'fa-cube';
const handleAskSitesmith = () => {
if (!selected || selected === 'ROOT') return;
const target = buildSitesmithTarget(query, selected);
if (target) openSitesmith(target);
};
return ( return (
<div className="guided-styles"> <div className="guided-styles">
{/* Component type badge */} {/* Component type badge + Sitesmith shortcut */}
<div className="guided-section guided-type-header"> <div className="guided-section guided-type-header" style={{ display: 'flex', alignItems: 'center', gap: 8 }}>
<span className="guided-type-badge"> <span className="guided-type-badge" style={{ flex: 1 }}>
<i className={`fa ${typeIcon}`} /> <i className={`fa ${typeIcon}`} />
{' '}{typeName} {' '}{typeName}
</span> </span>
<button
onClick={handleAskSitesmith}
title="Ask Sitesmith to edit this block"
style={{
display: 'inline-flex', alignItems: 'center', gap: 4,
padding: '4px 8px', fontSize: 11, fontWeight: 600,
color: '#a78bfa', background: 'rgba(139,92,246,0.12)',
border: '1px solid rgba(139,92,246,0.4)',
borderRadius: 'var(--radius-sm)', cursor: 'pointer',
}}
>
<i className="fa fa-magic" /> Ask Sitesmith
</button>
</div> </div>
{/* TEXT */} {/* TEXT */}

View File

@@ -4,6 +4,7 @@ import { useEditorConfig } from '../../state/EditorConfigContext';
import { useSitesmith } from '../../hooks/useSitesmith'; import { useSitesmith } from '../../hooks/useSitesmith';
import { useApplyAiResponse } from '../../utils/apply-ai-response'; import { useApplyAiResponse } from '../../utils/apply-ai-response';
import { summarizeCanvas } from '../../utils/canvas-summary'; import { summarizeCanvas } from '../../utils/canvas-summary';
import { SitesmithTarget } from '../../state/SitesmithContext';
import { UpgradeBanner } from './UpgradeBanner'; import { UpgradeBanner } from './UpgradeBanner';
import { ScopeConfirmDialog } from './ScopeConfirmDialog'; import { ScopeConfirmDialog } from './ScopeConfirmDialog';
import { MessageList } from './MessageList'; import { MessageList } from './MessageList';
@@ -11,9 +12,14 @@ import { ChatInput } from './ChatInput';
import { WorkingIndicator } from './WorkingIndicator'; import { WorkingIndicator } from './WorkingIndicator';
import { SitesmithResponse } from '../../types/sitesmith'; import { SitesmithResponse } from '../../types/sitesmith';
interface Props { onClose: () => void; } interface Props {
onClose: () => void;
/** When set, the chat is biased toward editing this specific node and the AI is
* instructed to return a `patch` op. The node's serialized tree is sent along. */
target?: SitesmithTarget | null;
}
export const SitesmithModal: React.FC<Props> = ({ onClose }) => { export const SitesmithModal: React.FC<Props> = ({ onClose, target }) => {
const cfg = useEditorConfig(); const cfg = useEditorConfig();
const siteId = cfg.whpConfig?.siteId ?? 0; const siteId = cfg.whpConfig?.siteId ?? 0;
const { query } = useEditor(); const { query } = useEditor();
@@ -38,13 +44,17 @@ export const SitesmithModal: React.FC<Props> = ({ onClose }) => {
setBusy(true); setError(null); setBusy(true); setError(null);
try { try {
const canvas = summarizeCanvas(query.getSerializedNodes()); const canvas = summarizeCanvas(query.getSerializedNodes());
const result = await send(text, canvas); const result = await send(text, canvas, target ? {
node_id: target.nodeId,
display_name: target.displayName,
tree_json: target.treeJson,
} : undefined);
if (!result.ok) { setError(result.message || 'Failed'); return; } if (!result.ok) { setError(result.message || 'Failed'); return; }
if (result.response.type === 'replace' && result.response.scope === 'site') { if (result.response.type === 'replace' && result.response.scope === 'site') {
setPendingReplace(result.response); setPendingReplace(result.response);
return; return;
} }
const applied = await apply(result.response); const applied = await apply(result.response, target?.nodeId);
if (!applied.ok) setError(applied.message || 'Apply failed'); if (!applied.ok) setError(applied.message || 'Apply failed');
} catch (e: any) { setError(String(e?.message ?? e)); } } catch (e: any) { setError(String(e?.message ?? e)); }
finally { setBusy(false); } finally { setBusy(false); }
@@ -86,6 +96,16 @@ export const SitesmithModal: React.FC<Props> = ({ onClose }) => {
</div> </div>
<div style={body}> <div style={body}>
<UpgradeBanner summary={summary} /> <UpgradeBanner summary={summary} />
{target && (
<div style={{
background: 'rgba(59,130,246,0.12)', border: '1px solid rgba(59,130,246,0.4)',
borderRadius: 6, padding: '8px 12px', marginBottom: 10, fontSize: 13, color: '#bfdbfe',
display: 'flex', alignItems: 'center', gap: 8,
}}>
<i className="fa fa-magic" style={{ color: '#60a5fa' }} />
<span>Editing <strong style={{ color: '#fff' }}>{target.displayName}</strong> describe the change you want and Sitesmith will modify just this block.</span>
</div>
)}
{error && <div role="alert" style={errBox}>{error}</div>} {error && <div role="alert" style={errBox}>{error}</div>}
{loading {loading
? <div style={{ color: '#71717a', textAlign: 'center', padding: 30 }}>Loading</div> ? <div style={{ color: '#71717a', textAlign: 'center', padding: 30 }}>Loading</div>

View File

@@ -7,7 +7,7 @@ import { DeviceMode } from '../../types';
import { TemplateModal } from './TemplateModal'; import { TemplateModal } from './TemplateModal';
import { HeadCodeModal } from './HeadCodeModal'; import { HeadCodeModal } from './HeadCodeModal';
import { SitesmithButton } from '../sitesmith/SitesmithButton'; import { SitesmithButton } from '../sitesmith/SitesmithButton';
import { SitesmithModal } from '../sitesmith/SitesmithModal'; import { useSitesmithModal } from '../../state/SitesmithContext';
interface TopBarProps { interface TopBarProps {
device: DeviceMode; device: DeviceMode;
@@ -28,7 +28,7 @@ export const TopBar: React.FC<TopBarProps> = ({ device, onDeviceChange }) => {
const [isDraft, setIsDraft] = useState(false); const [isDraft, setIsDraft] = useState(false);
const [templateModalOpen, setTemplateModalOpen] = useState(false); const [templateModalOpen, setTemplateModalOpen] = useState(false);
const [headCodeModalOpen, setHeadCodeModalOpen] = useState(false); const [headCodeModalOpen, setHeadCodeModalOpen] = useState(false);
const [sitesmithOpen, setSitesmithOpen] = useState(false); const { open: openSitesmith } = useSitesmithModal();
const saveTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null); const saveTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
const publishTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null); const publishTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
const hasLoadedRef = useRef(false); const hasLoadedRef = useRef(false);
@@ -242,7 +242,7 @@ export const TopBar: React.FC<TopBarProps> = ({ device, onDeviceChange }) => {
</span> </span>
)} )}
<SitesmithButton onClick={() => setSitesmithOpen(true)} /> <SitesmithButton onClick={() => openSitesmith()} />
<button <button
className="topbar-btn primary" className="topbar-btn primary"
@@ -274,7 +274,6 @@ export const TopBar: React.FC<TopBarProps> = ({ device, onDeviceChange }) => {
</div> </div>
<TemplateModal open={templateModalOpen} onClose={() => setTemplateModalOpen(false)} /> <TemplateModal open={templateModalOpen} onClose={() => setTemplateModalOpen(false)} />
<HeadCodeModal open={headCodeModalOpen} onClose={() => setHeadCodeModalOpen(false)} /> <HeadCodeModal open={headCodeModalOpen} onClose={() => setHeadCodeModalOpen(false)} />
{sitesmithOpen && <SitesmithModal onClose={() => setSitesmithOpen(false)} />}
</nav> </nav>
); );
}; };

View File

@@ -0,0 +1,55 @@
import React, { createContext, useCallback, useContext, useMemo, useState } from 'react';
/**
* Optional target for a Sitesmith chat session. When set, the modal renders a
* "Editing X" banner and the chat input is biased toward modifying just that
* subtree — the user's prompt is augmented server-side with the node's
* serialized tree, and the AI is instructed to return a `patch` op (typically
* `replace_node`) rather than a full-site replace.
*/
export interface SitesmithTarget {
/** Craft.js node id, used to find the node when applying the patch. */
nodeId: string;
/** Human-readable component name, shown in the modal header. */
displayName: string;
/** The component's serialized subtree (used to build a usable AI prompt). */
treeJson: string;
}
interface SitesmithContextValue {
isOpen: boolean;
target: SitesmithTarget | null;
open: (target?: SitesmithTarget) => void;
close: () => void;
}
const SitesmithCtx = createContext<SitesmithContextValue>({
isOpen: false,
target: null,
open: () => {},
close: () => {},
});
export const useSitesmithModal = () => useContext(SitesmithCtx);
export const SitesmithProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {
const [isOpen, setIsOpen] = useState(false);
const [target, setTarget] = useState<SitesmithTarget | null>(null);
const open = useCallback((t?: SitesmithTarget) => {
setTarget(t ?? null);
setIsOpen(true);
}, []);
const close = useCallback(() => {
setIsOpen(false);
setTarget(null);
}, []);
const value = useMemo<SitesmithContextValue>(
() => ({ isOpen, target, open, close }),
[isOpen, target, open, close],
);
return <SitesmithCtx.Provider value={value}>{children}</SitesmithCtx.Provider>;
};

View File

@@ -0,0 +1,81 @@
import React from 'react';
import { useNode, useEditor } from '@craftjs/core';
/**
* Reusable anchor-id input for any section/layout component. Lets the user
* set a stable URL fragment (e.g. #about) and auto-fills from the first
* heading found inside the node's subtree.
*
* Uses the editor query (more reliable than DOM lookup) to walk the Craft.js
* node tree and find the first Heading component's `text` prop.
*/
export const AnchorIdField: React.FC = () => {
const { id, actions: { setProp }, props, nodeName } = useNode((node) => ({
props: node.data.props as { anchorId?: string },
nodeName: node.data.displayName,
}));
const { query } = useEditor();
const value = (props.anchorId ?? '').toString();
const slugify = (s: string) =>
s.toLowerCase().trim().replace(/[^a-z0-9\s-]/g, '').replace(/\s+/g, '-').replace(/-+/g, '-').slice(0, 60);
// Walk the subtree via editor query looking for the first Heading's `text` prop.
const findFirstHeadingText = (): string | null => {
const walk = (nodeId: string): string | null => {
try {
const n = query.node(nodeId).get();
if (n.data.displayName === 'Heading') {
return ((n.data.props as any).text as string | undefined) ?? null;
}
for (const childId of n.data.nodes ?? []) {
const r = walk(childId);
if (r) return r;
}
for (const childId of Object.values(n.data.linkedNodes ?? {})) {
const r = walk(childId as string);
if (r) return r;
}
} catch {
return null;
}
return null;
};
try { return walk(id); } catch { return null; }
};
const autoFill = () => {
const txt = findFirstHeadingText();
if (txt) setProp((p: any) => { p.anchorId = slugify(txt); });
};
const labelStyle: React.CSSProperties = { display: 'block', fontSize: 11, color: 'var(--color-text-muted)', marginBottom: 4, fontWeight: 500 };
return (
<div style={{ marginBottom: 14, paddingBottom: 12, borderBottom: '1px solid var(--color-border)' }}>
<label style={labelStyle}>Anchor ID (URL fragment)</label>
<div style={{ display: 'flex', gap: 6 }}>
<span style={{ color: 'var(--color-text-dim)', fontSize: 13, padding: '6px 4px 6px 8px', background: 'var(--color-bg-base)', borderTopLeftRadius: 'var(--radius-sm)', borderBottomLeftRadius: 'var(--radius-sm)', border: '1px solid var(--color-border)', borderRight: 'none', fontFamily: 'monospace' }}>#</span>
<input
type="text"
value={value}
onChange={(e) => setProp((p: any) => { p.anchorId = slugify(e.target.value); })}
placeholder="optional"
className="control-input"
style={{ flex: 1, fontSize: 12, fontFamily: 'monospace', borderTopLeftRadius: 0, borderBottomLeftRadius: 0 }}
/>
<button
type="button"
onClick={autoFill}
title="Auto-fill from first heading inside this block"
style={{ padding: '4px 8px', fontSize: 11, background: 'var(--color-bg-base)', color: 'var(--color-text-muted)', border: '1px solid var(--color-border)', borderRadius: 'var(--radius-sm)', cursor: 'pointer', whiteSpace: 'nowrap' }}
>
From heading
</button>
</div>
<div style={{ fontSize: 10, color: 'var(--color-text-dim)', marginTop: 4 }}>
Link to this {nodeName?.toLowerCase() ?? 'block'} from anywhere with <code style={{ fontFamily: 'monospace' }}>#{value || 'your-anchor'}</code>
</div>
</div>
);
};

View File

@@ -167,7 +167,16 @@ export function useApplyAiResponse() {
const { actions, query } = useEditor(); const { actions, query } = useEditor();
const pages = usePages(); const pages = usePages();
return async function apply(resp: SitesmithResponse): Promise<{ ok: boolean; message?: string }> { /**
* @param targetNodeId If set and the AI returned a section-scoped replace
* instead of a patch, treat the first returned tree as a replacement for
* this node (the user said "edit this block" — they don't want a new
* section appended at the bottom).
*/
return async function apply(
resp: SitesmithResponse,
targetNodeId?: string,
): Promise<{ ok: boolean; message?: string }> {
// 'ask' type = AI wants clarification, nothing to apply // 'ask' type = AI wants clarification, nothing to apply
if (resp.type === 'ask') return { ok: true }; if (resp.type === 'ask') return { ok: true };
@@ -185,6 +194,21 @@ export function useApplyAiResponse() {
} }
if (resp.scope === 'section') { if (resp.scope === 'section') {
// When targeted at a specific node, the AI's tree replaces that node
// in place (vs appending a fresh section at the end of ROOT).
if (targetNodeId && resp.pages.length > 0) {
try {
const nodeTree = buildNodeTree(query, resp.pages[0].tree);
const parent: string = query.node(targetNodeId).get().data.parent ?? 'ROOT';
const siblings: string[] = query.node(parent).childNodes();
const index = siblings.indexOf(targetNodeId);
actions.delete(targetNodeId);
actions.addNodeTree(nodeTree, parent, index);
return { ok: true };
} catch (e) {
console.warn('sitesmith: targeted section replace failed, falling back to append', e);
}
}
// Insert each provided tree as a new node tree appended to ROOT // Insert each provided tree as a new node tree appended to ROOT
for (const p of resp.pages) { for (const p of resp.pages) {
try { try {

View File

@@ -0,0 +1,40 @@
import { SitesmithTarget } from '../state/SitesmithContext';
/**
* Build a Sitesmith target descriptor from a Craft.js node id. The returned
* `treeJson` is a flat node-map (compatible with the editor's serialized
* format) for just the selected subtree; the server includes it in the AI
* prompt so the model has the exact current shape of the block to modify.
*/
export function buildSitesmithTarget(query: any, nodeId: string): SitesmithTarget | null {
if (!nodeId || nodeId === 'ROOT') return null;
try {
const node = query.node(nodeId).get();
const displayName = node?.data?.displayName || node?.data?.type?.resolvedName || 'Block';
// Use Craft.js' own subtree serializer — toNodeTree gives a flat map keyed
// by node id, identical to what `actions.deserialize()` consumes.
const subtree = query.node(nodeId).toNodeTree();
const serializedMap: Record<string, unknown> = {};
for (const [id, n] of Object.entries(subtree.nodes ?? {}) as [string, any][]) {
serializedMap[id] = {
type: n.data.type,
props: n.data.props,
displayName: n.data.displayName,
isCanvas: n.data.isCanvas ?? false,
parent: n.data.parent,
nodes: n.data.nodes ?? [],
linkedNodes: n.data.linkedNodes ?? {},
hidden: n.data.hidden ?? false,
custom: n.data.custom ?? {},
};
}
return {
nodeId,
displayName,
treeJson: JSON.stringify({ root: subtree.rootNodeId, nodes: serializedMap }),
};
} catch (e) {
console.warn('buildSitesmithTarget failed:', e);
return null;
}
}