@roottale/cms-renderer-next 0.2.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.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,56 @@
1
1
  # @roottale/cms-renderer-next
2
2
 
3
+ ## 0.3.0
4
+
5
+ ### Minor Changes
6
+
7
+ - 1e738fd: Add design token auto-application for external sites.
8
+
9
+ admin `mysite.roottale.com → 디자인` 에서 저장한 색·글꼴·둥글기를 외부 사이트의
10
+ RootTale 블로그 컴포넌트가 자동으로 가져와 적용한다.
11
+
12
+ **`@roottale/cms-client`** — 신규 `fetchTheme()` + `RootTaleTheme` 타입 export.
13
+ `GET /v1/cms/public/theme` 엔드포인트 호출 (Bearer rtlk*cust*\*).
14
+
15
+ **`@roottale/cms-renderer-next`** — `<RootTaleBlogList>` / `<RootTaleBlogPost>` 가
16
+ 신규 `theme` prop 지원. 동시에 `<RootTaleThemeProvider>` 컴포넌트로 부모에서 1회
17
+ fetch 한 뒤 자식 컴포넌트가 변수를 상속하는 패턴도 지원.
18
+
19
+ ```tsx
20
+ // 자동 fetch (기본) — 별도 설정 없이 admin 토큰 적용
21
+ <RootTaleBlogList apiKey={process.env.ROOTTALE_API_KEY!} />
22
+
23
+ // 명시 override
24
+ <RootTaleBlogList
25
+ apiKey={process.env.ROOTTALE_API_KEY!}
26
+ theme={{ colors: { primary: "#0070f3" } }}
27
+ />
28
+
29
+ // 부모 1회 fetch → 자식들 상속 (다중 컴포넌트 페이지에서 호출 1회로 축소)
30
+ const theme = await fetchTheme({ apiKey });
31
+ <RootTaleThemeProvider theme={theme}>
32
+ <RootTaleBlogList apiKey={apiKey} theme={null} />
33
+ <RootTaleBlogPost apiKey={apiKey} slugOrId="..." theme={null} />
34
+ </RootTaleThemeProvider>
35
+ ```
36
+
37
+ **`@roottale/cms-renderer-astro`** — `renderBlogList` / `renderBlogPost` 에 동일
38
+ 의미의 `theme` 옵션. style="..." 속성으로 CSS 변수 주입 (서버 측 escape 보강).
39
+
40
+ `theme` prop semantics:
41
+
42
+ - `undefined` → 자동 fetch
43
+ - `RootTaleTheme` → 호출자 override
44
+ - `null` → 자동 fetch 건너뜀 (부모 `RootTaleThemeProvider` 활용 시)
45
+
46
+ fetch 실패는 비치명적 — CSS fallback 으로 graceful degrade.
47
+
48
+ ### Patch Changes
49
+
50
+ - Updated dependencies [a7444e2]
51
+ - Updated dependencies [1e738fd]
52
+ - @roottale/cms-client@0.2.0
53
+
3
54
  ## 0.2.1
4
55
 
5
56
  ### Patch Changes
@@ -12,6 +63,7 @@
12
63
 
13
64
  수정: `tsup` 으로 `dist/*.js` + `dist/*.d.ts` (ESM) 출력. `exports` 가 dist
14
65
  가리킴. customer site 측 `transpilePackages` 불필요.
66
+
15
67
  - `cms-client@0.1.1` — ESM `dist/server.js` + types
16
68
  - `cms-core@0.2.1` — ESM `dist/index.js` + types
17
69
  - `cms-renderer-next@0.2.1` — ESM `dist/{server,index}.js` + types + `dist/cms-public.css`
@@ -20,6 +72,7 @@
20
72
  - `cms-renderer-astro@0.2.1` — ESM `dist/index.js` + types
21
73
 
22
74
  후속 (customer site PR):
75
+
23
76
  - roottale-web / kjmtax / theoneulsan 의 `next.config` 의 `transpilePackages`
24
77
  에서 `@roottale/cms-*` 제거. `pnpm update @roottale/cms-client @roottale/cms-renderer-next` 로 patch 적용.
25
78
 
@@ -36,6 +89,7 @@
36
89
  ADR-0029 §0 amend (publish-only dormant) 는 design system 5 패키지 (tokens, ui-css, ui-react, ui-astro, ui-admin) 에 한정. cms-\* 는 별도 정책 — 실행 로직 + 5-20 외부 customer site 직접 의존 + schema 호환 + 보안 경계. Codex consult verdict (session `019e6703…`) 정합.
37
90
 
38
91
  변경:
92
+
39
93
  - 4 패키지 `private: true` 해제 + `publishConfig.access: "public"`
40
94
  - `@roottale/cms-renderer-next` 에서 `@roottale/tokens/tokens.css` import 제거 — 모든 `--rt-*` 변수에 static fallback 으로 self-contained. tokens dormant 와 무관하게 동작.
41
95
  - `RootTaleLeadForm` RSC 추가 — 외부 사이트 진단 폼 (`vertical`/`redirectUrl` props, medical 국외이전 동의 자동).
@@ -44,6 +98,7 @@
44
98
  - `cms-renderer-astro` 도 동등 surface 유지를 위해 동시 publish
45
99
 
46
100
  후속:
101
+
47
102
  - ADR 신규 — cms-\* publish 정책 (별 PR)
48
103
  - Astro 측 LeadForm 컴포넌트 (현재 `@roottale/ui-astro` 위치, `cms-renderer-astro` 로 이동 검토)
49
104
 
package/dist/server.d.ts CHANGED
@@ -1,5 +1,6 @@
1
- import { ReactElement } from 'react';
2
- import { CmsPostContent, CmsPostType } from '@roottale/cms-client/server';
1
+ import { ReactElement, ReactNode } from 'react';
2
+ import { CmsPostContent, CmsPostType, RootTaleTheme } from '@roottale/cms-client/server';
3
+ export { RootTaleTheme } from '@roottale/cms-client/server';
3
4
  import { Block } from '@roottale/cms-core';
4
5
 
5
6
  /**
@@ -93,7 +94,7 @@ declare function RootTaleFloatingCta(props: RootTaleFloatingCtaProps): ReactElem
93
94
  * `@roottale/cms-renderer-next` — RootTaleLeadForm RSC.
94
95
  *
95
96
  * 외부 고객 사이트(roottale.com, 고객사 외주)의 진단 신청 폼. HTML form POST →
96
- * admin-tenant `/api/lead-intake`. cross-origin POST 는 CORS 불필요 (HTML form
97
+ * tenant-api `/v1/public/inquiries`. cross-origin POST 는 CORS 불필요 (HTML form
97
98
  * submission). PII 는 server 측에서 암호화 후 `inquiries` insert, 302 redirect.
98
99
  *
99
100
  * vertical prop:
@@ -102,7 +103,7 @@ declare function RootTaleFloatingCta(props: RootTaleFloatingCtaProps): ReactElem
102
103
  * (medical = 국외이전 동의 — ADR-0018)
103
104
  *
104
105
  * redirectPath prop:
105
- * - 외부 사이트가 폼 제출 후 돌아갈 path (admin allowlist 검증, ?ok=1 / ?err=*)
106
+ * - 외부 사이트가 폼 제출 후 돌아갈 path (API allowlist 검증, ?ok=1 / ?err=*)
106
107
  *
107
108
  * 0 JS RSC. customer-facing className 전체 `.rt-cms-*` prefix, scoped via
108
109
  * `[data-roottale-cms]` (cms-public.css). customer override 자유.
@@ -110,7 +111,7 @@ declare function RootTaleFloatingCta(props: RootTaleFloatingCtaProps): ReactElem
110
111
 
111
112
  type LeadFormVertical = "consulting" | "medical" | "tax" | "legal";
112
113
  interface RootTaleLeadFormProps {
113
- /** admin-tenant lead-intake endpoint. 예: `https://mysite.roottale.com/api/lead-intake` */
114
+ /** lead intake endpoint. 예: `https://api.roottale.com/v1/public/inquiries` */
114
115
  action: string;
115
116
  /**
116
117
  * vertical 고정. 미지정 시 select 노출.
@@ -118,9 +119,9 @@ interface RootTaleLeadFormProps {
118
119
  */
119
120
  vertical?: LeadFormVertical;
120
121
  /**
121
- * 폼 제출 후 admin이 돌려보낼 절대 URL.
122
- * admin `LEAD_INTAKE_ALLOWED_ORIGINS` allowlist 검증. fail = fallback.
123
- * 미지정 시 admin env의 LEAD_INTAKE_REDIRECT_BASE 사용.
122
+ * 폼 제출 후 API가 돌려보낼 절대 URL.
123
+ * tenant-api `LEAD_INTAKE_ALLOWED_ORIGINS` allowlist 검증. fail = fallback.
124
+ * 미지정 시 tenant-api env의 LEAD_INTAKE_REDIRECT_BASE 사용.
124
125
  */
125
126
  redirectUrl?: string;
126
127
  heading?: string;
@@ -160,6 +161,33 @@ declare function renderBlocks(blocks: readonly Block[]): ReactElement[];
160
161
  * `:where()` to keep specificity at zero (ADR-0023 §5.1 #10).
161
162
  */
162
163
 
164
+ /**
165
+ * RootTaleBlogList/Post 가 받는 theme prop 의 평가 결과:
166
+ * - `RootTaleTheme` 객체 → 그대로 CSS 변수로 주입
167
+ * - `null` → override 없음 (cms-public.css fallback 그대로)
168
+ * - `undefined` → API key 로 자동 fetch (mysite.roottale.com 의 admin /design)
169
+ */
170
+ type RootTaleThemeInput = RootTaleTheme | null | undefined;
171
+ /**
172
+ * RootTale 디자인 토큰을 CSS 변수로 자식 트리에 주입한다. 외부 사이트가
173
+ * 여러 RootTale 컴포넌트를 한 레이아웃에 묶을 때, 컴포넌트마다 같은 토큰을
174
+ * fetch 하지 않도록 부모에서 한 번만 fetch 해서 전달하는 용도.
175
+ *
176
+ * ```tsx
177
+ * const theme = await fetchTheme({ apiKey });
178
+ * <RootTaleThemeProvider theme={theme}>
179
+ * <RootTaleBlogList apiKey={apiKey} theme={null} />
180
+ * <RootTaleBlogPost apiKey={apiKey} slugOrId="..." theme={null} />
181
+ * </RootTaleThemeProvider>
182
+ * ```
183
+ *
184
+ * 자식 RootTale 컴포넌트가 `theme={null}` 을 받으면 자체 주입을 건너뛰고
185
+ * 부모의 변수를 그대로 상속. 자식이 `theme` 을 지정하면 자식 변수가 우선.
186
+ */
187
+ declare function RootTaleThemeProvider(props: {
188
+ theme: RootTaleTheme | null | undefined;
189
+ children: ReactNode;
190
+ }): ReactElement;
163
191
  type RootTaleBlogListProps = {
164
192
  apiKey: string;
165
193
  baseUrl?: string;
@@ -170,6 +198,12 @@ type RootTaleBlogListProps = {
170
198
  postHref?: (post: CmsPostContent) => string;
171
199
  /** Rendered when the tenant has no published posts yet. */
172
200
  emptyMessage?: string;
201
+ /**
202
+ * RootTale 디자인 토큰. 생략 시 admin /design 에서 저장한 값을 자동 fetch.
203
+ * `null` 을 명시하면 자동 fetch 도 건너뛰고 CSS fallback 사용 (부모
204
+ * `<RootTaleThemeProvider>` 가 변수를 이미 설정한 경우 유용).
205
+ */
206
+ theme?: RootTaleThemeInput;
173
207
  };
174
208
  declare function RootTaleBlogList(props: RootTaleBlogListProps): Promise<ReactElement>;
175
209
  type RootTaleBlogPostProps = {
@@ -186,7 +220,9 @@ type RootTaleBlogPostProps = {
186
220
  showTableOfContents?: boolean;
187
221
  /** Title shown above the TOC (only when `showTableOfContents`). */
188
222
  tableOfContentsTitle?: string;
223
+ /** See `RootTaleBlogListProps['theme']`. */
224
+ theme?: RootTaleThemeInput;
189
225
  };
190
226
  declare function RootTaleBlogPost(props: RootTaleBlogPostProps): Promise<ReactElement>;
191
227
 
192
- export { type CtaButton, type CtaButtonType, type FloatingCtaPosition, type LeadFormVertical, RootTaleBlogList, type RootTaleBlogListProps, RootTaleBlogPost, type RootTaleBlogPostProps, RootTaleFloatingCta, type RootTaleFloatingCtaProps, RootTaleLeadForm, type RootTaleLeadFormProps, RootTalePostCard, type RootTalePostCardProps, RootTaleTableOfContents, type RootTaleTableOfContentsProps, type TocEntry, attachHeadingIds, extractToc, headingToId, renderBlock, renderBlocks };
228
+ export { type CtaButton, type CtaButtonType, type FloatingCtaPosition, type LeadFormVertical, RootTaleBlogList, type RootTaleBlogListProps, RootTaleBlogPost, type RootTaleBlogPostProps, RootTaleFloatingCta, type RootTaleFloatingCtaProps, RootTaleLeadForm, type RootTaleLeadFormProps, RootTalePostCard, type RootTalePostCardProps, RootTaleTableOfContents, type RootTaleTableOfContentsProps, type RootTaleThemeInput, RootTaleThemeProvider, type TocEntry, attachHeadingIds, extractToc, headingToId, renderBlock, renderBlocks };
package/dist/server.js CHANGED
@@ -1,7 +1,8 @@
1
1
  // src/server.tsx
2
2
  import {
3
3
  fetchPost,
4
- fetchPosts
4
+ fetchPosts,
5
+ fetchTheme
5
6
  } from "@roottale/cms-client/server";
6
7
 
7
8
  // src/PostCard.tsx
@@ -423,24 +424,13 @@ function renderBlock(block, key = 0) {
423
424
  }
424
425
  default: {
425
426
  const rawHtml = block.rawHtml;
426
- if (rawHtml) {
427
- return /* @__PURE__ */ jsx5(
428
- "div",
429
- {
430
- className: "rt-unknown-block",
431
- "data-block-id": dataBlockId,
432
- "data-block-name": block.name,
433
- dangerouslySetInnerHTML: htmlContent(rawHtml)
434
- },
435
- key
436
- );
437
- }
438
427
  return /* @__PURE__ */ jsx5(
439
428
  "div",
440
429
  {
441
430
  className: "rt-unknown-block",
442
431
  "data-block-id": dataBlockId,
443
432
  "data-block-name": block.name,
433
+ "data-rawhtml-suppressed": rawHtml ? "true" : void 0,
444
434
  children: renderInner(block.innerBlocks ?? [])
445
435
  },
446
436
  key
@@ -454,6 +444,11 @@ function renderBlocks(blocks) {
454
444
 
455
445
  // src/server.tsx
456
446
  import { Fragment as Fragment2, jsx as jsx6, jsxs as jsxs6 } from "react/jsx-runtime";
447
+ function RootTaleThemeProvider(props) {
448
+ const style = themeToCssVars(props.theme ?? null);
449
+ if (!style) return /* @__PURE__ */ jsx6(Fragment2, { children: props.children });
450
+ return /* @__PURE__ */ jsx6("div", { "data-roottale-cms": "theme-root", style, children: props.children });
451
+ }
457
452
  async function RootTaleBlogList(props) {
458
453
  const {
459
454
  apiKey,
@@ -461,13 +456,17 @@ async function RootTaleBlogList(props) {
461
456
  limit,
462
457
  type = "post",
463
458
  postHref = defaultPostHref,
464
- emptyMessage = "\uC544\uC9C1 \uBC1C\uD589\uB41C \uAE00\uC774 \uC5C6\uC2B5\uB2C8\uB2E4."
459
+ emptyMessage = "\uC544\uC9C1 \uBC1C\uD589\uB41C \uAE00\uC774 \uC5C6\uC2B5\uB2C8\uB2E4.",
460
+ theme: themeProp
465
461
  } = props;
466
- const page = await fetchPosts({ apiKey, baseUrl, limit, type });
462
+ const [page, themeStyle] = await Promise.all([
463
+ fetchPosts({ apiKey, baseUrl, limit, type }),
464
+ resolveThemeStyle({ themeProp, apiKey, baseUrl })
465
+ ]);
467
466
  if (page.items.length === 0) {
468
- return /* @__PURE__ */ jsx6("div", { "data-roottale-cms": "list", children: /* @__PURE__ */ jsx6("p", { className: "rt-cms-empty", children: emptyMessage }) });
467
+ return /* @__PURE__ */ jsx6("div", { "data-roottale-cms": "list", style: themeStyle ?? void 0, children: /* @__PURE__ */ jsx6("p", { className: "rt-cms-empty", children: emptyMessage }) });
469
468
  }
470
- return /* @__PURE__ */ jsx6("div", { "data-roottale-cms": "list", children: /* @__PURE__ */ jsx6("ul", { className: "rt-cms-list", children: page.items.map((post) => /* @__PURE__ */ jsx6("li", { className: "rt-cms-list-item", children: /* @__PURE__ */ jsx6(RootTalePostCard, { post, href: postHref }) }, post.id)) }) });
469
+ return /* @__PURE__ */ jsx6("div", { "data-roottale-cms": "list", style: themeStyle ?? void 0, children: /* @__PURE__ */ jsx6("ul", { className: "rt-cms-list", children: page.items.map((post) => /* @__PURE__ */ jsx6("li", { className: "rt-cms-list-item", children: /* @__PURE__ */ jsx6(RootTalePostCard, { post, href: postHref }) }, post.id)) }) });
471
470
  }
472
471
  async function RootTaleBlogPost(props) {
473
472
  const {
@@ -476,23 +475,65 @@ async function RootTaleBlogPost(props) {
476
475
  baseUrl,
477
476
  notFoundElement,
478
477
  showTableOfContents = false,
479
- tableOfContentsTitle
478
+ tableOfContentsTitle,
479
+ theme: themeProp
480
480
  } = props;
481
- const post = await fetchPost({ apiKey, slugOrId, baseUrl });
481
+ const [post, themeStyle] = await Promise.all([
482
+ fetchPost({ apiKey, slugOrId, baseUrl }),
483
+ resolveThemeStyle({ themeProp, apiKey, baseUrl })
484
+ ]);
482
485
  if (!post) {
483
- return notFoundElement ?? /* @__PURE__ */ jsx6("div", { "data-roottale-cms": "post-missing", children: /* @__PURE__ */ jsx6("p", { className: "rt-cms-empty", children: "\uAE00\uC744 \uCC3E\uC744 \uC218 \uC5C6\uC2B5\uB2C8\uB2E4." }) });
486
+ return notFoundElement ?? /* @__PURE__ */ jsx6("div", { "data-roottale-cms": "post-missing", style: themeStyle ?? void 0, children: /* @__PURE__ */ jsx6("p", { className: "rt-cms-empty", children: "\uAE00\uC744 \uCC3E\uC744 \uC218 \uC5C6\uC2B5\uB2C8\uB2E4." }) });
484
487
  }
485
488
  const toc = showTableOfContents ? extractToc(post.bodyJson) : [];
486
489
  const renderDoc = toc.length > 0 ? attachHeadingIds(post.bodyJson, toc) : post.bodyJson;
487
- return /* @__PURE__ */ jsxs6("article", { "data-roottale-cms": "post", className: "rt-cms-article", children: [
488
- /* @__PURE__ */ jsxs6("header", { children: [
489
- /* @__PURE__ */ jsx6("p", { className: "rt-cms-meta", children: formatPublishedDate2(post.publishedAt) }),
490
- /* @__PURE__ */ jsx6("h1", { className: "rt-cms-title", children: post.title }),
491
- post.excerpt ? /* @__PURE__ */ jsx6("p", { className: "rt-cms-excerpt", children: post.excerpt }) : null
492
- ] }),
493
- toc.length > 0 ? /* @__PURE__ */ jsx6(RootTaleTableOfContents, { entries: toc, title: tableOfContentsTitle }) : null,
494
- /* @__PURE__ */ jsx6(RenderTiptap, { doc: renderDoc })
495
- ] });
490
+ return /* @__PURE__ */ jsxs6(
491
+ "article",
492
+ {
493
+ "data-roottale-cms": "post",
494
+ className: "rt-cms-article",
495
+ style: themeStyle ?? void 0,
496
+ children: [
497
+ /* @__PURE__ */ jsxs6("header", { children: [
498
+ /* @__PURE__ */ jsx6("p", { className: "rt-cms-meta", children: formatPublishedDate2(post.publishedAt) }),
499
+ /* @__PURE__ */ jsx6("h1", { className: "rt-cms-title", children: post.title }),
500
+ post.excerpt ? /* @__PURE__ */ jsx6("p", { className: "rt-cms-excerpt", children: post.excerpt }) : null
501
+ ] }),
502
+ toc.length > 0 ? /* @__PURE__ */ jsx6(RootTaleTableOfContents, { entries: toc, title: tableOfContentsTitle }) : null,
503
+ /* @__PURE__ */ jsx6(RenderTiptap, { doc: renderDoc })
504
+ ]
505
+ }
506
+ );
507
+ }
508
+ async function resolveThemeStyle(input) {
509
+ const { themeProp, apiKey, baseUrl } = input;
510
+ if (themeProp === null) return null;
511
+ if (themeProp !== void 0) return themeToCssVars(themeProp);
512
+ try {
513
+ const fetched = await fetchTheme({ apiKey, baseUrl });
514
+ return themeToCssVars(fetched);
515
+ } catch {
516
+ return null;
517
+ }
518
+ }
519
+ function themeToCssVars(theme) {
520
+ if (!theme) return null;
521
+ const vars = {};
522
+ const c = theme.colors ?? {};
523
+ if (c.primary) vars["--rt-color-primary"] = c.primary;
524
+ if (c.primaryForeground) vars["--rt-color-primary-foreground"] = c.primaryForeground;
525
+ if (c.foreground) vars["--rt-color-foreground"] = c.foreground;
526
+ if (c.background) vars["--rt-color-background"] = c.background;
527
+ if (c.muted) vars["--rt-color-muted"] = c.muted;
528
+ if (c.mutedForeground) vars["--rt-color-muted-foreground"] = c.mutedForeground;
529
+ if (c.border) vars["--rt-color-border"] = c.border;
530
+ const f = theme.fonts ?? {};
531
+ if (f.body) vars["--rt-font-body"] = f.body;
532
+ if (f.display) vars["--rt-font-display"] = f.display;
533
+ const r = theme.radius ?? {};
534
+ if (r.md) vars["--rt-radius-md"] = r.md;
535
+ if (Object.keys(vars).length === 0) return null;
536
+ return vars;
496
537
  }
497
538
  function defaultPostHref(post) {
498
539
  return `/blog/${post.slug}`;
@@ -621,6 +662,7 @@ export {
621
662
  RootTaleLeadForm,
622
663
  RootTalePostCard,
623
664
  RootTaleTableOfContents,
665
+ RootTaleThemeProvider,
624
666
  attachHeadingIds,
625
667
  extractToc,
626
668
  headingToId,
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/server.tsx","../src/PostCard.tsx","../src/TableOfContents.tsx","../src/toc.ts","../src/FloatingCta.tsx","../src/LeadForm.tsx","../src/block-to-react.tsx"],"sourcesContent":["/**\n * `@roottale/cms-renderer-next/server` — RootTale CMS public-render React Server Components.\n *\n * Drop into any RSC-capable framework (Next.js App Router, Astro server islands,\n * React Router server). Single-line import, SSR-only, no `'use client'` boundary.\n *\n * ```tsx\n * import { RootTaleBlogList } from \"@roottale/cms-renderer-next/server\";\n *\n * export default function BlogPage() {\n * return <RootTaleBlogList apiKey={process.env.ROOTTALE_API_KEY!} />;\n * }\n * ```\n *\n * Customer site MUST keep `apiKey` server-side. Browser bundle ships zero\n * RootTale credentials (ADR-0023 §5.1 #15).\n *\n * Design tokens auto-apply via the CSS import below. Customers can override\n * any class with their own CSS or Tailwind utilities — scoped selectors use\n * `:where()` to keep specificity at zero (ADR-0023 §5.1 #10).\n */\n\nimport type { CSSProperties, ReactElement } from \"react\";\nimport {\n type CmsPostContent,\n type CmsPostType,\n fetchPost,\n fetchPosts,\n} from \"@roottale/cms-client/server\";\n// CSS는 customer 가 root layout 에서 명시적으로 import:\n// `import \"@roottale/cms-renderer-next/styles\";`\n// 본 파일에서 side-effect import 하지 않음 — tsup dts build 가 .css\n// side-effect import 처리 못 함 + customer 측 framework 마다 CSS 처리\n// 방식 다르므로 명시적 import 가 명확. README 참조.\n\nimport { RootTalePostCard } from \"./PostCard.js\";\nimport { RootTaleTableOfContents } from \"./TableOfContents.js\";\nimport { attachHeadingIds, extractToc } from \"./toc.js\";\n\nexport { RootTalePostCard, type RootTalePostCardProps } from \"./PostCard.js\";\nexport {\n RootTaleTableOfContents,\n type RootTaleTableOfContentsProps,\n} from \"./TableOfContents.js\";\nexport {\n RootTaleFloatingCta,\n type RootTaleFloatingCtaProps,\n type CtaButton,\n type CtaButtonType,\n type FloatingCtaPosition,\n} from \"./FloatingCta.js\";\nexport {\n RootTaleLeadForm,\n type RootTaleLeadFormProps,\n type LeadFormVertical,\n} from \"./LeadForm.js\";\nexport { renderBlock, renderBlocks } from \"./block-to-react.js\";\nexport {\n attachHeadingIds,\n extractToc,\n headingToId,\n type TocEntry,\n} from \"./toc.js\";\n\nexport type RootTaleBlogListProps = {\n apiKey: string;\n baseUrl?: string;\n limit?: number;\n /** Filter by post type (`post` or `page`). Default = `post` only (blog list 의도). */\n type?: CmsPostType;\n /** Build a customer-side URL for a single post. Defaults to `/blog/${slug}`. */\n postHref?: (post: CmsPostContent) => string;\n /** Rendered when the tenant has no published posts yet. */\n emptyMessage?: string;\n};\n\nexport async function RootTaleBlogList(\n props: RootTaleBlogListProps,\n): Promise<ReactElement> {\n const {\n apiKey,\n baseUrl,\n limit,\n type = \"post\",\n postHref = defaultPostHref,\n emptyMessage = \"아직 발행된 글이 없습니다.\",\n } = props;\n const page = await fetchPosts({ apiKey, baseUrl, limit, type });\n\n if (page.items.length === 0) {\n return (\n <div data-roottale-cms=\"list\">\n <p className=\"rt-cms-empty\">{emptyMessage}</p>\n </div>\n );\n }\n\n return (\n <div data-roottale-cms=\"list\">\n <ul className=\"rt-cms-list\">\n {page.items.map((post) => (\n <li key={post.id} className=\"rt-cms-list-item\">\n <RootTalePostCard post={post} href={postHref} />\n </li>\n ))}\n </ul>\n </div>\n );\n}\n\nexport type RootTaleBlogPostProps = {\n apiKey: string;\n slugOrId: string;\n baseUrl?: string;\n /** Rendered when the slug is not found or not published. */\n notFoundElement?: ReactElement;\n /**\n * When true, prepends an auto-derived `<RootTaleTableOfContents>` before\n * the article body (H2/H3 headings only). Default: false — keep markup\n * stable for customers who already control their own layout.\n */\n showTableOfContents?: boolean;\n /** Title shown above the TOC (only when `showTableOfContents`). */\n tableOfContentsTitle?: string;\n};\n\nexport async function RootTaleBlogPost(\n props: RootTaleBlogPostProps,\n): Promise<ReactElement> {\n const {\n apiKey,\n slugOrId,\n baseUrl,\n notFoundElement,\n showTableOfContents = false,\n tableOfContentsTitle,\n } = props;\n const post = await fetchPost({ apiKey, slugOrId, baseUrl });\n\n if (!post) {\n return (\n notFoundElement ?? (\n <div data-roottale-cms=\"post-missing\">\n <p className=\"rt-cms-empty\">글을 찾을 수 없습니다.</p>\n </div>\n )\n );\n }\n\n const toc = showTableOfContents ? extractToc(post.bodyJson) : [];\n const renderDoc =\n toc.length > 0 ? attachHeadingIds(post.bodyJson, toc) : post.bodyJson;\n\n return (\n <article data-roottale-cms=\"post\" className=\"rt-cms-article\">\n <header>\n <p className=\"rt-cms-meta\">{formatPublishedDate(post.publishedAt)}</p>\n <h1 className=\"rt-cms-title\">{post.title}</h1>\n {post.excerpt ? <p className=\"rt-cms-excerpt\">{post.excerpt}</p> : null}\n </header>\n {toc.length > 0 ? (\n <RootTaleTableOfContents entries={toc} title={tableOfContentsTitle} />\n ) : null}\n <RenderTiptap doc={renderDoc} />\n </article>\n );\n}\n\nfunction defaultPostHref(post: CmsPostContent): string {\n return `/blog/${post.slug}`;\n}\n\nfunction formatPublishedDate(iso: string): string {\n const d = new Date(iso);\n if (Number.isNaN(d.getTime())) return \"\";\n return new Intl.DateTimeFormat(\"ko-KR\", {\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n }).format(d);\n}\n\n/**\n * Minimal Tiptap doc renderer. Walks the Tiptap doc tree and emits semantic\n * HTML. This is intentionally tiny — richer extensions (image asset URL\n * resolution, TableOfContents, FloatingCta) belong in followup §5.7 #5.\n *\n * Browser-side sanitization is unnecessary here because output is rendered\n * server-side from data the customer-trusted API returned. If the customer\n * site federates with untrusted authors later, layer a sanitizer in front\n * of this component.\n */\nfunction RenderTiptap({ doc }: { doc: Record<string, unknown> }): ReactElement {\n const rawContent = (doc as TiptapDoc).content;\n const content: TiptapNode[] = Array.isArray(rawContent) ? rawContent : [];\n return <>{content.map((node, i) => renderNode(node, i))}</>;\n}\n\ntype TiptapDoc = { type?: string; content?: TiptapNode[] };\n\ntype TiptapMark = { type?: string; attrs?: Record<string, unknown> };\n\ntype TiptapNode = {\n type?: string;\n text?: string;\n attrs?: Record<string, unknown>;\n content?: TiptapNode[];\n marks?: TiptapMark[];\n};\n\n/**\n * Defense-in-depth: even though admin LinkButton.normalizeUrl filters input,\n * legacy imports / direct DB / external tenants can still emit unsafe href.\n * Allow only http(s)/mailto/tel/root-relative/fragment/query — falls back to \"#\".\n */\nfunction isSafeHref(value: unknown): value is string {\n if (typeof value !== \"string\") return false;\n const v = value.trim();\n if (v.length === 0) return false;\n if (v.startsWith(\"/\") || v.startsWith(\"#\") || v.startsWith(\"?\")) return true;\n return /^(https?:|mailto:|tel:)/i.test(v);\n}\n\nfunction alignStyle(node: TiptapNode): CSSProperties | undefined {\n const align = node.attrs?.textAlign;\n if (typeof align !== \"string\" || align === \"\" || align === \"left\") return undefined;\n if (align === \"center\" || align === \"right\" || align === \"justify\") {\n return { textAlign: align };\n }\n return undefined;\n}\n\nfunction renderNode(node: TiptapNode, key: number): ReactElement | string | null {\n if (!node || typeof node !== \"object\") return null;\n const children = Array.isArray(node.content)\n ? node.content.map((child, i) => renderNode(child, i))\n : undefined;\n\n switch (node.type) {\n case \"paragraph\":\n return <p key={key} style={alignStyle(node)}>{children}</p>;\n case \"heading\": {\n const level = Math.min(Math.max(Number(node.attrs?.level ?? 2), 1), 3);\n const idAttr = node.attrs?.id;\n const id = typeof idAttr === \"string\" && idAttr.length > 0 ? idAttr : undefined;\n const style = alignStyle(node);\n if (level === 1) return <h1 key={key} id={id} style={style}>{children}</h1>;\n if (level === 2) return <h2 key={key} id={id} style={style}>{children}</h2>;\n return <h3 key={key} id={id} style={style}>{children}</h3>;\n }\n case \"bulletList\":\n return <ul key={key}>{children}</ul>;\n case \"orderedList\":\n return <ol key={key}>{children}</ol>;\n case \"listItem\":\n return <li key={key}>{children}</li>;\n case \"blockquote\":\n return <blockquote key={key}>{children}</blockquote>;\n case \"codeBlock\":\n return (\n <pre key={key}>\n <code>{children}</code>\n </pre>\n );\n case \"horizontalRule\":\n return <hr key={key} />;\n case \"hardBreak\":\n return <br key={key} />;\n case \"image\": {\n const src = node.attrs?.src;\n if (typeof src !== \"string\" || src.length === 0) return null;\n const alt = typeof node.attrs?.alt === \"string\" ? node.attrs.alt : \"\";\n const title = typeof node.attrs?.title === \"string\" ? node.attrs.title : undefined;\n // eslint-disable-next-line @next/next/no-img-element\n return <img key={key} src={src} alt={alt} title={title} loading=\"lazy\" />;\n }\n case \"columns\":\n return <div key={key} className=\"rt-columns\">{children}</div>;\n case \"column\":\n return <div key={key} className=\"rt-column\">{children}</div>;\n case \"text\":\n return renderText(node, key);\n default:\n return children ? <span key={key}>{children}</span> : null;\n }\n}\n\nfunction renderText(node: TiptapNode, key: number): ReactElement | string {\n const text = node.text ?? \"\";\n const marks = node.marks ?? [];\n if (marks.length === 0) return text;\n let element: ReactElement | string = text;\n for (const mark of marks) {\n element = applyMark(mark, element);\n }\n return <span key={key}>{element}</span>;\n}\n\nfunction applyMark(mark: TiptapMark, child: ReactElement | string): ReactElement {\n switch (mark.type) {\n case \"bold\":\n return <strong>{child}</strong>;\n case \"italic\":\n return <em>{child}</em>;\n case \"underline\":\n return <u>{child}</u>;\n case \"strike\":\n return <s>{child}</s>;\n case \"code\":\n return <code>{child}</code>;\n case \"highlight\": {\n const color = mark.attrs?.color;\n const style =\n typeof color === \"string\" && color.length > 0\n ? { background: color }\n : undefined;\n return <mark style={style}>{child}</mark>;\n }\n case \"textStyle\": {\n const color = mark.attrs?.color;\n const style =\n typeof color === \"string\" && color.length > 0 ? { color } : undefined;\n if (!style) return <>{child}</>;\n return <span style={style}>{child}</span>;\n }\n case \"link\": {\n const raw = mark.attrs?.href;\n const href = isSafeHref(raw) ? raw : \"#\";\n return (\n <a href={href} rel=\"noopener noreferrer\" target=\"_blank\">\n {child}\n </a>\n );\n }\n default:\n return <span>{child}</span>;\n }\n}\n","/**\n * `<RootTalePostCard>` — server-only summary card for a single CmsPostContent.\n *\n * Building block for `<RootTaleBlogList>` (and customer-side custom grids).\n * Pure SSR — anchor + title + meta + excerpt. Featured image is left out for\n * now: `CmsPostContent` only carries `featuredMediaId`, not a public URL.\n * Media URL resolution belongs in a separate slice once `cms-client` exposes\n * it (ADR-0034 §4 media followup — CF Images variants).\n */\n\nimport type { ReactElement } from \"react\";\nimport type { CmsPostContent } from \"@roottale/cms-client/server\";\n\nexport interface RootTalePostCardProps {\n post: CmsPostContent;\n /** Customer-side URL builder. Defaults to `/blog/${slug}`. */\n href?: (post: CmsPostContent) => string;\n}\n\nexport function RootTalePostCard(props: RootTalePostCardProps): ReactElement {\n const { post, href = defaultHref } = props;\n return (\n <article data-roottale-cms=\"card\" className=\"rt-cms-card\">\n <a className=\"rt-cms-card-link\" href={href(post)}>\n <p className=\"rt-cms-meta\">{formatPublishedDate(post.publishedAt)}</p>\n <h2 className=\"rt-cms-title\">{post.title}</h2>\n {post.excerpt ? <p className=\"rt-cms-excerpt\">{post.excerpt}</p> : null}\n </a>\n </article>\n );\n}\n\nfunction defaultHref(post: CmsPostContent): string {\n return `/blog/${post.slug}`;\n}\n\nfunction formatPublishedDate(iso: string): string {\n const d = new Date(iso);\n if (Number.isNaN(d.getTime())) return \"\";\n return new Intl.DateTimeFormat(\"ko-KR\", {\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n }).format(d);\n}\n","/**\n * `<RootTaleTableOfContents>` — server-only anchor list for blog posts.\n *\n * Pure SSR. Renders `<nav>` + ordered anchors that link to `#<id>` slots\n * emitted by `attachHeadingIds`. No scroll-spy here — that requires an\n * `IntersectionObserver` which only exists in the browser; a client-island\n * variant can layer on top later without changing the data contract.\n *\n * Internal ancestor: `roottale-internal/packages/cms-renderer/src/TableOfContents.tsx`\n * (had `'use client'` + IntersectionObserver). Stripped to server-safe form per\n * the platform renderer contract (no `'use client'` boundary — ADR-0023 §5.3).\n */\n\nimport type { ReactElement } from \"react\";\nimport type { TocEntry } from \"./toc.js\";\n\nexport interface RootTaleTableOfContentsProps {\n entries: TocEntry[];\n title?: string;\n}\n\nexport function RootTaleTableOfContents(\n props: RootTaleTableOfContentsProps,\n): ReactElement | null {\n const { entries, title = \"목차\" } = props;\n if (entries.length === 0) return null;\n return (\n <nav data-roottale-cms=\"toc\" className=\"rt-cms-toc\" aria-label={title}>\n {title ? <p className=\"rt-cms-toc-title\">{title}</p> : null}\n <ol className=\"rt-cms-toc-list\">\n {entries.map((entry) => (\n <li\n key={entry.id}\n className={`rt-cms-toc-item rt-cms-toc-level-${entry.level}`}\n >\n <a href={`#${entry.id}`}>{entry.text}</a>\n </li>\n ))}\n </ol>\n </nav>\n );\n}\n","/**\n * `@roottale/cms-renderer/toc` — Tiptap doc → TOC entries + heading id injection.\n *\n * Server-only utility. Walks a Tiptap JSON doc, extracts H2/H3 headings, and\n * derives stable kebab-case ids (Korean-safe). The companion `attachHeadingIds`\n * mutates a shallow clone of the doc so renderers can emit `<h2 id=\"...\">`\n * anchors that match what `<RootTaleTableOfContents>` links to.\n *\n * Internal ancestor: `roottale-internal/packages/cms-renderer/src/toc.ts`\n * (HTML-string based, vendored snapshot). Ported to Tiptap-JSON because the\n * platform renderer no longer round-trips through HTML before render.\n */\n\nexport interface TocEntry {\n level: 2 | 3;\n text: string;\n id: string;\n}\n\ntype TiptapNode = {\n type?: string;\n text?: string;\n attrs?: Record<string, unknown>;\n content?: TiptapNode[];\n marks?: { type?: string }[];\n};\n\ntype TiptapDoc = { type?: string; content?: TiptapNode[] };\n\nexport function headingToId(text: string): string {\n return text\n .trim()\n .toLowerCase()\n .replace(/[^\\p{L}\\p{N}\\s-]/gu, \"\")\n .replace(/\\s+/g, \"-\")\n .replace(/-+/g, \"-\")\n .replace(/^-|-$/g, \"\");\n}\n\nfunction textOf(node: TiptapNode): string {\n if (typeof node.text === \"string\") return node.text;\n if (!Array.isArray(node.content)) return \"\";\n return node.content.map(textOf).join(\"\");\n}\n\nexport function extractToc(doc: Record<string, unknown>): TocEntry[] {\n const root = doc as TiptapDoc;\n const content = Array.isArray(root.content) ? root.content : [];\n const entries: TocEntry[] = [];\n const idCounts = new Map<string, number>();\n\n for (const node of content) {\n if (node?.type !== \"heading\") continue;\n const rawLevel = Number(node.attrs?.level ?? 2);\n if (rawLevel !== 2 && rawLevel !== 3) continue;\n const text = textOf(node).trim();\n if (!text) continue;\n\n const base = headingToId(text) || \"heading\";\n const count = idCounts.get(base) ?? 0;\n const id = count === 0 ? base : `${base}-${count}`;\n idCounts.set(base, count + 1);\n\n entries.push({ level: rawLevel, text, id });\n }\n\n return entries;\n}\n\n/**\n * Returns a shallow-cloned Tiptap doc where each H2/H3 carries the matching\n * `attrs.id` from `entries` (same order as `extractToc` returns). Existing\n * `id` attrs are overwritten so renderer + TOC always agree on anchors.\n */\nexport function attachHeadingIds(\n doc: Record<string, unknown>,\n entries: TocEntry[],\n): Record<string, unknown> {\n const root = doc as TiptapDoc;\n const content = Array.isArray(root.content) ? root.content : [];\n let cursor = 0;\n const nextContent = content.map((node) => {\n if (node?.type !== \"heading\") return node;\n const rawLevel = Number(node.attrs?.level ?? 2);\n if (rawLevel !== 2 && rawLevel !== 3) return node;\n if (!textOf(node).trim()) return node;\n const entry = entries[cursor++];\n if (!entry) return node;\n return {\n ...node,\n attrs: { ...(node.attrs ?? {}), id: entry.id },\n };\n });\n return { ...(root as Record<string, unknown>), content: nextContent };\n}\n","/**\n * `<RootTaleFloatingCta>` — fixed-position call-to-action stack.\n *\n * Server-only (anchors only, no state). External targets (`kakao` / `custom`)\n * open in a new tab with `rel=\"noopener noreferrer\"`; tel/contact stay in the\n * same tab so iOS/Android can trigger the dialer / mailto handler.\n *\n * Internal ancestor: `roottale-internal/packages/cms-renderer/src/FloatingCta.tsx`.\n */\n\nimport type { ReactElement } from \"react\";\n\nexport type CtaButtonType = \"phone\" | \"contact\" | \"kakao\" | \"custom\";\n\nexport interface CtaButton {\n type: CtaButtonType;\n label: string;\n href: string;\n /** Optional override icon (emoji or short text). */\n icon?: string;\n}\n\nexport type FloatingCtaPosition =\n | \"bottom-right\"\n | \"bottom-left\"\n | \"bottom-center\";\n\nexport interface RootTaleFloatingCtaProps {\n buttons: CtaButton[];\n position?: FloatingCtaPosition;\n}\n\nexport function RootTaleFloatingCta(\n props: RootTaleFloatingCtaProps,\n): ReactElement | null {\n const { buttons, position = \"bottom-right\" } = props;\n if (!buttons || buttons.length === 0) return null;\n return (\n <div\n data-roottale-cms=\"floating-cta\"\n className={`rt-cms-floating-cta rt-cms-floating-cta--${position}`}\n >\n {buttons.map((btn, idx) => {\n const isExternal = btn.type === \"kakao\" || btn.type === \"custom\";\n return (\n <a\n key={`${btn.type}-${idx}`}\n href={btn.href}\n className={`rt-cms-floating-cta__btn rt-cms-floating-cta__btn--${btn.type}`}\n target={isExternal ? \"_blank\" : undefined}\n rel={isExternal ? \"noopener noreferrer\" : undefined}\n >\n <span aria-hidden=\"true\" className=\"rt-cms-floating-cta__icon\">\n {btn.icon ?? defaultIcon(btn.type)}\n </span>\n <span className=\"rt-cms-floating-cta__label\">{btn.label}</span>\n </a>\n );\n })}\n </div>\n );\n}\n\nfunction defaultIcon(type: CtaButtonType): string {\n switch (type) {\n case \"phone\":\n return \"☎\";\n case \"contact\":\n return \"✉\";\n case \"kakao\":\n return \"💬\";\n case \"custom\":\n return \"→\";\n }\n}\n","/**\n * `@roottale/cms-renderer-next` — RootTaleLeadForm RSC.\n *\n * 외부 고객 사이트(roottale.com, 고객사 외주)의 진단 신청 폼. HTML form POST →\n * admin-tenant `/api/lead-intake`. cross-origin POST 는 CORS 불필요 (HTML form\n * submission). PII 는 server 측에서 암호화 후 `inquiries` insert, 302 redirect.\n *\n * vertical prop:\n * - 미지정 = \"분야 선택\" select 노출 (consulting/medical/tax/legal)\n * - 지정 = hidden input + 표시 라벨 + vertical 별 추가 필드\n * (medical = 국외이전 동의 — ADR-0018)\n *\n * redirectPath prop:\n * - 외부 사이트가 폼 제출 후 돌아갈 path (admin 이 allowlist 검증, ?ok=1 / ?err=*)\n *\n * 0 JS RSC. customer-facing className 전체 `.rt-cms-*` prefix, scoped via\n * `[data-roottale-cms]` (cms-public.css). customer override 자유.\n */\n\nimport type { ReactElement } from \"react\";\n\nexport type LeadFormVertical = \"consulting\" | \"medical\" | \"tax\" | \"legal\";\n\nexport interface RootTaleLeadFormProps {\n /** admin-tenant lead-intake endpoint. 예: `https://mysite.roottale.com/api/lead-intake` */\n action: string;\n /**\n * vertical 고정. 미지정 시 select 노출.\n * `medical` 일 때 국외이전 동의 체크박스 추가 (ADR-0018).\n */\n vertical?: LeadFormVertical;\n /**\n * 폼 제출 후 admin이 돌려보낼 절대 URL.\n * admin 측 `LEAD_INTAKE_ALLOWED_ORIGINS` allowlist 검증. fail = fallback.\n * 미지정 시 admin env의 LEAD_INTAKE_REDIRECT_BASE 사용.\n */\n redirectUrl?: string;\n heading?: string;\n description?: string;\n submitLabel?: string;\n /** 추가 클래스 (customer 디자인 hook). */\n className?: string;\n}\n\nconst VERTICAL_LABEL: Record<LeadFormVertical, string> = {\n consulting: \"컨설팅\",\n medical: \"의료\",\n tax: \"세무\",\n legal: \"법률\",\n};\n\nconst VERTICAL_OPTIONS: { value: LeadFormVertical; label: string }[] = [\n { value: \"consulting\", label: \"컨설팅 / 일반\" },\n { value: \"medical\", label: \"의료 (병의원·치과·한의원)\" },\n { value: \"tax\", label: \"세무 (세무사·회계사)\" },\n { value: \"legal\", label: \"법률 (법무법인·변호사)\" },\n];\n\nexport function RootTaleLeadForm(\n props: RootTaleLeadFormProps,\n): ReactElement {\n const {\n action,\n vertical,\n redirectUrl,\n heading,\n description,\n submitLabel = \"진단 신청\",\n className,\n } = props;\n\n const formClass = [\"rt-cms-lead-form\", className].filter(Boolean).join(\" \");\n\n return (\n <form\n method=\"post\"\n action={action}\n data-roottale-cms=\"lead-form\"\n className={formClass}\n >\n {heading ? (\n <h2 className=\"rt-cms-lead-heading\">{heading}</h2>\n ) : null}\n {description ? (\n <p className=\"rt-cms-lead-description\">{description}</p>\n ) : null}\n\n {redirectUrl ? (\n <input type=\"hidden\" name=\"_redirect_url\" value={redirectUrl} />\n ) : null}\n\n {vertical ? (\n <>\n <input type=\"hidden\" name=\"vertical\" value={vertical} />\n <p className=\"rt-cms-lead-vertical-label\">\n 분야: <strong>{VERTICAL_LABEL[vertical]}</strong>\n </p>\n </>\n ) : (\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 분야 <span aria-hidden=\"true\">*</span>\n </span>\n <select\n name=\"vertical\"\n required\n className=\"rt-cms-field-input rt-cms-field-select\"\n defaultValue=\"\"\n >\n <option value=\"\" disabled>\n 선택하세요\n </option>\n {VERTICAL_OPTIONS.map((opt) => (\n <option key={opt.value} value={opt.value}>\n {opt.label}\n </option>\n ))}\n </select>\n </label>\n )}\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 이름 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"text\"\n name=\"contact_name\"\n required\n autoComplete=\"name\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 사업체명 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"text\"\n name=\"business_name\"\n required\n autoComplete=\"organization\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 이메일 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"email\"\n name=\"email\"\n required\n autoComplete=\"email\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 전화번호 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"tel\"\n name=\"phone\"\n required\n autoComplete=\"tel\"\n placeholder=\"010-0000-0000\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">현재 사이트 URL (선택)</span>\n <input\n type=\"url\"\n name=\"current_site_url\"\n placeholder=\"https://\"\n className=\"rt-cms-field-input\"\n />\n <span className=\"rt-cms-field-hint\">없으면 비워두세요.</span>\n </label>\n\n <label className=\"rt-cms-lead-consent\">\n <input type=\"checkbox\" name=\"privacy_consent\" required />\n <span>\n <strong>(필수)</strong> 개인정보 수집·이용에 동의합니다. (수집 항목:\n 이름·연락처·이메일·사업체명. 보관 기간: 문의 종료 후 3년)\n </span>\n </label>\n\n {vertical === \"medical\" ? (\n <label className=\"rt-cms-lead-consent\">\n <input\n type=\"checkbox\"\n name=\"overseas_transfer_consent\"\n required\n />\n <span>\n <strong>(필수, 의료)</strong> 의료 PII 의 국외이전(보관·처리)에\n 동의합니다. (의료법 §21·개인정보보호법 §28 — ADR-0018)\n </span>\n </label>\n ) : null}\n\n <div className=\"rt-cms-lead-actions\">\n <button type=\"submit\" className=\"rt-cms-lead-submit\">\n {submitLabel}\n </button>\n </div>\n </form>\n );\n}\n","// ADR-0034 §1.5 amended — Block JSON → React elements\n//\n// Next/RSC public renderer 가 본 함수로 block tree 를 server-side React element\n// 로 변환. cms-renderer-astro 의 `block-to-html.ts` 와 1:1 대응 (같은 7 + α\n// block 핸들러). `BlockDefinition.nextRender` 정합.\n//\n// 알려지지 않은 block 은 rawHtml 출력 (codex v3 verdict #4, opaque atom\n// round-trip 보존).\n\nimport type { ReactElement, ReactNode } from \"react\";\n\nimport type { Block } from \"@roottale/cms-core\";\n\nfunction isString(value: unknown): value is string {\n return typeof value === \"string\";\n}\n\nfunction htmlContent(html: string): { __html: string } {\n // React 의 dangerouslySetInnerHTML 표면. Block 의 rawHtml/content 는 이미\n // 서버측에서 받은 후 별도 sanitize 가 필요할 수 있음 (cms-core sanitize).\n return { __html: html };\n}\n\nfunction renderInner(blocks: readonly Block[]): ReactElement[] {\n return blocks.map((b, i) => renderBlock(b, i));\n}\n\n/**\n * 단일 block 을 ReactElement 로 변환. `block-to-html.ts` 와 동일 분기.\n */\nexport function renderBlock(block: Block, key: number = 0): ReactElement {\n const dataBlockId = block._id;\n\n switch (block.name) {\n case \"core/paragraph\": {\n const text = isString(block.attributes.content)\n ? block.attributes.content\n : block.rawHtml ?? \"\";\n return (\n <p\n key={key}\n data-block-id={dataBlockId}\n dangerouslySetInnerHTML={htmlContent(text)}\n />\n );\n }\n case \"core/heading\": {\n const levelRaw = block.attributes.level;\n const level =\n typeof levelRaw === \"number\" && levelRaw >= 1 && levelRaw <= 6\n ? levelRaw\n : 2;\n const text = isString(block.attributes.content)\n ? block.attributes.content\n : block.rawHtml ?? \"\";\n const props = {\n \"data-block-id\": dataBlockId,\n dangerouslySetInnerHTML: htmlContent(text),\n } as const;\n if (level === 1) return <h1 key={key} {...props} />;\n if (level === 2) return <h2 key={key} {...props} />;\n if (level === 3) return <h3 key={key} {...props} />;\n if (level === 4) return <h4 key={key} {...props} />;\n if (level === 5) return <h5 key={key} {...props} />;\n return <h6 key={key} {...props} />;\n }\n case \"core/image\": {\n const src = isString(block.attributes.url) ? block.attributes.url : \"\";\n const alt = isString(block.attributes.alt) ? block.attributes.alt : \"\";\n const caption = isString(block.attributes.caption)\n ? block.attributes.caption\n : null;\n return (\n <figure key={key} data-block-id={dataBlockId}>\n {/* eslint-disable-next-line @next/next/no-img-element */}\n <img src={src} alt={alt} loading=\"lazy\" />\n {caption && <figcaption>{caption}</figcaption>}\n </figure>\n );\n }\n case \"core/list\": {\n const ordered = block.attributes.ordered === true;\n const items = (block.innerBlocks ?? []).map(\n (item, i): ReactNode =>\n item.rawHtml ? (\n <li\n key={i}\n dangerouslySetInnerHTML={htmlContent(item.rawHtml)}\n />\n ) : (\n <li key={i}>{renderInner(item.innerBlocks ?? [])}</li>\n ),\n );\n return ordered ? (\n <ol key={key} data-block-id={dataBlockId}>\n {items}\n </ol>\n ) : (\n <ul key={key} data-block-id={dataBlockId}>\n {items}\n </ul>\n );\n }\n case \"core/quote\": {\n const inner = renderInner(block.innerBlocks ?? []);\n const citation = isString(block.attributes.citation)\n ? block.attributes.citation\n : null;\n return (\n <blockquote key={key} data-block-id={dataBlockId}>\n {inner}\n {citation && <cite>{citation}</cite>}\n </blockquote>\n );\n }\n case \"core/code\": {\n const code = isString(block.attributes.content)\n ? block.attributes.content\n : block.rawHtml ?? \"\";\n const lang = isString(block.attributes.language)\n ? block.attributes.language\n : undefined;\n return (\n <pre\n key={key}\n data-block-id={dataBlockId}\n data-language={lang}\n >\n <code>{code}</code>\n </pre>\n );\n }\n case \"core/columns\": {\n return (\n <div\n key={key}\n className=\"rt-columns\"\n data-block-id={dataBlockId}\n >\n {renderInner(block.innerBlocks ?? [])}\n </div>\n );\n }\n case \"core/separator\":\n return <hr key={key} data-block-id={dataBlockId} />;\n case \"core/spacer\": {\n const heightRaw = block.attributes.height;\n const height = isString(heightRaw) ? heightRaw : \"32px\";\n return (\n <div\n key={key}\n className=\"rt-spacer\"\n data-block-id={dataBlockId}\n style={{ height }}\n />\n );\n }\n case \"core/group\": {\n return (\n <div\n key={key}\n className=\"rt-group\"\n data-block-id={dataBlockId}\n >\n {renderInner(block.innerBlocks ?? [])}\n </div>\n );\n }\n default: {\n const rawHtml = block.rawHtml;\n if (rawHtml) {\n return (\n <div\n key={key}\n className=\"rt-unknown-block\"\n data-block-id={dataBlockId}\n data-block-name={block.name}\n dangerouslySetInnerHTML={htmlContent(rawHtml)}\n />\n );\n }\n return (\n <div\n key={key}\n className=\"rt-unknown-block\"\n data-block-id={dataBlockId}\n data-block-name={block.name}\n >\n {renderInner(block.innerBlocks ?? [])}\n </div>\n );\n }\n }\n}\n\n/** Block tree 전체를 React element 배열로 직렬화. */\nexport function renderBlocks(blocks: readonly Block[]): ReactElement[] {\n return blocks.map((b, i) => renderBlock(b, i));\n}\n"],"mappings":"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","names":["jsx","jsxs","jsx","jsxs","jsx","jsxs","jsx","jsxs","Fragment","jsx","jsxs","formatPublishedDate"]}
1
+ {"version":3,"sources":["../src/server.tsx","../src/PostCard.tsx","../src/TableOfContents.tsx","../src/toc.ts","../src/FloatingCta.tsx","../src/LeadForm.tsx","../src/block-to-react.tsx"],"sourcesContent":["/**\n * `@roottale/cms-renderer-next/server` — RootTale CMS public-render React Server Components.\n *\n * Drop into any RSC-capable framework (Next.js App Router, Astro server islands,\n * React Router server). Single-line import, SSR-only, no `'use client'` boundary.\n *\n * ```tsx\n * import { RootTaleBlogList } from \"@roottale/cms-renderer-next/server\";\n *\n * export default function BlogPage() {\n * return <RootTaleBlogList apiKey={process.env.ROOTTALE_API_KEY!} />;\n * }\n * ```\n *\n * Customer site MUST keep `apiKey` server-side. Browser bundle ships zero\n * RootTale credentials (ADR-0023 §5.1 #15).\n *\n * Design tokens auto-apply via the CSS import below. Customers can override\n * any class with their own CSS or Tailwind utilities — scoped selectors use\n * `:where()` to keep specificity at zero (ADR-0023 §5.1 #10).\n */\n\nimport type { CSSProperties, ReactElement, ReactNode } from \"react\";\nimport {\n type CmsPostContent,\n type CmsPostType,\n type RootTaleTheme,\n fetchPost,\n fetchPosts,\n fetchTheme,\n} from \"@roottale/cms-client/server\";\n// CSS는 customer 가 root layout 에서 명시적으로 import:\n// `import \"@roottale/cms-renderer-next/styles\";`\n// 본 파일에서 side-effect import 하지 않음 — tsup dts build 가 .css\n// side-effect import 처리 못 함 + customer 측 framework 마다 CSS 처리\n// 방식 다르므로 명시적 import 가 명확. README 참조.\n\nimport { RootTalePostCard } from \"./PostCard.js\";\nimport { RootTaleTableOfContents } from \"./TableOfContents.js\";\nimport { attachHeadingIds, extractToc } from \"./toc.js\";\n\nexport { RootTalePostCard, type RootTalePostCardProps } from \"./PostCard.js\";\nexport {\n RootTaleTableOfContents,\n type RootTaleTableOfContentsProps,\n} from \"./TableOfContents.js\";\nexport {\n RootTaleFloatingCta,\n type RootTaleFloatingCtaProps,\n type CtaButton,\n type CtaButtonType,\n type FloatingCtaPosition,\n} from \"./FloatingCta.js\";\nexport {\n RootTaleLeadForm,\n type RootTaleLeadFormProps,\n type LeadFormVertical,\n} from \"./LeadForm.js\";\nexport { renderBlock, renderBlocks } from \"./block-to-react.js\";\nexport {\n attachHeadingIds,\n extractToc,\n headingToId,\n type TocEntry,\n} from \"./toc.js\";\nexport type { RootTaleTheme } from \"@roottale/cms-client/server\";\n\n/**\n * RootTaleBlogList/Post 가 받는 theme prop 의 평가 결과:\n * - `RootTaleTheme` 객체 → 그대로 CSS 변수로 주입\n * - `null` → override 없음 (cms-public.css fallback 그대로)\n * - `undefined` → API key 로 자동 fetch (mysite.roottale.com 의 admin /design)\n */\nexport type RootTaleThemeInput = RootTaleTheme | null | undefined;\n\n/**\n * RootTale 디자인 토큰을 CSS 변수로 자식 트리에 주입한다. 외부 사이트가\n * 여러 RootTale 컴포넌트를 한 레이아웃에 묶을 때, 컴포넌트마다 같은 토큰을\n * fetch 하지 않도록 부모에서 한 번만 fetch 해서 전달하는 용도.\n *\n * ```tsx\n * const theme = await fetchTheme({ apiKey });\n * <RootTaleThemeProvider theme={theme}>\n * <RootTaleBlogList apiKey={apiKey} theme={null} />\n * <RootTaleBlogPost apiKey={apiKey} slugOrId=\"...\" theme={null} />\n * </RootTaleThemeProvider>\n * ```\n *\n * 자식 RootTale 컴포넌트가 `theme={null}` 을 받으면 자체 주입을 건너뛰고\n * 부모의 변수를 그대로 상속. 자식이 `theme` 을 지정하면 자식 변수가 우선.\n */\nexport function RootTaleThemeProvider(props: {\n theme: RootTaleTheme | null | undefined;\n children: ReactNode;\n}): ReactElement {\n const style = themeToCssVars(props.theme ?? null);\n if (!style) return <>{props.children}</>;\n return (\n <div data-roottale-cms=\"theme-root\" style={style}>\n {props.children}\n </div>\n );\n}\n\nexport type RootTaleBlogListProps = {\n apiKey: string;\n baseUrl?: string;\n limit?: number;\n /** Filter by post type (`post` or `page`). Default = `post` only (blog list 의도). */\n type?: CmsPostType;\n /** Build a customer-side URL for a single post. Defaults to `/blog/${slug}`. */\n postHref?: (post: CmsPostContent) => string;\n /** Rendered when the tenant has no published posts yet. */\n emptyMessage?: string;\n /**\n * RootTale 디자인 토큰. 생략 시 admin /design 에서 저장한 값을 자동 fetch.\n * `null` 을 명시하면 자동 fetch 도 건너뛰고 CSS fallback 사용 (부모\n * `<RootTaleThemeProvider>` 가 변수를 이미 설정한 경우 유용).\n */\n theme?: RootTaleThemeInput;\n};\n\nexport async function RootTaleBlogList(\n props: RootTaleBlogListProps,\n): Promise<ReactElement> {\n const {\n apiKey,\n baseUrl,\n limit,\n type = \"post\",\n postHref = defaultPostHref,\n emptyMessage = \"아직 발행된 글이 없습니다.\",\n theme: themeProp,\n } = props;\n\n const [page, themeStyle] = await Promise.all([\n fetchPosts({ apiKey, baseUrl, limit, type }),\n resolveThemeStyle({ themeProp, apiKey, baseUrl }),\n ]);\n\n if (page.items.length === 0) {\n return (\n <div data-roottale-cms=\"list\" style={themeStyle ?? undefined}>\n <p className=\"rt-cms-empty\">{emptyMessage}</p>\n </div>\n );\n }\n\n return (\n <div data-roottale-cms=\"list\" style={themeStyle ?? undefined}>\n <ul className=\"rt-cms-list\">\n {page.items.map((post) => (\n <li key={post.id} className=\"rt-cms-list-item\">\n <RootTalePostCard post={post} href={postHref} />\n </li>\n ))}\n </ul>\n </div>\n );\n}\n\nexport type RootTaleBlogPostProps = {\n apiKey: string;\n slugOrId: string;\n baseUrl?: string;\n /** Rendered when the slug is not found or not published. */\n notFoundElement?: ReactElement;\n /**\n * When true, prepends an auto-derived `<RootTaleTableOfContents>` before\n * the article body (H2/H3 headings only). Default: false — keep markup\n * stable for customers who already control their own layout.\n */\n showTableOfContents?: boolean;\n /** Title shown above the TOC (only when `showTableOfContents`). */\n tableOfContentsTitle?: string;\n /** See `RootTaleBlogListProps['theme']`. */\n theme?: RootTaleThemeInput;\n};\n\nexport async function RootTaleBlogPost(\n props: RootTaleBlogPostProps,\n): Promise<ReactElement> {\n const {\n apiKey,\n slugOrId,\n baseUrl,\n notFoundElement,\n showTableOfContents = false,\n tableOfContentsTitle,\n theme: themeProp,\n } = props;\n const [post, themeStyle] = await Promise.all([\n fetchPost({ apiKey, slugOrId, baseUrl }),\n resolveThemeStyle({ themeProp, apiKey, baseUrl }),\n ]);\n\n if (!post) {\n return (\n notFoundElement ?? (\n <div data-roottale-cms=\"post-missing\" style={themeStyle ?? undefined}>\n <p className=\"rt-cms-empty\">글을 찾을 수 없습니다.</p>\n </div>\n )\n );\n }\n\n const toc = showTableOfContents ? extractToc(post.bodyJson) : [];\n const renderDoc =\n toc.length > 0 ? attachHeadingIds(post.bodyJson, toc) : post.bodyJson;\n\n return (\n <article\n data-roottale-cms=\"post\"\n className=\"rt-cms-article\"\n style={themeStyle ?? undefined}\n >\n <header>\n <p className=\"rt-cms-meta\">{formatPublishedDate(post.publishedAt)}</p>\n <h1 className=\"rt-cms-title\">{post.title}</h1>\n {post.excerpt ? <p className=\"rt-cms-excerpt\">{post.excerpt}</p> : null}\n </header>\n {toc.length > 0 ? (\n <RootTaleTableOfContents entries={toc} title={tableOfContentsTitle} />\n ) : null}\n <RenderTiptap doc={renderDoc} />\n </article>\n );\n}\n\n/**\n * theme prop 의 3가지 입력값 (object / null / undefined) 을 React inline style\n * 로 변환. `null` 또는 빈 객체는 style 자체를 생략 → 부모에서 이미 변수를\n * 설정했다면 자식이 덮어쓰지 않음.\n */\nasync function resolveThemeStyle(input: {\n themeProp: RootTaleThemeInput;\n apiKey: string;\n baseUrl?: string;\n}): Promise<CSSProperties | null> {\n const { themeProp, apiKey, baseUrl } = input;\n if (themeProp === null) return null;\n if (themeProp !== undefined) return themeToCssVars(themeProp);\n // undefined → auto-fetch. theme fetch 실패는 비치명적 — 빈 토큰으로 fallback.\n try {\n const fetched = await fetchTheme({ apiKey, baseUrl });\n return themeToCssVars(fetched);\n } catch {\n return null;\n }\n}\n\nfunction themeToCssVars(theme: RootTaleTheme | null): CSSProperties | null {\n if (!theme) return null;\n const vars: Record<string, string> = {};\n const c = theme.colors ?? {};\n if (c.primary) vars[\"--rt-color-primary\"] = c.primary;\n if (c.primaryForeground) vars[\"--rt-color-primary-foreground\"] = c.primaryForeground;\n if (c.foreground) vars[\"--rt-color-foreground\"] = c.foreground;\n if (c.background) vars[\"--rt-color-background\"] = c.background;\n if (c.muted) vars[\"--rt-color-muted\"] = c.muted;\n if (c.mutedForeground) vars[\"--rt-color-muted-foreground\"] = c.mutedForeground;\n if (c.border) vars[\"--rt-color-border\"] = c.border;\n const f = theme.fonts ?? {};\n if (f.body) vars[\"--rt-font-body\"] = f.body;\n if (f.display) vars[\"--rt-font-display\"] = f.display;\n const r = theme.radius ?? {};\n if (r.md) vars[\"--rt-radius-md\"] = r.md;\n if (Object.keys(vars).length === 0) return null;\n return vars as CSSProperties;\n}\n\nfunction defaultPostHref(post: CmsPostContent): string {\n return `/blog/${post.slug}`;\n}\n\nfunction formatPublishedDate(iso: string): string {\n const d = new Date(iso);\n if (Number.isNaN(d.getTime())) return \"\";\n return new Intl.DateTimeFormat(\"ko-KR\", {\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n }).format(d);\n}\n\n/**\n * Minimal Tiptap doc renderer. Walks the Tiptap doc tree and emits semantic\n * HTML. This is intentionally tiny — richer extensions (image asset URL\n * resolution, TableOfContents, FloatingCta) belong in followup §5.7 #5.\n *\n * Browser-side sanitization is unnecessary here because output is rendered\n * server-side from data the customer-trusted API returned. If the customer\n * site federates with untrusted authors later, layer a sanitizer in front\n * of this component.\n */\nfunction RenderTiptap({ doc }: { doc: Record<string, unknown> }): ReactElement {\n const rawContent = (doc as TiptapDoc).content;\n const content: TiptapNode[] = Array.isArray(rawContent) ? rawContent : [];\n return <>{content.map((node, i) => renderNode(node, i))}</>;\n}\n\ntype TiptapDoc = { type?: string; content?: TiptapNode[] };\n\ntype TiptapMark = { type?: string; attrs?: Record<string, unknown> };\n\ntype TiptapNode = {\n type?: string;\n text?: string;\n attrs?: Record<string, unknown>;\n content?: TiptapNode[];\n marks?: TiptapMark[];\n};\n\n/**\n * Defense-in-depth: even though admin LinkButton.normalizeUrl filters input,\n * legacy imports / direct DB / external tenants can still emit unsafe href.\n * Allow only http(s)/mailto/tel/root-relative/fragment/query — falls back to \"#\".\n */\nfunction isSafeHref(value: unknown): value is string {\n if (typeof value !== \"string\") return false;\n const v = value.trim();\n if (v.length === 0) return false;\n if (v.startsWith(\"/\") || v.startsWith(\"#\") || v.startsWith(\"?\")) return true;\n return /^(https?:|mailto:|tel:)/i.test(v);\n}\n\nfunction alignStyle(node: TiptapNode): CSSProperties | undefined {\n const align = node.attrs?.textAlign;\n if (typeof align !== \"string\" || align === \"\" || align === \"left\") return undefined;\n if (align === \"center\" || align === \"right\" || align === \"justify\") {\n return { textAlign: align };\n }\n return undefined;\n}\n\nfunction renderNode(node: TiptapNode, key: number): ReactElement | string | null {\n if (!node || typeof node !== \"object\") return null;\n const children = Array.isArray(node.content)\n ? node.content.map((child, i) => renderNode(child, i))\n : undefined;\n\n switch (node.type) {\n case \"paragraph\":\n return <p key={key} style={alignStyle(node)}>{children}</p>;\n case \"heading\": {\n const level = Math.min(Math.max(Number(node.attrs?.level ?? 2), 1), 3);\n const idAttr = node.attrs?.id;\n const id = typeof idAttr === \"string\" && idAttr.length > 0 ? idAttr : undefined;\n const style = alignStyle(node);\n if (level === 1) return <h1 key={key} id={id} style={style}>{children}</h1>;\n if (level === 2) return <h2 key={key} id={id} style={style}>{children}</h2>;\n return <h3 key={key} id={id} style={style}>{children}</h3>;\n }\n case \"bulletList\":\n return <ul key={key}>{children}</ul>;\n case \"orderedList\":\n return <ol key={key}>{children}</ol>;\n case \"listItem\":\n return <li key={key}>{children}</li>;\n case \"blockquote\":\n return <blockquote key={key}>{children}</blockquote>;\n case \"codeBlock\":\n return (\n <pre key={key}>\n <code>{children}</code>\n </pre>\n );\n case \"horizontalRule\":\n return <hr key={key} />;\n case \"hardBreak\":\n return <br key={key} />;\n case \"image\": {\n const src = node.attrs?.src;\n if (typeof src !== \"string\" || src.length === 0) return null;\n const alt = typeof node.attrs?.alt === \"string\" ? node.attrs.alt : \"\";\n const title = typeof node.attrs?.title === \"string\" ? node.attrs.title : undefined;\n // eslint-disable-next-line @next/next/no-img-element\n return <img key={key} src={src} alt={alt} title={title} loading=\"lazy\" />;\n }\n case \"columns\":\n return <div key={key} className=\"rt-columns\">{children}</div>;\n case \"column\":\n return <div key={key} className=\"rt-column\">{children}</div>;\n case \"text\":\n return renderText(node, key);\n default:\n return children ? <span key={key}>{children}</span> : null;\n }\n}\n\nfunction renderText(node: TiptapNode, key: number): ReactElement | string {\n const text = node.text ?? \"\";\n const marks = node.marks ?? [];\n if (marks.length === 0) return text;\n let element: ReactElement | string = text;\n for (const mark of marks) {\n element = applyMark(mark, element);\n }\n return <span key={key}>{element}</span>;\n}\n\nfunction applyMark(mark: TiptapMark, child: ReactElement | string): ReactElement {\n switch (mark.type) {\n case \"bold\":\n return <strong>{child}</strong>;\n case \"italic\":\n return <em>{child}</em>;\n case \"underline\":\n return <u>{child}</u>;\n case \"strike\":\n return <s>{child}</s>;\n case \"code\":\n return <code>{child}</code>;\n case \"highlight\": {\n const color = mark.attrs?.color;\n const style =\n typeof color === \"string\" && color.length > 0\n ? { background: color }\n : undefined;\n return <mark style={style}>{child}</mark>;\n }\n case \"textStyle\": {\n const color = mark.attrs?.color;\n const style =\n typeof color === \"string\" && color.length > 0 ? { color } : undefined;\n if (!style) return <>{child}</>;\n return <span style={style}>{child}</span>;\n }\n case \"link\": {\n const raw = mark.attrs?.href;\n const href = isSafeHref(raw) ? raw : \"#\";\n return (\n <a href={href} rel=\"noopener noreferrer\" target=\"_blank\">\n {child}\n </a>\n );\n }\n default:\n return <span>{child}</span>;\n }\n}\n","/**\n * `<RootTalePostCard>` — server-only summary card for a single CmsPostContent.\n *\n * Building block for `<RootTaleBlogList>` (and customer-side custom grids).\n * Pure SSR — anchor + title + meta + excerpt. Featured image is left out for\n * now: `CmsPostContent` only carries `featuredMediaId`, not a public URL.\n * Media URL resolution belongs in a separate slice once `cms-client` exposes\n * it (ADR-0034 §4 media followup — CF Images variants).\n */\n\nimport type { ReactElement } from \"react\";\nimport type { CmsPostContent } from \"@roottale/cms-client/server\";\n\nexport interface RootTalePostCardProps {\n post: CmsPostContent;\n /** Customer-side URL builder. Defaults to `/blog/${slug}`. */\n href?: (post: CmsPostContent) => string;\n}\n\nexport function RootTalePostCard(props: RootTalePostCardProps): ReactElement {\n const { post, href = defaultHref } = props;\n return (\n <article data-roottale-cms=\"card\" className=\"rt-cms-card\">\n <a className=\"rt-cms-card-link\" href={href(post)}>\n <p className=\"rt-cms-meta\">{formatPublishedDate(post.publishedAt)}</p>\n <h2 className=\"rt-cms-title\">{post.title}</h2>\n {post.excerpt ? <p className=\"rt-cms-excerpt\">{post.excerpt}</p> : null}\n </a>\n </article>\n );\n}\n\nfunction defaultHref(post: CmsPostContent): string {\n return `/blog/${post.slug}`;\n}\n\nfunction formatPublishedDate(iso: string): string {\n const d = new Date(iso);\n if (Number.isNaN(d.getTime())) return \"\";\n return new Intl.DateTimeFormat(\"ko-KR\", {\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n }).format(d);\n}\n","/**\n * `<RootTaleTableOfContents>` — server-only anchor list for blog posts.\n *\n * Pure SSR. Renders `<nav>` + ordered anchors that link to `#<id>` slots\n * emitted by `attachHeadingIds`. No scroll-spy here — that requires an\n * `IntersectionObserver` which only exists in the browser; a client-island\n * variant can layer on top later without changing the data contract.\n *\n * Internal ancestor: `roottale-internal/packages/cms-renderer/src/TableOfContents.tsx`\n * (had `'use client'` + IntersectionObserver). Stripped to server-safe form per\n * the platform renderer contract (no `'use client'` boundary — ADR-0023 §5.3).\n */\n\nimport type { ReactElement } from \"react\";\nimport type { TocEntry } from \"./toc.js\";\n\nexport interface RootTaleTableOfContentsProps {\n entries: TocEntry[];\n title?: string;\n}\n\nexport function RootTaleTableOfContents(\n props: RootTaleTableOfContentsProps,\n): ReactElement | null {\n const { entries, title = \"목차\" } = props;\n if (entries.length === 0) return null;\n return (\n <nav data-roottale-cms=\"toc\" className=\"rt-cms-toc\" aria-label={title}>\n {title ? <p className=\"rt-cms-toc-title\">{title}</p> : null}\n <ol className=\"rt-cms-toc-list\">\n {entries.map((entry) => (\n <li\n key={entry.id}\n className={`rt-cms-toc-item rt-cms-toc-level-${entry.level}`}\n >\n <a href={`#${entry.id}`}>{entry.text}</a>\n </li>\n ))}\n </ol>\n </nav>\n );\n}\n","/**\n * `@roottale/cms-renderer/toc` — Tiptap doc → TOC entries + heading id injection.\n *\n * Server-only utility. Walks a Tiptap JSON doc, extracts H2/H3 headings, and\n * derives stable kebab-case ids (Korean-safe). The companion `attachHeadingIds`\n * mutates a shallow clone of the doc so renderers can emit `<h2 id=\"...\">`\n * anchors that match what `<RootTaleTableOfContents>` links to.\n *\n * Internal ancestor: `roottale-internal/packages/cms-renderer/src/toc.ts`\n * (HTML-string based, vendored snapshot). Ported to Tiptap-JSON because the\n * platform renderer no longer round-trips through HTML before render.\n */\n\nexport interface TocEntry {\n level: 2 | 3;\n text: string;\n id: string;\n}\n\ntype TiptapNode = {\n type?: string;\n text?: string;\n attrs?: Record<string, unknown>;\n content?: TiptapNode[];\n marks?: { type?: string }[];\n};\n\ntype TiptapDoc = { type?: string; content?: TiptapNode[] };\n\nexport function headingToId(text: string): string {\n return text\n .trim()\n .toLowerCase()\n .replace(/[^\\p{L}\\p{N}\\s-]/gu, \"\")\n .replace(/\\s+/g, \"-\")\n .replace(/-+/g, \"-\")\n .replace(/^-|-$/g, \"\");\n}\n\nfunction textOf(node: TiptapNode): string {\n if (typeof node.text === \"string\") return node.text;\n if (!Array.isArray(node.content)) return \"\";\n return node.content.map(textOf).join(\"\");\n}\n\nexport function extractToc(doc: Record<string, unknown>): TocEntry[] {\n const root = doc as TiptapDoc;\n const content = Array.isArray(root.content) ? root.content : [];\n const entries: TocEntry[] = [];\n const idCounts = new Map<string, number>();\n\n for (const node of content) {\n if (node?.type !== \"heading\") continue;\n const rawLevel = Number(node.attrs?.level ?? 2);\n if (rawLevel !== 2 && rawLevel !== 3) continue;\n const text = textOf(node).trim();\n if (!text) continue;\n\n const base = headingToId(text) || \"heading\";\n const count = idCounts.get(base) ?? 0;\n const id = count === 0 ? base : `${base}-${count}`;\n idCounts.set(base, count + 1);\n\n entries.push({ level: rawLevel, text, id });\n }\n\n return entries;\n}\n\n/**\n * Returns a shallow-cloned Tiptap doc where each H2/H3 carries the matching\n * `attrs.id` from `entries` (same order as `extractToc` returns). Existing\n * `id` attrs are overwritten so renderer + TOC always agree on anchors.\n */\nexport function attachHeadingIds(\n doc: Record<string, unknown>,\n entries: TocEntry[],\n): Record<string, unknown> {\n const root = doc as TiptapDoc;\n const content = Array.isArray(root.content) ? root.content : [];\n let cursor = 0;\n const nextContent = content.map((node) => {\n if (node?.type !== \"heading\") return node;\n const rawLevel = Number(node.attrs?.level ?? 2);\n if (rawLevel !== 2 && rawLevel !== 3) return node;\n if (!textOf(node).trim()) return node;\n const entry = entries[cursor++];\n if (!entry) return node;\n return {\n ...node,\n attrs: { ...(node.attrs ?? {}), id: entry.id },\n };\n });\n return { ...(root as Record<string, unknown>), content: nextContent };\n}\n","/**\n * `<RootTaleFloatingCta>` — fixed-position call-to-action stack.\n *\n * Server-only (anchors only, no state). External targets (`kakao` / `custom`)\n * open in a new tab with `rel=\"noopener noreferrer\"`; tel/contact stay in the\n * same tab so iOS/Android can trigger the dialer / mailto handler.\n *\n * Internal ancestor: `roottale-internal/packages/cms-renderer/src/FloatingCta.tsx`.\n */\n\nimport type { ReactElement } from \"react\";\n\nexport type CtaButtonType = \"phone\" | \"contact\" | \"kakao\" | \"custom\";\n\nexport interface CtaButton {\n type: CtaButtonType;\n label: string;\n href: string;\n /** Optional override icon (emoji or short text). */\n icon?: string;\n}\n\nexport type FloatingCtaPosition =\n | \"bottom-right\"\n | \"bottom-left\"\n | \"bottom-center\";\n\nexport interface RootTaleFloatingCtaProps {\n buttons: CtaButton[];\n position?: FloatingCtaPosition;\n}\n\nexport function RootTaleFloatingCta(\n props: RootTaleFloatingCtaProps,\n): ReactElement | null {\n const { buttons, position = \"bottom-right\" } = props;\n if (!buttons || buttons.length === 0) return null;\n return (\n <div\n data-roottale-cms=\"floating-cta\"\n className={`rt-cms-floating-cta rt-cms-floating-cta--${position}`}\n >\n {buttons.map((btn, idx) => {\n const isExternal = btn.type === \"kakao\" || btn.type === \"custom\";\n return (\n <a\n key={`${btn.type}-${idx}`}\n href={btn.href}\n className={`rt-cms-floating-cta__btn rt-cms-floating-cta__btn--${btn.type}`}\n target={isExternal ? \"_blank\" : undefined}\n rel={isExternal ? \"noopener noreferrer\" : undefined}\n >\n <span aria-hidden=\"true\" className=\"rt-cms-floating-cta__icon\">\n {btn.icon ?? defaultIcon(btn.type)}\n </span>\n <span className=\"rt-cms-floating-cta__label\">{btn.label}</span>\n </a>\n );\n })}\n </div>\n );\n}\n\nfunction defaultIcon(type: CtaButtonType): string {\n switch (type) {\n case \"phone\":\n return \"☎\";\n case \"contact\":\n return \"✉\";\n case \"kakao\":\n return \"💬\";\n case \"custom\":\n return \"→\";\n }\n}\n","/**\n * `@roottale/cms-renderer-next` — RootTaleLeadForm RSC.\n *\n * 외부 고객 사이트(roottale.com, 고객사 외주)의 진단 신청 폼. HTML form POST →\n * tenant-api `/v1/public/inquiries`. cross-origin POST 는 CORS 불필요 (HTML form\n * submission). PII 는 server 측에서 암호화 후 `inquiries` insert, 302 redirect.\n *\n * vertical prop:\n * - 미지정 = \"분야 선택\" select 노출 (consulting/medical/tax/legal)\n * - 지정 = hidden input + 표시 라벨 + vertical 별 추가 필드\n * (medical = 국외이전 동의 — ADR-0018)\n *\n * redirectPath prop:\n * - 외부 사이트가 폼 제출 후 돌아갈 path (API 가 allowlist 검증, ?ok=1 / ?err=*)\n *\n * 0 JS RSC. customer-facing className 전체 `.rt-cms-*` prefix, scoped via\n * `[data-roottale-cms]` (cms-public.css). customer override 자유.\n */\n\nimport type { ReactElement } from \"react\";\n\nexport type LeadFormVertical = \"consulting\" | \"medical\" | \"tax\" | \"legal\";\n\nexport interface RootTaleLeadFormProps {\n /** lead intake endpoint. 예: `https://api.roottale.com/v1/public/inquiries` */\n action: string;\n /**\n * vertical 고정. 미지정 시 select 노출.\n * `medical` 일 때 국외이전 동의 체크박스 추가 (ADR-0018).\n */\n vertical?: LeadFormVertical;\n /**\n * 폼 제출 후 API가 돌려보낼 절대 URL.\n * tenant-api `LEAD_INTAKE_ALLOWED_ORIGINS` allowlist 검증. fail = fallback.\n * 미지정 시 tenant-api env의 LEAD_INTAKE_REDIRECT_BASE 사용.\n */\n redirectUrl?: string;\n heading?: string;\n description?: string;\n submitLabel?: string;\n /** 추가 클래스 (customer 디자인 hook). */\n className?: string;\n}\n\nconst VERTICAL_LABEL: Record<LeadFormVertical, string> = {\n consulting: \"컨설팅\",\n medical: \"의료\",\n tax: \"세무\",\n legal: \"법률\",\n};\n\nconst VERTICAL_OPTIONS: { value: LeadFormVertical; label: string }[] = [\n { value: \"consulting\", label: \"컨설팅 / 일반\" },\n { value: \"medical\", label: \"의료 (병의원·치과·한의원)\" },\n { value: \"tax\", label: \"세무 (세무사·회계사)\" },\n { value: \"legal\", label: \"법률 (법무법인·변호사)\" },\n];\n\nexport function RootTaleLeadForm(\n props: RootTaleLeadFormProps,\n): ReactElement {\n const {\n action,\n vertical,\n redirectUrl,\n heading,\n description,\n submitLabel = \"진단 신청\",\n className,\n } = props;\n\n const formClass = [\"rt-cms-lead-form\", className].filter(Boolean).join(\" \");\n\n return (\n <form\n method=\"post\"\n action={action}\n data-roottale-cms=\"lead-form\"\n className={formClass}\n >\n {heading ? (\n <h2 className=\"rt-cms-lead-heading\">{heading}</h2>\n ) : null}\n {description ? (\n <p className=\"rt-cms-lead-description\">{description}</p>\n ) : null}\n\n {redirectUrl ? (\n <input type=\"hidden\" name=\"_redirect_url\" value={redirectUrl} />\n ) : null}\n\n {vertical ? (\n <>\n <input type=\"hidden\" name=\"vertical\" value={vertical} />\n <p className=\"rt-cms-lead-vertical-label\">\n 분야: <strong>{VERTICAL_LABEL[vertical]}</strong>\n </p>\n </>\n ) : (\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 분야 <span aria-hidden=\"true\">*</span>\n </span>\n <select\n name=\"vertical\"\n required\n className=\"rt-cms-field-input rt-cms-field-select\"\n defaultValue=\"\"\n >\n <option value=\"\" disabled>\n 선택하세요\n </option>\n {VERTICAL_OPTIONS.map((opt) => (\n <option key={opt.value} value={opt.value}>\n {opt.label}\n </option>\n ))}\n </select>\n </label>\n )}\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 이름 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"text\"\n name=\"contact_name\"\n required\n autoComplete=\"name\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 사업체명 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"text\"\n name=\"business_name\"\n required\n autoComplete=\"organization\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 이메일 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"email\"\n name=\"email\"\n required\n autoComplete=\"email\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">\n 전화번호 <span aria-hidden=\"true\">*</span>\n </span>\n <input\n type=\"tel\"\n name=\"phone\"\n required\n autoComplete=\"tel\"\n placeholder=\"010-0000-0000\"\n className=\"rt-cms-field-input\"\n />\n </label>\n\n <label className=\"rt-cms-field\">\n <span className=\"rt-cms-field-label\">현재 사이트 URL (선택)</span>\n <input\n type=\"url\"\n name=\"current_site_url\"\n placeholder=\"https://\"\n className=\"rt-cms-field-input\"\n />\n <span className=\"rt-cms-field-hint\">없으면 비워두세요.</span>\n </label>\n\n <label className=\"rt-cms-lead-consent\">\n <input type=\"checkbox\" name=\"privacy_consent\" required />\n <span>\n <strong>(필수)</strong> 개인정보 수집·이용에 동의합니다. (수집 항목:\n 이름·연락처·이메일·사업체명. 보관 기간: 문의 종료 후 3년)\n </span>\n </label>\n\n {vertical === \"medical\" ? (\n <label className=\"rt-cms-lead-consent\">\n <input\n type=\"checkbox\"\n name=\"overseas_transfer_consent\"\n required\n />\n <span>\n <strong>(필수, 의료)</strong> 의료 PII 의 국외이전(보관·처리)에\n 동의합니다. (의료법 §21·개인정보보호법 §28 — ADR-0018)\n </span>\n </label>\n ) : null}\n\n <div className=\"rt-cms-lead-actions\">\n <button type=\"submit\" className=\"rt-cms-lead-submit\">\n {submitLabel}\n </button>\n </div>\n </form>\n );\n}\n","// ADR-0034 §1.5 amended — Block JSON → React elements\n//\n// Next/RSC public renderer 가 본 함수로 block tree 를 server-side React element\n// 로 변환. cms-renderer-astro 의 `block-to-html.ts` 와 1:1 대응 (같은 7 + α\n// block 핸들러). `BlockDefinition.nextRender` 정합.\n//\n// 알려지지 않은 block 은 rawHtml 출력 (codex v3 verdict #4, opaque atom\n// round-trip 보존).\n\nimport type { ReactElement, ReactNode } from \"react\";\n\nimport type { Block } from \"@roottale/cms-core\";\n\nfunction isString(value: unknown): value is string {\n return typeof value === \"string\";\n}\n\nfunction htmlContent(html: string): { __html: string } {\n // React 의 dangerouslySetInnerHTML 표면. Block 의 rawHtml/content 는 이미\n // 서버측에서 받은 후 별도 sanitize 가 필요할 수 있음 (cms-core sanitize).\n return { __html: html };\n}\n\nfunction renderInner(blocks: readonly Block[]): ReactElement[] {\n return blocks.map((b, i) => renderBlock(b, i));\n}\n\n/**\n * 단일 block 을 ReactElement 로 변환. `block-to-html.ts` 와 동일 분기.\n */\nexport function renderBlock(block: Block, key: number = 0): ReactElement {\n const dataBlockId = block._id;\n\n switch (block.name) {\n case \"core/paragraph\": {\n const text = isString(block.attributes.content)\n ? block.attributes.content\n : block.rawHtml ?? \"\";\n return (\n <p\n key={key}\n data-block-id={dataBlockId}\n dangerouslySetInnerHTML={htmlContent(text)}\n />\n );\n }\n case \"core/heading\": {\n const levelRaw = block.attributes.level;\n const level =\n typeof levelRaw === \"number\" && levelRaw >= 1 && levelRaw <= 6\n ? levelRaw\n : 2;\n const text = isString(block.attributes.content)\n ? block.attributes.content\n : block.rawHtml ?? \"\";\n const props = {\n \"data-block-id\": dataBlockId,\n dangerouslySetInnerHTML: htmlContent(text),\n } as const;\n if (level === 1) return <h1 key={key} {...props} />;\n if (level === 2) return <h2 key={key} {...props} />;\n if (level === 3) return <h3 key={key} {...props} />;\n if (level === 4) return <h4 key={key} {...props} />;\n if (level === 5) return <h5 key={key} {...props} />;\n return <h6 key={key} {...props} />;\n }\n case \"core/image\": {\n const src = isString(block.attributes.url) ? block.attributes.url : \"\";\n const alt = isString(block.attributes.alt) ? block.attributes.alt : \"\";\n const caption = isString(block.attributes.caption)\n ? block.attributes.caption\n : null;\n return (\n <figure key={key} data-block-id={dataBlockId}>\n {/* eslint-disable-next-line @next/next/no-img-element */}\n <img src={src} alt={alt} loading=\"lazy\" />\n {caption && <figcaption>{caption}</figcaption>}\n </figure>\n );\n }\n case \"core/list\": {\n const ordered = block.attributes.ordered === true;\n const items = (block.innerBlocks ?? []).map(\n (item, i): ReactNode =>\n item.rawHtml ? (\n <li\n key={i}\n dangerouslySetInnerHTML={htmlContent(item.rawHtml)}\n />\n ) : (\n <li key={i}>{renderInner(item.innerBlocks ?? [])}</li>\n ),\n );\n return ordered ? (\n <ol key={key} data-block-id={dataBlockId}>\n {items}\n </ol>\n ) : (\n <ul key={key} data-block-id={dataBlockId}>\n {items}\n </ul>\n );\n }\n case \"core/quote\": {\n const inner = renderInner(block.innerBlocks ?? []);\n const citation = isString(block.attributes.citation)\n ? block.attributes.citation\n : null;\n return (\n <blockquote key={key} data-block-id={dataBlockId}>\n {inner}\n {citation && <cite>{citation}</cite>}\n </blockquote>\n );\n }\n case \"core/code\": {\n const code = isString(block.attributes.content)\n ? block.attributes.content\n : block.rawHtml ?? \"\";\n const lang = isString(block.attributes.language)\n ? block.attributes.language\n : undefined;\n return (\n <pre\n key={key}\n data-block-id={dataBlockId}\n data-language={lang}\n >\n <code>{code}</code>\n </pre>\n );\n }\n case \"core/columns\": {\n return (\n <div\n key={key}\n className=\"rt-columns\"\n data-block-id={dataBlockId}\n >\n {renderInner(block.innerBlocks ?? [])}\n </div>\n );\n }\n case \"core/separator\":\n return <hr key={key} data-block-id={dataBlockId} />;\n case \"core/spacer\": {\n const heightRaw = block.attributes.height;\n const height = isString(heightRaw) ? heightRaw : \"32px\";\n return (\n <div\n key={key}\n className=\"rt-spacer\"\n data-block-id={dataBlockId}\n style={{ height }}\n />\n );\n }\n case \"core/group\": {\n return (\n <div\n key={key}\n className=\"rt-group\"\n data-block-id={dataBlockId}\n >\n {renderInner(block.innerBlocks ?? [])}\n </div>\n );\n }\n default: {\n // codex review §15 — 알려지지 않은 block 의 rawHtml 을 그대로\n // `dangerouslySetInnerHTML` 로 내보내면 API key 로 글을 작성하는 외부\n // 클라이언트가 임의 HTML/script 를 외부 사이트로 삽입할 수 있다 (XSS).\n // rawHtml 분기 자체를 제거 — 후속 PR 에서 cms-core 의 sanitize 모듈을\n // 도입한 뒤에만 raw 출력을 허용한다.\n const rawHtml = block.rawHtml;\n return (\n <div\n key={key}\n className=\"rt-unknown-block\"\n data-block-id={dataBlockId}\n data-block-name={block.name}\n data-rawhtml-suppressed={rawHtml ? \"true\" : undefined}\n >\n {renderInner(block.innerBlocks ?? [])}\n </div>\n );\n }\n }\n}\n\n/** Block tree 전체를 React element 배열로 직렬화. */\nexport function renderBlocks(blocks: readonly Block[]): ReactElement[] {\n return blocks.map((b, i) => renderBlock(b, i));\n}\n"],"mappings":"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","names":["jsx","jsxs","jsx","jsxs","jsx","jsxs","jsx","jsxs","Fragment","jsx","jsxs","formatPublishedDate"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@roottale/cms-renderer-next",
3
- "version": "0.2.1",
3
+ "version": "0.3.0",
4
4
  "type": "module",
5
5
  "description": "RootTale CMS public-render React/Next.js Server Components. SSR-only RSC components (RootTaleBlogList / RootTaleBlogPost / RootTaleLeadForm) for external customer sites. Companion of @roottale/cms-renderer-astro (ADR-0034 §1.5 amended).",
6
6
  "main": "./dist/index.js",
@@ -25,7 +25,7 @@
25
25
  "CHANGELOG.md"
26
26
  ],
27
27
  "dependencies": {
28
- "@roottale/cms-client": "^0.1.1",
28
+ "@roottale/cms-client": "^0.2.0",
29
29
  "@roottale/cms-core": "^0.2.1"
30
30
  },
31
31
  "peerDependencies": {