@ojiepermana/angular 21.0.0 → 21.0.3

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 (47) hide show
  1. package/collection.json +25 -0
  2. package/fesm2022/ojiepermana-angular-generator-api.mjs +67 -0
  3. package/fesm2022/ojiepermana-angular-generator-api.mjs.map +1 -0
  4. package/fesm2022/ojiepermana-angular-layout.mjs +76 -56
  5. package/fesm2022/ojiepermana-angular-layout.mjs.map +1 -1
  6. package/fesm2022/ojiepermana-angular.mjs +1 -0
  7. package/fesm2022/ojiepermana-angular.mjs.map +1 -1
  8. package/generator/api/README.md +218 -0
  9. package/generator/api/bin/schematics/init/index.js +88 -0
  10. package/generator/api/bin/schematics/sdk/index.js +58 -0
  11. package/generator/api/bin/src/config/loader.js +41 -0
  12. package/generator/api/bin/src/config/schema.js +56 -0
  13. package/generator/api/bin/src/emit/client.js +246 -0
  14. package/generator/api/bin/src/emit/metadata.js +295 -0
  15. package/generator/api/bin/src/emit/models.js +106 -0
  16. package/generator/api/bin/src/emit/navigation.js +56 -0
  17. package/generator/api/bin/src/emit/operations.js +122 -0
  18. package/generator/api/bin/src/emit/public-api.js +54 -0
  19. package/generator/api/bin/src/emit/services.js +87 -0
  20. package/generator/api/bin/src/engine.js +65 -0
  21. package/generator/api/bin/src/layout/per-domain.js +346 -0
  22. package/generator/api/bin/src/parser/bundle.js +25 -0
  23. package/generator/api/bin/src/parser/ir.js +320 -0
  24. package/generator/api/bin/src/parser/types.js +7 -0
  25. package/generator/api/bin/src/render/template.js +58 -0
  26. package/generator/api/bin/src/writer/index.js +69 -0
  27. package/generator/api/schematics/init/schema.json +19 -0
  28. package/generator/api/schematics/sdk/schema.json +19 -0
  29. package/generator/api/sdk.config.example.json +22 -0
  30. package/generator/guide/README.md +84 -0
  31. package/generator/guide/bin/schematics/build/index.js +35 -0
  32. package/generator/guide/bin/schematics/init/index.js +70 -0
  33. package/generator/guide/bin/src/config/loader.js +50 -0
  34. package/generator/guide/bin/src/config/schema.js +12 -0
  35. package/generator/guide/bin/src/engine/component.js +73 -0
  36. package/generator/guide/bin/src/engine/frontmatter.js +42 -0
  37. package/generator/guide/bin/src/engine/index.js +42 -0
  38. package/generator/guide/bin/src/engine/naming.js +39 -0
  39. package/generator/guide/bin/src/engine/render.js +18 -0
  40. package/generator/guide/bin/src/engine/routes.js +106 -0
  41. package/generator/guide/bin/src/engine/walk.js +35 -0
  42. package/generator/guide/guide.config.example.json +9 -0
  43. package/generator/guide/schematics/build/schema.json +14 -0
  44. package/generator/guide/schematics/init/schema.json +19 -0
  45. package/package.json +10 -3
  46. package/types/ojiepermana-angular-generator-api.d.ts +85 -0
  47. package/types/ojiepermana-angular-layout.d.ts +2 -0
@@ -0,0 +1,246 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.emitClient = emitClient;
4
+ const template_1 = require("../render/template");
5
+ /**
6
+ * Emit the tiny runtime client primitives:
7
+ * - `api-configuration.ts` — root-url config + provider
8
+ * - `base-service.ts` — per-service rootUrl override
9
+ * - `strict-http-response.ts`— HttpResponse alias
10
+ * - `request-builder.ts` — minimal (path / query / header / body / build)
11
+ * - `api.ts` — `Api` helper with invoke() / invoke$Response()
12
+ *
13
+ * These files are **static** — identical across specs — so we embed them as
14
+ * template literals rather than parsing the IR. The only dynamic bits are the
15
+ * banner comment and the configured default `rootUrl`.
16
+ */
17
+ function emitClient(target) {
18
+ const defaultRootUrl = target.rootUrl ?? '';
19
+ return [
20
+ { path: 'api-configuration.ts', content: (0, template_1.finalize)(apiConfigurationFile(target, defaultRootUrl)) },
21
+ { path: 'base-service.ts', content: (0, template_1.finalize)(baseServiceFile(target)) },
22
+ { path: 'strict-http-response.ts', content: (0, template_1.finalize)(strictResponseFile(target)) },
23
+ { path: 'request-builder.ts', content: (0, template_1.finalize)(requestBuilderFile(target)) },
24
+ { path: 'api.ts', content: (0, template_1.finalize)(apiFile(target)) },
25
+ ];
26
+ }
27
+ function apiConfigurationFile(target, rootUrl) {
28
+ return `${target.banner}
29
+
30
+ import { Injectable, makeEnvironmentProviders, type EnvironmentProviders } from '@angular/core';
31
+
32
+ @Injectable({ providedIn: 'root' })
33
+ export class ApiConfiguration {
34
+ rootUrl: string = ${JSON.stringify(rootUrl)};
35
+ }
36
+
37
+ export function provideApiConfiguration(rootUrl: string): EnvironmentProviders {
38
+ return makeEnvironmentProviders([
39
+ {
40
+ provide: ApiConfiguration,
41
+ useFactory: () => {
42
+ const c = new ApiConfiguration();
43
+ c.rootUrl = rootUrl;
44
+ return c;
45
+ },
46
+ },
47
+ ]);
48
+ }
49
+ `;
50
+ }
51
+ function baseServiceFile(target) {
52
+ return `${target.banner}
53
+
54
+ import { HttpClient } from '@angular/common/http';
55
+ import { Injectable, inject } from '@angular/core';
56
+
57
+ import { ApiConfiguration } from './api-configuration';
58
+
59
+ @Injectable()
60
+ export class BaseService {
61
+ protected readonly config = inject(ApiConfiguration);
62
+ protected readonly http = inject(HttpClient);
63
+
64
+ private _rootUrl?: string;
65
+
66
+ get rootUrl(): string {
67
+ return this._rootUrl ?? this.config.rootUrl;
68
+ }
69
+ set rootUrl(url: string) {
70
+ this._rootUrl = url;
71
+ }
72
+ }
73
+ `;
74
+ }
75
+ function strictResponseFile(target) {
76
+ return `${target.banner}
77
+
78
+ import { HttpResponse } from '@angular/common/http';
79
+
80
+ export type StrictHttpResponse<T> = HttpResponse<T> & { readonly body: T };
81
+ `;
82
+ }
83
+ /**
84
+ * Minimal request builder — intentionally drops the style/explode serializer
85
+ * from ng-openapi-gen's output. Covers the 99% case: path templating, simple
86
+ * query params (with array → repeated key), JSON body, and header params.
87
+ */
88
+ function requestBuilderFile(target) {
89
+ return `${target.banner}
90
+
91
+ import {
92
+ HttpContext,
93
+ HttpHeaders,
94
+ HttpParams,
95
+ HttpRequest,
96
+ } from '@angular/common/http';
97
+
98
+ export interface BuildOptions {
99
+ accept?: string;
100
+ responseType?: 'json' | 'text' | 'blob' | 'arraybuffer';
101
+ reportProgress?: boolean;
102
+ context?: HttpContext;
103
+ }
104
+
105
+ export class RequestBuilder {
106
+ private readonly _pathParams = new Map<string, unknown>();
107
+ private readonly _queryParams = new Map<string, unknown>();
108
+ private readonly _headerParams = new Map<string, unknown>();
109
+ private _body: unknown = undefined;
110
+ private _bodyContentType: string | undefined;
111
+
112
+ constructor(
113
+ public readonly rootUrl: string,
114
+ public readonly operationPath: string,
115
+ public readonly method: string,
116
+ ) {}
117
+
118
+ path(name: string, value: unknown): void {
119
+ this._pathParams.set(name, value);
120
+ }
121
+
122
+ query(name: string, value: unknown): void {
123
+ this._queryParams.set(name, value);
124
+ }
125
+
126
+ header(name: string, value: unknown): void {
127
+ this._headerParams.set(name, value);
128
+ }
129
+
130
+ body(value: unknown, contentType = 'application/json'): void {
131
+ if (value instanceof Blob) {
132
+ this._bodyContentType = value.type;
133
+ } else if (value instanceof FormData) {
134
+ // FormData sets its own Content-Type (with boundary) — leave it alone.
135
+ this._bodyContentType = undefined;
136
+ } else {
137
+ this._bodyContentType = contentType;
138
+ }
139
+ this._body = value;
140
+ }
141
+
142
+ build<T = unknown>(options: BuildOptions = {}): HttpRequest<T> {
143
+ let path = this.operationPath;
144
+ for (const [name, value] of this._pathParams) {
145
+ path = path.replace(
146
+ '{' + name + '}',
147
+ encodeURIComponent(value == null ? '' : String(value)),
148
+ );
149
+ }
150
+
151
+ let params = new HttpParams();
152
+ for (const [name, value] of this._queryParams) {
153
+ if (value === undefined || value === null) continue;
154
+ if (Array.isArray(value)) {
155
+ for (const v of value) {
156
+ if (v !== undefined && v !== null) params = params.append(name, String(v));
157
+ }
158
+ } else {
159
+ params = params.append(name, String(value));
160
+ }
161
+ }
162
+
163
+ let headers = new HttpHeaders();
164
+ if (options.accept) headers = headers.set('Accept', options.accept);
165
+ for (const [name, value] of this._headerParams) {
166
+ if (value === undefined || value === null) continue;
167
+ if (Array.isArray(value)) {
168
+ for (const v of value) headers = headers.append(name, String(v));
169
+ } else {
170
+ headers = headers.set(name, String(value));
171
+ }
172
+ }
173
+ if (this._bodyContentType) {
174
+ headers = headers.set('Content-Type', this._bodyContentType);
175
+ }
176
+
177
+ return new HttpRequest<T>(
178
+ this.method.toUpperCase(),
179
+ this.rootUrl + path,
180
+ this._body as T,
181
+ {
182
+ params,
183
+ headers,
184
+ responseType: options.responseType,
185
+ reportProgress: options.reportProgress,
186
+ context: options.context,
187
+ },
188
+ );
189
+ }
190
+ }
191
+ `;
192
+ }
193
+ function apiFile(target) {
194
+ return `${target.banner}
195
+
196
+ import { HttpClient, HttpContext, HttpResponse } from '@angular/common/http';
197
+ import { Injectable, inject } from '@angular/core';
198
+ import { Observable } from 'rxjs';
199
+ import { filter, map } from 'rxjs/operators';
200
+
201
+ import { ApiConfiguration } from './api-configuration';
202
+ import { StrictHttpResponse } from './strict-http-response';
203
+
204
+ export type ApiFn<P, R> = (
205
+ http: HttpClient,
206
+ rootUrl: string,
207
+ params: P,
208
+ context?: HttpContext,
209
+ ) => Observable<StrictHttpResponse<R>>;
210
+
211
+ /**
212
+ * Helper to invoke any tree-shakeable operation function directly.
213
+ */
214
+ @Injectable({ providedIn: 'root' })
215
+ export class ${target.clientName} {
216
+ private readonly config = inject(ApiConfiguration);
217
+ private readonly http = inject(HttpClient);
218
+
219
+ private _rootUrl?: string;
220
+
221
+ get rootUrl(): string {
222
+ return this._rootUrl ?? this.config.rootUrl;
223
+ }
224
+ set rootUrl(url: string) {
225
+ this._rootUrl = url;
226
+ }
227
+
228
+ /** Invoke an operation function and stream the response body. */
229
+ invoke<P, R>(fn: ApiFn<P, R>, params: P, context?: HttpContext): Observable<R> {
230
+ return this.invoke$Response(fn, params, context).pipe(map((r) => r.body));
231
+ }
232
+
233
+ /** Invoke an operation function and stream the full HTTP response. */
234
+ invoke$Response<P, R>(
235
+ fn: ApiFn<P, R>,
236
+ params: P,
237
+ context?: HttpContext,
238
+ ): Observable<StrictHttpResponse<R>> {
239
+ return fn(this.http, this.rootUrl, params, context).pipe(
240
+ filter((r: unknown): r is HttpResponse<unknown> => r instanceof HttpResponse),
241
+ map((r) => r as StrictHttpResponse<R>),
242
+ );
243
+ }
244
+ }
245
+ `;
246
+ }
@@ -0,0 +1,295 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.emitMetadata = emitMetadata;
4
+ const template_1 = require("../render/template");
5
+ /**
6
+ * Emit frontend-friendly metadata:
7
+ * - `metadata-types.ts` shared rule types
8
+ * - `permissions/<tag>.ts` operation rules (per tag) + `index.ts`
9
+ * - `validators/<tag>.ts` schema validation rules (per tag) + `index.ts`
10
+ * - `metadata.ts` aggregate export
11
+ * - `openapi-helpers.ts` small helpers (`getRequiredPermissions`, etc)
12
+ */
13
+ function emitMetadata(ir, target) {
14
+ const files = [];
15
+ files.push({ path: 'metadata-types.ts', content: (0, template_1.finalize)(metadataTypesFile(target)) });
16
+ const opsByTag = groupOps(ir.operations);
17
+ for (const [tag, ops] of opsByTag) {
18
+ files.push({
19
+ path: `permissions/${(0, template_1.kebabCase)(tag)}.ts`,
20
+ content: (0, template_1.finalize)(permissionFile(tag, ops, target)),
21
+ });
22
+ }
23
+ const permTags = Array.from(opsByTag.keys());
24
+ files.push({
25
+ path: 'permissions/index.ts',
26
+ content: (0, template_1.finalize)(buildAggregateFile(target, permTags, 'ApiOperationRule', 'apiOperationRules', (t) => `${toCamel(t)}OperationRules`)),
27
+ });
28
+ const schemasByPrefix = groupSchemas(ir.schemas);
29
+ for (const [group, schemas] of schemasByPrefix) {
30
+ files.push({
31
+ path: `validators/${(0, template_1.kebabCase)(group)}.ts`,
32
+ content: (0, template_1.finalize)(validatorFile(group, schemas, target)),
33
+ });
34
+ }
35
+ const valGroups = Array.from(schemasByPrefix.keys());
36
+ files.push({
37
+ path: 'validators/index.ts',
38
+ content: (0, template_1.finalize)(buildAggregateFile(target, valGroups, 'ApiSchemaRule', 'apiValidationSchemas', (g) => `${toCamel(g)}ValidationSchemas`)),
39
+ });
40
+ files.push({ path: 'metadata.ts', content: (0, template_1.finalize)(metadataFile(target)) });
41
+ files.push({ path: 'openapi-helpers.ts', content: (0, template_1.finalize)(helpersFile(target)) });
42
+ return files;
43
+ }
44
+ /**
45
+ * Emit a per-feature `index.ts` that re-exports each per-tag module plus a
46
+ * single aggregated record (`apiOperationRules`, `apiValidationSchemas`).
47
+ */
48
+ function buildAggregateFile(target, groups, typeName, aggregateName, varNameFor) {
49
+ const reexports = groups.map((g) => `export { ${varNameFor(g)} } from './${(0, template_1.kebabCase)(g)}';`).join('\n');
50
+ const imports = groups.map((g) => `import { ${varNameFor(g)} } from './${(0, template_1.kebabCase)(g)}';`).join('\n');
51
+ const spreads = groups.map((g) => ` ...${varNameFor(g)},`).join('\n');
52
+ return `${target.banner}
53
+
54
+ import type { ${typeName} } from '../metadata-types';
55
+ ${imports}
56
+
57
+ ${reexports}
58
+
59
+ export const ${aggregateName}: Record<string, ${typeName}> = {
60
+ ${spreads}
61
+ };
62
+ `;
63
+ }
64
+ function metadataTypesFile(target) {
65
+ return `${target.banner}
66
+
67
+ export interface ApiFieldRule {
68
+ required?: boolean;
69
+ type?: string;
70
+ format?: string;
71
+ description?: string;
72
+ nullable?: boolean;
73
+ minLength?: number;
74
+ maxLength?: number;
75
+ minimum?: number;
76
+ maximum?: number;
77
+ pattern?: string;
78
+ enumValues?: readonly (string | number)[];
79
+ ref?: string;
80
+ itemsRef?: string;
81
+ itemsType?: string;
82
+ }
83
+
84
+ export interface ApiSchemaRule {
85
+ required: readonly string[];
86
+ properties: Record<string, ApiFieldRule>;
87
+ }
88
+
89
+ export interface ApiOperationRule {
90
+ method: string;
91
+ path: string;
92
+ tags: readonly string[];
93
+ summary?: string;
94
+ description?: string;
95
+ requiredPermissions: readonly string[];
96
+ authorizationNotes: readonly string[];
97
+ securitySchemes: readonly string[];
98
+ pathParams: Record<string, ApiFieldRule>;
99
+ queryParams: Record<string, ApiFieldRule>;
100
+ bodySchema?: string;
101
+ responseSchemas: Record<string, string | undefined>;
102
+ }
103
+ `;
104
+ }
105
+ function permissionFile(tag, ops, target) {
106
+ const entries = ops
107
+ .slice()
108
+ .sort((a, b) => a.operationId.localeCompare(b.operationId))
109
+ .map((op) => renderOperationRule(op));
110
+ return `${target.banner}
111
+
112
+ import type { ApiOperationRule } from '../metadata-types';
113
+
114
+ export const ${toCamel(tag)}OperationRules: Record<string, ApiOperationRule> = {
115
+ ${entries.join(',\n')},
116
+ };
117
+ `;
118
+ }
119
+ function renderOperationRule(op) {
120
+ const pathParams = {};
121
+ const queryParams = {};
122
+ for (const p of op.params) {
123
+ const rule = trimRule(p.rule, p.required);
124
+ if (p.in === 'path')
125
+ pathParams[p.name] = rule;
126
+ else if (p.in === 'query')
127
+ queryParams[p.name] = rule;
128
+ }
129
+ const obj = {
130
+ method: op.method.toUpperCase(),
131
+ path: op.path,
132
+ tags: op.tags,
133
+ summary: op.summary,
134
+ description: op.description,
135
+ requiredPermissions: op.requiredPermissions,
136
+ authorizationNotes: op.authorizationNotes,
137
+ securitySchemes: op.securitySchemes,
138
+ pathParams,
139
+ queryParams,
140
+ bodySchema: op.bodySchemaRef,
141
+ responseSchemas: op.responses,
142
+ };
143
+ return ` ${JSON.stringify(op.operationId)}: ${toInlineJson(obj, 2)}`;
144
+ }
145
+ function validatorFile(group, schemas, target) {
146
+ const entries = schemas
147
+ .slice()
148
+ .sort((a, b) => a.name.localeCompare(b.name))
149
+ .map((s) => renderSchemaRule(s));
150
+ return `${target.banner}
151
+
152
+ import type { ApiSchemaRule } from '../metadata-types';
153
+
154
+ export const ${toCamel(group)}ValidationSchemas: Record<string, ApiSchemaRule> = {
155
+ ${entries.join(',\n')},
156
+ };
157
+ `;
158
+ }
159
+ function renderSchemaRule(schema) {
160
+ const properties = {};
161
+ for (const [name, rule] of Object.entries(schema.properties)) {
162
+ properties[name] = trimRule(rule, rule.required === true);
163
+ }
164
+ const obj = {
165
+ required: schema.required,
166
+ properties,
167
+ };
168
+ return ` ${JSON.stringify(schema.name)}: ${toInlineJson(obj, 2)}`;
169
+ }
170
+ function metadataFile(target) {
171
+ return `${target.banner}
172
+
173
+ export * from './metadata-types';
174
+ export * from './permissions';
175
+ export * from './validators';
176
+ `;
177
+ }
178
+ function helpersFile(target) {
179
+ return `${target.banner}
180
+
181
+ import { apiOperationRules, apiValidationSchemas } from './metadata';
182
+ import type { ApiFieldRule, ApiOperationRule, ApiSchemaRule } from './metadata-types';
183
+
184
+ export function getSchemaRule(schemaName: string): ApiSchemaRule | undefined {
185
+ return apiValidationSchemas[schemaName];
186
+ }
187
+
188
+ export function getFieldRule(schemaName: string, fieldName: string): ApiFieldRule | undefined {
189
+ return apiValidationSchemas[schemaName]?.properties[fieldName];
190
+ }
191
+
192
+ export function getOperationRule(operationId: string): ApiOperationRule | undefined {
193
+ return apiOperationRules[operationId];
194
+ }
195
+
196
+ export function getRequiredPermissions(operationId: string): readonly string[] {
197
+ return apiOperationRules[operationId]?.requiredPermissions ?? [];
198
+ }
199
+
200
+ export function hasRequiredPermissions(
201
+ operationId: string,
202
+ ownedPermissions: readonly string[],
203
+ ): boolean {
204
+ return getRequiredPermissions(operationId).every((p) => ownedPermissions.includes(p));
205
+ }
206
+ `;
207
+ }
208
+ // ---------------------------------------------------------------------------
209
+ // Helpers
210
+ // ---------------------------------------------------------------------------
211
+ function groupOps(operations) {
212
+ const map = new Map();
213
+ for (const op of operations) {
214
+ const bucket = map.get(op.tag) ?? [];
215
+ bucket.push(op);
216
+ map.set(op.tag, bucket);
217
+ }
218
+ return map;
219
+ }
220
+ /**
221
+ * Group schemas for validator output by a common prefix tag. We use a simple
222
+ * heuristic: first PascalCase word of the schema name (e.g. `UserListResponse`
223
+ * → `User`, `AuthTokenResponse` → `Auth`). Keeps file sizes small and mirrors
224
+ * the reference layout.
225
+ */
226
+ function groupSchemas(schemas) {
227
+ const map = new Map();
228
+ for (const s of schemas) {
229
+ const prefix = detectSchemaGroup(s.name);
230
+ const bucket = map.get(prefix) ?? [];
231
+ bucket.push(s);
232
+ map.set(prefix, bucket);
233
+ }
234
+ return map;
235
+ }
236
+ function detectSchemaGroup(name) {
237
+ const match = name.match(/^([A-Z][a-z]+)/);
238
+ return match ? match[1] : 'Shared';
239
+ }
240
+ function trimRule(rule, required) {
241
+ const out = {};
242
+ if (required)
243
+ out.required = true;
244
+ else
245
+ out.required = false;
246
+ if (rule.type)
247
+ out.type = rule.type;
248
+ if (rule.format)
249
+ out.format = rule.format;
250
+ if (rule.description)
251
+ out.description = rule.description;
252
+ if (rule.nullable)
253
+ out.nullable = true;
254
+ if (typeof rule.minLength === 'number')
255
+ out.minLength = rule.minLength;
256
+ if (typeof rule.maxLength === 'number')
257
+ out.maxLength = rule.maxLength;
258
+ if (typeof rule.minimum === 'number')
259
+ out.minimum = rule.minimum;
260
+ if (typeof rule.maximum === 'number')
261
+ out.maximum = rule.maximum;
262
+ if (rule.pattern)
263
+ out.pattern = rule.pattern;
264
+ if (rule.enumValues && rule.enumValues.length)
265
+ out.enumValues = rule.enumValues;
266
+ if (rule.ref)
267
+ out.ref = rule.ref;
268
+ if (rule.itemsRef)
269
+ out.itemsRef = rule.itemsRef;
270
+ if (rule.itemsType)
271
+ out.itemsType = rule.itemsType;
272
+ return out;
273
+ }
274
+ /** Stable JSON output with stripped `undefined` values. */
275
+ function toInlineJson(value, indent) {
276
+ const pretty = JSON.stringify(value, (_key, val) => (val === undefined ? undefined : val), 2);
277
+ if (!pretty)
278
+ return 'undefined';
279
+ return pretty
280
+ .split('\n')
281
+ .map((line, i) => (i === 0 ? line : ' '.repeat(indent) + line))
282
+ .join('\n');
283
+ }
284
+ function toCamel(input) {
285
+ const parts = String(input)
286
+ .replace(/[^a-zA-Z0-9]+/g, ' ')
287
+ .trim()
288
+ .split(/\s+/)
289
+ .filter(Boolean);
290
+ if (!parts.length)
291
+ return 'shared';
292
+ return parts
293
+ .map((w, i) => (i === 0 ? w[0].toLowerCase() + w.slice(1) : w[0].toUpperCase() + w.slice(1).toLowerCase()))
294
+ .join('');
295
+ }
@@ -0,0 +1,106 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.emitModels = emitModels;
4
+ exports.renderFieldType = renderFieldType;
5
+ const template_1 = require("../render/template");
6
+ /**
7
+ * Emit one `interface` per schema into `models/<kebab>.ts` — type-only, no
8
+ * runtime footprint.
9
+ */
10
+ function emitModels(ir, target) {
11
+ const files = [];
12
+ for (const schema of ir.schemas) {
13
+ const fileName = `${(0, template_1.kebabCase)(schema.name)}.ts`;
14
+ files.push({
15
+ path: `models/${fileName}`,
16
+ content: (0, template_1.finalize)(renderModel(schema, ir, target)),
17
+ });
18
+ }
19
+ return files;
20
+ }
21
+ function renderModel(schema, ir, target) {
22
+ const imports = collectModelImports(schema).filter((name) => name !== schema.name);
23
+ const importLines = imports.map((name) => `import type { ${name} } from './${(0, template_1.kebabCase)(name)}';`);
24
+ let body = '';
25
+ if (schema.enumValues) {
26
+ // Primitive enum alias.
27
+ body = `export type ${(0, template_1.pascalCase)(schema.name)} = ${schema.enumValues
28
+ .map((v) => (typeof v === 'string' ? `'${escapeSingle(v)}'` : String(v)))
29
+ .join(' | ')};`;
30
+ }
31
+ else if (schema.arrayItemRef || schema.arrayItemType) {
32
+ const itemType = schema.arrayItemRef ? (0, template_1.pascalCase)(schema.arrayItemRef) : tsPrimitive(schema.arrayItemType);
33
+ body = `export type ${(0, template_1.pascalCase)(schema.name)} = ${itemType}[];`;
34
+ }
35
+ else {
36
+ body = renderInterface(schema);
37
+ }
38
+ return [target.banner, '', ...importLines, importLines.length ? '' : '', body].join('\n');
39
+ }
40
+ function renderInterface(schema) {
41
+ const lines = [];
42
+ lines.push(`export interface ${(0, template_1.pascalCase)(schema.name)} {`);
43
+ for (const [propName, rule] of Object.entries(schema.properties)) {
44
+ const optional = rule.required ? '' : '?';
45
+ const type = renderFieldType(rule);
46
+ const safeName = /^[A-Za-z_$][A-Za-z0-9_$]*$/.test(propName) ? propName : `'${propName}'`;
47
+ lines.push(` ${safeName}${optional}: ${type};`);
48
+ }
49
+ lines.push(`}`);
50
+ return lines.join('\n');
51
+ }
52
+ function renderFieldType(rule) {
53
+ let base;
54
+ if (rule.ref) {
55
+ base = (0, template_1.pascalCase)(rule.ref);
56
+ }
57
+ else if (rule.type === 'array') {
58
+ const item = rule.itemsRef
59
+ ? (0, template_1.pascalCase)(rule.itemsRef)
60
+ : rule.enumValues
61
+ ? rule.enumValues.map((v) => (typeof v === 'string' ? `'${escapeSingle(v)}'` : String(v))).join(' | ')
62
+ : tsPrimitive(rule.itemsType);
63
+ base = rule.enumValues && !rule.itemsRef ? `(${item})[]` : `${item}[]`;
64
+ }
65
+ else if (rule.enumValues && rule.enumValues.length > 0) {
66
+ base = rule.enumValues.map((v) => (typeof v === 'string' ? `'${escapeSingle(v)}'` : String(v))).join(' | ');
67
+ }
68
+ else {
69
+ base = tsPrimitive(rule.type);
70
+ }
71
+ return rule.nullable ? `${base} | null` : base;
72
+ }
73
+ function tsPrimitive(type) {
74
+ switch (type) {
75
+ case 'string':
76
+ return 'string';
77
+ case 'integer':
78
+ case 'number':
79
+ return 'number';
80
+ case 'boolean':
81
+ return 'boolean';
82
+ case 'array':
83
+ return 'unknown[]';
84
+ case 'object':
85
+ return 'Record<string, unknown>';
86
+ case undefined:
87
+ case 'unknown':
88
+ default:
89
+ return 'unknown';
90
+ }
91
+ }
92
+ function escapeSingle(value) {
93
+ return value.replace(/\\/g, '\\\\').replace(/'/g, "\\'");
94
+ }
95
+ function collectModelImports(schema) {
96
+ const set = new Set();
97
+ if (schema.arrayItemRef)
98
+ set.add(schema.arrayItemRef);
99
+ for (const rule of Object.values(schema.properties)) {
100
+ if (rule.ref)
101
+ set.add(rule.ref);
102
+ if (rule.itemsRef)
103
+ set.add(rule.itemsRef);
104
+ }
105
+ return [...set].sort();
106
+ }
@@ -0,0 +1,56 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.emitNavigation = emitNavigation;
4
+ const template_1 = require("../render/template");
5
+ /**
6
+ * Emit `api.navigation.ts` exporting `NavigationItem[]` data that plugs
7
+ * directly into `NavigationService.registerItems(...)` from
8
+ * `@ojiepermana/angular/navigation`.
9
+ */
10
+ function emitNavigation(ir, target) {
11
+ return [
12
+ {
13
+ path: 'api.navigation.ts',
14
+ content: (0, template_1.finalize)(renderNavigation(ir.navigation, target)),
15
+ },
16
+ ];
17
+ }
18
+ function renderNavigation(nodes, target) {
19
+ return `${target.banner}
20
+
21
+ import type { NavigationItem } from '@ojiepermana/angular/navigation';
22
+
23
+ export const ApiNavigation: NavigationItem[] = ${stringifyNodes(nodes, 0, [])};
24
+ `;
25
+ }
26
+ function stringifyNodes(nodes, depth, lineage) {
27
+ if (!nodes.length)
28
+ return '[]';
29
+ const pad = ' '.repeat(depth + 1);
30
+ const closePad = ' '.repeat(depth);
31
+ return `[\n${nodes
32
+ .map((node) => pad + stringifyNode(node, depth + 1, [...lineage, node.name]))
33
+ .join(',\n')},\n${closePad}]`;
34
+ }
35
+ function stringifyNode(node, depth, lineage) {
36
+ const hasChildren = node.children.length > 0;
37
+ const pad = ' '.repeat(depth + 1);
38
+ const closePad = ' '.repeat(depth);
39
+ const fields = [
40
+ `${pad}id: ${JSON.stringify(lineage.map(template_1.kebabCase).filter(Boolean).join('-'))}`,
41
+ `${pad}title: ${JSON.stringify(node.name)}`,
42
+ `${pad}type: ${JSON.stringify(resolveItemType(lineage.length - 1, hasChildren))}`,
43
+ ];
44
+ if (node.description)
45
+ fields.push(`${pad}subtitle: ${JSON.stringify(node.description)}`);
46
+ if (node.xIcon)
47
+ fields.push(`${pad}icon: ${JSON.stringify(node.xIcon)}`);
48
+ if (hasChildren)
49
+ fields.push(`${pad}children: ${stringifyNodes(node.children, depth + 1, lineage)}`);
50
+ return `{\n${fields.join(',\n')},\n${closePad}}`;
51
+ }
52
+ function resolveItemType(depth, hasChildren) {
53
+ if (!hasChildren)
54
+ return 'basic';
55
+ return depth === 0 ? 'group' : 'collapsable';
56
+ }