valtech-components 2.0.434 → 2.0.435
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/esm2022/lib/components/atoms/fab/fab.component.mjs +5 -4
- package/esm2022/lib/components/molecules/comment-input/comment-input.component.mjs +13 -5
- package/esm2022/lib/components/molecules/comment-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +23 -4
- package/esm2022/lib/components/molecules/date-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/feedback-form/feedback-form.component.mjs +352 -0
- package/esm2022/lib/components/molecules/feedback-form/types.mjs +2 -0
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +7 -7
- package/esm2022/lib/components/molecules/file-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/number-from-to/number-from-to.component.mjs +23 -9
- package/esm2022/lib/components/molecules/number-from-to/types.mjs +2 -0
- package/esm2022/lib/components/molecules/pin-input/pin-input.component.mjs +2 -2
- package/esm2022/lib/components/molecules/pin-input/types.mjs +2 -0
- package/esm2022/lib/components/organisms/form/form.component.mjs +108 -30
- package/esm2022/lib/services/content/content-types/blog.mjs +275 -0
- package/esm2022/lib/services/content/content-types/documentation.mjs +303 -0
- package/esm2022/lib/services/content/content-types/news.mjs +277 -0
- package/esm2022/lib/services/content/index.mjs +51 -0
- package/esm2022/lib/services/content/transformer.mjs +265 -0
- package/esm2022/lib/services/content/types.mjs +41 -0
- package/esm2022/lib/services/feedback/config.mjs +49 -0
- package/esm2022/lib/services/feedback/feedback.service.mjs +174 -0
- package/esm2022/lib/services/feedback/index.mjs +44 -0
- package/esm2022/lib/services/feedback/types.mjs +30 -0
- package/esm2022/lib/services/firebase/index.mjs +3 -1
- package/esm2022/lib/services/firebase/shared-config.mjs +132 -0
- package/esm2022/public-api.mjs +14 -1
- package/fesm2022/valtech-components.mjs +2225 -177
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/fab/fab.component.d.ts +2 -0
- package/lib/components/molecules/comment-input/comment-input.component.d.ts +4 -4
- package/lib/components/molecules/comment-input/types.d.ts +59 -0
- package/lib/components/molecules/date-input/date-input.component.d.ts +4 -3
- package/lib/components/molecules/date-input/types.d.ts +74 -0
- package/lib/components/molecules/feedback-form/feedback-form.component.d.ts +56 -0
- package/lib/components/molecules/feedback-form/types.d.ts +54 -0
- package/lib/components/molecules/file-input/file-input.component.d.ts +5 -3
- package/lib/components/molecules/file-input/types.d.ts +72 -0
- package/lib/components/molecules/number-from-to/number-from-to.component.d.ts +8 -2
- package/lib/components/molecules/number-from-to/types.d.ts +76 -0
- package/lib/components/molecules/pin-input/pin-input.component.d.ts +4 -3
- package/lib/components/molecules/pin-input/types.d.ts +63 -0
- package/lib/components/organisms/form/form.component.d.ts +16 -2
- package/lib/services/content/content-types/blog.d.ts +148 -0
- package/lib/services/content/content-types/documentation.d.ts +183 -0
- package/lib/services/content/content-types/news.d.ts +162 -0
- package/lib/services/content/index.d.ts +49 -0
- package/lib/services/content/transformer.d.ts +96 -0
- package/lib/services/content/types.d.ts +220 -0
- package/lib/services/feedback/config.d.ts +35 -0
- package/lib/services/feedback/feedback.service.d.ts +76 -0
- package/lib/services/feedback/index.d.ts +40 -0
- package/lib/services/feedback/types.d.ts +107 -0
- package/lib/services/firebase/index.d.ts +1 -0
- package/lib/services/firebase/shared-config.d.ts +120 -0
- package/package.json +1 -1
- package/public-api.d.ts +9 -0
|
@@ -0,0 +1,277 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* News Article Content Type
|
|
3
|
+
*
|
|
4
|
+
* Represents a news article or announcement with headline,
|
|
5
|
+
* summary, featured image, and structured content blocks.
|
|
6
|
+
*/
|
|
7
|
+
import { ContentTransformer } from '../transformer';
|
|
8
|
+
/**
|
|
9
|
+
* Calculates estimated reading time based on word count
|
|
10
|
+
* @param content - Array of content blocks
|
|
11
|
+
* @returns Estimated minutes to read
|
|
12
|
+
*/
|
|
13
|
+
function calculateReadingTime(content) {
|
|
14
|
+
const wordsPerMinute = 200;
|
|
15
|
+
let wordCount = 0;
|
|
16
|
+
for (const block of content) {
|
|
17
|
+
if ('text' in block && typeof block.text === 'string') {
|
|
18
|
+
wordCount += block.text.split(/\s+/).length;
|
|
19
|
+
}
|
|
20
|
+
if (block.type === 'list') {
|
|
21
|
+
wordCount += block.items.join(' ').split(/\s+/).length;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
return Math.max(1, Math.ceil(wordCount / wordsPerMinute));
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* NewsBuilder provides a fluent API for creating news articles.
|
|
28
|
+
*
|
|
29
|
+
* @example
|
|
30
|
+
* ```typescript
|
|
31
|
+
* const news = new NewsBuilder()
|
|
32
|
+
* .headline('New Product Launch Announced')
|
|
33
|
+
* .summary('Company reveals groundbreaking new product...')
|
|
34
|
+
* .author('Jane Smith', '/avatars/jane.jpg', 'Technology Reporter')
|
|
35
|
+
* .publishedAt(new Date())
|
|
36
|
+
* .breaking()
|
|
37
|
+
* .featuredImage('/images/product-launch.jpg')
|
|
38
|
+
* .paragraph('In a surprise announcement today...')
|
|
39
|
+
* .quote('This is our most innovative product yet', 'CEO John Doe')
|
|
40
|
+
* .build();
|
|
41
|
+
*
|
|
42
|
+
* // Convert to ArticleMetadata
|
|
43
|
+
* const article = news.toArticle();
|
|
44
|
+
* ```
|
|
45
|
+
*/
|
|
46
|
+
export class NewsBuilder {
|
|
47
|
+
constructor() {
|
|
48
|
+
this.article = {
|
|
49
|
+
type: 'news',
|
|
50
|
+
content: [],
|
|
51
|
+
meta: {},
|
|
52
|
+
config: { showMeta: true },
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Sets the news headline
|
|
57
|
+
*/
|
|
58
|
+
headline(headline) {
|
|
59
|
+
this.article.headline = headline;
|
|
60
|
+
return this;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Sets the news summary
|
|
64
|
+
*/
|
|
65
|
+
summary(summary) {
|
|
66
|
+
this.article.summary = summary;
|
|
67
|
+
return this;
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Sets the featured image URL
|
|
71
|
+
*/
|
|
72
|
+
featuredImage(url) {
|
|
73
|
+
this.article.featuredImage = url;
|
|
74
|
+
return this;
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Marks the article as breaking news
|
|
78
|
+
*/
|
|
79
|
+
breaking(value = true) {
|
|
80
|
+
this.article.breaking = value;
|
|
81
|
+
return this;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Sets the news source/outlet
|
|
85
|
+
*/
|
|
86
|
+
source(source) {
|
|
87
|
+
this.article.source = source;
|
|
88
|
+
return this;
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Sets the article author
|
|
92
|
+
*/
|
|
93
|
+
author(name, avatar, role) {
|
|
94
|
+
this.article.meta = {
|
|
95
|
+
...this.article.meta,
|
|
96
|
+
author: { name, avatar, role },
|
|
97
|
+
};
|
|
98
|
+
return this;
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Sets the publication date
|
|
102
|
+
*/
|
|
103
|
+
publishedAt(date) {
|
|
104
|
+
this.article.meta = {
|
|
105
|
+
...this.article.meta,
|
|
106
|
+
publishedAt: date,
|
|
107
|
+
};
|
|
108
|
+
return this;
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Sets tags for the article
|
|
112
|
+
*/
|
|
113
|
+
tags(...tags) {
|
|
114
|
+
this.article.meta = {
|
|
115
|
+
...this.article.meta,
|
|
116
|
+
tags,
|
|
117
|
+
};
|
|
118
|
+
return this;
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Sets the category
|
|
122
|
+
*/
|
|
123
|
+
category(category) {
|
|
124
|
+
this.article.meta = {
|
|
125
|
+
...this.article.meta,
|
|
126
|
+
category,
|
|
127
|
+
};
|
|
128
|
+
return this;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Sets the slug
|
|
132
|
+
*/
|
|
133
|
+
slug(slug) {
|
|
134
|
+
this.article.meta = {
|
|
135
|
+
...this.article.meta,
|
|
136
|
+
slug,
|
|
137
|
+
};
|
|
138
|
+
return this;
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Sets the ID
|
|
142
|
+
*/
|
|
143
|
+
id(id) {
|
|
144
|
+
this.article.meta = {
|
|
145
|
+
...this.article.meta,
|
|
146
|
+
id,
|
|
147
|
+
};
|
|
148
|
+
return this;
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* Adds related articles
|
|
152
|
+
*/
|
|
153
|
+
relatedArticles(articles) {
|
|
154
|
+
this.article.relatedArticles = articles;
|
|
155
|
+
return this;
|
|
156
|
+
}
|
|
157
|
+
// Content block methods
|
|
158
|
+
/**
|
|
159
|
+
* Adds a heading block
|
|
160
|
+
*/
|
|
161
|
+
heading(text, level = 2) {
|
|
162
|
+
this.article.content.push({ type: 'heading', level, text });
|
|
163
|
+
return this;
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* Adds a paragraph block
|
|
167
|
+
*/
|
|
168
|
+
paragraph(text, emphasis) {
|
|
169
|
+
this.article.content.push({ type: 'paragraph', text, emphasis });
|
|
170
|
+
return this;
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Adds a quote block
|
|
174
|
+
*/
|
|
175
|
+
quote(text, author, source) {
|
|
176
|
+
this.article.content.push({ type: 'quote', text, author, source });
|
|
177
|
+
return this;
|
|
178
|
+
}
|
|
179
|
+
/**
|
|
180
|
+
* Adds an image block
|
|
181
|
+
*/
|
|
182
|
+
image(src, alt, caption) {
|
|
183
|
+
this.article.content.push({ type: 'image', src, alt, caption });
|
|
184
|
+
return this;
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Adds an unordered list
|
|
188
|
+
*/
|
|
189
|
+
list(items) {
|
|
190
|
+
this.article.content.push({ type: 'list', items });
|
|
191
|
+
return this;
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Adds an ordered/numbered list
|
|
195
|
+
*/
|
|
196
|
+
orderedList(items) {
|
|
197
|
+
this.article.content.push({ type: 'list', items, ordered: true });
|
|
198
|
+
return this;
|
|
199
|
+
}
|
|
200
|
+
/**
|
|
201
|
+
* Adds a callout/note block
|
|
202
|
+
*/
|
|
203
|
+
callout(text, variant = 'info', title) {
|
|
204
|
+
this.article.content.push({ type: 'callout', text, variant, title });
|
|
205
|
+
return this;
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* Adds a divider
|
|
209
|
+
*/
|
|
210
|
+
divider(style) {
|
|
211
|
+
this.article.content.push({ type: 'divider', style });
|
|
212
|
+
return this;
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Adds a button/CTA
|
|
216
|
+
*/
|
|
217
|
+
button(text, href, color) {
|
|
218
|
+
this.article.content.push({ type: 'button', text, href, color });
|
|
219
|
+
return this;
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Configures rendering options
|
|
223
|
+
*/
|
|
224
|
+
config(config) {
|
|
225
|
+
this.article.config = { ...this.article.config, ...config };
|
|
226
|
+
return this;
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* Builds the final NewsArticle object
|
|
230
|
+
*/
|
|
231
|
+
build() {
|
|
232
|
+
const content = this.article.content || [];
|
|
233
|
+
return {
|
|
234
|
+
type: 'news',
|
|
235
|
+
headline: this.article.headline || 'Untitled News',
|
|
236
|
+
summary: this.article.summary || '',
|
|
237
|
+
featuredImage: this.article.featuredImage,
|
|
238
|
+
breaking: this.article.breaking,
|
|
239
|
+
source: this.article.source,
|
|
240
|
+
relatedArticles: this.article.relatedArticles,
|
|
241
|
+
meta: this.article.meta,
|
|
242
|
+
content,
|
|
243
|
+
config: this.article.config,
|
|
244
|
+
};
|
|
245
|
+
}
|
|
246
|
+
/**
|
|
247
|
+
* Builds and converts to ArticleMetadata
|
|
248
|
+
*/
|
|
249
|
+
toArticle() {
|
|
250
|
+
return ContentTransformer.toArticle(this.build());
|
|
251
|
+
}
|
|
252
|
+
/**
|
|
253
|
+
* Gets the estimated reading time in minutes
|
|
254
|
+
*/
|
|
255
|
+
getReadingTime() {
|
|
256
|
+
return calculateReadingTime(this.article.content || []);
|
|
257
|
+
}
|
|
258
|
+
/**
|
|
259
|
+
* Resets the builder for reuse
|
|
260
|
+
*/
|
|
261
|
+
clear() {
|
|
262
|
+
this.article = {
|
|
263
|
+
type: 'news',
|
|
264
|
+
content: [],
|
|
265
|
+
meta: {},
|
|
266
|
+
config: { showMeta: true },
|
|
267
|
+
};
|
|
268
|
+
return this;
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
/**
|
|
272
|
+
* Creates a new NewsBuilder instance
|
|
273
|
+
*/
|
|
274
|
+
export function news() {
|
|
275
|
+
return new NewsBuilder();
|
|
276
|
+
}
|
|
277
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"news.js","sourceRoot":"","sources":["../../../../../../../src/lib/services/content/content-types/news.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * News Article Content Type\n *\n * Represents a news article or announcement with headline,\n * summary, featured image, and structured content blocks.\n */\n\nimport { ArticleMetadata } from '../../../components/organisms/article/types';\nimport { ContentDocument, ContentMeta, ContentBlock, ContentConfig } from '../types';\nimport { ContentTransformer } from '../transformer';\n\n/**\n * News article document interface\n */\nexport interface NewsArticle extends ContentDocument<'news'> {\n  /** News headline */\n  headline: string;\n  /** Short summary for listings */\n  summary: string;\n  /** Featured image URL */\n  featuredImage?: string;\n  /** Breaking news flag */\n  breaking?: boolean;\n  /** News source/outlet */\n  source?: string;\n  /** Related articles */\n  relatedArticles?: { title: string; slug: string }[];\n}\n\n/**\n * Calculates estimated reading time based on word count\n * @param content - Array of content blocks\n * @returns Estimated minutes to read\n */\nfunction calculateReadingTime(content: ContentBlock[]): number {\n  const wordsPerMinute = 200;\n  let wordCount = 0;\n\n  for (const block of content) {\n    if ('text' in block && typeof block.text === 'string') {\n      wordCount += block.text.split(/\\s+/).length;\n    }\n    if (block.type === 'list') {\n      wordCount += block.items.join(' ').split(/\\s+/).length;\n    }\n  }\n\n  return Math.max(1, Math.ceil(wordCount / wordsPerMinute));\n}\n\n/**\n * NewsBuilder provides a fluent API for creating news articles.\n *\n * @example\n * ```typescript\n * const news = new NewsBuilder()\n *   .headline('New Product Launch Announced')\n *   .summary('Company reveals groundbreaking new product...')\n *   .author('Jane Smith', '/avatars/jane.jpg', 'Technology Reporter')\n *   .publishedAt(new Date())\n *   .breaking()\n *   .featuredImage('/images/product-launch.jpg')\n *   .paragraph('In a surprise announcement today...')\n *   .quote('This is our most innovative product yet', 'CEO John Doe')\n *   .build();\n *\n * // Convert to ArticleMetadata\n * const article = news.toArticle();\n * ```\n */\nexport class NewsBuilder {\n  private article: Partial<NewsArticle> = {\n    type: 'news',\n    content: [],\n    meta: {},\n    config: { showMeta: true },\n  };\n\n  /**\n   * Sets the news headline\n   */\n  headline(headline: string): this {\n    this.article.headline = headline;\n    return this;\n  }\n\n  /**\n   * Sets the news summary\n   */\n  summary(summary: string): this {\n    this.article.summary = summary;\n    return this;\n  }\n\n  /**\n   * Sets the featured image URL\n   */\n  featuredImage(url: string): this {\n    this.article.featuredImage = url;\n    return this;\n  }\n\n  /**\n   * Marks the article as breaking news\n   */\n  breaking(value: boolean = true): this {\n    this.article.breaking = value;\n    return this;\n  }\n\n  /**\n   * Sets the news source/outlet\n   */\n  source(source: string): this {\n    this.article.source = source;\n    return this;\n  }\n\n  /**\n   * Sets the article author\n   */\n  author(name: string, avatar?: string, role?: string): this {\n    this.article.meta = {\n      ...this.article.meta,\n      author: { name, avatar, role },\n    };\n    return this;\n  }\n\n  /**\n   * Sets the publication date\n   */\n  publishedAt(date: Date | string): this {\n    this.article.meta = {\n      ...this.article.meta,\n      publishedAt: date,\n    };\n    return this;\n  }\n\n  /**\n   * Sets tags for the article\n   */\n  tags(...tags: string[]): this {\n    this.article.meta = {\n      ...this.article.meta,\n      tags,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the category\n   */\n  category(category: string): this {\n    this.article.meta = {\n      ...this.article.meta,\n      category,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the slug\n   */\n  slug(slug: string): this {\n    this.article.meta = {\n      ...this.article.meta,\n      slug,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the ID\n   */\n  id(id: string): this {\n    this.article.meta = {\n      ...this.article.meta,\n      id,\n    };\n    return this;\n  }\n\n  /**\n   * Adds related articles\n   */\n  relatedArticles(articles: { title: string; slug: string }[]): this {\n    this.article.relatedArticles = articles;\n    return this;\n  }\n\n  // Content block methods\n\n  /**\n   * Adds a heading block\n   */\n  heading(text: string, level: 1 | 2 | 3 = 2): this {\n    this.article.content!.push({ type: 'heading', level, text });\n    return this;\n  }\n\n  /**\n   * Adds a paragraph block\n   */\n  paragraph(text: string, emphasis?: boolean): this {\n    this.article.content!.push({ type: 'paragraph', text, emphasis });\n    return this;\n  }\n\n  /**\n   * Adds a quote block\n   */\n  quote(text: string, author?: string, source?: string): this {\n    this.article.content!.push({ type: 'quote', text, author, source });\n    return this;\n  }\n\n  /**\n   * Adds an image block\n   */\n  image(src: string, alt: string, caption?: string): this {\n    this.article.content!.push({ type: 'image', src, alt, caption });\n    return this;\n  }\n\n  /**\n   * Adds an unordered list\n   */\n  list(items: string[]): this {\n    this.article.content!.push({ type: 'list', items });\n    return this;\n  }\n\n  /**\n   * Adds an ordered/numbered list\n   */\n  orderedList(items: string[]): this {\n    this.article.content!.push({ type: 'list', items, ordered: true });\n    return this;\n  }\n\n  /**\n   * Adds a callout/note block\n   */\n  callout(text: string, variant: 'info' | 'warning' | 'success' | 'error' = 'info', title?: string): this {\n    this.article.content!.push({ type: 'callout', text, variant, title });\n    return this;\n  }\n\n  /**\n   * Adds a divider\n   */\n  divider(style?: 'line' | 'dots' | 'space'): this {\n    this.article.content!.push({ type: 'divider', style });\n    return this;\n  }\n\n  /**\n   * Adds a button/CTA\n   */\n  button(text: string, href?: string, color?: 'primary' | 'secondary' | 'success'): this {\n    this.article.content!.push({ type: 'button', text, href, color });\n    return this;\n  }\n\n  /**\n   * Configures rendering options\n   */\n  config(config: Partial<ContentConfig>): this {\n    this.article.config = { ...this.article.config, ...config };\n    return this;\n  }\n\n  /**\n   * Builds the final NewsArticle object\n   */\n  build(): NewsArticle {\n    const content = this.article.content || [];\n\n    return {\n      type: 'news',\n      headline: this.article.headline || 'Untitled News',\n      summary: this.article.summary || '',\n      featuredImage: this.article.featuredImage,\n      breaking: this.article.breaking,\n      source: this.article.source,\n      relatedArticles: this.article.relatedArticles,\n      meta: this.article.meta as ContentMeta,\n      content,\n      config: this.article.config,\n    };\n  }\n\n  /**\n   * Builds and converts to ArticleMetadata\n   */\n  toArticle(): ArticleMetadata {\n    return ContentTransformer.toArticle(this.build());\n  }\n\n  /**\n   * Gets the estimated reading time in minutes\n   */\n  getReadingTime(): number {\n    return calculateReadingTime(this.article.content || []);\n  }\n\n  /**\n   * Resets the builder for reuse\n   */\n  clear(): this {\n    this.article = {\n      type: 'news',\n      content: [],\n      meta: {},\n      config: { showMeta: true },\n    };\n    return this;\n  }\n}\n\n/**\n * Creates a new NewsBuilder instance\n */\nexport function news(): NewsBuilder {\n  return new NewsBuilder();\n}\n"]}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Content Types Module
|
|
3
|
+
*
|
|
4
|
+
* Provides a flexible content abstraction layer that transforms
|
|
5
|
+
* structured content documents into ArticleMetadata for rendering
|
|
6
|
+
* with the val-article component.
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```typescript
|
|
10
|
+
* // Using BlogPostBuilder
|
|
11
|
+
* import { blogPost } from 'valtech-components';
|
|
12
|
+
*
|
|
13
|
+
* const post = blogPost()
|
|
14
|
+
* .title('My First Post')
|
|
15
|
+
* .author('John Doe')
|
|
16
|
+
* .heading('Introduction')
|
|
17
|
+
* .paragraph('Welcome to my blog...')
|
|
18
|
+
* .toArticle();
|
|
19
|
+
*
|
|
20
|
+
* // Using DocsBuilder
|
|
21
|
+
* import { docs } from 'valtech-components';
|
|
22
|
+
*
|
|
23
|
+
* const page = docs()
|
|
24
|
+
* .title('Installation')
|
|
25
|
+
* .section('Getting Started')
|
|
26
|
+
* .code('npm install valtech-components', 'bash')
|
|
27
|
+
* .toArticle();
|
|
28
|
+
*
|
|
29
|
+
* // Using NewsBuilder
|
|
30
|
+
* import { news } from 'valtech-components';
|
|
31
|
+
*
|
|
32
|
+
* const article = news()
|
|
33
|
+
* .headline('Breaking News')
|
|
34
|
+
* .summary('Important announcement...')
|
|
35
|
+
* .breaking()
|
|
36
|
+
* .toArticle();
|
|
37
|
+
*
|
|
38
|
+
* // From JSON/API response
|
|
39
|
+
* import { ContentTransformer, BlogPost } from 'valtech-components';
|
|
40
|
+
*
|
|
41
|
+
* const json: BlogPost = await fetch('/api/posts/1').then(r => r.json());
|
|
42
|
+
* const article = ContentTransformer.toArticle(json);
|
|
43
|
+
* ```
|
|
44
|
+
*/
|
|
45
|
+
// Transformer
|
|
46
|
+
export { ContentTransformer, toArticle } from './transformer';
|
|
47
|
+
// Content types
|
|
48
|
+
export { BlogPostBuilder, blogPost } from './content-types/blog';
|
|
49
|
+
export { DocsBuilder, docs } from './content-types/documentation';
|
|
50
|
+
export { NewsBuilder, news } from './content-types/news';
|
|
51
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2NvbnRlbnQvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0EyQ0c7QUFxQkgsY0FBYztBQUNkLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFOUQsZ0JBQWdCO0FBQ2hCLE9BQU8sRUFBWSxlQUFlLEVBQUUsUUFBUSxFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDM0UsT0FBTyxFQUE2QixXQUFXLEVBQUUsSUFBSSxFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDN0YsT0FBTyxFQUFlLFdBQVcsRUFBRSxJQUFJLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29udGVudCBUeXBlcyBNb2R1bGVcbiAqXG4gKiBQcm92aWRlcyBhIGZsZXhpYmxlIGNvbnRlbnQgYWJzdHJhY3Rpb24gbGF5ZXIgdGhhdCB0cmFuc2Zvcm1zXG4gKiBzdHJ1Y3R1cmVkIGNvbnRlbnQgZG9jdW1lbnRzIGludG8gQXJ0aWNsZU1ldGFkYXRhIGZvciByZW5kZXJpbmdcbiAqIHdpdGggdGhlIHZhbC1hcnRpY2xlIGNvbXBvbmVudC5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gVXNpbmcgQmxvZ1Bvc3RCdWlsZGVyXG4gKiBpbXBvcnQgeyBibG9nUG9zdCB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKlxuICogY29uc3QgcG9zdCA9IGJsb2dQb3N0KClcbiAqICAgLnRpdGxlKCdNeSBGaXJzdCBQb3N0JylcbiAqICAgLmF1dGhvcignSm9obiBEb2UnKVxuICogICAuaGVhZGluZygnSW50cm9kdWN0aW9uJylcbiAqICAgLnBhcmFncmFwaCgnV2VsY29tZSB0byBteSBibG9nLi4uJylcbiAqICAgLnRvQXJ0aWNsZSgpO1xuICpcbiAqIC8vIFVzaW5nIERvY3NCdWlsZGVyXG4gKiBpbXBvcnQgeyBkb2NzIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBjb25zdCBwYWdlID0gZG9jcygpXG4gKiAgIC50aXRsZSgnSW5zdGFsbGF0aW9uJylcbiAqICAgLnNlY3Rpb24oJ0dldHRpbmcgU3RhcnRlZCcpXG4gKiAgIC5jb2RlKCducG0gaW5zdGFsbCB2YWx0ZWNoLWNvbXBvbmVudHMnLCAnYmFzaCcpXG4gKiAgIC50b0FydGljbGUoKTtcbiAqXG4gKiAvLyBVc2luZyBOZXdzQnVpbGRlclxuICogaW1wb3J0IHsgbmV3cyB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKlxuICogY29uc3QgYXJ0aWNsZSA9IG5ld3MoKVxuICogICAuaGVhZGxpbmUoJ0JyZWFraW5nIE5ld3MnKVxuICogICAuc3VtbWFyeSgnSW1wb3J0YW50IGFubm91bmNlbWVudC4uLicpXG4gKiAgIC5icmVha2luZygpXG4gKiAgIC50b0FydGljbGUoKTtcbiAqXG4gKiAvLyBGcm9tIEpTT04vQVBJIHJlc3BvbnNlXG4gKiBpbXBvcnQgeyBDb250ZW50VHJhbnNmb3JtZXIsIEJsb2dQb3N0IH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBjb25zdCBqc29uOiBCbG9nUG9zdCA9IGF3YWl0IGZldGNoKCcvYXBpL3Bvc3RzLzEnKS50aGVuKHIgPT4gci5qc29uKCkpO1xuICogY29uc3QgYXJ0aWNsZSA9IENvbnRlbnRUcmFuc2Zvcm1lci50b0FydGljbGUoanNvbik7XG4gKiBgYGBcbiAqL1xuXG4vLyBCYXNlIHR5cGVzXG5leHBvcnQge1xuICBDb250ZW50QXV0aG9yLFxuICBDb250ZW50TWV0YSxcbiAgQ29udGVudENvbmZpZyxcbiAgQ29udGVudERvY3VtZW50LFxuICBDb250ZW50QmxvY2ssXG4gIEhlYWRpbmdCbG9jayxcbiAgUGFyYWdyYXBoQmxvY2ssXG4gIFF1b3RlQmxvY2ssXG4gIENvZGVCbG9jayxcbiAgTGlzdEJsb2NrLFxuICBJbWFnZUJsb2NrLFxuICBDYWxsb3V0QmxvY2ssXG4gIERpdmlkZXJCbG9jayxcbiAgQnV0dG9uQmxvY2ssXG4gIENvbW1hbmRCbG9jayxcbn0gZnJvbSAnLi90eXBlcyc7XG5cbi8vIFRyYW5zZm9ybWVyXG5leHBvcnQgeyBDb250ZW50VHJhbnNmb3JtZXIsIHRvQXJ0aWNsZSB9IGZyb20gJy4vdHJhbnNmb3JtZXInO1xuXG4vLyBDb250ZW50IHR5cGVzXG5leHBvcnQgeyBCbG9nUG9zdCwgQmxvZ1Bvc3RCdWlsZGVyLCBibG9nUG9zdCB9IGZyb20gJy4vY29udGVudC10eXBlcy9ibG9nJztcbmV4cG9ydCB7IERvY3VtZW50YXRpb24sIERvY05hdkxpbmssIERvY3NCdWlsZGVyLCBkb2NzIH0gZnJvbSAnLi9jb250ZW50LXR5cGVzL2RvY3VtZW50YXRpb24nO1xuZXhwb3J0IHsgTmV3c0FydGljbGUsIE5ld3NCdWlsZGVyLCBuZXdzIH0gZnJvbSAnLi9jb250ZW50LXR5cGVzL25ld3MnO1xuIl19
|
|
@@ -0,0 +1,265 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Content Transformer
|
|
3
|
+
*
|
|
4
|
+
* Transforms ContentDocument instances into ArticleMetadata
|
|
5
|
+
* for rendering with the val-article component.
|
|
6
|
+
*/
|
|
7
|
+
import { ArticleBuilder } from '../../components/organisms/article/types';
|
|
8
|
+
import { ComponentStates } from '../../components/types';
|
|
9
|
+
/**
|
|
10
|
+
* Maps callout variants to Ionic colors
|
|
11
|
+
*/
|
|
12
|
+
const CALLOUT_COLOR_MAP = {
|
|
13
|
+
info: 'primary',
|
|
14
|
+
warning: 'warning',
|
|
15
|
+
success: 'success',
|
|
16
|
+
error: 'danger',
|
|
17
|
+
};
|
|
18
|
+
/**
|
|
19
|
+
* Maps callout variants to prefix text
|
|
20
|
+
*/
|
|
21
|
+
const CALLOUT_PREFIX_MAP = {
|
|
22
|
+
info: 'Info:',
|
|
23
|
+
warning: 'Aviso:',
|
|
24
|
+
success: 'Listo:',
|
|
25
|
+
error: 'Error:',
|
|
26
|
+
};
|
|
27
|
+
/**
|
|
28
|
+
* ContentTransformer converts ContentDocument objects into ArticleMetadata
|
|
29
|
+
* that can be rendered by the val-article component.
|
|
30
|
+
*
|
|
31
|
+
* @example
|
|
32
|
+
* ```typescript
|
|
33
|
+
* const doc: BlogPost = { ... };
|
|
34
|
+
* const article = ContentTransformer.toArticle(doc);
|
|
35
|
+
* // Use article with <val-article [props]="article">
|
|
36
|
+
* ```
|
|
37
|
+
*/
|
|
38
|
+
export class ContentTransformer {
|
|
39
|
+
/**
|
|
40
|
+
* Transforms a ContentDocument into ArticleMetadata
|
|
41
|
+
*
|
|
42
|
+
* @param doc - The content document to transform
|
|
43
|
+
* @returns ArticleMetadata ready for val-article component
|
|
44
|
+
*/
|
|
45
|
+
static toArticle(doc) {
|
|
46
|
+
const builder = new ArticleBuilder();
|
|
47
|
+
// Add header based on document type and meta
|
|
48
|
+
this.addHeader(builder, doc);
|
|
49
|
+
// Transform each content block
|
|
50
|
+
for (const block of doc.content) {
|
|
51
|
+
this.addBlock(builder, block);
|
|
52
|
+
}
|
|
53
|
+
// Add footer based on document type
|
|
54
|
+
this.addFooter(builder, doc);
|
|
55
|
+
return builder.build({
|
|
56
|
+
theme: doc.config?.theme ?? 'auto',
|
|
57
|
+
maxWidth: doc.config?.maxWidth ?? '800px',
|
|
58
|
+
centered: doc.config?.centered ?? true,
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Adds header elements based on document type and metadata
|
|
63
|
+
*/
|
|
64
|
+
static addHeader(builder, doc) {
|
|
65
|
+
// Add author/date info if showMeta is enabled
|
|
66
|
+
if (doc.config?.showMeta && doc.meta.author) {
|
|
67
|
+
this.addAuthorBlock(builder, doc.meta.author, doc.meta.publishedAt);
|
|
68
|
+
builder.separator('space', { spacing: { top: 'small', bottom: 'medium' } });
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Adds footer elements based on document type
|
|
73
|
+
*/
|
|
74
|
+
static addFooter(builder, doc) {
|
|
75
|
+
// Add tags if present
|
|
76
|
+
if (doc.meta.tags && doc.meta.tags.length > 0) {
|
|
77
|
+
builder.separator('line', { spacing: { top: 'large', bottom: 'medium' } });
|
|
78
|
+
builder.paragraph({
|
|
79
|
+
content: `Tags: ${doc.meta.tags.join(', ')}`,
|
|
80
|
+
size: 'small',
|
|
81
|
+
color: 'medium',
|
|
82
|
+
bold: false,
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Adds author information block
|
|
88
|
+
*/
|
|
89
|
+
static addAuthorBlock(builder, author, publishedAt) {
|
|
90
|
+
let authorText = author.name;
|
|
91
|
+
if (author.role) {
|
|
92
|
+
authorText += ` - ${author.role}`;
|
|
93
|
+
}
|
|
94
|
+
if (publishedAt) {
|
|
95
|
+
const date = typeof publishedAt === 'string' ? new Date(publishedAt) : publishedAt;
|
|
96
|
+
authorText += ` | ${date.toLocaleDateString()}`;
|
|
97
|
+
}
|
|
98
|
+
builder.paragraph({
|
|
99
|
+
content: authorText,
|
|
100
|
+
size: 'small',
|
|
101
|
+
color: 'medium',
|
|
102
|
+
bold: false,
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Transforms a single content block and adds it to the builder
|
|
107
|
+
*/
|
|
108
|
+
static addBlock(builder, block) {
|
|
109
|
+
switch (block.type) {
|
|
110
|
+
case 'heading':
|
|
111
|
+
this.addHeading(builder, block);
|
|
112
|
+
break;
|
|
113
|
+
case 'paragraph':
|
|
114
|
+
this.addParagraph(builder, block);
|
|
115
|
+
break;
|
|
116
|
+
case 'quote':
|
|
117
|
+
this.addQuote(builder, block);
|
|
118
|
+
break;
|
|
119
|
+
case 'code':
|
|
120
|
+
this.addCode(builder, block);
|
|
121
|
+
break;
|
|
122
|
+
case 'list':
|
|
123
|
+
this.addList(builder, block);
|
|
124
|
+
break;
|
|
125
|
+
case 'image':
|
|
126
|
+
this.addImage(builder, block);
|
|
127
|
+
break;
|
|
128
|
+
case 'callout':
|
|
129
|
+
this.addCallout(builder, block);
|
|
130
|
+
break;
|
|
131
|
+
case 'divider':
|
|
132
|
+
this.addDivider(builder, block);
|
|
133
|
+
break;
|
|
134
|
+
case 'button':
|
|
135
|
+
this.addButton(builder, block);
|
|
136
|
+
break;
|
|
137
|
+
case 'command':
|
|
138
|
+
this.addCommand(builder, block);
|
|
139
|
+
break;
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Adds a heading block
|
|
144
|
+
*/
|
|
145
|
+
static addHeading(builder, block) {
|
|
146
|
+
if (block.level === 1) {
|
|
147
|
+
builder.title({
|
|
148
|
+
content: block.text,
|
|
149
|
+
size: 'large',
|
|
150
|
+
color: 'dark',
|
|
151
|
+
bold: true,
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
else {
|
|
155
|
+
builder.subtitle({
|
|
156
|
+
content: block.text,
|
|
157
|
+
size: block.level === 2 ? 'medium' : 'small',
|
|
158
|
+
color: 'dark',
|
|
159
|
+
bold: true,
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Adds a paragraph block
|
|
165
|
+
*/
|
|
166
|
+
static addParagraph(builder, block) {
|
|
167
|
+
builder.paragraph({
|
|
168
|
+
content: block.text,
|
|
169
|
+
size: 'medium',
|
|
170
|
+
color: 'dark',
|
|
171
|
+
bold: block.emphasis ?? false,
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
/**
|
|
175
|
+
* Adds a quote block
|
|
176
|
+
*/
|
|
177
|
+
static addQuote(builder, block) {
|
|
178
|
+
builder.quote({
|
|
179
|
+
content: block.text,
|
|
180
|
+
size: 'medium',
|
|
181
|
+
color: 'dark',
|
|
182
|
+
bold: false,
|
|
183
|
+
author: block.author,
|
|
184
|
+
source: block.source,
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Adds a code block
|
|
189
|
+
*/
|
|
190
|
+
static addCode(builder, block) {
|
|
191
|
+
builder.code(block.code, block.language);
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Adds a list block
|
|
195
|
+
*/
|
|
196
|
+
static addList(builder, block) {
|
|
197
|
+
const items = block.items.map((text) => ({ text }));
|
|
198
|
+
const listType = block.checklist ? 'checklist' : block.ordered ? 'ordered' : 'unordered';
|
|
199
|
+
builder.list(items, listType);
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Adds an image block
|
|
203
|
+
*/
|
|
204
|
+
static addImage(builder, block) {
|
|
205
|
+
builder.image(block.src, block.alt, block.caption, {
|
|
206
|
+
// Note: alignment is handled differently in ArticleBuilder
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* Adds a callout/note block
|
|
211
|
+
*/
|
|
212
|
+
static addCallout(builder, block) {
|
|
213
|
+
const color = CALLOUT_COLOR_MAP[block.variant] ?? 'primary';
|
|
214
|
+
const prefix = block.title ?? CALLOUT_PREFIX_MAP[block.variant] ?? 'Nota:';
|
|
215
|
+
builder.note(block.text, prefix, color);
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Adds a divider/separator block
|
|
219
|
+
*/
|
|
220
|
+
static addDivider(builder, block) {
|
|
221
|
+
builder.separator(block.style ?? 'line');
|
|
222
|
+
}
|
|
223
|
+
/**
|
|
224
|
+
* Adds a button block
|
|
225
|
+
*/
|
|
226
|
+
static addButton(builder, block) {
|
|
227
|
+
const colorMap = {
|
|
228
|
+
primary: 'primary',
|
|
229
|
+
secondary: 'secondary',
|
|
230
|
+
success: 'success',
|
|
231
|
+
warning: 'warning',
|
|
232
|
+
danger: 'danger',
|
|
233
|
+
};
|
|
234
|
+
builder.button({
|
|
235
|
+
text: block.text,
|
|
236
|
+
color: colorMap[block.color ?? 'primary'] ?? 'primary',
|
|
237
|
+
fill: 'solid',
|
|
238
|
+
type: 'button',
|
|
239
|
+
state: ComponentStates.ENABLED,
|
|
240
|
+
href: block.href,
|
|
241
|
+
token: block.action,
|
|
242
|
+
}, block.alignment ?? 'center');
|
|
243
|
+
}
|
|
244
|
+
/**
|
|
245
|
+
* Adds a command/terminal block
|
|
246
|
+
*/
|
|
247
|
+
static addCommand(builder, block) {
|
|
248
|
+
builder.command(block.command);
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
/**
|
|
252
|
+
* Convenience function to transform a ContentDocument to ArticleMetadata
|
|
253
|
+
*
|
|
254
|
+
* @param doc - The content document to transform
|
|
255
|
+
* @returns ArticleMetadata ready for val-article component
|
|
256
|
+
*
|
|
257
|
+
* @example
|
|
258
|
+
* ```typescript
|
|
259
|
+
* const article = toArticle(myBlogPost);
|
|
260
|
+
* ```
|
|
261
|
+
*/
|
|
262
|
+
export function toArticle(doc) {
|
|
263
|
+
return ContentTransformer.toArticle(doc);
|
|
264
|
+
}
|
|
265
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transformer.js","sourceRoot":"","sources":["../../../../../../src/lib/services/content/transformer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Content Transformer\n *\n * Transforms ContentDocument instances into ArticleMetadata\n * for rendering with the val-article component.\n */\n\nimport { Color } from '@ionic/core';\nimport { ArticleBuilder, ArticleMetadata } from '../../components/organisms/article/types';\nimport { ComponentStates } from '../../components/types';\nimport {\n  ContentDocument,\n  ContentBlock,\n  ContentMeta,\n  ContentAuthor,\n  HeadingBlock,\n  ParagraphBlock,\n  QuoteBlock,\n  CodeBlock,\n  ListBlock,\n  ImageBlock,\n  CalloutBlock,\n  DividerBlock,\n  ButtonBlock,\n  CommandBlock,\n} from './types';\n\n/**\n * Maps callout variants to Ionic colors\n */\nconst CALLOUT_COLOR_MAP: Record<string, Color> = {\n  info: 'primary',\n  warning: 'warning',\n  success: 'success',\n  error: 'danger',\n};\n\n/**\n * Maps callout variants to prefix text\n */\nconst CALLOUT_PREFIX_MAP: Record<string, string> = {\n  info: 'Info:',\n  warning: 'Aviso:',\n  success: 'Listo:',\n  error: 'Error:',\n};\n\n/**\n * ContentTransformer converts ContentDocument objects into ArticleMetadata\n * that can be rendered by the val-article component.\n *\n * @example\n * ```typescript\n * const doc: BlogPost = { ... };\n * const article = ContentTransformer.toArticle(doc);\n * // Use article with <val-article [props]=\"article\">\n * ```\n */\nexport class ContentTransformer {\n  /**\n   * Transforms a ContentDocument into ArticleMetadata\n   *\n   * @param doc - The content document to transform\n   * @returns ArticleMetadata ready for val-article component\n   */\n  static toArticle(doc: ContentDocument): ArticleMetadata {\n    const builder = new ArticleBuilder();\n\n    // Add header based on document type and meta\n    this.addHeader(builder, doc);\n\n    // Transform each content block\n    for (const block of doc.content) {\n      this.addBlock(builder, block);\n    }\n\n    // Add footer based on document type\n    this.addFooter(builder, doc);\n\n    return builder.build({\n      theme: doc.config?.theme ?? 'auto',\n      maxWidth: doc.config?.maxWidth ?? '800px',\n      centered: doc.config?.centered ?? true,\n    });\n  }\n\n  /**\n   * Adds header elements based on document type and metadata\n   */\n  private static addHeader(builder: ArticleBuilder, doc: ContentDocument): void {\n    // Add author/date info if showMeta is enabled\n    if (doc.config?.showMeta && doc.meta.author) {\n      this.addAuthorBlock(builder, doc.meta.author, doc.meta.publishedAt);\n      builder.separator('space', { spacing: { top: 'small', bottom: 'medium' } });\n    }\n  }\n\n  /**\n   * Adds footer elements based on document type\n   */\n  private static addFooter(builder: ArticleBuilder, doc: ContentDocument): void {\n    // Add tags if present\n    if (doc.meta.tags && doc.meta.tags.length > 0) {\n      builder.separator('line', { spacing: { top: 'large', bottom: 'medium' } });\n      builder.paragraph({\n        content: `Tags: ${doc.meta.tags.join(', ')}`,\n        size: 'small',\n        color: 'medium',\n        bold: false,\n      });\n    }\n  }\n\n  /**\n   * Adds author information block\n   */\n  private static addAuthorBlock(\n    builder: ArticleBuilder,\n    author: ContentAuthor,\n    publishedAt?: Date | string\n  ): void {\n    let authorText = author.name;\n    if (author.role) {\n      authorText += ` - ${author.role}`;\n    }\n    if (publishedAt) {\n      const date = typeof publishedAt === 'string' ? new Date(publishedAt) : publishedAt;\n      authorText += ` | ${date.toLocaleDateString()}`;\n    }\n\n    builder.paragraph({\n      content: authorText,\n      size: 'small',\n      color: 'medium',\n      bold: false,\n    });\n  }\n\n  /**\n   * Transforms a single content block and adds it to the builder\n   */\n  private static addBlock(builder: ArticleBuilder, block: ContentBlock): void {\n    switch (block.type) {\n      case 'heading':\n        this.addHeading(builder, block);\n        break;\n      case 'paragraph':\n        this.addParagraph(builder, block);\n        break;\n      case 'quote':\n        this.addQuote(builder, block);\n        break;\n      case 'code':\n        this.addCode(builder, block);\n        break;\n      case 'list':\n        this.addList(builder, block);\n        break;\n      case 'image':\n        this.addImage(builder, block);\n        break;\n      case 'callout':\n        this.addCallout(builder, block);\n        break;\n      case 'divider':\n        this.addDivider(builder, block);\n        break;\n      case 'button':\n        this.addButton(builder, block);\n        break;\n      case 'command':\n        this.addCommand(builder, block);\n        break;\n    }\n  }\n\n  /**\n   * Adds a heading block\n   */\n  private static addHeading(builder: ArticleBuilder, block: HeadingBlock): void {\n    if (block.level === 1) {\n      builder.title({\n        content: block.text,\n        size: 'large',\n        color: 'dark',\n        bold: true,\n      });\n    } else {\n      builder.subtitle({\n        content: block.text,\n        size: block.level === 2 ? 'medium' : 'small',\n        color: 'dark',\n        bold: true,\n      });\n    }\n  }\n\n  /**\n   * Adds a paragraph block\n   */\n  private static addParagraph(builder: ArticleBuilder, block: ParagraphBlock): void {\n    builder.paragraph({\n      content: block.text,\n      size: 'medium',\n      color: 'dark',\n      bold: block.emphasis ?? false,\n    });\n  }\n\n  /**\n   * Adds a quote block\n   */\n  private static addQuote(builder: ArticleBuilder, block: QuoteBlock): void {\n    builder.quote({\n      content: block.text,\n      size: 'medium',\n      color: 'dark',\n      bold: false,\n      author: block.author,\n      source: block.source,\n    });\n  }\n\n  /**\n   * Adds a code block\n   */\n  private static addCode(builder: ArticleBuilder, block: CodeBlock): void {\n    builder.code(block.code, block.language);\n  }\n\n  /**\n   * Adds a list block\n   */\n  private static addList(builder: ArticleBuilder, block: ListBlock): void {\n    const items = block.items.map((text) => ({ text }));\n    const listType = block.checklist ? 'checklist' : block.ordered ? 'ordered' : 'unordered';\n    builder.list(items, listType);\n  }\n\n  /**\n   * Adds an image block\n   */\n  private static addImage(builder: ArticleBuilder, block: ImageBlock): void {\n    builder.image(block.src, block.alt, block.caption, {\n      // Note: alignment is handled differently in ArticleBuilder\n    });\n  }\n\n  /**\n   * Adds a callout/note block\n   */\n  private static addCallout(builder: ArticleBuilder, block: CalloutBlock): void {\n    const color = CALLOUT_COLOR_MAP[block.variant] ?? 'primary';\n    const prefix = block.title ?? CALLOUT_PREFIX_MAP[block.variant] ?? 'Nota:';\n    builder.note(block.text, prefix, color);\n  }\n\n  /**\n   * Adds a divider/separator block\n   */\n  private static addDivider(builder: ArticleBuilder, block: DividerBlock): void {\n    builder.separator(block.style ?? 'line');\n  }\n\n  /**\n   * Adds a button block\n   */\n  private static addButton(builder: ArticleBuilder, block: ButtonBlock): void {\n    const colorMap: Record<string, Color> = {\n      primary: 'primary',\n      secondary: 'secondary',\n      success: 'success',\n      warning: 'warning',\n      danger: 'danger',\n    };\n\n    builder.button(\n      {\n        text: block.text,\n        color: colorMap[block.color ?? 'primary'] ?? 'primary',\n        fill: 'solid',\n        type: 'button',\n        state: ComponentStates.ENABLED,\n        href: block.href,\n        token: block.action,\n      },\n      block.alignment ?? 'center'\n    );\n  }\n\n  /**\n   * Adds a command/terminal block\n   */\n  private static addCommand(builder: ArticleBuilder, block: CommandBlock): void {\n    builder.command(block.command);\n  }\n}\n\n/**\n * Convenience function to transform a ContentDocument to ArticleMetadata\n *\n * @param doc - The content document to transform\n * @returns ArticleMetadata ready for val-article component\n *\n * @example\n * ```typescript\n * const article = toArticle(myBlogPost);\n * ```\n */\nexport function toArticle(doc: ContentDocument): ArticleMetadata {\n  return ContentTransformer.toArticle(doc);\n}\n"]}
|