valtech-components 2.0.529 → 2.0.531
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/molecules/docs-api-table/docs-api-table.component.mjs +218 -0
- package/esm2022/lib/components/molecules/docs-api-table/types.mjs +11 -0
- package/esm2022/lib/components/molecules/docs-code-example/docs-code-example.component.mjs +217 -0
- package/esm2022/lib/components/molecules/docs-code-example/types.mjs +2 -0
- package/esm2022/lib/components/molecules/docs-nav-links/docs-nav-links.component.mjs +133 -0
- package/esm2022/lib/components/molecules/docs-nav-links/types.mjs +2 -0
- package/esm2022/lib/components/organisms/comment-section/comment-section.component.mjs +15 -10
- package/esm2022/lib/components/organisms/docs-sidebar/docs-sidebar.component.mjs +276 -0
- package/esm2022/lib/components/organisms/docs-sidebar/types.mjs +2 -0
- package/esm2022/lib/components/organisms/docs-toc/docs-toc.component.mjs +255 -0
- package/esm2022/lib/components/organisms/docs-toc/types.mjs +2 -0
- package/esm2022/lib/components/organisms/infinite-list/infinite-list.component.mjs +20 -18
- package/esm2022/lib/components/templates/docs-layout/docs-layout.component.mjs +203 -0
- package/esm2022/lib/components/templates/docs-layout/types.mjs +2 -0
- package/esm2022/lib/services/i18n/default-content.mjs +5 -1
- package/esm2022/public-api.mjs +15 -1
- package/fesm2022/valtech-components.mjs +1313 -27
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/docs-api-table/docs-api-table.component.d.ts +41 -0
- package/lib/components/molecules/docs-api-table/types.d.ts +62 -0
- package/lib/components/molecules/docs-code-example/docs-code-example.component.d.ts +35 -0
- package/lib/components/molecules/docs-code-example/types.d.ts +63 -0
- package/lib/components/molecules/docs-nav-links/docs-nav-links.component.d.ts +29 -0
- package/lib/components/molecules/docs-nav-links/types.d.ts +31 -0
- package/lib/components/organisms/comment-section/comment-section.component.d.ts +2 -0
- package/lib/components/organisms/docs-sidebar/docs-sidebar.component.d.ts +51 -0
- package/lib/components/organisms/docs-sidebar/types.d.ts +93 -0
- package/lib/components/organisms/docs-toc/docs-toc.component.d.ts +58 -0
- package/lib/components/organisms/docs-toc/types.d.ts +62 -0
- package/lib/components/organisms/infinite-list/infinite-list.component.d.ts +2 -0
- package/lib/components/templates/docs-layout/docs-layout.component.d.ts +41 -0
- package/lib/components/templates/docs-layout/types.d.ts +39 -0
- package/package.json +1 -1
- package/public-api.d.ts +12 -0
|
@@ -0,0 +1,218 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, Input } from '@angular/core';
|
|
3
|
+
import { API_TABLE_COLUMN_LABELS } from './types';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
/**
|
|
6
|
+
* val-docs-api-table
|
|
7
|
+
*
|
|
8
|
+
* A table component for documenting component APIs (inputs, outputs, methods).
|
|
9
|
+
*
|
|
10
|
+
* @example Properties table
|
|
11
|
+
* ```html
|
|
12
|
+
* <val-docs-api-table
|
|
13
|
+
* [props]="{
|
|
14
|
+
* title: 'Properties',
|
|
15
|
+
* items: [
|
|
16
|
+
* { name: 'disabled', type: 'boolean', default: 'false', description: 'Disables the button' },
|
|
17
|
+
* { name: 'color', type: 'string', default: 'primary', description: 'Button color variant' }
|
|
18
|
+
* ]
|
|
19
|
+
* }"
|
|
20
|
+
* ></val-docs-api-table>
|
|
21
|
+
* ```
|
|
22
|
+
*
|
|
23
|
+
* @example Events table
|
|
24
|
+
* ```html
|
|
25
|
+
* <val-docs-api-table
|
|
26
|
+
* [props]="{
|
|
27
|
+
* title: 'Events',
|
|
28
|
+
* columns: ['name', 'type', 'description'],
|
|
29
|
+
* items: [
|
|
30
|
+
* { name: 'click', type: 'EventEmitter<MouseEvent>', description: 'Emitted when clicked' }
|
|
31
|
+
* ]
|
|
32
|
+
* }"
|
|
33
|
+
* ></val-docs-api-table>
|
|
34
|
+
* ```
|
|
35
|
+
*/
|
|
36
|
+
export class DocsApiTableComponent {
|
|
37
|
+
constructor() {
|
|
38
|
+
this.props = { items: [] };
|
|
39
|
+
this.columnLabels = API_TABLE_COLUMN_LABELS;
|
|
40
|
+
}
|
|
41
|
+
get displayColumns() {
|
|
42
|
+
return this.props.columns || ['name', 'type', 'default', 'description'];
|
|
43
|
+
}
|
|
44
|
+
getColumnLabel(col) {
|
|
45
|
+
return this.columnLabels[col];
|
|
46
|
+
}
|
|
47
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DocsApiTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
48
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: DocsApiTableComponent, isStandalone: true, selector: "val-docs-api-table", inputs: { props: "props" }, ngImport: i0, template: `
|
|
49
|
+
<div
|
|
50
|
+
class="docs-api-table"
|
|
51
|
+
[class]="props.cssClass"
|
|
52
|
+
[class.docs-api-table--bordered]="props.bordered !== false"
|
|
53
|
+
>
|
|
54
|
+
@if (props.title) {
|
|
55
|
+
<h3 class="docs-api-table__title">{{ props.title }}</h3>
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
<div class="docs-api-table__wrapper">
|
|
59
|
+
<table class="docs-api-table__table">
|
|
60
|
+
<thead>
|
|
61
|
+
<tr>
|
|
62
|
+
@for (col of displayColumns; track col) {
|
|
63
|
+
<th
|
|
64
|
+
class="docs-api-table__th"
|
|
65
|
+
[class.docs-api-table__th--name]="col === 'name'"
|
|
66
|
+
[class.docs-api-table__th--type]="col === 'type'"
|
|
67
|
+
[class.docs-api-table__th--default]="col === 'default'"
|
|
68
|
+
[class.docs-api-table__th--description]="col === 'description'"
|
|
69
|
+
>
|
|
70
|
+
{{ getColumnLabel(col) }}
|
|
71
|
+
</th>
|
|
72
|
+
}
|
|
73
|
+
</tr>
|
|
74
|
+
</thead>
|
|
75
|
+
<tbody>
|
|
76
|
+
@for (item of props.items; track item.name) {
|
|
77
|
+
<tr
|
|
78
|
+
class="docs-api-table__row"
|
|
79
|
+
[class.docs-api-table__row--deprecated]="item.deprecated"
|
|
80
|
+
>
|
|
81
|
+
@for (col of displayColumns; track col) {
|
|
82
|
+
<td
|
|
83
|
+
class="docs-api-table__td"
|
|
84
|
+
[class.docs-api-table__td--name]="col === 'name'"
|
|
85
|
+
[class.docs-api-table__td--type]="col === 'type'"
|
|
86
|
+
[class.docs-api-table__td--default]="col === 'default'"
|
|
87
|
+
[class.docs-api-table__td--description]="col === 'description'"
|
|
88
|
+
>
|
|
89
|
+
@switch (col) {
|
|
90
|
+
@case ('name') {
|
|
91
|
+
<code class="docs-api-table__code">{{ item.name }}</code>
|
|
92
|
+
@if (item.required) {
|
|
93
|
+
<span class="docs-api-table__required">*</span>
|
|
94
|
+
}
|
|
95
|
+
@if (item.deprecated) {
|
|
96
|
+
<span class="docs-api-table__badge docs-api-table__badge--deprecated">Deprecated</span>
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
@case ('type') {
|
|
100
|
+
<code class="docs-api-table__type">{{ item.type || '-' }}</code>
|
|
101
|
+
}
|
|
102
|
+
@case ('default') {
|
|
103
|
+
<code class="docs-api-table__default">{{ item.default || '-' }}</code>
|
|
104
|
+
}
|
|
105
|
+
@case ('description') {
|
|
106
|
+
<span>{{ item.description || '-' }}</span>
|
|
107
|
+
@if (item.deprecated && item.deprecationNote) {
|
|
108
|
+
<div class="docs-api-table__deprecation-note">
|
|
109
|
+
{{ item.deprecationNote }}
|
|
110
|
+
</div>
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
@case ('required') {
|
|
114
|
+
{{ item.required ? 'Yes' : 'No' }}
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
</td>
|
|
118
|
+
}
|
|
119
|
+
</tr>
|
|
120
|
+
}
|
|
121
|
+
</tbody>
|
|
122
|
+
</table>
|
|
123
|
+
</div>
|
|
124
|
+
|
|
125
|
+
@if (props.items.length === 0) {
|
|
126
|
+
<p class="docs-api-table__empty">No items to display.</p>
|
|
127
|
+
}
|
|
128
|
+
</div>
|
|
129
|
+
`, isInline: true, styles: [".docs-api-table{margin:1.5rem 0;&__title{font-size:1.125rem;font-weight:600;margin:0 0 1rem;color:var(--ion-text-color, #1a1a1a)}&__wrapper{overflow-x:auto}&__table{width:100%;border-collapse:collapse;font-size:.875rem}&__th{text-align:left;padding:.75rem 1rem;font-weight:600;color:var(--ion-color-medium-shade, #555);background:var(--ion-color-light, #f4f5f8);white-space:nowrap;&--name{width:20%;min-width:140px}&--type{width:20%;min-width:120px}&--default{width:15%;min-width:100px}&--description{width:45%}}&__td{padding:.75rem 1rem;border-bottom:1px solid var(--ion-border-color, #e0e0e0);vertical-align:top;&--name{font-weight:500}}&__row{&--deprecated{opacity:.7;.docs-api-table__code{text-decoration:line-through}}}&__code{font-family:SF Mono,Monaco,Courier New,monospace;font-size:.8125rem;color:var(--ion-color-primary, #3880ff);background:#3880ff1a;padding:.125rem .375rem;border-radius:4px}&__type,&__default{font-family:SF Mono,Monaco,Courier New,monospace;font-size:.75rem;color:var(--ion-color-medium, #666);background:var(--ion-color-light, #f4f5f8);padding:.125rem .375rem;border-radius:4px}&__required{color:var(--ion-color-danger, #eb445a);font-weight:700;margin-left:.25rem}&__badge{display:inline-block;margin-left:.5rem;padding:.125rem .5rem;font-size:.625rem;font-weight:600;text-transform:uppercase;border-radius:10px;&--deprecated{background:var(--ion-color-warning-tint, rgba(255, 196, 9, .2));color:var(--ion-color-warning-shade, #e0ac08)}}&__deprecation-note{margin-top:.5rem;font-size:.75rem;font-style:italic;color:var(--ion-color-warning-shade, #e0ac08)}&__empty{padding:2rem;text-align:center;color:var(--ion-color-medium, #666)}// Bordered variant &--bordered{.docs-api-table__wrapper{border:1px solid var(--ion-border-color, #e0e0e0);border-radius:8px;overflow:hidden}.docs-api-table__th:first-child{border-top-left-radius:8px}.docs-api-table__th:last-child{border-top-right-radius:8px}}}// Dark mode :host-context(.dark),:host-context([color-scheme=\"dark\"]){.docs-api-table{&__title{color:#f4f5f8}&__th{background:#ffffff0d;color:#a0a0a0}&__td{border-color:#3a3a3a}&__type,&__default{background:#ffffff0d}&--bordered .docs-api-table__wrapper{border-color:#3a3a3a}}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
|
|
130
|
+
}
|
|
131
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DocsApiTableComponent, decorators: [{
|
|
132
|
+
type: Component,
|
|
133
|
+
args: [{ selector: 'val-docs-api-table', standalone: true, imports: [CommonModule], template: `
|
|
134
|
+
<div
|
|
135
|
+
class="docs-api-table"
|
|
136
|
+
[class]="props.cssClass"
|
|
137
|
+
[class.docs-api-table--bordered]="props.bordered !== false"
|
|
138
|
+
>
|
|
139
|
+
@if (props.title) {
|
|
140
|
+
<h3 class="docs-api-table__title">{{ props.title }}</h3>
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
<div class="docs-api-table__wrapper">
|
|
144
|
+
<table class="docs-api-table__table">
|
|
145
|
+
<thead>
|
|
146
|
+
<tr>
|
|
147
|
+
@for (col of displayColumns; track col) {
|
|
148
|
+
<th
|
|
149
|
+
class="docs-api-table__th"
|
|
150
|
+
[class.docs-api-table__th--name]="col === 'name'"
|
|
151
|
+
[class.docs-api-table__th--type]="col === 'type'"
|
|
152
|
+
[class.docs-api-table__th--default]="col === 'default'"
|
|
153
|
+
[class.docs-api-table__th--description]="col === 'description'"
|
|
154
|
+
>
|
|
155
|
+
{{ getColumnLabel(col) }}
|
|
156
|
+
</th>
|
|
157
|
+
}
|
|
158
|
+
</tr>
|
|
159
|
+
</thead>
|
|
160
|
+
<tbody>
|
|
161
|
+
@for (item of props.items; track item.name) {
|
|
162
|
+
<tr
|
|
163
|
+
class="docs-api-table__row"
|
|
164
|
+
[class.docs-api-table__row--deprecated]="item.deprecated"
|
|
165
|
+
>
|
|
166
|
+
@for (col of displayColumns; track col) {
|
|
167
|
+
<td
|
|
168
|
+
class="docs-api-table__td"
|
|
169
|
+
[class.docs-api-table__td--name]="col === 'name'"
|
|
170
|
+
[class.docs-api-table__td--type]="col === 'type'"
|
|
171
|
+
[class.docs-api-table__td--default]="col === 'default'"
|
|
172
|
+
[class.docs-api-table__td--description]="col === 'description'"
|
|
173
|
+
>
|
|
174
|
+
@switch (col) {
|
|
175
|
+
@case ('name') {
|
|
176
|
+
<code class="docs-api-table__code">{{ item.name }}</code>
|
|
177
|
+
@if (item.required) {
|
|
178
|
+
<span class="docs-api-table__required">*</span>
|
|
179
|
+
}
|
|
180
|
+
@if (item.deprecated) {
|
|
181
|
+
<span class="docs-api-table__badge docs-api-table__badge--deprecated">Deprecated</span>
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
@case ('type') {
|
|
185
|
+
<code class="docs-api-table__type">{{ item.type || '-' }}</code>
|
|
186
|
+
}
|
|
187
|
+
@case ('default') {
|
|
188
|
+
<code class="docs-api-table__default">{{ item.default || '-' }}</code>
|
|
189
|
+
}
|
|
190
|
+
@case ('description') {
|
|
191
|
+
<span>{{ item.description || '-' }}</span>
|
|
192
|
+
@if (item.deprecated && item.deprecationNote) {
|
|
193
|
+
<div class="docs-api-table__deprecation-note">
|
|
194
|
+
{{ item.deprecationNote }}
|
|
195
|
+
</div>
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
@case ('required') {
|
|
199
|
+
{{ item.required ? 'Yes' : 'No' }}
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
</td>
|
|
203
|
+
}
|
|
204
|
+
</tr>
|
|
205
|
+
}
|
|
206
|
+
</tbody>
|
|
207
|
+
</table>
|
|
208
|
+
</div>
|
|
209
|
+
|
|
210
|
+
@if (props.items.length === 0) {
|
|
211
|
+
<p class="docs-api-table__empty">No items to display.</p>
|
|
212
|
+
}
|
|
213
|
+
</div>
|
|
214
|
+
`, styles: [".docs-api-table{margin:1.5rem 0;&__title{font-size:1.125rem;font-weight:600;margin:0 0 1rem;color:var(--ion-text-color, #1a1a1a)}&__wrapper{overflow-x:auto}&__table{width:100%;border-collapse:collapse;font-size:.875rem}&__th{text-align:left;padding:.75rem 1rem;font-weight:600;color:var(--ion-color-medium-shade, #555);background:var(--ion-color-light, #f4f5f8);white-space:nowrap;&--name{width:20%;min-width:140px}&--type{width:20%;min-width:120px}&--default{width:15%;min-width:100px}&--description{width:45%}}&__td{padding:.75rem 1rem;border-bottom:1px solid var(--ion-border-color, #e0e0e0);vertical-align:top;&--name{font-weight:500}}&__row{&--deprecated{opacity:.7;.docs-api-table__code{text-decoration:line-through}}}&__code{font-family:SF Mono,Monaco,Courier New,monospace;font-size:.8125rem;color:var(--ion-color-primary, #3880ff);background:#3880ff1a;padding:.125rem .375rem;border-radius:4px}&__type,&__default{font-family:SF Mono,Monaco,Courier New,monospace;font-size:.75rem;color:var(--ion-color-medium, #666);background:var(--ion-color-light, #f4f5f8);padding:.125rem .375rem;border-radius:4px}&__required{color:var(--ion-color-danger, #eb445a);font-weight:700;margin-left:.25rem}&__badge{display:inline-block;margin-left:.5rem;padding:.125rem .5rem;font-size:.625rem;font-weight:600;text-transform:uppercase;border-radius:10px;&--deprecated{background:var(--ion-color-warning-tint, rgba(255, 196, 9, .2));color:var(--ion-color-warning-shade, #e0ac08)}}&__deprecation-note{margin-top:.5rem;font-size:.75rem;font-style:italic;color:var(--ion-color-warning-shade, #e0ac08)}&__empty{padding:2rem;text-align:center;color:var(--ion-color-medium, #666)}// Bordered variant &--bordered{.docs-api-table__wrapper{border:1px solid var(--ion-border-color, #e0e0e0);border-radius:8px;overflow:hidden}.docs-api-table__th:first-child{border-top-left-radius:8px}.docs-api-table__th:last-child{border-top-right-radius:8px}}}// Dark mode :host-context(.dark),:host-context([color-scheme=\"dark\"]){.docs-api-table{&__title{color:#f4f5f8}&__th{background:#ffffff0d;color:#a0a0a0}&__td{border-color:#3a3a3a}&__type,&__default{background:#ffffff0d}&--bordered .docs-api-table__wrapper{border-color:#3a3a3a}}}\n"] }]
|
|
215
|
+
}], propDecorators: { props: [{
|
|
216
|
+
type: Input
|
|
217
|
+
}] } });
|
|
218
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-api-table.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/docs-api-table/docs-api-table.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AACjD,OAAO,EAAoD,uBAAuB,EAAE,MAAM,SAAS,CAAC;;AAEpG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AAkPH,MAAM,OAAO,qBAAqB;IAjPlC;QAkPW,UAAK,GAAyB,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC;QAE5C,iBAAY,GAAG,uBAAuB,CAAC;KASjD;IAPC,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,aAAa,CAAC,CAAC;IAC1E,CAAC;IAED,cAAc,CAAC,GAAkB;QAC/B,OAAO,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;IAChC,CAAC;+GAXU,qBAAqB;mGAArB,qBAAqB,0GA7OtB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiFT,suEAlFS,YAAY;;4FA8OX,qBAAqB;kBAjPjC,SAAS;+BACE,oBAAoB,cAClB,IAAI,WACP,CAAC,YAAY,CAAC,YACb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiFT;8BA6JQ,KAAK;sBAAb,KAAK","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input } from '@angular/core';\nimport { DocsApiTableMetadata, DocsApiColumn, DocsApiItem, API_TABLE_COLUMN_LABELS } from './types';\n\n/**\n * val-docs-api-table\n *\n * A table component for documenting component APIs (inputs, outputs, methods).\n *\n * @example Properties table\n * ```html\n * <val-docs-api-table\n *   [props]=\"{\n *     title: 'Properties',\n *     items: [\n *       { name: 'disabled', type: 'boolean', default: 'false', description: 'Disables the button' },\n *       { name: 'color', type: 'string', default: 'primary', description: 'Button color variant' }\n *     ]\n *   }\"\n * ></val-docs-api-table>\n * ```\n *\n * @example Events table\n * ```html\n * <val-docs-api-table\n *   [props]=\"{\n *     title: 'Events',\n *     columns: ['name', 'type', 'description'],\n *     items: [\n *       { name: 'click', type: 'EventEmitter<MouseEvent>', description: 'Emitted when clicked' }\n *     ]\n *   }\"\n * ></val-docs-api-table>\n * ```\n */\n@Component({\n  selector: 'val-docs-api-table',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <div\n      class=\"docs-api-table\"\n      [class]=\"props.cssClass\"\n      [class.docs-api-table--bordered]=\"props.bordered !== false\"\n    >\n      @if (props.title) {\n        <h3 class=\"docs-api-table__title\">{{ props.title }}</h3>\n      }\n\n      <div class=\"docs-api-table__wrapper\">\n        <table class=\"docs-api-table__table\">\n          <thead>\n            <tr>\n              @for (col of displayColumns; track col) {\n                <th\n                  class=\"docs-api-table__th\"\n                  [class.docs-api-table__th--name]=\"col === 'name'\"\n                  [class.docs-api-table__th--type]=\"col === 'type'\"\n                  [class.docs-api-table__th--default]=\"col === 'default'\"\n                  [class.docs-api-table__th--description]=\"col === 'description'\"\n                >\n                  {{ getColumnLabel(col) }}\n                </th>\n              }\n            </tr>\n          </thead>\n          <tbody>\n            @for (item of props.items; track item.name) {\n              <tr\n                class=\"docs-api-table__row\"\n                [class.docs-api-table__row--deprecated]=\"item.deprecated\"\n              >\n                @for (col of displayColumns; track col) {\n                  <td\n                    class=\"docs-api-table__td\"\n                    [class.docs-api-table__td--name]=\"col === 'name'\"\n                    [class.docs-api-table__td--type]=\"col === 'type'\"\n                    [class.docs-api-table__td--default]=\"col === 'default'\"\n                    [class.docs-api-table__td--description]=\"col === 'description'\"\n                  >\n                    @switch (col) {\n                      @case ('name') {\n                        <code class=\"docs-api-table__code\">{{ item.name }}</code>\n                        @if (item.required) {\n                          <span class=\"docs-api-table__required\">*</span>\n                        }\n                        @if (item.deprecated) {\n                          <span class=\"docs-api-table__badge docs-api-table__badge--deprecated\">Deprecated</span>\n                        }\n                      }\n                      @case ('type') {\n                        <code class=\"docs-api-table__type\">{{ item.type || '-' }}</code>\n                      }\n                      @case ('default') {\n                        <code class=\"docs-api-table__default\">{{ item.default || '-' }}</code>\n                      }\n                      @case ('description') {\n                        <span>{{ item.description || '-' }}</span>\n                        @if (item.deprecated && item.deprecationNote) {\n                          <div class=\"docs-api-table__deprecation-note\">\n                            {{ item.deprecationNote }}\n                          </div>\n                        }\n                      }\n                      @case ('required') {\n                        {{ item.required ? 'Yes' : 'No' }}\n                      }\n                    }\n                  </td>\n                }\n              </tr>\n            }\n          </tbody>\n        </table>\n      </div>\n\n      @if (props.items.length === 0) {\n        <p class=\"docs-api-table__empty\">No items to display.</p>\n      }\n    </div>\n  `,\n  styles: [`\n    .docs-api-table {\n      margin: 1.5rem 0;\n\n      &__title {\n        font-size: 1.125rem;\n        font-weight: 600;\n        margin: 0 0 1rem 0;\n        color: var(--ion-text-color, #1a1a1a);\n      }\n\n      &__wrapper {\n        overflow-x: auto;\n      }\n\n      &__table {\n        width: 100%;\n        border-collapse: collapse;\n        font-size: 0.875rem;\n      }\n\n      &__th {\n        text-align: left;\n        padding: 0.75rem 1rem;\n        font-weight: 600;\n        color: var(--ion-color-medium-shade, #555);\n        background: var(--ion-color-light, #f4f5f8);\n        white-space: nowrap;\n\n        &--name { width: 20%; min-width: 140px; }\n        &--type { width: 20%; min-width: 120px; }\n        &--default { width: 15%; min-width: 100px; }\n        &--description { width: 45%; }\n      }\n\n      &__td {\n        padding: 0.75rem 1rem;\n        border-bottom: 1px solid var(--ion-border-color, #e0e0e0);\n        vertical-align: top;\n\n        &--name {\n          font-weight: 500;\n        }\n      }\n\n      &__row {\n        &--deprecated {\n          opacity: 0.7;\n\n          .docs-api-table__code {\n            text-decoration: line-through;\n          }\n        }\n      }\n\n      &__code {\n        font-family: 'SF Mono', Monaco, 'Courier New', monospace;\n        font-size: 0.8125rem;\n        color: var(--ion-color-primary, #3880ff);\n        background: rgba(56, 128, 255, 0.1);\n        padding: 0.125rem 0.375rem;\n        border-radius: 4px;\n      }\n\n      &__type,\n      &__default {\n        font-family: 'SF Mono', Monaco, 'Courier New', monospace;\n        font-size: 0.75rem;\n        color: var(--ion-color-medium, #666);\n        background: var(--ion-color-light, #f4f5f8);\n        padding: 0.125rem 0.375rem;\n        border-radius: 4px;\n      }\n\n      &__required {\n        color: var(--ion-color-danger, #eb445a);\n        font-weight: bold;\n        margin-left: 0.25rem;\n      }\n\n      &__badge {\n        display: inline-block;\n        margin-left: 0.5rem;\n        padding: 0.125rem 0.5rem;\n        font-size: 0.625rem;\n        font-weight: 600;\n        text-transform: uppercase;\n        border-radius: 10px;\n\n        &--deprecated {\n          background: var(--ion-color-warning-tint, rgba(255, 196, 9, 0.2));\n          color: var(--ion-color-warning-shade, #e0ac08);\n        }\n      }\n\n      &__deprecation-note {\n        margin-top: 0.5rem;\n        font-size: 0.75rem;\n        font-style: italic;\n        color: var(--ion-color-warning-shade, #e0ac08);\n      }\n\n      &__empty {\n        padding: 2rem;\n        text-align: center;\n        color: var(--ion-color-medium, #666);\n      }\n\n      // Bordered variant\n      &--bordered {\n        .docs-api-table__wrapper {\n          border: 1px solid var(--ion-border-color, #e0e0e0);\n          border-radius: 8px;\n          overflow: hidden;\n        }\n\n        .docs-api-table__th:first-child {\n          border-top-left-radius: 8px;\n        }\n\n        .docs-api-table__th:last-child {\n          border-top-right-radius: 8px;\n        }\n      }\n    }\n\n    // Dark mode\n    :host-context(.dark),\n    :host-context([color-scheme='dark']) {\n      .docs-api-table {\n        &__title {\n          color: #f4f5f8;\n        }\n\n        &__th {\n          background: rgba(255, 255, 255, 0.05);\n          color: #a0a0a0;\n        }\n\n        &__td {\n          border-color: #3a3a3a;\n        }\n\n        &__type,\n        &__default {\n          background: rgba(255, 255, 255, 0.05);\n        }\n\n        &--bordered .docs-api-table__wrapper {\n          border-color: #3a3a3a;\n        }\n      }\n    }\n  `],\n})\nexport class DocsApiTableComponent {\n  @Input() props: DocsApiTableMetadata = { items: [] };\n\n  readonly columnLabels = API_TABLE_COLUMN_LABELS;\n\n  get displayColumns(): DocsApiColumn[] {\n    return this.props.columns || ['name', 'type', 'default', 'description'];\n  }\n\n  getColumnLabel(col: DocsApiColumn): string {\n    return this.columnLabels[col];\n  }\n}\n"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Column labels configuration.
|
|
3
|
+
*/
|
|
4
|
+
export const API_TABLE_COLUMN_LABELS = {
|
|
5
|
+
name: 'Name',
|
|
6
|
+
type: 'Type',
|
|
7
|
+
default: 'Default',
|
|
8
|
+
description: 'Description',
|
|
9
|
+
required: 'Required',
|
|
10
|
+
};
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2RvY3MtYXBpLXRhYmxlL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQXVFQTs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLHVCQUF1QixHQUFrQztJQUNwRSxJQUFJLEVBQUUsTUFBTTtJQUNaLElBQUksRUFBRSxNQUFNO0lBQ1osT0FBTyxFQUFFLFNBQVM7SUFDbEIsV0FBVyxFQUFFLGFBQWE7SUFDMUIsUUFBUSxFQUFFLFVBQVU7Q0FDckIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29uZmlndXJhdGlvbiBmb3IgdGhlIGRvY3MtYXBpLXRhYmxlIGNvbXBvbmVudC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBEb2NzQXBpVGFibGVNZXRhZGF0YSB7XG4gIC8qKlxuICAgKiBUYWJsZSB0aXRsZSAoZS5nLiwgXCJQcm9wZXJ0aWVzXCIsIFwiRXZlbnRzXCIsIFwiTWV0aG9kc1wiKS5cbiAgICovXG4gIHRpdGxlPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBBUEkgaXRlbXMgdG8gZGlzcGxheS5cbiAgICovXG4gIGl0ZW1zOiBEb2NzQXBpSXRlbVtdO1xuXG4gIC8qKlxuICAgKiBDb2x1bW5zIHRvIHNob3cgKHN1YnNldCBvZiBhdmFpbGFibGUgY29sdW1ucykuXG4gICAqIEBkZWZhdWx0IFsnbmFtZScsICd0eXBlJywgJ2RlZmF1bHQnLCAnZGVzY3JpcHRpb24nXVxuICAgKi9cbiAgY29sdW1ucz86IERvY3NBcGlDb2x1bW5bXTtcblxuICAvKipcbiAgICogQ3VzdG9tIENTUyBjbGFzcy5cbiAgICovXG4gIGNzc0NsYXNzPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBTaG93IGJvcmRlciBhcm91bmQgdGhlIHRhYmxlLlxuICAgKiBAZGVmYXVsdCB0cnVlXG4gICAqL1xuICBib3JkZXJlZD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCB0eXBlIERvY3NBcGlDb2x1bW4gPSAnbmFtZScgfCAndHlwZScgfCAnZGVmYXVsdCcgfCAnZGVzY3JpcHRpb24nIHwgJ3JlcXVpcmVkJztcblxuZXhwb3J0IGludGVyZmFjZSBEb2NzQXBpSXRlbSB7XG4gIC8qKlxuICAgKiBQcm9wZXJ0eS9ldmVudC9tZXRob2QgbmFtZS5cbiAgICovXG4gIG5hbWU6IHN0cmluZztcblxuICAvKipcbiAgICogVHlwZVNjcmlwdCB0eXBlIChlLmcuLCBcInN0cmluZ1wiLCBcImJvb2xlYW5cIiwgXCJFdmVudEVtaXR0ZXI8VD5cIikuXG4gICAqL1xuICB0eXBlPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBEZWZhdWx0IHZhbHVlLlxuICAgKi9cbiAgZGVmYXVsdD86IHN0cmluZztcblxuICAvKipcbiAgICogRGVzY3JpcHRpb24gb2YgdGhlIHByb3BlcnR5L2V2ZW50L21ldGhvZC5cbiAgICovXG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBXaGV0aGVyIGl0J3MgcmVxdWlyZWQuXG4gICAqL1xuICByZXF1aXJlZD86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgaXQncyBkZXByZWNhdGVkLlxuICAgKi9cbiAgZGVwcmVjYXRlZD86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIERlcHJlY2F0aW9uIG1lc3NhZ2Ugb3IgYWx0ZXJuYXRpdmUuXG4gICAqL1xuICBkZXByZWNhdGlvbk5vdGU/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogQ29sdW1uIGxhYmVscyBjb25maWd1cmF0aW9uLlxuICovXG5leHBvcnQgY29uc3QgQVBJX1RBQkxFX0NPTFVNTl9MQUJFTFM6IFJlY29yZDxEb2NzQXBpQ29sdW1uLCBzdHJpbmc+ID0ge1xuICBuYW1lOiAnTmFtZScsXG4gIHR5cGU6ICdUeXBlJyxcbiAgZGVmYXVsdDogJ0RlZmF1bHQnLFxuICBkZXNjcmlwdGlvbjogJ0Rlc2NyaXB0aW9uJyxcbiAgcmVxdWlyZWQ6ICdSZXF1aXJlZCcsXG59O1xuIl19
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, Input, signal, ContentChild } from '@angular/core';
|
|
3
|
+
import { IonIcon, IonButton } from '@ionic/angular/standalone';
|
|
4
|
+
import { addIcons } from 'ionicons';
|
|
5
|
+
import { copyOutline, checkmarkOutline, openOutline } from 'ionicons/icons';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/common";
|
|
8
|
+
addIcons({ copyOutline, checkmarkOutline, openOutline });
|
|
9
|
+
/**
|
|
10
|
+
* val-docs-code-example
|
|
11
|
+
*
|
|
12
|
+
* A code example component with tabs, preview, and copy functionality.
|
|
13
|
+
*
|
|
14
|
+
* @example Basic usage
|
|
15
|
+
* ```html
|
|
16
|
+
* <val-docs-code-example
|
|
17
|
+
* [props]="{
|
|
18
|
+
* title: 'Basic Button',
|
|
19
|
+
* tabs: [
|
|
20
|
+
* { label: 'HTML', language: 'html', code: '<val-button>Click me</val-button>' },
|
|
21
|
+
* { label: 'TypeScript', language: 'typescript', code: 'import { ButtonComponent } from ...' }
|
|
22
|
+
* ]
|
|
23
|
+
* }"
|
|
24
|
+
* >
|
|
25
|
+
* <ng-template #preview>
|
|
26
|
+
* <val-button [props]="{ label: 'Click me' }"></val-button>
|
|
27
|
+
* </ng-template>
|
|
28
|
+
* </val-docs-code-example>
|
|
29
|
+
* ```
|
|
30
|
+
*/
|
|
31
|
+
export class DocsCodeExampleComponent {
|
|
32
|
+
constructor() {
|
|
33
|
+
this.props = { tabs: [] };
|
|
34
|
+
this.activeTab = signal(0);
|
|
35
|
+
this.copied = signal(false);
|
|
36
|
+
}
|
|
37
|
+
currentTab() {
|
|
38
|
+
return this.props.tabs[this.activeTab()] || { label: '', code: '', language: 'text' };
|
|
39
|
+
}
|
|
40
|
+
async copyCode() {
|
|
41
|
+
const code = this.currentTab().code;
|
|
42
|
+
try {
|
|
43
|
+
await navigator.clipboard.writeText(code);
|
|
44
|
+
this.copied.set(true);
|
|
45
|
+
setTimeout(() => this.copied.set(false), 2000);
|
|
46
|
+
}
|
|
47
|
+
catch (err) {
|
|
48
|
+
console.error('Failed to copy code:', err);
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DocsCodeExampleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
52
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: DocsCodeExampleComponent, isStandalone: true, selector: "val-docs-code-example", inputs: { props: "props" }, queries: [{ propertyName: "previewTpl", first: true, predicate: ["preview"], descendants: true }], ngImport: i0, template: `
|
|
53
|
+
<div class="docs-code-example" [class]="props.cssClass">
|
|
54
|
+
<!-- Header -->
|
|
55
|
+
@if (props.title || props.description || props.externalLink) {
|
|
56
|
+
<div class="docs-code-example__header">
|
|
57
|
+
@if (props.title) {
|
|
58
|
+
<h4 class="docs-code-example__title">{{ props.title }}</h4>
|
|
59
|
+
}
|
|
60
|
+
@if (props.description) {
|
|
61
|
+
<p class="docs-code-example__description">{{ props.description }}</p>
|
|
62
|
+
}
|
|
63
|
+
@if (props.externalLink) {
|
|
64
|
+
<a
|
|
65
|
+
class="docs-code-example__external-link"
|
|
66
|
+
[href]="props.externalLink.url"
|
|
67
|
+
target="_blank"
|
|
68
|
+
rel="noopener noreferrer"
|
|
69
|
+
>
|
|
70
|
+
{{ props.externalLink.label || 'Open in StackBlitz' }}
|
|
71
|
+
<ion-icon name="open-outline"></ion-icon>
|
|
72
|
+
</a>
|
|
73
|
+
}
|
|
74
|
+
</div>
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
<!-- Preview -->
|
|
78
|
+
@if (previewTpl) {
|
|
79
|
+
<div class="docs-code-example__preview">
|
|
80
|
+
<ng-container *ngTemplateOutlet="previewTpl"></ng-container>
|
|
81
|
+
</div>
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
<!-- Tabs -->
|
|
85
|
+
@if (props.tabs?.length) {
|
|
86
|
+
<div class="docs-code-example__tabs">
|
|
87
|
+
<div class="docs-code-example__tab-list" role="tablist">
|
|
88
|
+
@for (tab of props.tabs; track tab.label; let i = $index) {
|
|
89
|
+
<button
|
|
90
|
+
class="docs-code-example__tab"
|
|
91
|
+
[class.docs-code-example__tab--active]="activeTab() === i"
|
|
92
|
+
(click)="activeTab.set(i)"
|
|
93
|
+
role="tab"
|
|
94
|
+
[attr.aria-selected]="activeTab() === i"
|
|
95
|
+
>
|
|
96
|
+
{{ tab.label }}
|
|
97
|
+
</button>
|
|
98
|
+
}
|
|
99
|
+
</div>
|
|
100
|
+
|
|
101
|
+
<!-- Code Panel -->
|
|
102
|
+
<div class="docs-code-example__code-panel">
|
|
103
|
+
<div class="docs-code-example__code-header">
|
|
104
|
+
@if (props.showCopyButton !== false) {
|
|
105
|
+
<ion-button
|
|
106
|
+
fill="clear"
|
|
107
|
+
size="small"
|
|
108
|
+
class="docs-code-example__copy-btn"
|
|
109
|
+
(click)="copyCode()"
|
|
110
|
+
>
|
|
111
|
+
<ion-icon
|
|
112
|
+
slot="icon-only"
|
|
113
|
+
[name]="copied() ? 'checkmark-outline' : 'copy-outline'"
|
|
114
|
+
></ion-icon>
|
|
115
|
+
</ion-button>
|
|
116
|
+
}
|
|
117
|
+
</div>
|
|
118
|
+
<pre
|
|
119
|
+
class="docs-code-example__pre"
|
|
120
|
+
[class.docs-code-example__pre--line-numbers]="props.showLineNumbers"
|
|
121
|
+
><code
|
|
122
|
+
class="docs-code-example__code"
|
|
123
|
+
[class]="'language-' + currentTab().language"
|
|
124
|
+
>{{ currentTab().code }}</code></pre>
|
|
125
|
+
</div>
|
|
126
|
+
</div>
|
|
127
|
+
}
|
|
128
|
+
</div>
|
|
129
|
+
`, isInline: true, styles: [".docs-code-example{margin:1.5rem 0;border:1px solid var(--ion-border-color, #e0e0e0);border-radius:8px;overflow:hidden;&__header{padding:1rem 1.25rem;border-bottom:1px solid var(--ion-border-color, #e0e0e0)}&__title{font-size:1rem;font-weight:600;margin:0 0 .25rem;color:var(--ion-text-color, #1a1a1a)}&__description{font-size:.875rem;color:var(--ion-color-medium, #666);margin:0}&__external-link{display:inline-flex;align-items:center;gap:.25rem;margin-top:.5rem;font-size:.8125rem;color:var(--ion-color-primary, #3880ff);text-decoration:none;&:hover{text-decoration:underline}ion-icon{font-size:.875rem}}&__preview{padding:1.5rem;background:var(--ion-background-color, #fff);border-bottom:1px solid var(--ion-border-color, #e0e0e0)}&__tabs{background:var(--ion-color-light, #f4f5f8)}&__tab-list{display:flex;gap:0;border-bottom:1px solid var(--ion-border-color, #e0e0e0)}&__tab{padding:.75rem 1rem;font-size:.8125rem;font-weight:500;color:var(--ion-color-medium, #666);background:transparent;border:none;cursor:pointer;transition:all .15s ease;border-bottom:2px solid transparent;margin-bottom:-1px;&:hover{color:var(--ion-text-color, #1a1a1a)}&--active{color:var(--ion-color-primary, #3880ff);border-bottom-color:var(--ion-color-primary, #3880ff);background:var(--ion-background-color, #fff)}}&__code-panel{position:relative}&__code-header{position:absolute;top:.5rem;right:.5rem;z-index:1}&__copy-btn{--padding-start: .5rem;--padding-end: .5rem;--color: var(--ion-color-medium);&:hover{--color: var(--ion-color-primary)}}&__pre{margin:0;padding:1rem 1.25rem;overflow-x:auto;background:#1e1e1e;&--line-numbers{padding-left:3rem;counter-reset:line;.docs-code-example__code{counter-increment:line;&:before{content:counter(line);position:absolute;left:1rem;color:#666;font-size:.75rem}}}}&__code{font-family:SF Mono,Monaco,Courier New,monospace;font-size:.8125rem;line-height:1.6;color:#d4d4d4;white-space:pre}}// Dark mode :host-context(.dark),:host-context([color-scheme=\"dark\"]){.docs-code-example{border-color:#3a3a3a;&__header{border-color:#3a3a3a}&__title{color:#f4f5f8}&__preview{border-color:#3a3a3a}&__tabs{background:#0003}&__tab-list{border-color:#3a3a3a}&__tab--active{background:#0000004d}}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }] }); }
|
|
130
|
+
}
|
|
131
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DocsCodeExampleComponent, decorators: [{
|
|
132
|
+
type: Component,
|
|
133
|
+
args: [{ selector: 'val-docs-code-example', standalone: true, imports: [CommonModule, IonIcon, IonButton], template: `
|
|
134
|
+
<div class="docs-code-example" [class]="props.cssClass">
|
|
135
|
+
<!-- Header -->
|
|
136
|
+
@if (props.title || props.description || props.externalLink) {
|
|
137
|
+
<div class="docs-code-example__header">
|
|
138
|
+
@if (props.title) {
|
|
139
|
+
<h4 class="docs-code-example__title">{{ props.title }}</h4>
|
|
140
|
+
}
|
|
141
|
+
@if (props.description) {
|
|
142
|
+
<p class="docs-code-example__description">{{ props.description }}</p>
|
|
143
|
+
}
|
|
144
|
+
@if (props.externalLink) {
|
|
145
|
+
<a
|
|
146
|
+
class="docs-code-example__external-link"
|
|
147
|
+
[href]="props.externalLink.url"
|
|
148
|
+
target="_blank"
|
|
149
|
+
rel="noopener noreferrer"
|
|
150
|
+
>
|
|
151
|
+
{{ props.externalLink.label || 'Open in StackBlitz' }}
|
|
152
|
+
<ion-icon name="open-outline"></ion-icon>
|
|
153
|
+
</a>
|
|
154
|
+
}
|
|
155
|
+
</div>
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
<!-- Preview -->
|
|
159
|
+
@if (previewTpl) {
|
|
160
|
+
<div class="docs-code-example__preview">
|
|
161
|
+
<ng-container *ngTemplateOutlet="previewTpl"></ng-container>
|
|
162
|
+
</div>
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
<!-- Tabs -->
|
|
166
|
+
@if (props.tabs?.length) {
|
|
167
|
+
<div class="docs-code-example__tabs">
|
|
168
|
+
<div class="docs-code-example__tab-list" role="tablist">
|
|
169
|
+
@for (tab of props.tabs; track tab.label; let i = $index) {
|
|
170
|
+
<button
|
|
171
|
+
class="docs-code-example__tab"
|
|
172
|
+
[class.docs-code-example__tab--active]="activeTab() === i"
|
|
173
|
+
(click)="activeTab.set(i)"
|
|
174
|
+
role="tab"
|
|
175
|
+
[attr.aria-selected]="activeTab() === i"
|
|
176
|
+
>
|
|
177
|
+
{{ tab.label }}
|
|
178
|
+
</button>
|
|
179
|
+
}
|
|
180
|
+
</div>
|
|
181
|
+
|
|
182
|
+
<!-- Code Panel -->
|
|
183
|
+
<div class="docs-code-example__code-panel">
|
|
184
|
+
<div class="docs-code-example__code-header">
|
|
185
|
+
@if (props.showCopyButton !== false) {
|
|
186
|
+
<ion-button
|
|
187
|
+
fill="clear"
|
|
188
|
+
size="small"
|
|
189
|
+
class="docs-code-example__copy-btn"
|
|
190
|
+
(click)="copyCode()"
|
|
191
|
+
>
|
|
192
|
+
<ion-icon
|
|
193
|
+
slot="icon-only"
|
|
194
|
+
[name]="copied() ? 'checkmark-outline' : 'copy-outline'"
|
|
195
|
+
></ion-icon>
|
|
196
|
+
</ion-button>
|
|
197
|
+
}
|
|
198
|
+
</div>
|
|
199
|
+
<pre
|
|
200
|
+
class="docs-code-example__pre"
|
|
201
|
+
[class.docs-code-example__pre--line-numbers]="props.showLineNumbers"
|
|
202
|
+
><code
|
|
203
|
+
class="docs-code-example__code"
|
|
204
|
+
[class]="'language-' + currentTab().language"
|
|
205
|
+
>{{ currentTab().code }}</code></pre>
|
|
206
|
+
</div>
|
|
207
|
+
</div>
|
|
208
|
+
}
|
|
209
|
+
</div>
|
|
210
|
+
`, styles: [".docs-code-example{margin:1.5rem 0;border:1px solid var(--ion-border-color, #e0e0e0);border-radius:8px;overflow:hidden;&__header{padding:1rem 1.25rem;border-bottom:1px solid var(--ion-border-color, #e0e0e0)}&__title{font-size:1rem;font-weight:600;margin:0 0 .25rem;color:var(--ion-text-color, #1a1a1a)}&__description{font-size:.875rem;color:var(--ion-color-medium, #666);margin:0}&__external-link{display:inline-flex;align-items:center;gap:.25rem;margin-top:.5rem;font-size:.8125rem;color:var(--ion-color-primary, #3880ff);text-decoration:none;&:hover{text-decoration:underline}ion-icon{font-size:.875rem}}&__preview{padding:1.5rem;background:var(--ion-background-color, #fff);border-bottom:1px solid var(--ion-border-color, #e0e0e0)}&__tabs{background:var(--ion-color-light, #f4f5f8)}&__tab-list{display:flex;gap:0;border-bottom:1px solid var(--ion-border-color, #e0e0e0)}&__tab{padding:.75rem 1rem;font-size:.8125rem;font-weight:500;color:var(--ion-color-medium, #666);background:transparent;border:none;cursor:pointer;transition:all .15s ease;border-bottom:2px solid transparent;margin-bottom:-1px;&:hover{color:var(--ion-text-color, #1a1a1a)}&--active{color:var(--ion-color-primary, #3880ff);border-bottom-color:var(--ion-color-primary, #3880ff);background:var(--ion-background-color, #fff)}}&__code-panel{position:relative}&__code-header{position:absolute;top:.5rem;right:.5rem;z-index:1}&__copy-btn{--padding-start: .5rem;--padding-end: .5rem;--color: var(--ion-color-medium);&:hover{--color: var(--ion-color-primary)}}&__pre{margin:0;padding:1rem 1.25rem;overflow-x:auto;background:#1e1e1e;&--line-numbers{padding-left:3rem;counter-reset:line;.docs-code-example__code{counter-increment:line;&:before{content:counter(line);position:absolute;left:1rem;color:#666;font-size:.75rem}}}}&__code{font-family:SF Mono,Monaco,Courier New,monospace;font-size:.8125rem;line-height:1.6;color:#d4d4d4;white-space:pre}}// Dark mode :host-context(.dark),:host-context([color-scheme=\"dark\"]){.docs-code-example{border-color:#3a3a3a;&__header{border-color:#3a3a3a}&__title{color:#f4f5f8}&__preview{border-color:#3a3a3a}&__tabs{background:#0003}&__tab-list{border-color:#3a3a3a}&__tab--active{background:#0000004d}}}\n"] }]
|
|
211
|
+
}], propDecorators: { props: [{
|
|
212
|
+
type: Input
|
|
213
|
+
}], previewTpl: [{
|
|
214
|
+
type: ContentChild,
|
|
215
|
+
args: ['preview']
|
|
216
|
+
}] } });
|
|
217
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-code-example.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/docs-code-example/docs-code-example.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAe,YAAY,EAAE,MAAM,eAAe,CAAC;AACpF,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AAC/D,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;;;AAG5E,QAAQ,CAAC,EAAE,WAAW,EAAE,gBAAgB,EAAE,WAAW,EAAE,CAAC,CAAC;AAEzD;;;;;;;;;;;;;;;;;;;;;GAqBG;AA4PH,MAAM,OAAO,wBAAwB;IA3PrC;QA4PW,UAAK,GAA4B,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC;QAI7C,cAAS,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QACtB,WAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;KAgBlC;IAdW,UAAU;QAClB,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,IAAI,EAAE,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC;IACxF,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC;QACpC,IAAI,CAAC;YACH,MAAM,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACtB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,IAAI,CAAC,CAAC;QACjD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,CAAC,KAAK,CAAC,sBAAsB,EAAE,GAAG,CAAC,CAAC;QAC7C,CAAC;IACH,CAAC;+GArBU,wBAAwB;mGAAxB,wBAAwB,gNAvPzB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6ET,yuEA9ES,YAAY,sMAAE,OAAO,2JAAE,SAAS;;4FAwP/B,wBAAwB;kBA3PpC,SAAS;+BACE,uBAAuB,cACrB,IAAI,WACP,CAAC,YAAY,EAAE,OAAO,EAAE,SAAS,CAAC,YACjC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6ET;8BA2KQ,KAAK;sBAAb,KAAK;gBAEmB,UAAU;sBAAlC,YAAY;uBAAC,SAAS","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, signal, TemplateRef, ContentChild } from '@angular/core';\nimport { IonIcon, IonButton } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { copyOutline, checkmarkOutline, openOutline } from 'ionicons/icons';\nimport { DocsCodeExampleMetadata, DocsCodeTab } from './types';\n\naddIcons({ copyOutline, checkmarkOutline, openOutline });\n\n/**\n * val-docs-code-example\n *\n * A code example component with tabs, preview, and copy functionality.\n *\n * @example Basic usage\n * ```html\n * <val-docs-code-example\n *   [props]=\"{\n *     title: 'Basic Button',\n *     tabs: [\n *       { label: 'HTML', language: 'html', code: '<val-button>Click me</val-button>' },\n *       { label: 'TypeScript', language: 'typescript', code: 'import { ButtonComponent } from ...' }\n *     ]\n *   }\"\n * >\n *   <ng-template #preview>\n *     <val-button [props]=\"{ label: 'Click me' }\"></val-button>\n *   </ng-template>\n * </val-docs-code-example>\n * ```\n */\n@Component({\n  selector: 'val-docs-code-example',\n  standalone: true,\n  imports: [CommonModule, IonIcon, IonButton],\n  template: `\n    <div class=\"docs-code-example\" [class]=\"props.cssClass\">\n      <!-- Header -->\n      @if (props.title || props.description || props.externalLink) {\n        <div class=\"docs-code-example__header\">\n          @if (props.title) {\n            <h4 class=\"docs-code-example__title\">{{ props.title }}</h4>\n          }\n          @if (props.description) {\n            <p class=\"docs-code-example__description\">{{ props.description }}</p>\n          }\n          @if (props.externalLink) {\n            <a\n              class=\"docs-code-example__external-link\"\n              [href]=\"props.externalLink.url\"\n              target=\"_blank\"\n              rel=\"noopener noreferrer\"\n            >\n              {{ props.externalLink.label || 'Open in StackBlitz' }}\n              <ion-icon name=\"open-outline\"></ion-icon>\n            </a>\n          }\n        </div>\n      }\n\n      <!-- Preview -->\n      @if (previewTpl) {\n        <div class=\"docs-code-example__preview\">\n          <ng-container *ngTemplateOutlet=\"previewTpl\"></ng-container>\n        </div>\n      }\n\n      <!-- Tabs -->\n      @if (props.tabs?.length) {\n        <div class=\"docs-code-example__tabs\">\n          <div class=\"docs-code-example__tab-list\" role=\"tablist\">\n            @for (tab of props.tabs; track tab.label; let i = $index) {\n              <button\n                class=\"docs-code-example__tab\"\n                [class.docs-code-example__tab--active]=\"activeTab() === i\"\n                (click)=\"activeTab.set(i)\"\n                role=\"tab\"\n                [attr.aria-selected]=\"activeTab() === i\"\n              >\n                {{ tab.label }}\n              </button>\n            }\n          </div>\n\n          <!-- Code Panel -->\n          <div class=\"docs-code-example__code-panel\">\n            <div class=\"docs-code-example__code-header\">\n              @if (props.showCopyButton !== false) {\n                <ion-button\n                  fill=\"clear\"\n                  size=\"small\"\n                  class=\"docs-code-example__copy-btn\"\n                  (click)=\"copyCode()\"\n                >\n                  <ion-icon\n                    slot=\"icon-only\"\n                    [name]=\"copied() ? 'checkmark-outline' : 'copy-outline'\"\n                  ></ion-icon>\n                </ion-button>\n              }\n            </div>\n            <pre\n              class=\"docs-code-example__pre\"\n              [class.docs-code-example__pre--line-numbers]=\"props.showLineNumbers\"\n            ><code\n              class=\"docs-code-example__code\"\n              [class]=\"'language-' + currentTab().language\"\n            >{{ currentTab().code }}</code></pre>\n          </div>\n        </div>\n      }\n    </div>\n  `,\n  styles: [`\n    .docs-code-example {\n      margin: 1.5rem 0;\n      border: 1px solid var(--ion-border-color, #e0e0e0);\n      border-radius: 8px;\n      overflow: hidden;\n\n      &__header {\n        padding: 1rem 1.25rem;\n        border-bottom: 1px solid var(--ion-border-color, #e0e0e0);\n      }\n\n      &__title {\n        font-size: 1rem;\n        font-weight: 600;\n        margin: 0 0 0.25rem 0;\n        color: var(--ion-text-color, #1a1a1a);\n      }\n\n      &__description {\n        font-size: 0.875rem;\n        color: var(--ion-color-medium, #666);\n        margin: 0;\n      }\n\n      &__external-link {\n        display: inline-flex;\n        align-items: center;\n        gap: 0.25rem;\n        margin-top: 0.5rem;\n        font-size: 0.8125rem;\n        color: var(--ion-color-primary, #3880ff);\n        text-decoration: none;\n\n        &:hover {\n          text-decoration: underline;\n        }\n\n        ion-icon {\n          font-size: 0.875rem;\n        }\n      }\n\n      &__preview {\n        padding: 1.5rem;\n        background: var(--ion-background-color, #fff);\n        border-bottom: 1px solid var(--ion-border-color, #e0e0e0);\n      }\n\n      &__tabs {\n        background: var(--ion-color-light, #f4f5f8);\n      }\n\n      &__tab-list {\n        display: flex;\n        gap: 0;\n        border-bottom: 1px solid var(--ion-border-color, #e0e0e0);\n      }\n\n      &__tab {\n        padding: 0.75rem 1rem;\n        font-size: 0.8125rem;\n        font-weight: 500;\n        color: var(--ion-color-medium, #666);\n        background: transparent;\n        border: none;\n        cursor: pointer;\n        transition: all 0.15s ease;\n        border-bottom: 2px solid transparent;\n        margin-bottom: -1px;\n\n        &:hover {\n          color: var(--ion-text-color, #1a1a1a);\n        }\n\n        &--active {\n          color: var(--ion-color-primary, #3880ff);\n          border-bottom-color: var(--ion-color-primary, #3880ff);\n          background: var(--ion-background-color, #fff);\n        }\n      }\n\n      &__code-panel {\n        position: relative;\n      }\n\n      &__code-header {\n        position: absolute;\n        top: 0.5rem;\n        right: 0.5rem;\n        z-index: 1;\n      }\n\n      &__copy-btn {\n        --padding-start: 0.5rem;\n        --padding-end: 0.5rem;\n        --color: var(--ion-color-medium);\n\n        &:hover {\n          --color: var(--ion-color-primary);\n        }\n      }\n\n      &__pre {\n        margin: 0;\n        padding: 1rem 1.25rem;\n        overflow-x: auto;\n        background: #1e1e1e;\n\n        &--line-numbers {\n          padding-left: 3rem;\n          counter-reset: line;\n\n          .docs-code-example__code {\n            counter-increment: line;\n\n            &::before {\n              content: counter(line);\n              position: absolute;\n              left: 1rem;\n              color: #666;\n              font-size: 0.75rem;\n            }\n          }\n        }\n      }\n\n      &__code {\n        font-family: 'SF Mono', Monaco, 'Courier New', monospace;\n        font-size: 0.8125rem;\n        line-height: 1.6;\n        color: #d4d4d4;\n        white-space: pre;\n      }\n    }\n\n    // Dark mode\n    :host-context(.dark),\n    :host-context([color-scheme='dark']) {\n      .docs-code-example {\n        border-color: #3a3a3a;\n\n        &__header {\n          border-color: #3a3a3a;\n        }\n\n        &__title {\n          color: #f4f5f8;\n        }\n\n        &__preview {\n          border-color: #3a3a3a;\n        }\n\n        &__tabs {\n          background: rgba(0, 0, 0, 0.2);\n        }\n\n        &__tab-list {\n          border-color: #3a3a3a;\n        }\n\n        &__tab--active {\n          background: rgba(0, 0, 0, 0.3);\n        }\n      }\n    }\n  `],\n})\nexport class DocsCodeExampleComponent {\n  @Input() props: DocsCodeExampleMetadata = { tabs: [] };\n\n  @ContentChild('preview') previewTpl?: TemplateRef<any>;\n\n  protected activeTab = signal(0);\n  protected copied = signal(false);\n\n  protected currentTab(): DocsCodeTab {\n    return this.props.tabs[this.activeTab()] || { label: '', code: '', language: 'text' };\n  }\n\n  async copyCode(): Promise<void> {\n    const code = this.currentTab().code;\n    try {\n      await navigator.clipboard.writeText(code);\n      this.copied.set(true);\n      setTimeout(() => this.copied.set(false), 2000);\n    } catch (err) {\n      console.error('Failed to copy code:', err);\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2RvY3MtY29kZS1leGFtcGxlL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUZW1wbGF0ZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG4vKipcbiAqIENvbmZpZ3VyYXRpb24gZm9yIHRoZSBkb2NzLWNvZGUtZXhhbXBsZSBjb21wb25lbnQuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRG9jc0NvZGVFeGFtcGxlTWV0YWRhdGEge1xuICAvKipcbiAgICogRXhhbXBsZSB0aXRsZS5cbiAgICovXG4gIHRpdGxlPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBFeGFtcGxlIGRlc2NyaXB0aW9uLlxuICAgKi9cbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIENvZGUgdGFicyB0byBkaXNwbGF5LlxuICAgKi9cbiAgdGFiczogRG9jc0NvZGVUYWJbXTtcblxuICAvKipcbiAgICogVGVtcGxhdGUgZm9yIGxpdmUgcHJldmlldy5cbiAgICogVXNlIG5nLXRlbXBsYXRlIHdpdGggI3ByZXZpZXcgcmVmZXJlbmNlLlxuICAgKi9cbiAgcHJldmlld1RlbXBsYXRlPzogVGVtcGxhdGVSZWY8YW55PjtcblxuICAvKipcbiAgICogRXh0ZXJuYWwgbGluayAoZS5nLiwgU3RhY2tCbGl0eiwgQ29kZVNhbmRib3gpLlxuICAgKi9cbiAgZXh0ZXJuYWxMaW5rPzoge1xuICAgIHVybDogc3RyaW5nO1xuICAgIGxhYmVsPzogc3RyaW5nO1xuICB9O1xuXG4gIC8qKlxuICAgKiBTaG93IGNvcHkgYnV0dG9uIG9uIGNvZGUgYmxvY2tzLlxuICAgKiBAZGVmYXVsdCB0cnVlXG4gICAqL1xuICBzaG93Q29weUJ1dHRvbj86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIFNob3cgbGluZSBudW1iZXJzIGluIGNvZGUuXG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqL1xuICBzaG93TGluZU51bWJlcnM/OiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBDdXN0b20gQ1NTIGNsYXNzLlxuICAgKi9cbiAgY3NzQ2xhc3M/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIERlZmF1bHQgYWN0aXZlIHRhYiBpbmRleC5cbiAgICogQGRlZmF1bHQgMFxuICAgKi9cbiAgZGVmYXVsdFRhYj86IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBEb2NzQ29kZVRhYiB7XG4gIC8qKlxuICAgKiBUYWIgbGFiZWwgKGUuZy4sIFwiSFRNTFwiLCBcIlR5cGVTY3JpcHRcIiwgXCJTQ1NTXCIpLlxuICAgKi9cbiAgbGFiZWw6IHN0cmluZztcblxuICAvKipcbiAgICogQ29kZSBjb250ZW50LlxuICAgKi9cbiAgY29kZTogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBMYW5ndWFnZSBmb3Igc3ludGF4IGhpZ2hsaWdodGluZy5cbiAgICovXG4gIGxhbmd1YWdlOiAnaHRtbCcgfCAndHlwZXNjcmlwdCcgfCAnc2NzcycgfCAnY3NzJyB8ICdqc29uJyB8ICdiYXNoJyB8ICd0ZXh0Jztcbn1cbiJdfQ==
|