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,275 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Blog Post Content Type
|
|
3
|
+
*
|
|
4
|
+
* Represents a blog post with title, excerpt, cover image,
|
|
5
|
+
* 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
|
+
if (block.type === 'code') {
|
|
24
|
+
wordCount += block.code.split(/\s+/).length * 0.5; // Code reads faster
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
return Math.max(1, Math.ceil(wordCount / wordsPerMinute));
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* BlogPostBuilder provides a fluent API for creating blog posts.
|
|
31
|
+
*
|
|
32
|
+
* @example
|
|
33
|
+
* ```typescript
|
|
34
|
+
* const post = new BlogPostBuilder()
|
|
35
|
+
* .title('My First Post')
|
|
36
|
+
* .excerpt('A brief introduction...')
|
|
37
|
+
* .author('John Doe', '/avatars/john.jpg')
|
|
38
|
+
* .coverImage('/images/post-cover.jpg')
|
|
39
|
+
* .heading('Introduction')
|
|
40
|
+
* .paragraph('Welcome to my blog...')
|
|
41
|
+
* .build();
|
|
42
|
+
*
|
|
43
|
+
* // Convert to ArticleMetadata
|
|
44
|
+
* const article = post.toArticle();
|
|
45
|
+
* ```
|
|
46
|
+
*/
|
|
47
|
+
export class BlogPostBuilder {
|
|
48
|
+
constructor() {
|
|
49
|
+
this.post = {
|
|
50
|
+
type: 'blog',
|
|
51
|
+
content: [],
|
|
52
|
+
meta: {},
|
|
53
|
+
config: { showMeta: true },
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Sets the blog post title
|
|
58
|
+
*/
|
|
59
|
+
title(title) {
|
|
60
|
+
this.post.title = title;
|
|
61
|
+
return this;
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Sets the blog post excerpt
|
|
65
|
+
*/
|
|
66
|
+
excerpt(excerpt) {
|
|
67
|
+
this.post.excerpt = excerpt;
|
|
68
|
+
return this;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Sets the cover image URL
|
|
72
|
+
*/
|
|
73
|
+
coverImage(url) {
|
|
74
|
+
this.post.coverImage = url;
|
|
75
|
+
return this;
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Marks the post as featured
|
|
79
|
+
*/
|
|
80
|
+
featured(value = true) {
|
|
81
|
+
this.post.featured = value;
|
|
82
|
+
return this;
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Sets the post author
|
|
86
|
+
*/
|
|
87
|
+
author(name, avatar, role) {
|
|
88
|
+
this.post.meta = {
|
|
89
|
+
...this.post.meta,
|
|
90
|
+
author: { name, avatar, role },
|
|
91
|
+
};
|
|
92
|
+
return this;
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Sets the publication date
|
|
96
|
+
*/
|
|
97
|
+
publishedAt(date) {
|
|
98
|
+
this.post.meta = {
|
|
99
|
+
...this.post.meta,
|
|
100
|
+
publishedAt: date,
|
|
101
|
+
};
|
|
102
|
+
return this;
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Sets tags for the post
|
|
106
|
+
*/
|
|
107
|
+
tags(...tags) {
|
|
108
|
+
this.post.meta = {
|
|
109
|
+
...this.post.meta,
|
|
110
|
+
tags,
|
|
111
|
+
};
|
|
112
|
+
return this;
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Sets the category
|
|
116
|
+
*/
|
|
117
|
+
category(category) {
|
|
118
|
+
this.post.meta = {
|
|
119
|
+
...this.post.meta,
|
|
120
|
+
category,
|
|
121
|
+
};
|
|
122
|
+
return this;
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Sets the slug
|
|
126
|
+
*/
|
|
127
|
+
slug(slug) {
|
|
128
|
+
this.post.meta = {
|
|
129
|
+
...this.post.meta,
|
|
130
|
+
slug,
|
|
131
|
+
};
|
|
132
|
+
return this;
|
|
133
|
+
}
|
|
134
|
+
/**
|
|
135
|
+
* Sets the ID
|
|
136
|
+
*/
|
|
137
|
+
id(id) {
|
|
138
|
+
this.post.meta = {
|
|
139
|
+
...this.post.meta,
|
|
140
|
+
id,
|
|
141
|
+
};
|
|
142
|
+
return this;
|
|
143
|
+
}
|
|
144
|
+
// Content block methods
|
|
145
|
+
/**
|
|
146
|
+
* Adds a heading block
|
|
147
|
+
*/
|
|
148
|
+
heading(text, level = 2) {
|
|
149
|
+
this.post.content.push({ type: 'heading', level, text });
|
|
150
|
+
return this;
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Adds a paragraph block
|
|
154
|
+
*/
|
|
155
|
+
paragraph(text, emphasis) {
|
|
156
|
+
this.post.content.push({ type: 'paragraph', text, emphasis });
|
|
157
|
+
return this;
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* Adds a quote block
|
|
161
|
+
*/
|
|
162
|
+
quote(text, author, source) {
|
|
163
|
+
this.post.content.push({ type: 'quote', text, author, source });
|
|
164
|
+
return this;
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Adds a code block
|
|
168
|
+
*/
|
|
169
|
+
code(code, language) {
|
|
170
|
+
this.post.content.push({ type: 'code', code, language });
|
|
171
|
+
return this;
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Adds an unordered list
|
|
175
|
+
*/
|
|
176
|
+
list(items) {
|
|
177
|
+
this.post.content.push({ type: 'list', items });
|
|
178
|
+
return this;
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Adds an ordered/numbered list
|
|
182
|
+
*/
|
|
183
|
+
orderedList(items) {
|
|
184
|
+
this.post.content.push({ type: 'list', items, ordered: true });
|
|
185
|
+
return this;
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Adds a checklist
|
|
189
|
+
*/
|
|
190
|
+
checklist(items) {
|
|
191
|
+
this.post.content.push({ type: 'list', items, checklist: true });
|
|
192
|
+
return this;
|
|
193
|
+
}
|
|
194
|
+
/**
|
|
195
|
+
* Adds an image block
|
|
196
|
+
*/
|
|
197
|
+
image(src, alt, caption) {
|
|
198
|
+
this.post.content.push({ type: 'image', src, alt, caption });
|
|
199
|
+
return this;
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Adds a callout/note block
|
|
203
|
+
*/
|
|
204
|
+
callout(text, variant = 'info', title) {
|
|
205
|
+
this.post.content.push({ type: 'callout', text, variant, title });
|
|
206
|
+
return this;
|
|
207
|
+
}
|
|
208
|
+
/**
|
|
209
|
+
* Adds a divider
|
|
210
|
+
*/
|
|
211
|
+
divider(style) {
|
|
212
|
+
this.post.content.push({ type: 'divider', style });
|
|
213
|
+
return this;
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* Adds a button/CTA
|
|
217
|
+
*/
|
|
218
|
+
button(text, href, color) {
|
|
219
|
+
this.post.content.push({ type: 'button', text, href, color });
|
|
220
|
+
return this;
|
|
221
|
+
}
|
|
222
|
+
/**
|
|
223
|
+
* Configures rendering options
|
|
224
|
+
*/
|
|
225
|
+
config(config) {
|
|
226
|
+
this.post.config = { ...this.post.config, ...config };
|
|
227
|
+
return this;
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Builds the final BlogPost object
|
|
231
|
+
*/
|
|
232
|
+
build() {
|
|
233
|
+
const content = this.post.content || [];
|
|
234
|
+
// Auto-calculate reading time if not set
|
|
235
|
+
if (!this.post.readingTime) {
|
|
236
|
+
this.post.readingTime = calculateReadingTime(content);
|
|
237
|
+
}
|
|
238
|
+
return {
|
|
239
|
+
type: 'blog',
|
|
240
|
+
title: this.post.title || 'Untitled Post',
|
|
241
|
+
excerpt: this.post.excerpt || '',
|
|
242
|
+
coverImage: this.post.coverImage,
|
|
243
|
+
readingTime: this.post.readingTime,
|
|
244
|
+
featured: this.post.featured,
|
|
245
|
+
meta: this.post.meta,
|
|
246
|
+
content,
|
|
247
|
+
config: this.post.config,
|
|
248
|
+
};
|
|
249
|
+
}
|
|
250
|
+
/**
|
|
251
|
+
* Builds and converts to ArticleMetadata
|
|
252
|
+
*/
|
|
253
|
+
toArticle() {
|
|
254
|
+
return ContentTransformer.toArticle(this.build());
|
|
255
|
+
}
|
|
256
|
+
/**
|
|
257
|
+
* Resets the builder for reuse
|
|
258
|
+
*/
|
|
259
|
+
clear() {
|
|
260
|
+
this.post = {
|
|
261
|
+
type: 'blog',
|
|
262
|
+
content: [],
|
|
263
|
+
meta: {},
|
|
264
|
+
config: { showMeta: true },
|
|
265
|
+
};
|
|
266
|
+
return this;
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
/**
|
|
270
|
+
* Creates a new BlogPostBuilder instance
|
|
271
|
+
*/
|
|
272
|
+
export function blogPost() {
|
|
273
|
+
return new BlogPostBuilder();
|
|
274
|
+
}
|
|
275
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"blog.js","sourceRoot":"","sources":["../../../../../../../src/lib/services/content/content-types/blog.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Blog Post Content Type\n *\n * Represents a blog post with title, excerpt, cover image,\n * 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 * Blog post document interface\n */\nexport interface BlogPost extends ContentDocument<'blog'> {\n  /** Blog post title */\n  title: string;\n  /** Short excerpt/summary for listings */\n  excerpt: string;\n  /** Cover image URL */\n  coverImage?: string;\n  /** Estimated reading time in minutes */\n  readingTime?: number;\n  /** Featured post flag */\n  featured?: boolean;\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    if (block.type === 'code') {\n      wordCount += block.code.split(/\\s+/).length * 0.5; // Code reads faster\n    }\n  }\n\n  return Math.max(1, Math.ceil(wordCount / wordsPerMinute));\n}\n\n/**\n * BlogPostBuilder provides a fluent API for creating blog posts.\n *\n * @example\n * ```typescript\n * const post = new BlogPostBuilder()\n *   .title('My First Post')\n *   .excerpt('A brief introduction...')\n *   .author('John Doe', '/avatars/john.jpg')\n *   .coverImage('/images/post-cover.jpg')\n *   .heading('Introduction')\n *   .paragraph('Welcome to my blog...')\n *   .build();\n *\n * // Convert to ArticleMetadata\n * const article = post.toArticle();\n * ```\n */\nexport class BlogPostBuilder {\n  private post: Partial<BlogPost> = {\n    type: 'blog',\n    content: [],\n    meta: {},\n    config: { showMeta: true },\n  };\n\n  /**\n   * Sets the blog post title\n   */\n  title(title: string): this {\n    this.post.title = title;\n    return this;\n  }\n\n  /**\n   * Sets the blog post excerpt\n   */\n  excerpt(excerpt: string): this {\n    this.post.excerpt = excerpt;\n    return this;\n  }\n\n  /**\n   * Sets the cover image URL\n   */\n  coverImage(url: string): this {\n    this.post.coverImage = url;\n    return this;\n  }\n\n  /**\n   * Marks the post as featured\n   */\n  featured(value: boolean = true): this {\n    this.post.featured = value;\n    return this;\n  }\n\n  /**\n   * Sets the post author\n   */\n  author(name: string, avatar?: string, role?: string): this {\n    this.post.meta = {\n      ...this.post.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.post.meta = {\n      ...this.post.meta,\n      publishedAt: date,\n    };\n    return this;\n  }\n\n  /**\n   * Sets tags for the post\n   */\n  tags(...tags: string[]): this {\n    this.post.meta = {\n      ...this.post.meta,\n      tags,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the category\n   */\n  category(category: string): this {\n    this.post.meta = {\n      ...this.post.meta,\n      category,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the slug\n   */\n  slug(slug: string): this {\n    this.post.meta = {\n      ...this.post.meta,\n      slug,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the ID\n   */\n  id(id: string): this {\n    this.post.meta = {\n      ...this.post.meta,\n      id,\n    };\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.post.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.post.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.post.content!.push({ type: 'quote', text, author, source });\n    return this;\n  }\n\n  /**\n   * Adds a code block\n   */\n  code(code: string, language?: string): this {\n    this.post.content!.push({ type: 'code', code, language });\n    return this;\n  }\n\n  /**\n   * Adds an unordered list\n   */\n  list(items: string[]): this {\n    this.post.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.post.content!.push({ type: 'list', items, ordered: true });\n    return this;\n  }\n\n  /**\n   * Adds a checklist\n   */\n  checklist(items: string[]): this {\n    this.post.content!.push({ type: 'list', items, checklist: true });\n    return this;\n  }\n\n  /**\n   * Adds an image block\n   */\n  image(src: string, alt: string, caption?: string): this {\n    this.post.content!.push({ type: 'image', src, alt, caption });\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.post.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.post.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.post.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.post.config = { ...this.post.config, ...config };\n    return this;\n  }\n\n  /**\n   * Builds the final BlogPost object\n   */\n  build(): BlogPost {\n    const content = this.post.content || [];\n\n    // Auto-calculate reading time if not set\n    if (!this.post.readingTime) {\n      this.post.readingTime = calculateReadingTime(content);\n    }\n\n    return {\n      type: 'blog',\n      title: this.post.title || 'Untitled Post',\n      excerpt: this.post.excerpt || '',\n      coverImage: this.post.coverImage,\n      readingTime: this.post.readingTime,\n      featured: this.post.featured,\n      meta: this.post.meta as ContentMeta,\n      content,\n      config: this.post.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   * Resets the builder for reuse\n   */\n  clear(): this {\n    this.post = {\n      type: 'blog',\n      content: [],\n      meta: {},\n      config: { showMeta: true },\n    };\n    return this;\n  }\n}\n\n/**\n * Creates a new BlogPostBuilder instance\n */\nexport function blogPost(): BlogPostBuilder {\n  return new BlogPostBuilder();\n}\n"]}
|
|
@@ -0,0 +1,303 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Documentation Content Type
|
|
3
|
+
*
|
|
4
|
+
* Represents technical documentation with navigation,
|
|
5
|
+
* sections, and structured content blocks.
|
|
6
|
+
*/
|
|
7
|
+
import { ContentTransformer } from '../transformer';
|
|
8
|
+
/**
|
|
9
|
+
* Generates a URL-friendly ID from text
|
|
10
|
+
*/
|
|
11
|
+
function slugify(text) {
|
|
12
|
+
return text
|
|
13
|
+
.toLowerCase()
|
|
14
|
+
.replace(/[^\w\s-]/g, '')
|
|
15
|
+
.replace(/\s+/g, '-')
|
|
16
|
+
.trim();
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Extracts table of contents from content blocks
|
|
20
|
+
*/
|
|
21
|
+
function extractToc(content) {
|
|
22
|
+
const toc = [];
|
|
23
|
+
for (const block of content) {
|
|
24
|
+
if (block.type === 'heading') {
|
|
25
|
+
toc.push({
|
|
26
|
+
text: block.text,
|
|
27
|
+
level: block.level,
|
|
28
|
+
id: slugify(block.text),
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
return toc;
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* DocsBuilder provides a fluent API for creating documentation pages.
|
|
36
|
+
*
|
|
37
|
+
* @example
|
|
38
|
+
* ```typescript
|
|
39
|
+
* const docs = new DocsBuilder()
|
|
40
|
+
* .title('Installation Guide')
|
|
41
|
+
* .section('Getting Started')
|
|
42
|
+
* .version('1.0.0')
|
|
43
|
+
* .heading('Prerequisites', 1)
|
|
44
|
+
* .paragraph('Before you begin, ensure you have...')
|
|
45
|
+
* .callout('Node.js 18+ is required', 'warning')
|
|
46
|
+
* .heading('Installation', 1)
|
|
47
|
+
* .code('npm install valtech-components', 'bash')
|
|
48
|
+
* .prevPage('Introduction', '/docs/intro')
|
|
49
|
+
* .nextPage('Configuration', '/docs/config')
|
|
50
|
+
* .build();
|
|
51
|
+
*
|
|
52
|
+
* // Convert to ArticleMetadata
|
|
53
|
+
* const article = docs.toArticle();
|
|
54
|
+
* ```
|
|
55
|
+
*/
|
|
56
|
+
export class DocsBuilder {
|
|
57
|
+
constructor() {
|
|
58
|
+
this.doc = {
|
|
59
|
+
type: 'docs',
|
|
60
|
+
content: [],
|
|
61
|
+
meta: {},
|
|
62
|
+
config: { showMeta: false, showTableOfContents: true },
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Sets the documentation page title
|
|
67
|
+
*/
|
|
68
|
+
title(title) {
|
|
69
|
+
this.doc.title = title;
|
|
70
|
+
return this;
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Sets the section name
|
|
74
|
+
*/
|
|
75
|
+
section(section) {
|
|
76
|
+
this.doc.section = section;
|
|
77
|
+
return this;
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Sets the documentation version
|
|
81
|
+
*/
|
|
82
|
+
version(version) {
|
|
83
|
+
this.doc.version = version;
|
|
84
|
+
return this;
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Sets the previous page navigation
|
|
88
|
+
*/
|
|
89
|
+
prevPage(title, slug) {
|
|
90
|
+
this.doc.prevPage = { title, slug };
|
|
91
|
+
return this;
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Sets the next page navigation
|
|
95
|
+
*/
|
|
96
|
+
nextPage(title, slug) {
|
|
97
|
+
this.doc.nextPage = { title, slug };
|
|
98
|
+
return this;
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Sets the slug for the page
|
|
102
|
+
*/
|
|
103
|
+
slug(slug) {
|
|
104
|
+
this.doc.meta = {
|
|
105
|
+
...this.doc.meta,
|
|
106
|
+
slug,
|
|
107
|
+
};
|
|
108
|
+
return this;
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Sets the ID for the page
|
|
112
|
+
*/
|
|
113
|
+
id(id) {
|
|
114
|
+
this.doc.meta = {
|
|
115
|
+
...this.doc.meta,
|
|
116
|
+
id,
|
|
117
|
+
};
|
|
118
|
+
return this;
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Sets tags for the documentation page
|
|
122
|
+
*/
|
|
123
|
+
tags(...tags) {
|
|
124
|
+
this.doc.meta = {
|
|
125
|
+
...this.doc.meta,
|
|
126
|
+
tags,
|
|
127
|
+
};
|
|
128
|
+
return this;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Sets the category
|
|
132
|
+
*/
|
|
133
|
+
category(category) {
|
|
134
|
+
this.doc.meta = {
|
|
135
|
+
...this.doc.meta,
|
|
136
|
+
category,
|
|
137
|
+
};
|
|
138
|
+
return this;
|
|
139
|
+
}
|
|
140
|
+
// Content block methods
|
|
141
|
+
/**
|
|
142
|
+
* Adds a heading block
|
|
143
|
+
*/
|
|
144
|
+
heading(text, level = 2) {
|
|
145
|
+
this.doc.content.push({ type: 'heading', level, text });
|
|
146
|
+
return this;
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Adds a paragraph block
|
|
150
|
+
*/
|
|
151
|
+
paragraph(text, emphasis) {
|
|
152
|
+
this.doc.content.push({ type: 'paragraph', text, emphasis });
|
|
153
|
+
return this;
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Adds a code block
|
|
157
|
+
*/
|
|
158
|
+
code(code, language) {
|
|
159
|
+
this.doc.content.push({ type: 'code', code, language });
|
|
160
|
+
return this;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Adds a command/terminal block
|
|
164
|
+
*/
|
|
165
|
+
command(command) {
|
|
166
|
+
this.doc.content.push({ type: 'command', command });
|
|
167
|
+
return this;
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Adds a callout/note block
|
|
171
|
+
*/
|
|
172
|
+
callout(text, variant = 'info', title) {
|
|
173
|
+
this.doc.content.push({ type: 'callout', text, variant, title });
|
|
174
|
+
return this;
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* Adds an info callout (alias for callout with info variant)
|
|
178
|
+
*/
|
|
179
|
+
info(text, title) {
|
|
180
|
+
return this.callout(text, 'info', title);
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Adds a warning callout
|
|
184
|
+
*/
|
|
185
|
+
warning(text, title) {
|
|
186
|
+
return this.callout(text, 'warning', title);
|
|
187
|
+
}
|
|
188
|
+
/**
|
|
189
|
+
* Adds an error callout
|
|
190
|
+
*/
|
|
191
|
+
error(text, title) {
|
|
192
|
+
return this.callout(text, 'error', title);
|
|
193
|
+
}
|
|
194
|
+
/**
|
|
195
|
+
* Adds a success callout
|
|
196
|
+
*/
|
|
197
|
+
success(text, title) {
|
|
198
|
+
return this.callout(text, 'success', title);
|
|
199
|
+
}
|
|
200
|
+
/**
|
|
201
|
+
* Adds an unordered list
|
|
202
|
+
*/
|
|
203
|
+
list(items) {
|
|
204
|
+
this.doc.content.push({ type: 'list', items });
|
|
205
|
+
return this;
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* Adds an ordered/numbered list
|
|
209
|
+
*/
|
|
210
|
+
orderedList(items) {
|
|
211
|
+
this.doc.content.push({ type: 'list', items, ordered: true });
|
|
212
|
+
return this;
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Adds a checklist
|
|
216
|
+
*/
|
|
217
|
+
checklist(items) {
|
|
218
|
+
this.doc.content.push({ type: 'list', items, checklist: true });
|
|
219
|
+
return this;
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Adds an image block
|
|
223
|
+
*/
|
|
224
|
+
image(src, alt, caption) {
|
|
225
|
+
this.doc.content.push({ type: 'image', src, alt, caption });
|
|
226
|
+
return this;
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* Adds a quote block
|
|
230
|
+
*/
|
|
231
|
+
quote(text, author, source) {
|
|
232
|
+
this.doc.content.push({ type: 'quote', text, author, source });
|
|
233
|
+
return this;
|
|
234
|
+
}
|
|
235
|
+
/**
|
|
236
|
+
* Adds a divider
|
|
237
|
+
*/
|
|
238
|
+
divider(style) {
|
|
239
|
+
this.doc.content.push({ type: 'divider', style });
|
|
240
|
+
return this;
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* Adds a button/CTA
|
|
244
|
+
*/
|
|
245
|
+
button(text, href, color) {
|
|
246
|
+
this.doc.content.push({ type: 'button', text, href, color });
|
|
247
|
+
return this;
|
|
248
|
+
}
|
|
249
|
+
/**
|
|
250
|
+
* Configures rendering options
|
|
251
|
+
*/
|
|
252
|
+
config(config) {
|
|
253
|
+
this.doc.config = { ...this.doc.config, ...config };
|
|
254
|
+
return this;
|
|
255
|
+
}
|
|
256
|
+
/**
|
|
257
|
+
* Builds the final Documentation object
|
|
258
|
+
*/
|
|
259
|
+
build() {
|
|
260
|
+
const content = this.doc.content || [];
|
|
261
|
+
// Auto-generate TOC if not provided and showTableOfContents is enabled
|
|
262
|
+
if (!this.doc.toc && this.doc.config?.showTableOfContents) {
|
|
263
|
+
this.doc.toc = extractToc(content);
|
|
264
|
+
}
|
|
265
|
+
return {
|
|
266
|
+
type: 'docs',
|
|
267
|
+
title: this.doc.title || 'Untitled Documentation',
|
|
268
|
+
section: this.doc.section,
|
|
269
|
+
version: this.doc.version,
|
|
270
|
+
prevPage: this.doc.prevPage,
|
|
271
|
+
nextPage: this.doc.nextPage,
|
|
272
|
+
toc: this.doc.toc,
|
|
273
|
+
meta: this.doc.meta,
|
|
274
|
+
content,
|
|
275
|
+
config: this.doc.config,
|
|
276
|
+
};
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Builds and converts to ArticleMetadata
|
|
280
|
+
*/
|
|
281
|
+
toArticle() {
|
|
282
|
+
return ContentTransformer.toArticle(this.build());
|
|
283
|
+
}
|
|
284
|
+
/**
|
|
285
|
+
* Resets the builder for reuse
|
|
286
|
+
*/
|
|
287
|
+
clear() {
|
|
288
|
+
this.doc = {
|
|
289
|
+
type: 'docs',
|
|
290
|
+
content: [],
|
|
291
|
+
meta: {},
|
|
292
|
+
config: { showMeta: false, showTableOfContents: true },
|
|
293
|
+
};
|
|
294
|
+
return this;
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
/**
|
|
298
|
+
* Creates a new DocsBuilder instance
|
|
299
|
+
*/
|
|
300
|
+
export function docs() {
|
|
301
|
+
return new DocsBuilder();
|
|
302
|
+
}
|
|
303
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"documentation.js","sourceRoot":"","sources":["../../../../../../../src/lib/services/content/content-types/documentation.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Documentation Content Type\n *\n * Represents technical documentation with navigation,\n * sections, 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 * Navigation link for prev/next page navigation\n */\nexport interface DocNavLink {\n  /** Page title */\n  title: string;\n  /** Page slug/URL */\n  slug: string;\n}\n\n/**\n * Documentation document interface\n */\nexport interface Documentation extends ContentDocument<'docs'> {\n  /** Documentation page title */\n  title: string;\n  /** Section name (e.g., \"Getting Started\", \"API Reference\") */\n  section?: string;\n  /** Version of the documentation */\n  version?: string;\n  /** Previous page navigation */\n  prevPage?: DocNavLink;\n  /** Next page navigation */\n  nextPage?: DocNavLink;\n  /** Table of contents items (auto-generated from headings if not provided) */\n  toc?: { text: string; level: number; id: string }[];\n}\n\n/**\n * Generates a URL-friendly ID from text\n */\nfunction slugify(text: string): string {\n  return text\n    .toLowerCase()\n    .replace(/[^\\w\\s-]/g, '')\n    .replace(/\\s+/g, '-')\n    .trim();\n}\n\n/**\n * Extracts table of contents from content blocks\n */\nfunction extractToc(content: ContentBlock[]): { text: string; level: number; id: string }[] {\n  const toc: { text: string; level: number; id: string }[] = [];\n\n  for (const block of content) {\n    if (block.type === 'heading') {\n      toc.push({\n        text: block.text,\n        level: block.level,\n        id: slugify(block.text),\n      });\n    }\n  }\n\n  return toc;\n}\n\n/**\n * DocsBuilder provides a fluent API for creating documentation pages.\n *\n * @example\n * ```typescript\n * const docs = new DocsBuilder()\n *   .title('Installation Guide')\n *   .section('Getting Started')\n *   .version('1.0.0')\n *   .heading('Prerequisites', 1)\n *   .paragraph('Before you begin, ensure you have...')\n *   .callout('Node.js 18+ is required', 'warning')\n *   .heading('Installation', 1)\n *   .code('npm install valtech-components', 'bash')\n *   .prevPage('Introduction', '/docs/intro')\n *   .nextPage('Configuration', '/docs/config')\n *   .build();\n *\n * // Convert to ArticleMetadata\n * const article = docs.toArticle();\n * ```\n */\nexport class DocsBuilder {\n  private doc: Partial<Documentation> = {\n    type: 'docs',\n    content: [],\n    meta: {},\n    config: { showMeta: false, showTableOfContents: true },\n  };\n\n  /**\n   * Sets the documentation page title\n   */\n  title(title: string): this {\n    this.doc.title = title;\n    return this;\n  }\n\n  /**\n   * Sets the section name\n   */\n  section(section: string): this {\n    this.doc.section = section;\n    return this;\n  }\n\n  /**\n   * Sets the documentation version\n   */\n  version(version: string): this {\n    this.doc.version = version;\n    return this;\n  }\n\n  /**\n   * Sets the previous page navigation\n   */\n  prevPage(title: string, slug: string): this {\n    this.doc.prevPage = { title, slug };\n    return this;\n  }\n\n  /**\n   * Sets the next page navigation\n   */\n  nextPage(title: string, slug: string): this {\n    this.doc.nextPage = { title, slug };\n    return this;\n  }\n\n  /**\n   * Sets the slug for the page\n   */\n  slug(slug: string): this {\n    this.doc.meta = {\n      ...this.doc.meta,\n      slug,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the ID for the page\n   */\n  id(id: string): this {\n    this.doc.meta = {\n      ...this.doc.meta,\n      id,\n    };\n    return this;\n  }\n\n  /**\n   * Sets tags for the documentation page\n   */\n  tags(...tags: string[]): this {\n    this.doc.meta = {\n      ...this.doc.meta,\n      tags,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the category\n   */\n  category(category: string): this {\n    this.doc.meta = {\n      ...this.doc.meta,\n      category,\n    };\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.doc.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.doc.content!.push({ type: 'paragraph', text, emphasis });\n    return this;\n  }\n\n  /**\n   * Adds a code block\n   */\n  code(code: string, language?: string): this {\n    this.doc.content!.push({ type: 'code', code, language });\n    return this;\n  }\n\n  /**\n   * Adds a command/terminal block\n   */\n  command(command: string): this {\n    this.doc.content!.push({ type: 'command', command });\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.doc.content!.push({ type: 'callout', text, variant, title });\n    return this;\n  }\n\n  /**\n   * Adds an info callout (alias for callout with info variant)\n   */\n  info(text: string, title?: string): this {\n    return this.callout(text, 'info', title);\n  }\n\n  /**\n   * Adds a warning callout\n   */\n  warning(text: string, title?: string): this {\n    return this.callout(text, 'warning', title);\n  }\n\n  /**\n   * Adds an error callout\n   */\n  error(text: string, title?: string): this {\n    return this.callout(text, 'error', title);\n  }\n\n  /**\n   * Adds a success callout\n   */\n  success(text: string, title?: string): this {\n    return this.callout(text, 'success', title);\n  }\n\n  /**\n   * Adds an unordered list\n   */\n  list(items: string[]): this {\n    this.doc.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.doc.content!.push({ type: 'list', items, ordered: true });\n    return this;\n  }\n\n  /**\n   * Adds a checklist\n   */\n  checklist(items: string[]): this {\n    this.doc.content!.push({ type: 'list', items, checklist: true });\n    return this;\n  }\n\n  /**\n   * Adds an image block\n   */\n  image(src: string, alt: string, caption?: string): this {\n    this.doc.content!.push({ type: 'image', src, alt, caption });\n    return this;\n  }\n\n  /**\n   * Adds a quote block\n   */\n  quote(text: string, author?: string, source?: string): this {\n    this.doc.content!.push({ type: 'quote', text, author, source });\n    return this;\n  }\n\n  /**\n   * Adds a divider\n   */\n  divider(style?: 'line' | 'dots' | 'space'): this {\n    this.doc.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.doc.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.doc.config = { ...this.doc.config, ...config };\n    return this;\n  }\n\n  /**\n   * Builds the final Documentation object\n   */\n  build(): Documentation {\n    const content = this.doc.content || [];\n\n    // Auto-generate TOC if not provided and showTableOfContents is enabled\n    if (!this.doc.toc && this.doc.config?.showTableOfContents) {\n      this.doc.toc = extractToc(content);\n    }\n\n    return {\n      type: 'docs',\n      title: this.doc.title || 'Untitled Documentation',\n      section: this.doc.section,\n      version: this.doc.version,\n      prevPage: this.doc.prevPage,\n      nextPage: this.doc.nextPage,\n      toc: this.doc.toc,\n      meta: this.doc.meta as ContentMeta,\n      content,\n      config: this.doc.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   * Resets the builder for reuse\n   */\n  clear(): this {\n    this.doc = {\n      type: 'docs',\n      content: [],\n      meta: {},\n      config: { showMeta: false, showTableOfContents: true },\n    };\n    return this;\n  }\n}\n\n/**\n * Creates a new DocsBuilder instance\n */\nexport function docs(): DocsBuilder {\n  return new DocsBuilder();\n}\n"]}
|