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.
Files changed (57) hide show
  1. package/esm2022/lib/components/atoms/fab/fab.component.mjs +5 -4
  2. package/esm2022/lib/components/molecules/comment-input/comment-input.component.mjs +13 -5
  3. package/esm2022/lib/components/molecules/comment-input/types.mjs +2 -0
  4. package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +23 -4
  5. package/esm2022/lib/components/molecules/date-input/types.mjs +2 -0
  6. package/esm2022/lib/components/molecules/feedback-form/feedback-form.component.mjs +352 -0
  7. package/esm2022/lib/components/molecules/feedback-form/types.mjs +2 -0
  8. package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +7 -7
  9. package/esm2022/lib/components/molecules/file-input/types.mjs +2 -0
  10. package/esm2022/lib/components/molecules/number-from-to/number-from-to.component.mjs +23 -9
  11. package/esm2022/lib/components/molecules/number-from-to/types.mjs +2 -0
  12. package/esm2022/lib/components/molecules/pin-input/pin-input.component.mjs +2 -2
  13. package/esm2022/lib/components/molecules/pin-input/types.mjs +2 -0
  14. package/esm2022/lib/components/organisms/form/form.component.mjs +108 -30
  15. package/esm2022/lib/services/content/content-types/blog.mjs +275 -0
  16. package/esm2022/lib/services/content/content-types/documentation.mjs +303 -0
  17. package/esm2022/lib/services/content/content-types/news.mjs +277 -0
  18. package/esm2022/lib/services/content/index.mjs +51 -0
  19. package/esm2022/lib/services/content/transformer.mjs +265 -0
  20. package/esm2022/lib/services/content/types.mjs +41 -0
  21. package/esm2022/lib/services/feedback/config.mjs +49 -0
  22. package/esm2022/lib/services/feedback/feedback.service.mjs +174 -0
  23. package/esm2022/lib/services/feedback/index.mjs +44 -0
  24. package/esm2022/lib/services/feedback/types.mjs +30 -0
  25. package/esm2022/lib/services/firebase/index.mjs +3 -1
  26. package/esm2022/lib/services/firebase/shared-config.mjs +132 -0
  27. package/esm2022/public-api.mjs +14 -1
  28. package/fesm2022/valtech-components.mjs +2225 -177
  29. package/fesm2022/valtech-components.mjs.map +1 -1
  30. package/lib/components/atoms/fab/fab.component.d.ts +2 -0
  31. package/lib/components/molecules/comment-input/comment-input.component.d.ts +4 -4
  32. package/lib/components/molecules/comment-input/types.d.ts +59 -0
  33. package/lib/components/molecules/date-input/date-input.component.d.ts +4 -3
  34. package/lib/components/molecules/date-input/types.d.ts +74 -0
  35. package/lib/components/molecules/feedback-form/feedback-form.component.d.ts +56 -0
  36. package/lib/components/molecules/feedback-form/types.d.ts +54 -0
  37. package/lib/components/molecules/file-input/file-input.component.d.ts +5 -3
  38. package/lib/components/molecules/file-input/types.d.ts +72 -0
  39. package/lib/components/molecules/number-from-to/number-from-to.component.d.ts +8 -2
  40. package/lib/components/molecules/number-from-to/types.d.ts +76 -0
  41. package/lib/components/molecules/pin-input/pin-input.component.d.ts +4 -3
  42. package/lib/components/molecules/pin-input/types.d.ts +63 -0
  43. package/lib/components/organisms/form/form.component.d.ts +16 -2
  44. package/lib/services/content/content-types/blog.d.ts +148 -0
  45. package/lib/services/content/content-types/documentation.d.ts +183 -0
  46. package/lib/services/content/content-types/news.d.ts +162 -0
  47. package/lib/services/content/index.d.ts +49 -0
  48. package/lib/services/content/transformer.d.ts +96 -0
  49. package/lib/services/content/types.d.ts +220 -0
  50. package/lib/services/feedback/config.d.ts +35 -0
  51. package/lib/services/feedback/feedback.service.d.ts +76 -0
  52. package/lib/services/feedback/index.d.ts +40 -0
  53. package/lib/services/feedback/types.d.ts +107 -0
  54. package/lib/services/firebase/index.d.ts +1 -0
  55. package/lib/services/firebase/shared-config.d.ts +120 -0
  56. package/package.json +1 -1
  57. 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"]}