@paulpaulstudio/strapi-render 0.1.1 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/EditModeContext.tsx","../src/StrapiField.tsx","../src/StrapiText.tsx","../src/StrapiImage.tsx","../src/StrapiList.tsx"],"names":["createContext","useContext","useState","useEffect","useMemo","jsx","useCallback","Fragment","Children","isValidElement","cloneElement","createElement"],"mappings":"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","file":"index.cjs","sourcesContent":["\"use client\";\n\nimport { createContext, useContext, useEffect, useMemo, useState } from \"react\";\n\ninterface EditModeState {\n enabled: boolean;\n token: string | null;\n}\n\nconst EditModeContext = createContext<EditModeState>({ enabled: false, token: null });\n\nexport function useStrapiEditMode(): EditModeState {\n return useContext(EditModeContext);\n}\n\ninterface ProviderProps {\n children: React.ReactNode;\n /**\n * Override automatic URL-based detection. If `true`/`false` ist gesetzt,\n * wird der URL-Parameter ignoriert.\n */\n enabled?: boolean;\n}\n\n/**\n * Provider, der den Edit-Mode aus `?__pp_edit=1&__pp_token=<jwt>` der URL\n * liest und an Kinder via Context weitergibt. Token wird sicherheitshalber\n * NICHT ans CSR exponiert über Props oder DOM — er bleibt nur im Context.\n *\n * Im Edit-Mode wird ausserdem eine kleine globale CSS-Klasse `pp-edit-active`\n * am `<html>` gesetzt, sodass Customer-Sites optional CSS-Hooks setzen können.\n */\nexport function StrapiEditModeProvider({ children, enabled: enabledOverride }: ProviderProps) {\n const [urlState, setUrlState] = useState<EditModeState>({ enabled: false, token: null });\n\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n if (enabledOverride !== undefined) {\n setUrlState({ enabled: enabledOverride, token: null });\n return;\n }\n try {\n const params = new URLSearchParams(window.location.search);\n const edit = params.get(\"__pp_edit\") === \"1\";\n const token = params.get(\"__pp_token\");\n setUrlState({ enabled: edit, token });\n } catch {\n // ignore\n }\n }, [enabledOverride]);\n\n useEffect(() => {\n if (typeof document === \"undefined\") return;\n document.documentElement.classList.toggle(\"pp-edit-active\", urlState.enabled);\n }, [urlState.enabled]);\n\n // postMessage \"ready\" wenn Edit-Mode aktiv ist und Iframe geladen\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n if (!urlState.enabled) return;\n if (window.parent === window) return; // nicht im Iframe\n try {\n window.parent.postMessage(\n { type: \"pp:edit:ready\", href: window.location.href },\n \"*\",\n );\n } catch {\n // ignore\n }\n }, [urlState.enabled]);\n\n // Reload-Listener vom Parent (nach Save)\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n if (!urlState.enabled) return;\n function onMsg(e: MessageEvent) {\n const d = e.data;\n if (d && typeof d === \"object\" && d.type === \"pp:edit:reload\") {\n // Soft-Reload — re-loaded mit gleichem Suchparameter\n window.location.reload();\n }\n }\n window.addEventListener(\"message\", onMsg);\n return () => window.removeEventListener(\"message\", onMsg);\n }, [urlState.enabled]);\n\n const value = useMemo(() => urlState, [urlState]);\n return <EditModeContext.Provider value={value}>{children}</EditModeContext.Provider>;\n}\n","\"use client\";\n\nimport { Children, cloneElement, isValidElement, useCallback, type ReactElement, type ReactNode } from \"react\";\nimport { useStrapiEditMode } from \"./EditModeContext\";\nimport type { StrapiFieldType } from \"./types\";\n\ninterface StrapiFieldProps {\n /** Dot-notation path im Strapi-Document, z.B. \"hero.title\" oder \"features.0.image\" */\n path: string;\n /** Strapi field type — bestimmt welche Edit-UI im Parent geöffnet wird */\n type: StrapiFieldType;\n /** Aktueller Wert. Wird beim Click-Event mitgesendet, damit der Editor initial befüllt ist */\n value?: unknown;\n /** Inhalt (das echte gerenderte HTML). Wird im Edit-Mode mit data-attrs + Hover/Click annotiert */\n children: ReactNode;\n /** Optional: zusätzliche CSS-Klassen die im Edit-Mode angehängt werden */\n className?: string;\n}\n\nconst EDIT_ATTR_STYLE = {\n outline: \"1px dashed transparent\",\n outlineOffset: \"2px\",\n cursor: \"pointer\",\n transition: \"outline-color 0.15s ease\",\n} as const;\n\nconst EDIT_HOVER_STYLE = {\n outlineColor: \"#FA501E\",\n};\n\n/**\n * Universeller Wrapper für editierbare Strapi-Felder.\n *\n * Im normalen Mode rendert er nur die `children` — keine zusätzlichen DOM-\n * Nodes, kein Overhead. Im Edit-Mode wird das einzige Top-Level-Child mit\n * `data-pp-edit=\"<path>\"` annotiert + Click-Handler bekommt.\n *\n * Falls children mehrere Top-Level-Knoten haben, wird automatisch ein\n * `<span>` als Wrapper eingefügt.\n */\nexport function StrapiField({ path, type, value, children, className }: StrapiFieldProps) {\n const { enabled } = useStrapiEditMode();\n\n const onClick = useCallback(\n (e: React.MouseEvent) => {\n e.preventDefault();\n e.stopPropagation();\n if (typeof window === \"undefined\" || window.parent === window) return;\n const target = e.currentTarget as HTMLElement;\n const rect = target.getBoundingClientRect();\n try {\n window.parent.postMessage(\n {\n type: \"pp:edit:click\",\n path,\n fieldType: type,\n rect: {\n top: rect.top,\n left: rect.left,\n width: rect.width,\n height: rect.height,\n },\n currentValue: value,\n },\n \"*\",\n );\n } catch {\n // ignore\n }\n },\n [path, type, value],\n );\n\n if (!enabled) {\n return <>{children}</>;\n }\n\n // Wenn genau ein React-Element drin ist, klonen wir es und reichen die Edit-Attrs durch.\n const arr = Children.toArray(children);\n if (arr.length === 1 && isValidElement(arr[0])) {\n const el = arr[0] as ReactElement<Record<string, unknown>>;\n const existing = (el.props.style as React.CSSProperties) ?? {};\n const existingClass = (el.props.className as string) ?? \"\";\n return cloneElement(el, {\n \"data-pp-edit\": path,\n \"data-pp-type\": type,\n onClick,\n onMouseEnter: (e: React.MouseEvent) => {\n (e.currentTarget as HTMLElement).style.outlineColor = EDIT_HOVER_STYLE.outlineColor;\n },\n onMouseLeave: (e: React.MouseEvent) => {\n (e.currentTarget as HTMLElement).style.outlineColor = \"transparent\";\n },\n style: { ...EDIT_ATTR_STYLE, ...existing },\n className: [existingClass, className, \"pp-edit-target\"].filter(Boolean).join(\" \"),\n });\n }\n\n // Mehrere Kinder: in einen span wrappen\n return (\n <span\n data-pp-edit={path}\n data-pp-type={type}\n onClick={onClick}\n onMouseEnter={(e) => { (e.currentTarget).style.outlineColor = EDIT_HOVER_STYLE.outlineColor; }}\n onMouseLeave={(e) => { (e.currentTarget).style.outlineColor = \"transparent\"; }}\n style={EDIT_ATTR_STYLE}\n className={[\"pp-edit-target\", className].filter(Boolean).join(\" \")}\n >\n {children}\n </span>\n );\n}\n","\"use client\";\n\nimport { createElement, type ElementType, type ReactNode } from \"react\";\nimport { StrapiField } from \"./StrapiField\";\nimport type { StrapiFieldType } from \"./types\";\n\ninterface StrapiTextProps {\n path: string;\n value: string | number | null | undefined;\n /** Welcher Strapi-Feldtyp ist das? Default \"text\". Für mehrzeilige Felder \"textarea\", für Markdown \"richText\". */\n fieldType?: Extract<StrapiFieldType, \"text\" | \"textarea\" | \"richText\" | \"email\" | \"number\">;\n /** HTML-Tag um den Wert zu rendern. Default \"span\". */\n as?: ElementType;\n className?: string;\n children?: ReactNode;\n}\n\n/**\n * Inline-Text wrapper. Im normalen Mode rendert es einfach `value` im\n * gewählten Tag — identisch zu einem rohen `<h1>{value}</h1>`. Im Edit-Mode\n * wird ein Click-Handler aktiviert.\n *\n * Wenn `children` mitgegeben sind, werden diese statt `value` als Inhalt\n * verwendet — nützlich für custom Formatierung. Der Click-Wert bleibt\n * `value`.\n */\nexport function StrapiText({ path, value, fieldType = \"text\", as = \"span\", className, children }: StrapiTextProps) {\n const content = children ?? (value === null || value === undefined ? \"\" : String(value));\n const el = createElement(as, { className }, content);\n return (\n <StrapiField path={path} type={fieldType} value={value}>\n {el}\n </StrapiField>\n );\n}\n","\"use client\";\n\nimport { StrapiField } from \"./StrapiField\";\nimport type { StrapiMediaValue } from \"./types\";\n\ninterface StrapiImageProps {\n path: string;\n value: StrapiMediaValue | null | undefined;\n /** Optional: Strapi-Basis-URL für relative URLs (z.B. \"https://cms.paulpaul.studio\") */\n baseUrl?: string;\n /** Override alt-Text (default: alternativeText vom Strapi-Objekt, dann name) */\n alt?: string;\n className?: string;\n style?: React.CSSProperties;\n loading?: \"lazy\" | \"eager\";\n /** Fallback wenn value null/undefined ist */\n fallback?: React.ReactNode;\n}\n\nfunction resolveUrl(value: StrapiMediaValue, baseUrl?: string): string {\n const u = value.url;\n if (u.startsWith(\"http\")) return u;\n if (baseUrl) return `${baseUrl.replace(/\\/$/, \"\")}${u.startsWith(\"/\") ? \"\" : \"/\"}${u}`;\n return u;\n}\n\n/**\n * Wrapper für ein einzelnes Strapi-Media-Feld. Rendert `<img>` mit der\n * abgeleiteten URL. Im Edit-Mode klickbar — der Editor öffnet einen\n * MediaPicker.\n *\n * Wenn `value` null ist, wird der `fallback` gerendert (oder ein leeres\n * `<span>`).\n */\nexport function StrapiImage({ path, value, baseUrl, alt, className, style, loading = \"lazy\", fallback }: StrapiImageProps) {\n if (!value) {\n return (\n <StrapiField path={path} type=\"media\" value={null}>\n <span className={className} style={style}>{fallback ?? \"\"}</span>\n </StrapiField>\n );\n }\n return (\n <StrapiField path={path} type=\"media\" value={value}>\n <img\n src={resolveUrl(value, baseUrl)}\n alt={alt ?? value.alternativeText ?? value.name ?? \"\"}\n width={value.width}\n height={value.height}\n loading={loading}\n className={className}\n style={style}\n />\n </StrapiField>\n );\n}\n","\"use client\";\n\nimport { Fragment, type ReactNode } from \"react\";\n\ninterface StrapiListProps<T> {\n /** Pfad zum Array-Feld, z.B. \"features\" */\n path: string;\n value: T[] | null | undefined;\n /**\n * Render-Funktion pro Item. Bekommt einen `itemPath` als zweites Argument,\n * der für nested-StrapiText/Image-Calls als path-Prefix dient.\n */\n renderItem: (item: T, itemPath: string, index: number) => ReactNode;\n /** Optional: Fallback wenn value null/leer ist */\n fallback?: ReactNode;\n}\n\n/**\n * Iteriert über ein Strapi-Repeatable-Field oder eine Relation-Liste. Generiert\n * pro Item den vollständigen Edit-Pfad (z.B. \"features.0\", \"features.1\"), den\n * der Caller in nested StrapiText/Image weitergibt.\n *\n * Beispiel:\n * <StrapiList path=\"features\" value={page.features} renderItem={(f, fp) => (\n * <div>\n * <StrapiText path={`${fp}.title`} value={f.title} as=\"h3\" />\n * <StrapiImage path={`${fp}.icon`} value={f.icon} />\n * </div>\n * )} />\n */\nexport function StrapiList<T>({ path, value, renderItem, fallback }: StrapiListProps<T>) {\n if (!Array.isArray(value) || value.length === 0) {\n return <>{fallback ?? null}</>;\n }\n return (\n <>\n {value.map((item, i) => (\n <Fragment key={i}>{renderItem(item, `${path}.${i}`, i)}</Fragment>\n ))}\n </>\n );\n}\n"]}
1
+ {"version":3,"sources":["../src/EditModeContext.tsx","../src/StrapiField.tsx","../src/StrapiText.tsx","../src/StrapiImage.tsx","../src/StrapiList.tsx","../src/StrapiBlocks.tsx"],"names":["createContext","useContext","useState","useRef","useEffect","useCallback","value","useMemo","jsxs","jsx","Fragment","Children","isValidElement","cloneElement","createElement","HOVER_OUTLINE_COLOR","inner"],"mappings":"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","file":"index.cjs","sourcesContent":["\"use client\";\n\nimport { createContext, useCallback, useContext, useEffect, useMemo, useRef, useState } from \"react\";\nimport type { StrapiFieldType } from \"./types\";\n\ninterface PendingChange {\n path: string;\n fieldType: StrapiFieldType;\n value: unknown;\n}\n\ninterface EditModeState {\n enabled: boolean;\n token: string | null;\n changes: Map<string, PendingChange>;\n isDirty: boolean;\n isCommitting: boolean;\n lastCommitError: string | null;\n setChange: (path: string, fieldType: StrapiFieldType, value: unknown) => void;\n clearChange: (path: string) => void;\n commit: () => void;\n openMediaPicker: (path: string, multiple: boolean, allowedTypes: string[] | undefined, currentValue: unknown) => void;\n onMediaPicked: (path: string, listener: (value: unknown) => void) => () => void;\n}\n\nconst noop = () => {};\nconst EditModeContext = createContext<EditModeState>({\n enabled: false,\n token: null,\n changes: new Map(),\n isDirty: false,\n isCommitting: false,\n lastCommitError: null,\n setChange: noop,\n clearChange: noop,\n commit: noop,\n openMediaPicker: noop,\n onMediaPicked: () => noop,\n});\n\nexport function useStrapiEditMode(): EditModeState {\n return useContext(EditModeContext);\n}\n\ninterface ProviderProps {\n children: React.ReactNode;\n enabled?: boolean;\n}\n\nexport function StrapiEditModeProvider({ children, enabled: enabledOverride }: ProviderProps) {\n const [urlState, setUrlState] = useState<{ enabled: boolean; token: string | null }>({ enabled: false, token: null });\n const [changes, setChanges] = useState<Map<string, PendingChange>>(new Map());\n const [isCommitting, setIsCommitting] = useState(false);\n const [lastCommitError, setLastCommitError] = useState<string | null>(null);\n const mediaPickListeners = useRef<Map<string, (v: unknown) => void>>(new Map());\n\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n if (enabledOverride !== undefined) {\n setUrlState({ enabled: enabledOverride, token: null });\n return;\n }\n try {\n const params = new URLSearchParams(window.location.search);\n const edit = params.get(\"__pp_edit\") === \"1\";\n const token = params.get(\"__pp_token\");\n setUrlState({ enabled: edit, token });\n } catch {\n // ignore\n }\n }, [enabledOverride]);\n\n useEffect(() => {\n if (typeof document === \"undefined\") return;\n document.documentElement.classList.toggle(\"pp-edit-active\", urlState.enabled);\n }, [urlState.enabled]);\n\n // Ready ping to parent\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n if (!urlState.enabled) return;\n if (window.parent === window) return;\n try {\n window.parent.postMessage({ type: \"pp:edit:ready\", href: window.location.href }, \"*\");\n } catch { /* ignore */ }\n }, [urlState.enabled]);\n\n // Listen for parent → iframe messages\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n if (!urlState.enabled) return;\n function onMsg(e: MessageEvent) {\n const d = e.data;\n if (!d || typeof d !== \"object\") return;\n if (d.type === \"pp:edit:reload\") {\n window.location.reload();\n return;\n }\n if (d.type === \"pp:edit:commit-result\") {\n setIsCommitting(false);\n if (d.ok) {\n setChanges(new Map());\n setLastCommitError(null);\n } else {\n setLastCommitError(d.error || \"Speichern fehlgeschlagen\");\n }\n return;\n }\n if (d.type === \"pp:edit:media-picked\") {\n const listener = mediaPickListeners.current.get(d.path);\n if (listener) {\n listener(d.value);\n mediaPickListeners.current.delete(d.path);\n }\n return;\n }\n }\n window.addEventListener(\"message\", onMsg);\n return () => window.removeEventListener(\"message\", onMsg);\n }, [urlState.enabled]);\n\n const setChange = useCallback((path: string, fieldType: StrapiFieldType, value: unknown) => {\n setChanges((prev) => {\n const next = new Map(prev);\n next.set(path, { path, fieldType, value });\n return next;\n });\n setLastCommitError(null);\n // Optional: send live change event for future \"show preview on parent\"\n if (typeof window !== \"undefined\" && window.parent !== window) {\n try {\n window.parent.postMessage({ type: \"pp:edit:change\", path, fieldType, value }, \"*\");\n } catch { /* ignore */ }\n }\n }, []);\n\n const clearChange = useCallback((path: string) => {\n setChanges((prev) => {\n const next = new Map(prev);\n next.delete(path);\n return next;\n });\n }, []);\n\n const commit = useCallback(() => {\n if (changes.size === 0) return;\n if (typeof window === \"undefined\" || window.parent === window) return;\n setIsCommitting(true);\n setLastCommitError(null);\n try {\n window.parent.postMessage(\n {\n type: \"pp:edit:commit-request\",\n changes: Array.from(changes.values()),\n },\n \"*\",\n );\n } catch (err) {\n setIsCommitting(false);\n setLastCommitError(err instanceof Error ? err.message : String(err));\n }\n }, [changes]);\n\n const openMediaPicker = useCallback((path: string, multiple: boolean, allowedTypes: string[] | undefined, currentValue: unknown) => {\n if (typeof window === \"undefined\" || window.parent === window) return;\n try {\n window.parent.postMessage(\n {\n type: \"pp:edit:open-media-picker\",\n path, multiple, allowedTypes, currentValue,\n },\n \"*\",\n );\n } catch { /* ignore */ }\n }, []);\n\n const onMediaPicked = useCallback((path: string, listener: (v: unknown) => void) => {\n mediaPickListeners.current.set(path, listener);\n return () => { mediaPickListeners.current.delete(path); };\n }, []);\n\n const value = useMemo<EditModeState>(() => ({\n enabled: urlState.enabled,\n token: urlState.token,\n changes,\n isDirty: changes.size > 0,\n isCommitting,\n lastCommitError,\n setChange,\n clearChange,\n commit,\n openMediaPicker,\n onMediaPicked,\n }), [urlState, changes, isCommitting, lastCommitError, setChange, clearChange, commit, openMediaPicker, onMediaPicked]);\n\n return (\n <EditModeContext.Provider value={value}>\n {children}\n {urlState.enabled && <SaveBar />}\n </EditModeContext.Provider>\n );\n}\n\nfunction SaveBar() {\n const { isDirty, isCommitting, lastCommitError, changes, commit } = useStrapiEditMode();\n if (!isDirty && !lastCommitError && !isCommitting) return null;\n\n return (\n <div\n style={{\n position: \"fixed\",\n bottom: 24,\n right: 24,\n zIndex: 999999,\n fontFamily: \"system-ui, -apple-system, sans-serif\",\n }}\n >\n {lastCommitError && (\n <div style={{\n background: \"#FA501E\",\n color: \"#000\",\n padding: \"8px 14px\",\n marginBottom: 8,\n fontSize: 13,\n fontWeight: 600,\n maxWidth: 320,\n }}>\n {lastCommitError}\n </div>\n )}\n <button\n type=\"button\"\n onClick={commit}\n disabled={isCommitting || !isDirty}\n style={{\n background: isCommitting ? \"#666\" : \"#000\",\n color: \"#EBC6DF\",\n border: \"2px solid #000\",\n padding: \"14px 22px\",\n fontSize: 13,\n fontWeight: 700,\n textTransform: \"uppercase\",\n letterSpacing: \"0.2em\",\n cursor: isCommitting ? \"wait\" : \"pointer\",\n display: \"flex\",\n alignItems: \"center\",\n gap: 10,\n boxShadow: \"0 4px 12px rgba(0,0,0,0.3)\",\n transition: \"background 0.15s ease\",\n }}\n onMouseEnter={(e) => {\n if (!isCommitting && isDirty) {\n e.currentTarget.style.background = \"#FA501E\";\n e.currentTarget.style.color = \"#000\";\n }\n }}\n onMouseLeave={(e) => {\n if (!isCommitting && isDirty) {\n e.currentTarget.style.background = \"#000\";\n e.currentTarget.style.color = \"#EBC6DF\";\n }\n }}\n >\n {isCommitting ? (\n <span>Speichere…</span>\n ) : (\n <>\n <span style={{\n background: \"#FA501E\", color: \"#000\",\n width: 22, height: 22, borderRadius: \"50%\",\n display: \"inline-flex\", alignItems: \"center\", justifyContent: \"center\",\n fontSize: 11, fontWeight: 700,\n }}>{changes.size}</span>\n <span>Speichern</span>\n </>\n )}\n </button>\n </div>\n );\n}\n","\"use client\";\n\nimport { Children, cloneElement, isValidElement, useEffect, useRef, type ReactElement, type ReactNode } from \"react\";\nimport { useStrapiEditMode } from \"./EditModeContext\";\nimport type { StrapiFieldType } from \"./types\";\n\ninterface StrapiFieldProps {\n path: string;\n type: StrapiFieldType;\n value?: unknown;\n children: ReactNode;\n className?: string;\n}\n\nconst EDITABLE_TYPES = new Set<StrapiFieldType>([\n \"text\",\n \"textarea\",\n \"richText\",\n \"email\",\n \"number\",\n]);\n\nconst HOVER_OUTLINE_COLOR = \"#FA501E\";\n\n/**\n * Universeller Wrapper. Im Normal-Mode: rendert children unverändert.\n *\n * Im Edit-Mode:\n * - Für text/textarea/richText/email/number: macht das Top-Level-Child\n * `contenteditable`, mit dünnem orange Hover-Outline, sammelt Wert-Änderungen\n * in den globalen Edit-Context, der dann beim Klick auf \"Speichern\" alle\n * Änderungen in einem Roundtrip ans Parent schickt.\n * - Für andere Typen (media/select/checkbox): trägt ein Click-Handler bzw.\n * ein Overlay (Implementierung in jeweiligen Sugar-Komponenten).\n */\nexport function StrapiField({ path, type, value, children, className }: StrapiFieldProps) {\n const ctx = useStrapiEditMode();\n const ref = useRef<HTMLElement | null>(null);\n const initialText = useRef<string | undefined>(undefined);\n\n // Captured initial textContent on first mount in edit-mode (for revert-detection)\n useEffect(() => {\n if (!ctx.enabled) return;\n if (!EDITABLE_TYPES.has(type)) return;\n if (initialText.current === undefined && ref.current) {\n initialText.current = ref.current.textContent ?? \"\";\n }\n }, [ctx.enabled, type]);\n\n if (!ctx.enabled) {\n return <>{children}</>;\n }\n\n const arr = Children.toArray(children);\n const isTextual = EDITABLE_TYPES.has(type);\n\n if (arr.length === 1 && isValidElement(arr[0])) {\n const el = arr[0] as ReactElement<Record<string, unknown>>;\n const existingStyle = (el.props.style as React.CSSProperties) ?? {};\n const existingClass = (el.props.className as string) ?? \"\";\n\n const inlineStyle: React.CSSProperties = {\n outline: \"1px dashed transparent\",\n outlineOffset: 2,\n transition: \"outline-color 0.15s ease\",\n cursor: isTextual ? \"text\" : \"pointer\",\n ...existingStyle,\n };\n\n const props: Record<string, unknown> = {\n \"data-pp-edit\": path,\n \"data-pp-type\": type,\n ref: (node: HTMLElement | null) => { ref.current = node; },\n className: [existingClass, className, \"pp-edit-target\"].filter(Boolean).join(\" \"),\n style: inlineStyle,\n onMouseEnter: (e: React.MouseEvent) => {\n (e.currentTarget as HTMLElement).style.outlineColor = HOVER_OUTLINE_COLOR;\n },\n onMouseLeave: (e: React.MouseEvent) => {\n (e.currentTarget as HTMLElement).style.outlineColor = \"transparent\";\n },\n };\n\n if (isTextual) {\n props.contentEditable = \"plaintext-only\";\n props.suppressContentEditableWarning = true;\n props.onFocus = (e: React.FocusEvent) => {\n (e.currentTarget as HTMLElement).style.outlineColor = HOVER_OUTLINE_COLOR;\n (e.currentTarget as HTMLElement).style.outlineStyle = \"solid\";\n };\n props.onBlur = (e: React.FocusEvent) => {\n const target = e.currentTarget as HTMLElement;\n target.style.outlineColor = \"transparent\";\n target.style.outlineStyle = \"dashed\";\n const newText = target.textContent ?? \"\";\n const original = typeof value === \"string\" ? value : (initialText.current ?? \"\");\n if (newText !== original) {\n const newValue = type === \"number\" ? (newText === \"\" ? null : Number(newText)) : newText;\n ctx.setChange(path, type, newValue);\n }\n };\n props.onKeyDown = (e: React.KeyboardEvent) => {\n // Enter ohne Shift = blur (commit local). Für richText/textarea kann user mit Shift+Enter Zeilenumbruch erzeugen.\n if (e.key === \"Enter\" && !e.shiftKey && type === \"text\") {\n e.preventDefault();\n (e.currentTarget as HTMLElement).blur();\n }\n if (e.key === \"Escape\") {\n e.preventDefault();\n (e.currentTarget as HTMLElement).blur();\n }\n };\n } else {\n // Non-editable types: click handler wird von Sugar-Komponenten (StrapiImage etc.) gesetzt\n }\n\n return cloneElement(el, props);\n }\n\n // Fallback: span wrap\n return (\n <span\n data-pp-edit={path}\n data-pp-type={type}\n className={[\"pp-edit-target\", className].filter(Boolean).join(\" \")}\n style={{ outline: \"1px dashed transparent\", outlineOffset: 2 }}\n >\n {children}\n </span>\n );\n}\n","\"use client\";\n\nimport { createElement, type ElementType, type ReactNode } from \"react\";\nimport { StrapiField } from \"./StrapiField\";\nimport type { StrapiFieldType } from \"./types\";\n\ninterface StrapiTextProps {\n path: string;\n value: string | number | null | undefined;\n fieldType?: Extract<StrapiFieldType, \"text\" | \"textarea\" | \"richText\" | \"email\" | \"number\">;\n as?: ElementType;\n className?: string;\n children?: ReactNode;\n}\n\nexport function StrapiText({ path, value, fieldType = \"text\", as = \"span\", className, children }: StrapiTextProps) {\n const content = children ?? (value === null || value === undefined ? \"\" : String(value));\n const el = createElement(as, { className }, content);\n return (\n <StrapiField path={path} type={fieldType} value={value}>\n {el}\n </StrapiField>\n );\n}\n","\"use client\";\n\nimport { useRef } from \"react\";\nimport { useStrapiEditMode } from \"./EditModeContext\";\nimport type { StrapiMediaValue } from \"./types\";\n\ninterface StrapiImageProps {\n path: string;\n value: StrapiMediaValue | null | undefined;\n baseUrl?: string;\n alt?: string;\n className?: string;\n style?: React.CSSProperties;\n loading?: \"lazy\" | \"eager\";\n fallback?: React.ReactNode;\n allowedTypes?: string[];\n multiple?: boolean;\n}\n\nfunction resolveUrl(value: StrapiMediaValue, baseUrl?: string): string {\n if (value.url.startsWith(\"http\")) return value.url;\n if (baseUrl) return `${baseUrl.replace(/\\/$/, \"\")}${value.url.startsWith(\"/\") ? \"\" : \"/\"}${value.url}`;\n return value.url;\n}\n\nconst HOVER_OUTLINE_COLOR = \"#FA501E\";\n\nexport function StrapiImage({ path, value, baseUrl, alt, className, style, loading = \"lazy\", fallback, allowedTypes, multiple = false }: StrapiImageProps) {\n const ctx = useStrapiEditMode();\n const containerRef = useRef<HTMLSpanElement>(null);\n const imgRef = useRef<HTMLImageElement>(null);\n\n const handleClick = (e: React.MouseEvent) => {\n if (!ctx.enabled) return;\n e.preventDefault();\n e.stopPropagation();\n const unsubscribe = ctx.onMediaPicked(path, (picked) => {\n ctx.setChange(path, \"media\", picked);\n unsubscribe();\n // Optimistic UI: swap src in-place wenn picked = single file mit url\n if (imgRef.current && picked && typeof picked === \"object\" && !Array.isArray(picked) && \"url\" in (picked as Record<string, unknown>)) {\n imgRef.current.src = resolveUrl(picked as StrapiMediaValue, baseUrl);\n }\n });\n ctx.openMediaPicker(path, multiple, allowedTypes, value);\n };\n\n if (!value) {\n if (!ctx.enabled) return <>{fallback ?? null}</>;\n return (\n <span\n ref={containerRef}\n data-pp-edit={path}\n data-pp-type=\"media\"\n onClick={handleClick}\n onMouseEnter={(e) => { (e.currentTarget as HTMLElement).style.outlineColor = HOVER_OUTLINE_COLOR; }}\n onMouseLeave={(e) => { (e.currentTarget as HTMLElement).style.outlineColor = \"transparent\"; }}\n style={{\n display: \"inline-flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n background: \"rgba(0,0,0,0.05)\",\n border: \"1px dashed rgba(0,0,0,0.3)\",\n outline: \"1px dashed transparent\",\n outlineOffset: 2,\n cursor: \"pointer\",\n padding: \"20px 28px\",\n fontSize: 12,\n fontWeight: 700,\n textTransform: \"uppercase\",\n letterSpacing: \"0.15em\",\n color: \"#000\",\n transition: \"outline-color 0.15s ease\",\n }}\n >\n + Bild hinzufügen\n </span>\n );\n }\n\n const imgEl = (\n <img\n ref={imgRef}\n src={resolveUrl(value, baseUrl)}\n alt={alt ?? value.alternativeText ?? value.name ?? \"\"}\n width={value.width}\n height={value.height}\n loading={loading}\n className={className}\n style={style}\n />\n );\n\n if (!ctx.enabled) return imgEl;\n\n return (\n <span\n ref={containerRef}\n data-pp-edit={path}\n data-pp-type=\"media\"\n style={{ position: \"relative\", display: \"inline-block\", outline: \"1px dashed transparent\", outlineOffset: 2, transition: \"outline-color 0.15s ease\" }}\n onMouseEnter={(e) => {\n (e.currentTarget as HTMLElement).style.outlineColor = HOVER_OUTLINE_COLOR;\n const btn = (e.currentTarget as HTMLElement).querySelector(\"[data-pp-edit-btn]\") as HTMLElement | null;\n if (btn) btn.style.opacity = \"1\";\n }}\n onMouseLeave={(e) => {\n (e.currentTarget as HTMLElement).style.outlineColor = \"transparent\";\n const btn = (e.currentTarget as HTMLElement).querySelector(\"[data-pp-edit-btn]\") as HTMLElement | null;\n if (btn) btn.style.opacity = \"0\";\n }}\n >\n {imgEl}\n <button\n type=\"button\"\n data-pp-edit-btn\n onClick={handleClick}\n style={{\n position: \"absolute\",\n top: 8,\n right: 8,\n background: \"#000\",\n color: \"#EBC6DF\",\n border: \"2px solid #000\",\n padding: \"6px 10px\",\n fontSize: 11,\n fontWeight: 700,\n textTransform: \"uppercase\",\n letterSpacing: \"0.15em\",\n cursor: \"pointer\",\n opacity: 0,\n transition: \"opacity 0.15s ease, background 0.15s ease\",\n fontFamily: \"system-ui, -apple-system, sans-serif\",\n }}\n onMouseEnter={(e) => { e.currentTarget.style.background = \"#FA501E\"; e.currentTarget.style.color = \"#000\"; }}\n onMouseLeave={(e) => { e.currentTarget.style.background = \"#000\"; e.currentTarget.style.color = \"#EBC6DF\"; }}\n >\n Ändern\n </button>\n </span>\n );\n}\n","\"use client\";\n\nimport { Fragment, type ReactNode } from \"react\";\n\ninterface StrapiListProps<T> {\n /** Pfad zum Array-Feld, z.B. \"features\" */\n path: string;\n value: T[] | null | undefined;\n /**\n * Render-Funktion pro Item. Bekommt einen `itemPath` als zweites Argument,\n * der für nested-StrapiText/Image-Calls als path-Prefix dient.\n */\n renderItem: (item: T, itemPath: string, index: number) => ReactNode;\n /** Optional: Fallback wenn value null/leer ist */\n fallback?: ReactNode;\n}\n\n/**\n * Iteriert über ein Strapi-Repeatable-Field oder eine Relation-Liste. Generiert\n * pro Item den vollständigen Edit-Pfad (z.B. \"features.0\", \"features.1\"), den\n * der Caller in nested StrapiText/Image weitergibt.\n *\n * Beispiel:\n * <StrapiList path=\"features\" value={page.features} renderItem={(f, fp) => (\n * <div>\n * <StrapiText path={`${fp}.title`} value={f.title} as=\"h3\" />\n * <StrapiImage path={`${fp}.icon`} value={f.icon} />\n * </div>\n * )} />\n */\nexport function StrapiList<T>({ path, value, renderItem, fallback }: StrapiListProps<T>) {\n if (!Array.isArray(value) || value.length === 0) {\n return <>{fallback ?? null}</>;\n }\n return (\n <>\n {value.map((item, i) => (\n <Fragment key={i}>{renderItem(item, `${path}.${i}`, i)}</Fragment>\n ))}\n </>\n );\n}\n","\"use client\";\n\nimport { Fragment, useEffect, useRef, type ReactNode } from \"react\";\nimport { useStrapiEditMode } from \"./EditModeContext\";\n\n// ── Strapi v5 Blocks types ──────────────────────────────────────────\ninterface TextLeaf {\n type: \"text\";\n text: string;\n bold?: boolean;\n italic?: boolean;\n underline?: boolean;\n strikethrough?: boolean;\n code?: boolean;\n}\n\ninterface LinkNode {\n type: \"link\";\n url: string;\n children: TextLeaf[];\n}\n\ntype InlineNode = TextLeaf | LinkNode;\n\ninterface ListItemBlock {\n type: \"list-item\";\n children: InlineNode[];\n}\n\ninterface ParagraphBlock { type: \"paragraph\"; children: InlineNode[]; }\ninterface HeadingBlock { type: \"heading\"; level: 1 | 2 | 3 | 4 | 5 | 6; children: InlineNode[]; }\ninterface QuoteBlock { type: \"quote\"; children: InlineNode[]; }\ninterface ListBlock { type: \"list\"; format: \"ordered\" | \"unordered\"; children: ListItemBlock[]; }\ninterface CodeBlock { type: \"code\"; children: InlineNode[] }\ninterface ImageBlock { type: \"image\"; image: { url: string; alternativeText?: string } }\n\ntype Block = ParagraphBlock | HeadingBlock | QuoteBlock | ListBlock | CodeBlock | ImageBlock;\n\ninterface Props {\n path: string;\n value: unknown;\n className?: string;\n}\n\nconst HOVER_OUTLINE = \"#FA501E\";\n\n// ── Read-Mode renderer ──────────────────────────────────────────────\nfunction renderInline(node: InlineNode, key: number): ReactNode {\n if (node.type === \"link\") {\n return (\n <a key={key} href={node.url} style={{ fontWeight: 700, textDecoration: \"underline\", textUnderlineOffset: 4 }}>\n {(node.children ?? []).map((c, i) => renderInline(c, i))}\n </a>\n );\n }\n let el: ReactNode = node.text;\n if (node.code) el = <code style={{ background: \"rgba(0,0,0,0.08)\", padding: \"0 4px\", fontFamily: \"ui-monospace, monospace\", fontSize: \"0.9em\" }}>{el}</code>;\n if (node.bold) el = <strong>{el}</strong>;\n if (node.italic) el = <em>{el}</em>;\n if (node.underline) el = <u>{el}</u>;\n if (node.strikethrough) el = <s>{el}</s>;\n return <Fragment key={key}>{el}</Fragment>;\n}\n\nfunction renderBlock(b: Block, key: number): ReactNode {\n if (b.type === \"paragraph\") {\n return <p key={key}>{(b.children ?? []).map((c, i) => renderInline(c, i))}</p>;\n }\n if (b.type === \"heading\") {\n const Tag = (`h${Math.min(Math.max(b.level ?? 2, 1), 6)}`) as \"h1\" | \"h2\" | \"h3\" | \"h4\" | \"h5\" | \"h6\";\n return <Tag key={key}>{(b.children ?? []).map((c, i) => renderInline(c, i))}</Tag>;\n }\n if (b.type === \"quote\") {\n return <blockquote key={key}>{(b.children ?? []).map((c, i) => renderInline(c, i))}</blockquote>;\n }\n if (b.type === \"list\") {\n const ListTag = b.format === \"ordered\" ? \"ol\" : \"ul\";\n return (\n <ListTag key={key}>\n {(b.children ?? []).map((li, i) => (\n <li key={i}>{(li.children ?? []).map((c, j) => renderInline(c, j))}</li>\n ))}\n </ListTag>\n );\n }\n if (b.type === \"code\") {\n return <pre key={key}><code>{(b.children ?? []).map((c) => (c as TextLeaf).text).join(\"\")}</code></pre>;\n }\n if (b.type === \"image\") {\n return <img key={key} src={b.image.url} alt={b.image.alternativeText ?? \"\"} />;\n }\n return null;\n}\n\n// ── DOM → Blocks parser (für Edit-Mode commit) ─────────────────────\nfunction parseInline(node: Node): InlineNode[] {\n const out: InlineNode[] = [];\n for (const child of Array.from(node.childNodes)) {\n if (child.nodeType === Node.TEXT_NODE) {\n const text = child.textContent ?? \"\";\n if (text) out.push({ type: \"text\", text });\n } else if (child.nodeType === Node.ELEMENT_NODE) {\n const el = child as HTMLElement;\n const tag = el.tagName.toLowerCase();\n if (tag === \"br\") {\n out.push({ type: \"text\", text: \"\\n\" });\n continue;\n }\n if (tag === \"a\") {\n const inner = parseInline(el).filter((n): n is TextLeaf => n.type === \"text\");\n out.push({ type: \"link\", url: el.getAttribute(\"href\") ?? \"\", children: inner.length > 0 ? inner : [{ type: \"text\", text: el.textContent ?? \"\" }] });\n continue;\n }\n // Inline formatting wrappers\n const inner = parseInline(el);\n const flagged = inner.map((n) => {\n if (n.type !== \"text\") return n;\n const t = { ...n };\n if (tag === \"strong\" || tag === \"b\") t.bold = true;\n if (tag === \"em\" || tag === \"i\") t.italic = true;\n if (tag === \"u\") t.underline = true;\n if (tag === \"s\" || tag === \"strike\" || tag === \"del\") t.strikethrough = true;\n if (tag === \"code\") t.code = true;\n return t;\n });\n out.push(...flagged);\n }\n }\n return out;\n}\n\nfunction parseBlocksFromDOM(root: HTMLElement): Block[] {\n const blocks: Block[] = [];\n for (const child of Array.from(root.children)) {\n const el = child as HTMLElement;\n const tag = el.tagName.toLowerCase();\n if (tag === \"p\") {\n blocks.push({ type: \"paragraph\", children: parseInline(el) });\n } else if (/^h[1-6]$/.test(tag)) {\n const level = parseInt(tag.slice(1), 10) as 1 | 2 | 3 | 4 | 5 | 6;\n blocks.push({ type: \"heading\", level, children: parseInline(el) });\n } else if (tag === \"blockquote\") {\n blocks.push({ type: \"quote\", children: parseInline(el) });\n } else if (tag === \"ul\" || tag === \"ol\") {\n const items: ListItemBlock[] = [];\n for (const li of Array.from(el.querySelectorAll(\":scope > li\"))) {\n items.push({ type: \"list-item\", children: parseInline(li) });\n }\n blocks.push({ type: \"list\", format: tag === \"ol\" ? \"ordered\" : \"unordered\", children: items });\n } else if (tag === \"pre\") {\n blocks.push({ type: \"code\", children: [{ type: \"text\", text: el.textContent ?? \"\" }] });\n } else if (tag === \"img\") {\n blocks.push({ type: \"image\", image: { url: el.getAttribute(\"src\") ?? \"\", alternativeText: el.getAttribute(\"alt\") ?? \"\" } });\n } else {\n // unknown tag → paragraph mit text\n blocks.push({ type: \"paragraph\", children: parseInline(el) });\n }\n }\n return blocks;\n}\n\nexport function StrapiBlocks({ path, value, className }: Props) {\n const ctx = useStrapiEditMode();\n const containerRef = useRef<HTMLDivElement>(null);\n const initialValue = useRef<Block[] | null>(null);\n\n const blocks: Block[] = Array.isArray(value) ? (value as Block[]) : [];\n\n useEffect(() => {\n if (initialValue.current === null) initialValue.current = blocks;\n }, [blocks]);\n\n if (!ctx.enabled) {\n return (\n <div className={className}>\n {blocks.map((b, i) => renderBlock(b, i))}\n </div>\n );\n }\n\n return (\n <div\n ref={containerRef}\n className={className}\n contentEditable\n suppressContentEditableWarning\n data-pp-edit={path}\n data-pp-type=\"blocks\"\n style={{\n outline: \"1px dashed transparent\",\n outlineOffset: 4,\n transition: \"outline-color 0.15s ease\",\n minHeight: \"1em\",\n }}\n onMouseEnter={(e) => { (e.currentTarget as HTMLElement).style.outlineColor = HOVER_OUTLINE; }}\n onMouseLeave={(e) => { (e.currentTarget as HTMLElement).style.outlineColor = \"transparent\"; }}\n onFocus={(e) => { (e.currentTarget as HTMLElement).style.outlineColor = HOVER_OUTLINE; (e.currentTarget as HTMLElement).style.outlineStyle = \"solid\"; }}\n onBlur={(e) => {\n const root = e.currentTarget;\n const parsed = parseBlocksFromDOM(root);\n const before = JSON.stringify(initialValue.current);\n const after = JSON.stringify(parsed);\n if (before !== after) {\n ctx.setChange(path, \"blocks\", parsed);\n }\n (e.currentTarget as HTMLElement).style.outlineColor = \"transparent\";\n (e.currentTarget as HTMLElement).style.outlineStyle = \"dashed\";\n }}\n >\n {blocks.map((b, i) => renderBlock(b, i))}\n </div>\n );\n}\n"]}
package/dist/index.d.cts CHANGED
@@ -1,29 +1,6 @@
1
1
  import * as react_jsx_runtime from 'react/jsx-runtime';
2
2
  import { ReactNode, ElementType } from 'react';
3
3
 
4
- interface EditModeState {
5
- enabled: boolean;
6
- token: string | null;
7
- }
8
- declare function useStrapiEditMode(): EditModeState;
9
- interface ProviderProps {
10
- children: React.ReactNode;
11
- /**
12
- * Override automatic URL-based detection. If `true`/`false` ist gesetzt,
13
- * wird der URL-Parameter ignoriert.
14
- */
15
- enabled?: boolean;
16
- }
17
- /**
18
- * Provider, der den Edit-Mode aus `?__pp_edit=1&__pp_token=<jwt>` der URL
19
- * liest und an Kinder via Context weitergibt. Token wird sicherheitshalber
20
- * NICHT ans CSR exponiert über Props oder DOM — er bleibt nur im Context.
21
- *
22
- * Im Edit-Mode wird ausserdem eine kleine globale CSS-Klasse `pp-edit-active`
23
- * am `<html>` gesetzt, sodass Customer-Sites optional CSS-Hooks setzen können.
24
- */
25
- declare function StrapiEditModeProvider({ children, enabled: enabledOverride }: ProviderProps): react_jsx_runtime.JSX.Element;
26
-
27
4
  type StrapiFieldType = "text" | "textarea" | "richText" | "number" | "checkbox" | "date" | "email" | "media" | "select" | "relation" | "group" | "array" | "blocks" | "json" | "unknown";
28
5
  interface StrapiMediaValue {
29
6
  id?: number;
@@ -35,94 +12,115 @@ interface StrapiMediaValue {
35
12
  width?: number;
36
13
  height?: number;
37
14
  }
38
- interface StrapiEditReadyMessage {
15
+ interface MsgEditReady {
39
16
  type: "pp:edit:ready";
40
17
  href: string;
41
18
  }
42
- interface StrapiEditClickMessage {
43
- type: "pp:edit:click";
19
+ interface MsgEditChange {
20
+ type: "pp:edit:change";
44
21
  path: string;
45
22
  fieldType: StrapiFieldType;
46
- rect: {
47
- top: number;
48
- left: number;
49
- width: number;
50
- height: number;
51
- };
23
+ value: unknown;
24
+ }
25
+ interface MsgEditCommitRequest {
26
+ type: "pp:edit:commit-request";
27
+ changes: Array<{
28
+ path: string;
29
+ fieldType: StrapiFieldType;
30
+ value: unknown;
31
+ }>;
32
+ }
33
+ interface MsgEditOpenMediaPicker {
34
+ type: "pp:edit:open-media-picker";
35
+ path: string;
36
+ multiple: boolean;
37
+ allowedTypes?: string[];
52
38
  currentValue: unknown;
53
39
  }
54
- interface StrapiEditReloadMessage {
40
+ interface MsgEditReload {
55
41
  type: "pp:edit:reload";
56
42
  }
57
- type StrapiEditMessage = StrapiEditReadyMessage | StrapiEditClickMessage | StrapiEditReloadMessage;
43
+ interface MsgEditCommitResult {
44
+ type: "pp:edit:commit-result";
45
+ ok: boolean;
46
+ error?: string;
47
+ savedPaths?: string[];
48
+ }
49
+ interface MsgEditMediaPicked {
50
+ type: "pp:edit:media-picked";
51
+ path: string;
52
+ value: StrapiMediaValue | StrapiMediaValue[] | null;
53
+ }
54
+ type StrapiEditMessage = MsgEditReady | MsgEditChange | MsgEditCommitRequest | MsgEditOpenMediaPicker | MsgEditReload | MsgEditCommitResult | MsgEditMediaPicked;
55
+
56
+ interface PendingChange {
57
+ path: string;
58
+ fieldType: StrapiFieldType;
59
+ value: unknown;
60
+ }
61
+ interface EditModeState {
62
+ enabled: boolean;
63
+ token: string | null;
64
+ changes: Map<string, PendingChange>;
65
+ isDirty: boolean;
66
+ isCommitting: boolean;
67
+ lastCommitError: string | null;
68
+ setChange: (path: string, fieldType: StrapiFieldType, value: unknown) => void;
69
+ clearChange: (path: string) => void;
70
+ commit: () => void;
71
+ openMediaPicker: (path: string, multiple: boolean, allowedTypes: string[] | undefined, currentValue: unknown) => void;
72
+ onMediaPicked: (path: string, listener: (value: unknown) => void) => () => void;
73
+ }
74
+ declare function useStrapiEditMode(): EditModeState;
75
+ interface ProviderProps {
76
+ children: React.ReactNode;
77
+ enabled?: boolean;
78
+ }
79
+ declare function StrapiEditModeProvider({ children, enabled: enabledOverride }: ProviderProps): react_jsx_runtime.JSX.Element;
58
80
 
59
81
  interface StrapiFieldProps {
60
- /** Dot-notation path im Strapi-Document, z.B. "hero.title" oder "features.0.image" */
61
82
  path: string;
62
- /** Strapi field type — bestimmt welche Edit-UI im Parent geöffnet wird */
63
83
  type: StrapiFieldType;
64
- /** Aktueller Wert. Wird beim Click-Event mitgesendet, damit der Editor initial befüllt ist */
65
84
  value?: unknown;
66
- /** Inhalt (das echte gerenderte HTML). Wird im Edit-Mode mit data-attrs + Hover/Click annotiert */
67
85
  children: ReactNode;
68
- /** Optional: zusätzliche CSS-Klassen die im Edit-Mode angehängt werden */
69
86
  className?: string;
70
87
  }
71
88
  /**
72
- * Universeller Wrapper für editierbare Strapi-Felder.
89
+ * Universeller Wrapper. Im Normal-Mode: rendert children unverändert.
73
90
  *
74
- * Im normalen Mode rendert er nur die `children` — keine zusätzlichen DOM-
75
- * Nodes, kein Overhead. Im Edit-Mode wird das einzige Top-Level-Child mit
76
- * `data-pp-edit="<path>"` annotiert + Click-Handler bekommt.
77
- *
78
- * Falls children mehrere Top-Level-Knoten haben, wird automatisch ein
79
- * `<span>` als Wrapper eingefügt.
91
+ * Im Edit-Mode:
92
+ * - Für text/textarea/richText/email/number: macht das Top-Level-Child
93
+ * `contenteditable`, mit dünnem orange Hover-Outline, sammelt Wert-Änderungen
94
+ * in den globalen Edit-Context, der dann beim Klick auf "Speichern" alle
95
+ * Änderungen in einem Roundtrip ans Parent schickt.
96
+ * - Für andere Typen (media/select/checkbox): trägt ein Click-Handler bzw.
97
+ * ein Overlay (Implementierung in jeweiligen Sugar-Komponenten).
80
98
  */
81
99
  declare function StrapiField({ path, type, value, children, className }: StrapiFieldProps): react_jsx_runtime.JSX.Element;
82
100
 
83
101
  interface StrapiTextProps {
84
102
  path: string;
85
103
  value: string | number | null | undefined;
86
- /** Welcher Strapi-Feldtyp ist das? Default "text". Für mehrzeilige Felder "textarea", für Markdown "richText". */
87
104
  fieldType?: Extract<StrapiFieldType, "text" | "textarea" | "richText" | "email" | "number">;
88
- /** HTML-Tag um den Wert zu rendern. Default "span". */
89
105
  as?: ElementType;
90
106
  className?: string;
91
107
  children?: ReactNode;
92
108
  }
93
- /**
94
- * Inline-Text wrapper. Im normalen Mode rendert es einfach `value` im
95
- * gewählten Tag — identisch zu einem rohen `<h1>{value}</h1>`. Im Edit-Mode
96
- * wird ein Click-Handler aktiviert.
97
- *
98
- * Wenn `children` mitgegeben sind, werden diese statt `value` als Inhalt
99
- * verwendet — nützlich für custom Formatierung. Der Click-Wert bleibt
100
- * `value`.
101
- */
102
109
  declare function StrapiText({ path, value, fieldType, as, className, children }: StrapiTextProps): react_jsx_runtime.JSX.Element;
103
110
 
104
111
  interface StrapiImageProps {
105
112
  path: string;
106
113
  value: StrapiMediaValue | null | undefined;
107
- /** Optional: Strapi-Basis-URL für relative URLs (z.B. "https://cms.paulpaul.studio") */
108
114
  baseUrl?: string;
109
- /** Override alt-Text (default: alternativeText vom Strapi-Objekt, dann name) */
110
115
  alt?: string;
111
116
  className?: string;
112
117
  style?: React.CSSProperties;
113
118
  loading?: "lazy" | "eager";
114
- /** Fallback wenn value null/undefined ist */
115
119
  fallback?: React.ReactNode;
120
+ allowedTypes?: string[];
121
+ multiple?: boolean;
116
122
  }
117
- /**
118
- * Wrapper für ein einzelnes Strapi-Media-Feld. Rendert `<img>` mit der
119
- * abgeleiteten URL. Im Edit-Mode klickbar — der Editor öffnet einen
120
- * MediaPicker.
121
- *
122
- * Wenn `value` null ist, wird der `fallback` gerendert (oder ein leeres
123
- * `<span>`).
124
- */
125
- declare function StrapiImage({ path, value, baseUrl, alt, className, style, loading, fallback }: StrapiImageProps): react_jsx_runtime.JSX.Element;
123
+ declare function StrapiImage({ path, value, baseUrl, alt, className, style, loading, fallback, allowedTypes, multiple }: StrapiImageProps): react_jsx_runtime.JSX.Element;
126
124
 
127
125
  interface StrapiListProps<T> {
128
126
  /** Pfad zum Array-Feld, z.B. "features" */
@@ -151,4 +149,11 @@ interface StrapiListProps<T> {
151
149
  */
152
150
  declare function StrapiList<T>({ path, value, renderItem, fallback }: StrapiListProps<T>): react_jsx_runtime.JSX.Element;
153
151
 
154
- export { type StrapiEditClickMessage, type StrapiEditMessage, StrapiEditModeProvider, StrapiField, type StrapiFieldType, StrapiImage, StrapiList, type StrapiMediaValue, StrapiText, useStrapiEditMode };
152
+ interface Props {
153
+ path: string;
154
+ value: unknown;
155
+ className?: string;
156
+ }
157
+ declare function StrapiBlocks({ path, value, className }: Props): react_jsx_runtime.JSX.Element;
158
+
159
+ export { type MsgEditChange, type MsgEditCommitRequest, type MsgEditCommitResult, type MsgEditMediaPicked, type MsgEditOpenMediaPicker, type MsgEditReady, type MsgEditReload, StrapiBlocks, type StrapiEditMessage, StrapiEditModeProvider, StrapiField, type StrapiFieldType, StrapiImage, StrapiList, type StrapiMediaValue, StrapiText, useStrapiEditMode };
package/dist/index.d.ts CHANGED
@@ -1,29 +1,6 @@
1
1
  import * as react_jsx_runtime from 'react/jsx-runtime';
2
2
  import { ReactNode, ElementType } from 'react';
3
3
 
4
- interface EditModeState {
5
- enabled: boolean;
6
- token: string | null;
7
- }
8
- declare function useStrapiEditMode(): EditModeState;
9
- interface ProviderProps {
10
- children: React.ReactNode;
11
- /**
12
- * Override automatic URL-based detection. If `true`/`false` ist gesetzt,
13
- * wird der URL-Parameter ignoriert.
14
- */
15
- enabled?: boolean;
16
- }
17
- /**
18
- * Provider, der den Edit-Mode aus `?__pp_edit=1&__pp_token=<jwt>` der URL
19
- * liest und an Kinder via Context weitergibt. Token wird sicherheitshalber
20
- * NICHT ans CSR exponiert über Props oder DOM — er bleibt nur im Context.
21
- *
22
- * Im Edit-Mode wird ausserdem eine kleine globale CSS-Klasse `pp-edit-active`
23
- * am `<html>` gesetzt, sodass Customer-Sites optional CSS-Hooks setzen können.
24
- */
25
- declare function StrapiEditModeProvider({ children, enabled: enabledOverride }: ProviderProps): react_jsx_runtime.JSX.Element;
26
-
27
4
  type StrapiFieldType = "text" | "textarea" | "richText" | "number" | "checkbox" | "date" | "email" | "media" | "select" | "relation" | "group" | "array" | "blocks" | "json" | "unknown";
28
5
  interface StrapiMediaValue {
29
6
  id?: number;
@@ -35,94 +12,115 @@ interface StrapiMediaValue {
35
12
  width?: number;
36
13
  height?: number;
37
14
  }
38
- interface StrapiEditReadyMessage {
15
+ interface MsgEditReady {
39
16
  type: "pp:edit:ready";
40
17
  href: string;
41
18
  }
42
- interface StrapiEditClickMessage {
43
- type: "pp:edit:click";
19
+ interface MsgEditChange {
20
+ type: "pp:edit:change";
44
21
  path: string;
45
22
  fieldType: StrapiFieldType;
46
- rect: {
47
- top: number;
48
- left: number;
49
- width: number;
50
- height: number;
51
- };
23
+ value: unknown;
24
+ }
25
+ interface MsgEditCommitRequest {
26
+ type: "pp:edit:commit-request";
27
+ changes: Array<{
28
+ path: string;
29
+ fieldType: StrapiFieldType;
30
+ value: unknown;
31
+ }>;
32
+ }
33
+ interface MsgEditOpenMediaPicker {
34
+ type: "pp:edit:open-media-picker";
35
+ path: string;
36
+ multiple: boolean;
37
+ allowedTypes?: string[];
52
38
  currentValue: unknown;
53
39
  }
54
- interface StrapiEditReloadMessage {
40
+ interface MsgEditReload {
55
41
  type: "pp:edit:reload";
56
42
  }
57
- type StrapiEditMessage = StrapiEditReadyMessage | StrapiEditClickMessage | StrapiEditReloadMessage;
43
+ interface MsgEditCommitResult {
44
+ type: "pp:edit:commit-result";
45
+ ok: boolean;
46
+ error?: string;
47
+ savedPaths?: string[];
48
+ }
49
+ interface MsgEditMediaPicked {
50
+ type: "pp:edit:media-picked";
51
+ path: string;
52
+ value: StrapiMediaValue | StrapiMediaValue[] | null;
53
+ }
54
+ type StrapiEditMessage = MsgEditReady | MsgEditChange | MsgEditCommitRequest | MsgEditOpenMediaPicker | MsgEditReload | MsgEditCommitResult | MsgEditMediaPicked;
55
+
56
+ interface PendingChange {
57
+ path: string;
58
+ fieldType: StrapiFieldType;
59
+ value: unknown;
60
+ }
61
+ interface EditModeState {
62
+ enabled: boolean;
63
+ token: string | null;
64
+ changes: Map<string, PendingChange>;
65
+ isDirty: boolean;
66
+ isCommitting: boolean;
67
+ lastCommitError: string | null;
68
+ setChange: (path: string, fieldType: StrapiFieldType, value: unknown) => void;
69
+ clearChange: (path: string) => void;
70
+ commit: () => void;
71
+ openMediaPicker: (path: string, multiple: boolean, allowedTypes: string[] | undefined, currentValue: unknown) => void;
72
+ onMediaPicked: (path: string, listener: (value: unknown) => void) => () => void;
73
+ }
74
+ declare function useStrapiEditMode(): EditModeState;
75
+ interface ProviderProps {
76
+ children: React.ReactNode;
77
+ enabled?: boolean;
78
+ }
79
+ declare function StrapiEditModeProvider({ children, enabled: enabledOverride }: ProviderProps): react_jsx_runtime.JSX.Element;
58
80
 
59
81
  interface StrapiFieldProps {
60
- /** Dot-notation path im Strapi-Document, z.B. "hero.title" oder "features.0.image" */
61
82
  path: string;
62
- /** Strapi field type — bestimmt welche Edit-UI im Parent geöffnet wird */
63
83
  type: StrapiFieldType;
64
- /** Aktueller Wert. Wird beim Click-Event mitgesendet, damit der Editor initial befüllt ist */
65
84
  value?: unknown;
66
- /** Inhalt (das echte gerenderte HTML). Wird im Edit-Mode mit data-attrs + Hover/Click annotiert */
67
85
  children: ReactNode;
68
- /** Optional: zusätzliche CSS-Klassen die im Edit-Mode angehängt werden */
69
86
  className?: string;
70
87
  }
71
88
  /**
72
- * Universeller Wrapper für editierbare Strapi-Felder.
89
+ * Universeller Wrapper. Im Normal-Mode: rendert children unverändert.
73
90
  *
74
- * Im normalen Mode rendert er nur die `children` — keine zusätzlichen DOM-
75
- * Nodes, kein Overhead. Im Edit-Mode wird das einzige Top-Level-Child mit
76
- * `data-pp-edit="<path>"` annotiert + Click-Handler bekommt.
77
- *
78
- * Falls children mehrere Top-Level-Knoten haben, wird automatisch ein
79
- * `<span>` als Wrapper eingefügt.
91
+ * Im Edit-Mode:
92
+ * - Für text/textarea/richText/email/number: macht das Top-Level-Child
93
+ * `contenteditable`, mit dünnem orange Hover-Outline, sammelt Wert-Änderungen
94
+ * in den globalen Edit-Context, der dann beim Klick auf "Speichern" alle
95
+ * Änderungen in einem Roundtrip ans Parent schickt.
96
+ * - Für andere Typen (media/select/checkbox): trägt ein Click-Handler bzw.
97
+ * ein Overlay (Implementierung in jeweiligen Sugar-Komponenten).
80
98
  */
81
99
  declare function StrapiField({ path, type, value, children, className }: StrapiFieldProps): react_jsx_runtime.JSX.Element;
82
100
 
83
101
  interface StrapiTextProps {
84
102
  path: string;
85
103
  value: string | number | null | undefined;
86
- /** Welcher Strapi-Feldtyp ist das? Default "text". Für mehrzeilige Felder "textarea", für Markdown "richText". */
87
104
  fieldType?: Extract<StrapiFieldType, "text" | "textarea" | "richText" | "email" | "number">;
88
- /** HTML-Tag um den Wert zu rendern. Default "span". */
89
105
  as?: ElementType;
90
106
  className?: string;
91
107
  children?: ReactNode;
92
108
  }
93
- /**
94
- * Inline-Text wrapper. Im normalen Mode rendert es einfach `value` im
95
- * gewählten Tag — identisch zu einem rohen `<h1>{value}</h1>`. Im Edit-Mode
96
- * wird ein Click-Handler aktiviert.
97
- *
98
- * Wenn `children` mitgegeben sind, werden diese statt `value` als Inhalt
99
- * verwendet — nützlich für custom Formatierung. Der Click-Wert bleibt
100
- * `value`.
101
- */
102
109
  declare function StrapiText({ path, value, fieldType, as, className, children }: StrapiTextProps): react_jsx_runtime.JSX.Element;
103
110
 
104
111
  interface StrapiImageProps {
105
112
  path: string;
106
113
  value: StrapiMediaValue | null | undefined;
107
- /** Optional: Strapi-Basis-URL für relative URLs (z.B. "https://cms.paulpaul.studio") */
108
114
  baseUrl?: string;
109
- /** Override alt-Text (default: alternativeText vom Strapi-Objekt, dann name) */
110
115
  alt?: string;
111
116
  className?: string;
112
117
  style?: React.CSSProperties;
113
118
  loading?: "lazy" | "eager";
114
- /** Fallback wenn value null/undefined ist */
115
119
  fallback?: React.ReactNode;
120
+ allowedTypes?: string[];
121
+ multiple?: boolean;
116
122
  }
117
- /**
118
- * Wrapper für ein einzelnes Strapi-Media-Feld. Rendert `<img>` mit der
119
- * abgeleiteten URL. Im Edit-Mode klickbar — der Editor öffnet einen
120
- * MediaPicker.
121
- *
122
- * Wenn `value` null ist, wird der `fallback` gerendert (oder ein leeres
123
- * `<span>`).
124
- */
125
- declare function StrapiImage({ path, value, baseUrl, alt, className, style, loading, fallback }: StrapiImageProps): react_jsx_runtime.JSX.Element;
123
+ declare function StrapiImage({ path, value, baseUrl, alt, className, style, loading, fallback, allowedTypes, multiple }: StrapiImageProps): react_jsx_runtime.JSX.Element;
126
124
 
127
125
  interface StrapiListProps<T> {
128
126
  /** Pfad zum Array-Feld, z.B. "features" */
@@ -151,4 +149,11 @@ interface StrapiListProps<T> {
151
149
  */
152
150
  declare function StrapiList<T>({ path, value, renderItem, fallback }: StrapiListProps<T>): react_jsx_runtime.JSX.Element;
153
151
 
154
- export { type StrapiEditClickMessage, type StrapiEditMessage, StrapiEditModeProvider, StrapiField, type StrapiFieldType, StrapiImage, StrapiList, type StrapiMediaValue, StrapiText, useStrapiEditMode };
152
+ interface Props {
153
+ path: string;
154
+ value: unknown;
155
+ className?: string;
156
+ }
157
+ declare function StrapiBlocks({ path, value, className }: Props): react_jsx_runtime.JSX.Element;
158
+
159
+ export { type MsgEditChange, type MsgEditCommitRequest, type MsgEditCommitResult, type MsgEditMediaPicked, type MsgEditOpenMediaPicker, type MsgEditReady, type MsgEditReload, StrapiBlocks, type StrapiEditMessage, StrapiEditModeProvider, StrapiField, type StrapiFieldType, StrapiImage, StrapiList, type StrapiMediaValue, StrapiText, useStrapiEditMode };