@decaf-ts/ui-decorators 0.5.21 → 0.5.23
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/LICENSE.md +0 -53
- package/README.md +1 -1
- package/dist/ui-decorators.cjs +50 -30
- package/dist/ui-decorators.esm.cjs +50 -30
- package/lib/esm/index.d.ts +1 -1
- package/lib/esm/index.js +1 -1
- package/lib/esm/ui/Rendering.js +39 -26
- package/lib/esm/ui/types.d.ts +2 -1
- package/lib/esm/ui/types.js +1 -1
- package/lib/esm/ui/utils.js +11 -4
- package/lib/index.cjs +1 -1
- package/lib/index.d.ts +1 -1
- package/lib/ui/Rendering.cjs +39 -26
- package/lib/ui/types.cjs +1 -1
- package/lib/ui/types.d.ts +2 -1
- package/lib/ui/utils.cjs +11 -4
- package/package.json +2 -2
package/lib/ui/Rendering.cjs
CHANGED
|
@@ -198,7 +198,10 @@ class RenderingEngine {
|
|
|
198
198
|
if (types?.length > 1)
|
|
199
199
|
throw new errors_1.RenderingError(`Only one type of decoration is allowed. Please choose between @uiprop, @uichild or @uielement`);
|
|
200
200
|
decs.shift();
|
|
201
|
-
decs.
|
|
201
|
+
const sorted = decs.sort((a, b) => {
|
|
202
|
+
return a.key === constants_1.UIKeys.ELEMENT ? -1 : 1;
|
|
203
|
+
});
|
|
204
|
+
sorted.forEach((dec) => {
|
|
202
205
|
if (!dec)
|
|
203
206
|
throw new errors_1.RenderingError(`No decorator found`);
|
|
204
207
|
switch (dec.key) {
|
|
@@ -230,7 +233,6 @@ class RenderingEngine {
|
|
|
230
233
|
children.push(childDefinition);
|
|
231
234
|
break;
|
|
232
235
|
}
|
|
233
|
-
case constants_1.UIKeys.HIDDEN:
|
|
234
236
|
case constants_1.UIKeys.UILISTPROP: {
|
|
235
237
|
mapper = mapper || {};
|
|
236
238
|
if (dec.props?.name)
|
|
@@ -238,44 +240,55 @@ class RenderingEngine {
|
|
|
238
240
|
const props = Object.assign({}, classDecorator.props?.item || {}, item?.props || {}, dec.props?.props || {}, globalProps);
|
|
239
241
|
childProps = {
|
|
240
242
|
tag: item?.tag || props.render || "",
|
|
241
|
-
props: Object.assign({}, childProps?.props,
|
|
243
|
+
props: Object.assign({}, childProps?.props, { mapper }, props),
|
|
242
244
|
};
|
|
243
245
|
break;
|
|
244
246
|
}
|
|
247
|
+
case constants_1.UIKeys.HIDDEN:
|
|
245
248
|
case constants_1.UIKeys.ELEMENT: {
|
|
246
249
|
children = children || [];
|
|
247
250
|
const uiProps = dec.props;
|
|
248
|
-
const props = Object.assign({}, uiProps.props, {
|
|
251
|
+
const props = Object.assign({}, childProps?.props, uiProps.props || {}, (uiProps?.props?.name ? {
|
|
249
252
|
path: getPath(globalProps?.childOf, uiProps.props.name),
|
|
250
253
|
childOf: undefined, // The childOf prop is passed by globalProps when it is a nested prop
|
|
251
|
-
}
|
|
254
|
+
} : {}), globalProps);
|
|
255
|
+
const tag = uiProps.tag || childProps?.tag;
|
|
252
256
|
const childDefinition = {
|
|
253
|
-
tag
|
|
257
|
+
tag,
|
|
254
258
|
props,
|
|
255
259
|
};
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
this.toAttributeValue(dec.key, dec.props);
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
260
|
+
if (dec.key === constants_1.UIKeys.ELEMENT) {
|
|
261
|
+
const validationDecs = validationDecorators[key];
|
|
262
|
+
const typeDec = validationDecs.shift();
|
|
263
|
+
for (const dec of validationDecs) {
|
|
264
|
+
if (this.isValidatableByAttribute(dec.key)) {
|
|
265
|
+
childDefinition.props[this.translate(dec.key)] = this.toAttributeValue(dec.key, dec.props);
|
|
266
|
+
continue;
|
|
267
|
+
}
|
|
268
|
+
if (this.isValidatableByType(dec.key)) {
|
|
269
|
+
if (dec.key === constants_1.HTML5InputTypes.DATE) {
|
|
270
|
+
childDefinition.props[constants_1.UIKeys.FORMAT] = dec.props.format || constants_1.HTML5DateFormat;
|
|
271
|
+
}
|
|
272
|
+
childDefinition.props[constants_1.UIKeys.TYPE] = dec.key;
|
|
273
|
+
continue;
|
|
268
274
|
}
|
|
269
|
-
childDefinition.props[constants_1.UIKeys.TYPE] = dec.key;
|
|
270
|
-
continue;
|
|
271
275
|
}
|
|
276
|
+
if (!childDefinition.props[constants_1.UIKeys.TYPE]) {
|
|
277
|
+
const basicType = typeDec.props.name;
|
|
278
|
+
childDefinition.props[constants_1.UIKeys.TYPE] = this.translate(basicType.toLowerCase(), true);
|
|
279
|
+
}
|
|
280
|
+
childDefinition.props.value = (0, utils_1.formatByType)(childDefinition.props[constants_1.UIKeys.TYPE], model[key], childDefinition.props[constants_1.UIKeys.FORMAT]);
|
|
281
|
+
children.push(childDefinition);
|
|
272
282
|
}
|
|
273
|
-
if (
|
|
274
|
-
const
|
|
275
|
-
|
|
283
|
+
if (dec.key === constants_1.UIKeys.HIDDEN) {
|
|
284
|
+
const child = children.find(c => c.props?.name === key);
|
|
285
|
+
if (child) {
|
|
286
|
+
child.props = Object.assign({}, child.props, { [dec.key]: uiProps });
|
|
287
|
+
}
|
|
288
|
+
else {
|
|
289
|
+
children.push(childDefinition);
|
|
290
|
+
}
|
|
276
291
|
}
|
|
277
|
-
childDefinition.props.value = (0, utils_1.formatByType)(childDefinition.props[constants_1.UIKeys.TYPE], model[key], childDefinition.props[constants_1.UIKeys.FORMAT]);
|
|
278
|
-
children.push(childDefinition);
|
|
279
292
|
break;
|
|
280
293
|
}
|
|
281
294
|
case constants_1.UIKeys.UILAYOUTITEM:
|
|
@@ -424,4 +437,4 @@ class RenderingEngine {
|
|
|
424
437
|
}
|
|
425
438
|
}
|
|
426
439
|
exports.RenderingEngine = RenderingEngine;
|
|
427
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Rendering.js","sourceRoot":"","sources":["../../src/ui/Rendering.ts"],"names":[],"mappings":"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","sourcesContent":["import { InternalError } from \"@decaf-ts/db-decorators\";\nimport {\n  Constructor,\n  Model,\n  ModelConstructor,\n  ReservedModels,\n  ValidationKeys,\n  ValidationMetadata,\n} from \"@decaf-ts/decorator-validation\";\nimport {\n  HTML5DateFormat,\n  HTML5InputTypes,\n  UIKeys,\n  ValidatableByAttribute,\n  ValidatableByType,\n} from \"./constants\";\nimport {\n  FieldDefinition,\n  FieldProperties,\n  UIClassMetadata,\n  UIElementMetadata,\n  UILayoutItemMetadata,\n  UIListItemElementMetadata,\n  UIModelMetadata,\n  UIPropMetadata,\n} from \"./types\";\nimport { RenderingError } from \"./errors\";\nimport { DecoratorMetadata, Reflection } from \"@decaf-ts/reflection\";\nimport { formatByType, generateUIModelID } from \"./utils\";\n\n/**\n * @description Abstract class for rendering UI components based on model metadata.\n * @summary The RenderingEngine class provides a framework for converting model metadata into UI field definitions.\n * It handles the translation of model properties to UI elements, applies validation rules, and manages different rendering flavors.\n * This class is designed to be extended by specific rendering implementations.\n *\n * @template T The type of the rendering result, defaults to void\n * @template R The type of the field definition, defaults to FieldDefinition<T>\n *\n * @param {string} flavour - The flavor of the rendering engine.\n *\n * @class RenderingEngine\n */\nexport abstract class RenderingEngine<T = void, R = FieldDefinition<T>> {\n  /**\n   * @description Cache for storing rendering engine instances or constructors.\n   * @private\n   * @static\n   */\n  private static cache: Record<\n    string,\n    | Constructor<RenderingEngine<unknown, unknown>>\n    | RenderingEngine<unknown, unknown>\n  > = {};\n\n  /**\n   * @description The currently active rendering engine.\n   * @private\n   * @static\n   */\n  private static current:\n    | Constructor<RenderingEngine<unknown, unknown>>\n    | RenderingEngine<unknown, unknown>;\n\n  /**\n   * Flag indicating whether the rendering engine has been initialized.\n   */\n  protected initialized: boolean = false;\n\n  protected constructor(readonly flavour: string) {\n    RenderingEngine.register(this);\n    console.log(`decaf's ${flavour} rendering engine loaded`);\n  }\n\n  /**\n   * @description Initializes the rendering engine.\n   * @summary Abstract method to be implemented by subclasses for specific initialization logic.\n   *\n   * @param {...any[]} args - Any additional arguments needed for initialization.\n   * @returns {Promise<void>} A promise that resolves when initialization is complete.\n   *\n   * @abstract\n   */\n  abstract initialize(...args: any[]): Promise<void>;\n\n  /**\n   * @description Translates between model types and HTML input types.\n   * @summary Converts model data types to appropriate HTML input types and vice versa.\n   *\n   * @param {string} key - The key to translate.\n   * @param {boolean} [toView=true] - Direction of translation (true for model to view, false for view to model).\n   * @returns {string} The translated type.\n   */\n  translate(key: string, toView: boolean = true): string {\n    if (toView) {\n      switch (key) {\n        case ReservedModels.STRING:\n          return HTML5InputTypes.TEXT;\n        case ReservedModels.NUMBER:\n        case ReservedModels.BIGINT:\n          return HTML5InputTypes.NUMBER;\n        case ReservedModels.BOOLEAN:\n          return HTML5InputTypes.CHECKBOX;\n        case ReservedModels.DATE:\n          return HTML5InputTypes.DATE;\n      }\n    } else {\n      switch (key) {\n        case HTML5InputTypes.SELECT:\n        case HTML5InputTypes.TEXT:\n        case HTML5InputTypes.EMAIL:\n        case HTML5InputTypes.COLOR:\n        case HTML5InputTypes.PASSWORD:\n        case HTML5InputTypes.TEL:\n        case HTML5InputTypes.URL:\n        case HTML5InputTypes.SEARCH:\n        case HTML5InputTypes.HIDDEN:\n        case HTML5InputTypes.TEXTAREA:\n        case HTML5InputTypes.RADIO:\n          return ReservedModels.STRING;\n        case HTML5InputTypes.NUMBER:\n          return ReservedModels.NUMBER;\n        case HTML5InputTypes.CHECKBOX:\n          return ReservedModels.BOOLEAN;\n        case HTML5InputTypes.DATE:\n        case HTML5InputTypes.DATETIME_LOCAL:\n        case HTML5InputTypes.TIME:\n          return ReservedModels.DATE;\n      }\n    }\n    return key;\n  }\n\n  /**\n   * @description Retrieves class decorator metadata for a model instance\n   * @summary Extracts UI-related class decorators from a model and returns them as an array\n   * This method collects metadata from various UI class decorators including @uimodel,\n   * @uilistitem, @uihandlers, and @uilayout applied to the model class.\n   *\n   * @template M Type extending Model\n   * @param {M} model - The model instance to extract metadata from\n   * @returns {UIClassMetadata[]} Array of UI class metadata objects\n   *\n   * @private\n   */\n  private getClassDecoratorsMetadata<M extends Model>(model: M): UIClassMetadata[]  {\n    return [\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UIMODEL),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UIMODEL),\n        Model.get(model.constructor.name) as any\n      ),\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILISTITEM),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILISTITEM),\n        Model.get(model.constructor.name) as any\n      ),\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.HANDLERS),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.HANDLERS),\n        Model.get(model.constructor.name) as any\n      ),\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILAYOUT),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILAYOUT),\n        Model.get(model.constructor.name) as any\n      ),\n    ].filter(Boolean);;\n  }\n\n  /**\n   * @description Checks if a type is validatable by its nature.\n   * @summary Determines if a given UI key represents a type that is inherently validatable.\n   *\n   * @param {string} key - The UI key to check.\n   * @returns {boolean} True if the type is validatable, false otherwise.\n   */\n  protected isValidatableByType(key: string): boolean {\n    return Object.keys(ValidatableByType).includes(key);\n  }\n\n  /**\n   * @description Checks if a type is validatable by attribute.\n   * @summary Determines if a given UI key represents a validation that can be applied as an attribute.\n   *\n   * @param {string} key - The UI key to check.\n   * @returns {boolean} True if the type is validatable by attribute, false otherwise.\n   */\n  protected isValidatableByAttribute(key: string): boolean {\n    return Object.keys(ValidatableByAttribute).includes(key);\n  }\n\n  /**\n   * @description Converts validation metadata to an attribute value.\n   * @summary Transforms validation metadata into a value suitable for use as an HTML attribute.\n   *\n   * @param {string} key - The validation key.\n   * @param {ValidationMetadata} value - The validation metadata.\n   * @returns {string | number | boolean} The converted attribute value.\n   * @throws {Error} If the given key is not validatable by attribute.\n   */\n  protected toAttributeValue(\n    key: string,\n    value: ValidationMetadata\n  ): string | number | boolean {\n    if (!Object.keys(ValidatableByAttribute).includes(key))\n      throw new Error(\n        `Invalid attribute key \"${key}\". Expected one of: ${Object.keys(ValidatableByAttribute).join(\", \")}.`\n      );\n\n    return key === UIKeys.REQUIRED ? true : value[key];\n  }\n\n  /**\n   * @description Converts a model to a field definition.\n   * @summary Processes a model instance, extracting UI-related metadata and validation rules to create a field definition.\n   *\n   * @template M Type extending Model\n   * @template T Type referencing the specific Rendering engine field properties/inputs\n   * @param {M} model - The model instance to convert.\n   * @param {Record<string, unknown>} [globalProps={}] - Global properties to apply to all child elements.\n   * @param {boolean} [generateId=true] - Flag indicating whether to populate the rendererId property.\n   * @returns {FieldDefinition<T>} A field definition object representing the UI structure of the model.\n   * @throws {RenderingError} If no UI definitions are set for the model or if there are invalid decorators.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *  participant C as Client\n   *  participant RE as RenderingEngine\n   *  participant R as Reflection\n   *  participant M as Model\n   *  C->>RE: toFieldDefinition(model, globalProps)\n   *  RE->>R: getMetadata(UIKeys.UIMODEL, model.constructor)\n   *  R-->>RE: UIModelMetadata\n   *  RE->>R: getAllPropertyDecorators(model, UIKeys.REFLECT)\n   *  R-->>RE: Record<string, DecoratorMetadata[]>\n   *  RE->>R: getAllPropertyDecorators(model, ValidationKeys.REFLECT)\n   *  R-->>RE: Record<string, DecoratorMetadata<ValidationMetadata>[]>\n   *  loop For each property\n   *    RE->>RE: Process UI decorators\n   *    RE->>RE: Apply validation rules\n   *  end\n   *  RE-->>C: FieldDefinition<T>\n   */\n  protected toFieldDefinition<M extends Model>(\n    model: M,\n    globalProps: Record<string, unknown> = {},\n    generateId: boolean = true\n  ): FieldDefinition<T> {\n    \n    const { inheritProps, ...globalPropsWithoutInherits } = globalProps;\n    globalProps = globalPropsWithoutInherits;\n\n    const classDecorators = this.getClassDecoratorsMetadata<M>(model);\n\n    if (!classDecorators.length)\n      throw new RenderingError(\n        `No ui definitions set for model ${model.constructor.name}. Did you use @uimodel?`\n      );\n\n    const classDecorator = Object.assign(\n      {},\n      ...classDecorators,\n      inheritProps ? inheritProps : {} // override tag and properties when it is a component that should inherit properties from its parent.\n    );\n    const { tag, props, item, handlers } = classDecorator;\n\n    const uiDecorators: Record<string, DecoratorMetadata[]> =\n      Reflection.getAllPropertyDecorators(model, UIKeys.REFLECT) as Record<\n        string,\n        DecoratorMetadata[]\n      >;\n    let children: FieldDefinition<Record<string, any>>[] | undefined;\n    let childProps: Record<string, any> = item?.props || {};\n    let mapper: Record<string, string> = {};\n    const getPath = (parent: string | undefined, prop: string) => {\n      return parent ? [parent, prop].join(\".\") : prop;\n    };\n\n    if (uiDecorators) {\n      const validationDecorators: Record<\n        string,\n        DecoratorMetadata<ValidationMetadata>[]\n      > = Reflection.getAllPropertyDecorators(\n        model,\n        ValidationKeys.REFLECT\n      ) as Record<string, DecoratorMetadata<ValidationMetadata>[]>;\n      for (const key in uiDecorators) {\n        const decs = uiDecorators[key];\n        const types = Object.values(decs).filter(({key}) => [UIKeys.PROP, UIKeys.ELEMENT, UIKeys.CHILD].includes(key));\n        if (types?.length > 1)\n          throw new RenderingError(\n            `Only one type of decoration is allowed. Please choose between @uiprop, @uichild or @uielement`\n          );\n        decs.shift();\n        decs.forEach((dec) => {\n          if (!dec) throw new RenderingError(`No decorator found`);\n\n          switch (dec.key) {\n            case UIKeys.PROP: {\n              childProps[key] = dec.props as UIPropMetadata;\n              break;\n            }\n            case UIKeys.CHILD: {\n              if (!Model.isPropertyModel(model, key))\n                throw new RenderingError(`Child \"${key}\" must be a model.`);\n\n              let Clazz;\n              const submodel = (model as Record<string, any>)[key] as Model;\n              const constructable =\n                typeof submodel === \"object\" &&\n                submodel !== null &&\n                !Array.isArray(submodel);\n              // create instance if undefined\n              if (!constructable) {\n                const clazzName = (dec.props.props as Record<string, any>)\n                  ?.name as string;\n                Clazz = new (Model.get(clazzName) as ModelConstructor<Model>)();\n              }\n\n              children = children || [];\n              const childrenGlobalProps = Object.assign({}, globalProps || {}, {\"model\": Clazz} ,{\n                inheritProps: dec.props as UIModelMetadata,\n                childOf: getPath(globalProps?.childOf as string, key),\n              });\n              const childDefinition = this.toFieldDefinition(\n                submodel || Clazz, // Must avoid undefined values — an instance is required to retrieve properties.\n                childrenGlobalProps,\n                false\n              );\n              children.push(\n                childDefinition as FieldDefinition<Record<string, any>>\n              );\n              break;\n            }\n            case UIKeys.HIDDEN: \n            case UIKeys.UILISTPROP: {\n              mapper = mapper || {};\n              if(dec.props?.name)\n                mapper[dec.props?.name as string] = key;\n              const props = Object.assign(\n                {},\n                classDecorator.props?.item || {},\n                item?.props || {},\n                dec.props?.props || {},\n                globalProps\n              );\n              childProps = {\n                tag: item?.tag || props.render || \"\",\n                props: Object.assign(\n                  {}, \n                  childProps?.props, \n                  dec.key === UIKeys.UILISTPROP ? { mapper } : {[dec.key]: dec.props}, \n                  props),\n              };\n\n              break;\n            }\n            case UIKeys.ELEMENT: {\n              children = children || [];\n\n              const uiProps: UIElementMetadata = dec.props as UIElementMetadata;\n              const props = Object.assign(\n                {},\n                uiProps.props as any,\n                {\n                  path: getPath(\n                    globalProps?.childOf as string,\n                    uiProps.props!.name\n                  ),\n                  childOf: undefined, // The childOf prop is passed by globalProps when it is a nested prop\n                },\n                childProps?.props,\n                globalProps\n              );\n\n              const childDefinition: FieldDefinition<Record<string, any>> = {\n                tag: uiProps.tag,\n                props,\n              };\n\n              const validationDecs: DecoratorMetadata<ValidationMetadata>[] =\n                validationDecorators[\n                  key\n                ] as DecoratorMetadata<ValidationMetadata>[];\n\n              const typeDec: DecoratorMetadataObject =\n                validationDecs.shift() as DecoratorMetadata;\n              for (const dec of validationDecs) {\n                if (this.isValidatableByAttribute(dec.key)) {\n                  childDefinition.props[this.translate(dec.key)] =\n                    this.toAttributeValue(dec.key, dec.props);\n                  continue;\n                }\n                if (this.isValidatableByType(dec.key)) {\n                  if (dec.key === HTML5InputTypes.DATE) {\n                    childDefinition.props[UIKeys.FORMAT] =\n                      dec.props.format || HTML5DateFormat;\n                  }\n                  childDefinition.props[UIKeys.TYPE] = dec.key;\n                  continue;\n                }\n              }\n\n              if (!childDefinition.props[UIKeys.TYPE]) {\n                const basicType = (typeDec.props as { name: string }).name;\n                childDefinition.props[UIKeys.TYPE] = this.translate(\n                  basicType.toLowerCase(),\n                  true\n                );\n              }\n\n              childDefinition.props.value = formatByType(\n                childDefinition.props[UIKeys.TYPE],\n                model[key as keyof M],\n                childDefinition.props[UIKeys.FORMAT]\n              );\n\n              children.push(childDefinition);\n              break;\n            }\n            case UIKeys.UILAYOUTITEM: \n            break;\n            default:\n              throw new RenderingError(`Invalid key: ${dec.key}`);\n          }\n        });\n      }\n    }\n\n    globalProps = Object.assign({}, props, globalProps, {\n      handlers: handlers || {},\n    });\n    const result: FieldDefinition<T> = {\n      tag: tag,\n      item: childProps as UIListItemElementMetadata,\n      props: globalProps as T & FieldProperties,\n      children: ((Object.keys(uiDecorators)?.length && children?.length) ? \n        this.getLayoutItems(children, uiDecorators) : children),\n    \n    };\n\n    if (generateId) result.rendererId = generateUIModelID(model);\n    return result;\n  }\n\n  /**\n   * @description Processes layout items for grid positioning\n   * @summary Maps child field definitions to their corresponding layout positions\n   * This method iterates through child field definitions and applies layout metadata\n   * from @uilayoutitem decorators to position them correctly in a grid layout.\n   *\n   * @param {FieldDefinition[]} children - Array of child field definitions to process\n   * @param {Record<string, any>} uiDecorators - UI decorator metadata keyed by property name\n   * @returns {FieldDefinition[]} Array of field definitions with layout positioning applied\n   *\n   * @example\n   * // Internal usage - positions children in grid layout\n   * const layoutChildren = this.getLayoutItems(childDefinitions, decoratorMetadata);\n   */\n  getLayoutItems(children: FieldDefinition<any>[], uiDecorators: Record<string, any>): FieldDefinition<any>[] {\n    return children.map((child) => {\n      let updatedChild = child;\n      for (const key in uiDecorators) {\n        const decs = uiDecorators[key];\n        for (const dec of decs) {\n          if (\n            dec.key === UIKeys.UILAYOUTITEM &&\n            (dec.props?.name === child.props?.name || dec.props?.name === child.props?.childOf)\n          ) {\n            const { col, props, row } = dec.props as UILayoutItemMetadata;\n            updatedChild = {\n              row,\n              col,\n              ...child,\n              props: {\n                ...child.props,\n                ...props,\n              },\n            };\n            break;\n          }\n        }\n      }\n      return updatedChild;\n    });\n  }\n\n  /**\n   * @description Renders a model with global properties and additional arguments.\n   * @summary Abstract method to be implemented by subclasses to define specific rendering behavior.\n   *\n   * @template M Type extending Model\n   * @template R Rendering engine implementation specific output type\n   * @param {M} model - The model to be rendered.\n   * @param {Record<string, unknown>} globalProps - Global properties to be applied to all elements during rendering.\n   * @param {...any[]} args - Additional arguments that may be required for specific rendering implementations.\n   * @returns {R} The rendered result, type depends on the specific implementation.\n   *\n   * @abstract\n   */\n  abstract render<M extends Model>(\n    model: M,\n    globalProps: Record<string, unknown>,\n    ...args: any[]\n  ): R;\n\n  /**\n   * @description Registers a rendering engine instance.\n   * @summary Adds a rendering engine to the static cache and sets it as the current engine.\n   *\n   * @param {RenderingEngine<unknown, unknown>} engine - The rendering engine to register.\n   * @throws {InternalError} If an engine with the same flavor already exists.\n   *\n   * @static\n   */\n  static register(engine: RenderingEngine<unknown, unknown>) {\n    if (engine.flavour in this.cache)\n      throw new InternalError(\n        `Rendering engine under ${engine.flavour} already exists`\n      );\n    this.cache[engine.flavour] = engine;\n    this.current = engine;\n  }\n\n  /**\n   * @description Retrieves or initializes a rendering engine.\n   * @summary Gets an existing engine instance or creates and initializes a new one if given a constructor.\n   *\n   * @template O The type of the rendering engine output\n   * @param {Constructor<RenderingEngine<O>> | RenderingEngine<O>} obj - The engine instance or constructor.\n   * @returns {RenderingEngine<O>} The initialized rendering engine.\n   *\n   * @private\n   * @static\n   */\n  private static getOrBoot<O>(\n    obj: Constructor<RenderingEngine<O>> | RenderingEngine<O>\n  ): RenderingEngine<O> {\n    if (obj instanceof RenderingEngine) return obj as RenderingEngine<O>;\n    const engine: RenderingEngine<O> = new obj();\n    engine.initialize(); // make the booting async. use the initialized flag to control it\n    return engine as RenderingEngine<O>;\n  }\n\n  /**\n   * @description Retrieves a rendering engine by flavor.\n   * @summary Gets the current rendering engine or a specific one by flavor.\n   *\n   * @template O The type of the rendering engine output\n   * @param {string} [flavour] - The flavor of the rendering engine to retrieve.\n   * @returns {RenderingEngine<O>} The requested rendering engine.\n   * @throws {InternalError} If the requested flavor does not exist.\n   *\n   * @static\n   */\n  static get<O>(flavour?: string): RenderingEngine<O> {\n    if (!flavour)\n      return this.getOrBoot<O>(\n        this.current as Constructor<RenderingEngine<O>> | RenderingEngine<O>\n      );\n    if (!(flavour in this.cache))\n      throw new InternalError(\n        `Rendering engine under ${flavour} does not exist`\n      );\n    return this.getOrBoot<O>(\n      this.cache[flavour] as\n        | Constructor<RenderingEngine<O>>\n        | RenderingEngine<O>\n    );\n  }\n\n  /**\n   * @description Renders a model using the appropriate rendering engine.\n   * @summary Determines the correct rendering engine for a model and invokes its render method.\n   *\n   * @template M Type extending Model\n   * @param {M} model - The model to render.\n   * @param {...any[]} args - Additional arguments to pass to the render method.\n   * @returns {any} The result of the rendering process.\n   * @throws {InternalError} If no registered model is found.\n   *\n   * @static\n   */\n  static render<M extends Model>(model: M, ...args: any[]): any {\n    const constructor =\n      Model.get(model.constructor.name) || Model.fromObject(model);\n    if (!constructor) throw new InternalError(\"No model registered found\");\n    const flavour = Reflect.getMetadata(\n      RenderingEngine.key(UIKeys.RENDERED_BY),\n      constructor as ModelConstructor<Model>\n    );\n\n    // @ts-expect-error for the var args type check\n    return RenderingEngine.get(flavour).render(model, ...args);\n  }\n\n  /**\n   * @description Generates a metadata key for UI-related properties.\n   * @summary Prefixes a given key with the UI reflection prefix.\n   *\n   * @param {string} key - The key to prefix.\n   * @returns {string} The prefixed key.\n   *\n   * @static\n   */\n  static key(key: string): string {\n    return `${UIKeys.REFLECT}${key}`;\n  }\n}\n"]}
|
|
440
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Rendering.js","sourceRoot":"","sources":["../../src/ui/Rendering.ts"],"names":[],"mappings":"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","sourcesContent":["import { InternalError } from \"@decaf-ts/db-decorators\";\nimport {\n  Constructor,\n  Model,\n  ModelConstructor,\n  ReservedModels,\n  ValidationKeys,\n  ValidationMetadata,\n} from \"@decaf-ts/decorator-validation\";\nimport {\n  HTML5DateFormat,\n  HTML5InputTypes,\n  UIKeys,\n  ValidatableByAttribute,\n  ValidatableByType,\n} from \"./constants\";\nimport {\n  FieldDefinition,\n  FieldProperties,\n  UIClassMetadata,\n  UIElementMetadata,\n  UILayoutItemMetadata,\n  UIListItemElementMetadata,\n  UIModelMetadata,\n  UIPropMetadata,\n} from \"./types\";\nimport { RenderingError } from \"./errors\";\nimport { DecoratorMetadata, Reflection } from \"@decaf-ts/reflection\";\nimport { formatByType, generateUIModelID } from \"./utils\";\n\n/**\n * @description Abstract class for rendering UI components based on model metadata.\n * @summary The RenderingEngine class provides a framework for converting model metadata into UI field definitions.\n * It handles the translation of model properties to UI elements, applies validation rules, and manages different rendering flavors.\n * This class is designed to be extended by specific rendering implementations.\n *\n * @template T The type of the rendering result, defaults to void\n * @template R The type of the field definition, defaults to FieldDefinition<T>\n *\n * @param {string} flavour - The flavor of the rendering engine.\n *\n * @class RenderingEngine\n */\nexport abstract class RenderingEngine<T = void, R = FieldDefinition<T>> {\n  /**\n   * @description Cache for storing rendering engine instances or constructors.\n   * @private\n   * @static\n   */\n  private static cache: Record<\n    string,\n    | Constructor<RenderingEngine<unknown, unknown>>\n    | RenderingEngine<unknown, unknown>\n  > = {};\n\n  /**\n   * @description The currently active rendering engine.\n   * @private\n   * @static\n   */\n  private static current:\n    | Constructor<RenderingEngine<unknown, unknown>>\n    | RenderingEngine<unknown, unknown>;\n\n  /**\n   * Flag indicating whether the rendering engine has been initialized.\n   */\n  protected initialized: boolean = false;\n\n  protected constructor(readonly flavour: string) {\n    RenderingEngine.register(this);\n    console.log(`decaf's ${flavour} rendering engine loaded`);\n  }\n\n  /**\n   * @description Initializes the rendering engine.\n   * @summary Abstract method to be implemented by subclasses for specific initialization logic.\n   *\n   * @param {...any[]} args - Any additional arguments needed for initialization.\n   * @returns {Promise<void>} A promise that resolves when initialization is complete.\n   *\n   * @abstract\n   */\n  abstract initialize(...args: any[]): Promise<void>;\n\n  /**\n   * @description Translates between model types and HTML input types.\n   * @summary Converts model data types to appropriate HTML input types and vice versa.\n   *\n   * @param {string} key - The key to translate.\n   * @param {boolean} [toView=true] - Direction of translation (true for model to view, false for view to model).\n   * @returns {string} The translated type.\n   */\n  translate(key: string, toView: boolean = true): string {\n    if (toView) {\n      switch (key) {\n        case ReservedModels.STRING:\n          return HTML5InputTypes.TEXT;\n        case ReservedModels.NUMBER:\n        case ReservedModels.BIGINT:\n          return HTML5InputTypes.NUMBER;\n        case ReservedModels.BOOLEAN:\n          return HTML5InputTypes.CHECKBOX;\n        case ReservedModels.DATE:\n          return HTML5InputTypes.DATE;\n      }\n    } else {\n      switch (key) {\n        case HTML5InputTypes.SELECT:\n        case HTML5InputTypes.TEXT:\n        case HTML5InputTypes.EMAIL:\n        case HTML5InputTypes.COLOR:\n        case HTML5InputTypes.PASSWORD:\n        case HTML5InputTypes.TEL:\n        case HTML5InputTypes.URL:\n        case HTML5InputTypes.SEARCH:\n        case HTML5InputTypes.HIDDEN:\n        case HTML5InputTypes.TEXTAREA:\n        case HTML5InputTypes.RADIO:\n          return ReservedModels.STRING;\n        case HTML5InputTypes.NUMBER:\n          return ReservedModels.NUMBER;\n        case HTML5InputTypes.CHECKBOX: \n          return ReservedModels.BOOLEAN;\n        case HTML5InputTypes.DATE:\n        case HTML5InputTypes.DATETIME_LOCAL:\n        case HTML5InputTypes.TIME:\n          return ReservedModels.DATE;\n      }\n    }\n    return key;\n  }\n\n  /**\n   * @description Retrieves class decorator metadata for a model instance\n   * @summary Extracts UI-related class decorators from a model and returns them as an array\n   * This method collects metadata from various UI class decorators including @uimodel,\n   * @uilistitem, @uihandlers, and @uilayout applied to the model class.\n   *\n   * @template M Type extending Model\n   * @param {M} model - The model instance to extract metadata from\n   * @returns {UIClassMetadata[]} Array of UI class metadata objects\n   *\n   * @private\n   */\n  private getClassDecoratorsMetadata<M extends Model>(model: M): UIClassMetadata[]  {\n    return [\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UIMODEL),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UIMODEL),\n        Model.get(model.constructor.name) as any\n      ),\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILISTITEM),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILISTITEM),\n        Model.get(model.constructor.name) as any\n      ),\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.HANDLERS),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.HANDLERS),\n        Model.get(model.constructor.name) as any\n      ),\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILAYOUT),\n        model.constructor\n      ) ||\n      Reflect.getMetadata(\n        RenderingEngine.key(UIKeys.UILAYOUT),\n        Model.get(model.constructor.name) as any\n      ),\n    ].filter(Boolean);;\n  }\n\n  /**\n   * @description Checks if a type is validatable by its nature.\n   * @summary Determines if a given UI key represents a type that is inherently validatable.\n   *\n   * @param {string} key - The UI key to check.\n   * @returns {boolean} True if the type is validatable, false otherwise.\n   */\n  protected isValidatableByType(key: string): boolean {\n    return Object.keys(ValidatableByType).includes(key);\n  }\n\n  /**\n   * @description Checks if a type is validatable by attribute.\n   * @summary Determines if a given UI key represents a validation that can be applied as an attribute.\n   *\n   * @param {string} key - The UI key to check.\n   * @returns {boolean} True if the type is validatable by attribute, false otherwise.\n   */\n  protected isValidatableByAttribute(key: string): boolean {\n    return Object.keys(ValidatableByAttribute).includes(key);\n  }\n\n  /**\n   * @description Converts validation metadata to an attribute value.\n   * @summary Transforms validation metadata into a value suitable for use as an HTML attribute.\n   *\n   * @param {string} key - The validation key.\n   * @param {ValidationMetadata} value - The validation metadata.\n   * @returns {string | number | boolean} The converted attribute value.\n   * @throws {Error} If the given key is not validatable by attribute.\n   */\n  protected toAttributeValue(\n    key: string,\n    value: ValidationMetadata\n  ): string | number | boolean {\n    if (!Object.keys(ValidatableByAttribute).includes(key))\n      throw new Error(\n        `Invalid attribute key \"${key}\". Expected one of: ${Object.keys(ValidatableByAttribute).join(\", \")}.`\n      );\n\n    return key === UIKeys.REQUIRED ? true : value[key];\n  }\n\n  /**\n   * @description Converts a model to a field definition.\n   * @summary Processes a model instance, extracting UI-related metadata and validation rules to create a field definition.\n   *\n   * @template M Type extending Model\n   * @template T Type referencing the specific Rendering engine field properties/inputs\n   * @param {M} model - The model instance to convert.\n   * @param {Record<string, unknown>} [globalProps={}] - Global properties to apply to all child elements.\n   * @param {boolean} [generateId=true] - Flag indicating whether to populate the rendererId property.\n   * @returns {FieldDefinition<T>} A field definition object representing the UI structure of the model.\n   * @throws {RenderingError} If no UI definitions are set for the model or if there are invalid decorators.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *  participant C as Client\n   *  participant RE as RenderingEngine\n   *  participant R as Reflection\n   *  participant M as Model\n   *  C->>RE: toFieldDefinition(model, globalProps)\n   *  RE->>R: getMetadata(UIKeys.UIMODEL, model.constructor)\n   *  R-->>RE: UIModelMetadata\n   *  RE->>R: getAllPropertyDecorators(model, UIKeys.REFLECT)\n   *  R-->>RE: Record<string, DecoratorMetadata[]>\n   *  RE->>R: getAllPropertyDecorators(model, ValidationKeys.REFLECT)\n   *  R-->>RE: Record<string, DecoratorMetadata<ValidationMetadata>[]>\n   *  loop For each property\n   *    RE->>RE: Process UI decorators\n   *    RE->>RE: Apply validation rules\n   *  end\n   *  RE-->>C: FieldDefinition<T>\n   */\n  protected toFieldDefinition<M extends Model>(\n    model: M,\n    globalProps: Record<string, unknown> = {},\n    generateId: boolean = true\n  ): FieldDefinition<T> {\n    \n    const { inheritProps, ...globalPropsWithoutInherits } = globalProps;\n    globalProps = globalPropsWithoutInherits;\n\n    const classDecorators = this.getClassDecoratorsMetadata<M>(model);\n\n    if (!classDecorators.length)\n      throw new RenderingError(\n        `No ui definitions set for model ${model.constructor.name}. Did you use @uimodel?`\n      );\n\n    const classDecorator = Object.assign(\n      {},\n      ...classDecorators,\n      inheritProps ? inheritProps : {} // override tag and properties when it is a component that should inherit properties from its parent.\n    );\n    const { tag, props, item, handlers } = classDecorator;\n\n    const uiDecorators: Record<string, DecoratorMetadata[]> =\n      Reflection.getAllPropertyDecorators(model, UIKeys.REFLECT) as Record<\n        string,\n        DecoratorMetadata[]\n      >;\n    let children: FieldDefinition<Record<string, any>>[] | undefined;\n    let childProps: Record<string, any> = item?.props || {};\n    let mapper: Record<string, string> = {};\n    const getPath = (parent: string | undefined, prop: string) => {\n      return parent ? [parent, prop].join(\".\") : prop;\n    };\n\n    if (uiDecorators) {\n      const validationDecorators: Record<\n        string,\n        DecoratorMetadata<ValidationMetadata>[]\n      > = Reflection.getAllPropertyDecorators(\n        model,\n        ValidationKeys.REFLECT\n      ) as Record<string, DecoratorMetadata<ValidationMetadata>[]>;\n      for (const key in uiDecorators) {\n        const decs = uiDecorators[key];\n        const types = Object.values(decs).filter(({key}) => [UIKeys.PROP, UIKeys.ELEMENT, UIKeys.CHILD].includes(key));\n        if (types?.length > 1)\n          throw new RenderingError(\n            `Only one type of decoration is allowed. Please choose between @uiprop, @uichild or @uielement`\n          );\n        decs.shift();\n        const sorted = decs.sort((a, b) => {\n          return a.key === UIKeys.ELEMENT ? -1 : 1;\n        });\n        sorted.forEach((dec) => {\n          if (!dec) throw new RenderingError(`No decorator found`);\n\n          switch (dec.key) {\n            case UIKeys.PROP: {\n              childProps[key] = dec.props as UIPropMetadata;\n              break;\n            }\n            case UIKeys.CHILD: {\n              if (!Model.isPropertyModel(model, key))\n                throw new RenderingError(`Child \"${key}\" must be a model.`);\n\n              let Clazz;\n              const submodel = (model as Record<string, any>)[key] as Model;\n              const constructable =\n                typeof submodel === \"object\" &&\n                submodel !== null &&\n                !Array.isArray(submodel);\n              // create instance if undefined\n              if (!constructable) {\n                const clazzName = (dec.props.props as Record<string, any>)\n                  ?.name as string;\n                Clazz = new (Model.get(clazzName) as ModelConstructor<Model>)();\n              }\n\n              children = children || [];\n              const childrenGlobalProps = Object.assign({}, globalProps || {}, {\"model\": Clazz} ,{\n                inheritProps: dec.props as UIModelMetadata,\n                childOf: getPath(globalProps?.childOf as string, key),\n              });\n              const childDefinition = this.toFieldDefinition(\n                submodel || Clazz, // Must avoid undefined values — an instance is required to retrieve properties.\n                childrenGlobalProps,\n                false\n              );\n              children.push(\n                childDefinition as FieldDefinition<Record<string, any>>\n              );\n              break;\n            }\n            case UIKeys.UILISTPROP: {\n              mapper = mapper || {};\n              if(dec.props?.name)\n                mapper[dec.props?.name as string] = key;\n              const props = Object.assign(\n                {},\n                classDecorator.props?.item || {},\n                item?.props || {},\n                dec.props?.props || {},\n                globalProps\n              );\n              childProps = {\n                tag: item?.tag || props.render || \"\",\n                props: Object.assign(\n                  {}, \n                  childProps?.props, \n                  { mapper }, \n                  props),\n              };\n\n              break;\n            }\n            case UIKeys.HIDDEN: \n            case UIKeys.ELEMENT: {\n              children = children || [];\n              const uiProps: UIElementMetadata = dec.props as UIElementMetadata;\n              const props = Object.assign(\n                  {},\n                  childProps?.props,\n                  uiProps.props || {},\n                  (uiProps?.props?.name ? {\n                    path: getPath(\n                      globalProps?.childOf as string,\n                      uiProps.props!.name\n                    ),\n                    childOf: undefined, // The childOf prop is passed by globalProps when it is a nested prop\n                  } : {}),\n                  globalProps\n                );\n                const tag = uiProps.tag || childProps?.tag;\n                const childDefinition: FieldDefinition<Record<string, any>> = {\n                  tag,\n                  props,\n                };\n              if(dec.key === UIKeys.ELEMENT) {\n                const validationDecs = validationDecorators[key] as DecoratorMetadata<ValidationMetadata>[];\n                const typeDec = validationDecs.shift() as DecoratorMetadata;\n                for (const dec of validationDecs) {\n                  if (this.isValidatableByAttribute(dec.key)) {\n                    childDefinition.props[this.translate(dec.key)] = this.toAttributeValue(dec.key, dec.props);\n                    continue;\n                  }\n                  if (this.isValidatableByType(dec.key)) {\n                    if (dec.key === HTML5InputTypes.DATE) {\n                      childDefinition.props[UIKeys.FORMAT] = dec.props.format || HTML5DateFormat;\n                    }\n                    childDefinition.props[UIKeys.TYPE] = dec.key;\n                    continue;\n                  }\n                }\n\n                if (!childDefinition.props[UIKeys.TYPE]) {\n                  const basicType = (typeDec.props as { name: string }).name;\n                  childDefinition.props[UIKeys.TYPE] = this.translate(\n                    basicType.toLowerCase(),\n                    true\n                  );\n                }\n\n                childDefinition.props.value = formatByType(\n                  childDefinition.props[UIKeys.TYPE],\n                  model[key as keyof M],\n                  childDefinition.props[UIKeys.FORMAT]\n                );\n                children.push(childDefinition);\n              }\n              if(dec.key === UIKeys.HIDDEN) {\n                const child = children.find(c => c.props?.name === key);\n                if (child) {\n                  child.props = Object.assign({}, child.props, { [dec.key]: uiProps });\n                } else {\n                  children.push(childDefinition);\n                }\n              }\n              break;\n            }\n            case UIKeys.UILAYOUTITEM: \n            break;\n            default:\n              throw new RenderingError(`Invalid key: ${dec.key}`);\n          }\n        });\n      }\n    }\n\n    globalProps = Object.assign({}, props, globalProps, {\n      handlers: handlers || {},\n    });\n    const result: FieldDefinition<T> = {\n      tag: tag,\n      item: childProps as UIListItemElementMetadata,\n      props: globalProps as T & FieldProperties,\n      children: ((Object.keys(uiDecorators)?.length && children?.length) ? \n        this.getLayoutItems(children, uiDecorators) : children),\n    \n    };\n\n    if (generateId) result.rendererId = generateUIModelID(model);\n    return result;\n  }\n\n  /**\n   * @description Processes layout items for grid positioning\n   * @summary Maps child field definitions to their corresponding layout positions\n   * This method iterates through child field definitions and applies layout metadata\n   * from @uilayoutitem decorators to position them correctly in a grid layout.\n   *\n   * @param {FieldDefinition[]} children - Array of child field definitions to process\n   * @param {Record<string, any>} uiDecorators - UI decorator metadata keyed by property name\n   * @returns {FieldDefinition[]} Array of field definitions with layout positioning applied\n   *\n   * @example\n   * // Internal usage - positions children in grid layout\n   * const layoutChildren = this.getLayoutItems(childDefinitions, decoratorMetadata);\n   */\n  getLayoutItems(children: FieldDefinition<any>[], uiDecorators: Record<string, any>): FieldDefinition<any>[] {\n    return children.map((child) => {\n      let updatedChild = child;\n      for (const key in uiDecorators) {\n        const decs = uiDecorators[key];\n        for (const dec of decs) {\n          if (\n            dec.key === UIKeys.UILAYOUTITEM &&\n            (dec.props?.name === child.props?.name || dec.props?.name === child.props?.childOf)\n          ) {\n            const { col, props, row } = dec.props as UILayoutItemMetadata;\n            updatedChild = {\n              row,\n              col,\n              ...child,\n              props: {\n                ...child.props,\n                ...props,\n              },\n            };\n            break;\n          }\n        }\n      }\n      return updatedChild;\n    });\n  }\n\n  /**\n   * @description Renders a model with global properties and additional arguments.\n   * @summary Abstract method to be implemented by subclasses to define specific rendering behavior.\n   *\n   * @template M Type extending Model\n   * @template R Rendering engine implementation specific output type\n   * @param {M} model - The model to be rendered.\n   * @param {Record<string, unknown>} globalProps - Global properties to be applied to all elements during rendering.\n   * @param {...any[]} args - Additional arguments that may be required for specific rendering implementations.\n   * @returns {R} The rendered result, type depends on the specific implementation.\n   *\n   * @abstract\n   */\n  abstract render<M extends Model>(\n    model: M,\n    globalProps: Record<string, unknown>,\n    ...args: any[]\n  ): R;\n\n  /**\n   * @description Registers a rendering engine instance.\n   * @summary Adds a rendering engine to the static cache and sets it as the current engine.\n   *\n   * @param {RenderingEngine<unknown, unknown>} engine - The rendering engine to register.\n   * @throws {InternalError} If an engine with the same flavor already exists.\n   *\n   * @static\n   */\n  static register(engine: RenderingEngine<unknown, unknown>) {\n    if (engine.flavour in this.cache)\n      throw new InternalError(\n        `Rendering engine under ${engine.flavour} already exists`\n      );\n    this.cache[engine.flavour] = engine;\n    this.current = engine;\n  }\n\n  /**\n   * @description Retrieves or initializes a rendering engine.\n   * @summary Gets an existing engine instance or creates and initializes a new one if given a constructor.\n   *\n   * @template O The type of the rendering engine output\n   * @param {Constructor<RenderingEngine<O>> | RenderingEngine<O>} obj - The engine instance or constructor.\n   * @returns {RenderingEngine<O>} The initialized rendering engine.\n   *\n   * @private\n   * @static\n   */\n  private static getOrBoot<O>(\n    obj: Constructor<RenderingEngine<O>> | RenderingEngine<O>\n  ): RenderingEngine<O> {\n    if (obj instanceof RenderingEngine) return obj as RenderingEngine<O>;\n    const engine: RenderingEngine<O> = new obj();\n    engine.initialize(); // make the booting async. use the initialized flag to control it\n    return engine as RenderingEngine<O>;\n  }\n\n  /**\n   * @description Retrieves a rendering engine by flavor.\n   * @summary Gets the current rendering engine or a specific one by flavor.\n   *\n   * @template O The type of the rendering engine output\n   * @param {string} [flavour] - The flavor of the rendering engine to retrieve.\n   * @returns {RenderingEngine<O>} The requested rendering engine.\n   * @throws {InternalError} If the requested flavor does not exist.\n   *\n   * @static\n   */\n  static get<O>(flavour?: string): RenderingEngine<O> {\n    if (!flavour)\n      return this.getOrBoot<O>(\n        this.current as Constructor<RenderingEngine<O>> | RenderingEngine<O>\n      );\n    if (!(flavour in this.cache))\n      throw new InternalError(\n        `Rendering engine under ${flavour} does not exist`\n      );\n    return this.getOrBoot<O>(\n      this.cache[flavour] as\n        | Constructor<RenderingEngine<O>>\n        | RenderingEngine<O>\n    );\n  }\n\n  /**\n   * @description Renders a model using the appropriate rendering engine.\n   * @summary Determines the correct rendering engine for a model and invokes its render method.\n   *\n   * @template M Type extending Model\n   * @param {M} model - The model to render.\n   * @param {...any[]} args - Additional arguments to pass to the render method.\n   * @returns {any} The result of the rendering process.\n   * @throws {InternalError} If no registered model is found.\n   *\n   * @static\n   */\n  static render<M extends Model>(model: M, ...args: any[]): any {\n    const constructor =\n      Model.get(model.constructor.name) || Model.fromObject(model);\n    if (!constructor) throw new InternalError(\"No model registered found\");\n    const flavour = Reflect.getMetadata(\n      RenderingEngine.key(UIKeys.RENDERED_BY),\n      constructor as ModelConstructor<Model>\n    );\n\n    // @ts-expect-error for the var args type check\n    return RenderingEngine.get(flavour).render(model, ...args);\n  }\n\n  /**\n   * @description Generates a metadata key for UI-related properties.\n   * @summary Prefixes a given key with the UI reflection prefix.\n   *\n   * @param {string} key - The key to prefix.\n   * @returns {string} The prefixed key.\n   *\n   * @static\n   */\n  static key(key: string): string {\n    return `${UIKeys.REFLECT}${key}`;\n  }\n}\n"]}
|
package/lib/ui/types.cjs
CHANGED
|
@@ -9,4 +9,4 @@
|
|
|
9
9
|
*/
|
|
10
10
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
11
11
|
const constants_1 = require("./constants.cjs");
|
|
12
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdWkvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7O0dBT0c7O0FBR0gsK0NBQXFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAZGVzY3JpcHRpb24gVHlwZSBkZWZpbml0aW9ucyBmb3IgVUkgY29tcG9uZW50cyBhbmQgcmVuZGVyaW5nXG4gKiBAc3VtbWFyeSBEZWZpbmVzIHR5cGVzIGFuZCBpbnRlcmZhY2VzIHVzZWQgdGhyb3VnaG91dCB0aGUgVUkgZGVjb3JhdG9ycyBsaWJyYXJ5XG4gKiBUaGlzIG1vZHVsZSBjb250YWlucyB0eXBlIGRlZmluaXRpb25zIGZvciBmaWVsZCBwcm9wZXJ0aWVzLCBVSSBtZXRhZGF0YSxcbiAqIGFuZCBvdGhlciBzdHJ1Y3R1cmVzIHVzZWQgaW4gcmVuZGVyaW5nIFVJIGNvbXBvbmVudHMuXG4gKiBAbW9kdWxlIHVpL3R5cGVzXG4gKiBAbWVtYmVyT2YgbW9kdWxlOnVpLWRlY29yYXRvcnNcbiAqL1xuXG5pbXBvcnQgeyBPcGVyYXRpb25LZXlzIH0gZnJvbSBcIkBkZWNhZi10cy9kYi1kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBVSUtleXMgfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gSW50ZXJmYWNlIGZvciBkZWZpbmluZyBhIFVJIGZpZWxkIG9yIGNvbXBvbmVudFxuICogQHN1bW1hcnkgUmVwcmVzZW50cyBhIHJlbmRlcmFibGUgVUkgZWxlbWVudCB3aXRoIHByb3BlcnRpZXMgYW5kIGNoaWxkcmVuXG4gKiBUaGlzIGludGVyZmFjZSBkZWZpbmVzIHRoZSBzdHJ1Y3R1cmUgb2YgYSBVSSBmaWVsZCBvciBjb21wb25lbnQsIGluY2x1ZGluZ1xuICogaXRzIHRhZyBuYW1lLCBwcm9wZXJ0aWVzLCBhbmQgb3B0aW9uYWwgY2hpbGRyZW4gZWxlbWVudHMuXG4gKlxuICogQGludGVyZmFjZSBGaWVsZERlZmluaXRpb25cbiAqIEB0ZW1wbGF0ZSBUIEFkZGl0aW9uYWwgcHJvcGVydGllcyB0eXBlIChkZWZhdWx0cyB0byB2b2lkKVxuICogQG1lbWJlck9mIG1vZHVsZTp1aS1kZWNvcmF0b3JzXG4gKlxuICogQHByb3BlcnR5IHtzdHJpbmd9IHRhZyAtIFRoZSBIVE1MIGVsZW1lbnQgb3IgY29tcG9uZW50IHRhZyBuYW1lXG4gKiBAcHJvcGVydHkge3N0cmluZ30gW3JlbmRlcmVySWRdIC0gT3B0aW9uYWwgSUQgb2YgdGhlIHJlbmRlcmVyIHRvIHVzZVxuICogQHByb3BlcnR5IHByb3BzIC0gQ29tYmluZWQgcHJvcGVydGllcyBmb3IgdGhlIGZpZWxkXG4gKiBAcHJvcGVydHkge0ZpZWxkRGVmaW5pdGlvbltdfSBbY2hpbGRyZW5dIC0gT3B0aW9uYWwgY2hpbGQgZWxlbWVudHNcbiAqIEBwcm9wZXJ0eSB7VUlMaXN0SXRlbUVsZW1lbnRNZXRhZGF0YX0gW2l0ZW1dIC0gT3B0aW9uYWwgbGlzdCBpdGVtIG1ldGFkYXRhXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRmllbGREZWZpbml0aW9uPFQgPSB2b2lkPiB7XG4gIHRhZzogc3RyaW5nO1xuICByZW5kZXJlcklkPzogc3RyaW5nO1xuICBwcm9wczogVCAmIEZpZWxkUHJvcGVydGllcztcbiAgY2hpbGRyZW4/OiBGaWVsZERlZmluaXRpb248VD5bXTtcbiAgaXRlbT86IFVJTGlzdEl0ZW1FbGVtZW50TWV0YWRhdGE7XG4gIGNvbD86IG51bWJlciB8IHN0cmluZ1tdO1xuICByb3c/
|
|
12
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/ui/types.ts"],"names":[],"mappings":";AAAA;;;;;;;GAOG;;AAGH,+CAAqC","sourcesContent":["/**\n * @description Type definitions for UI components and rendering\n * @summary Defines types and interfaces used throughout the UI decorators library\n * This module contains type definitions for field properties, UI metadata,\n * and other structures used in rendering UI components.\n * @module ui/types\n * @memberOf module:ui-decorators\n */\n\nimport { OperationKeys } from \"@decaf-ts/db-decorators\";\nimport { UIKeys } from \"./constants\";\n\n/**\n * @description Interface for defining a UI field or component\n * @summary Represents a renderable UI element with properties and children\n * This interface defines the structure of a UI field or component, including\n * its tag name, properties, and optional children elements.\n *\n * @interface FieldDefinition\n * @template T Additional properties type (defaults to void)\n * @memberOf module:ui-decorators\n *\n * @property {string} tag - The HTML element or component tag name\n * @property {string} [rendererId] - Optional ID of the renderer to use\n * @property props - Combined properties for the field\n * @property {FieldDefinition[]} [children] - Optional child elements\n * @property {UIListItemElementMetadata} [item] - Optional list item metadata\n */\nexport interface FieldDefinition<T = void> {\n  tag: string;\n  rendererId?: string;\n  props: T & FieldProperties;\n  children?: FieldDefinition<T>[];\n  item?: UIListItemElementMetadata;\n  col?: number | string[];\n  row?: number | string[];\n}\n\n/**\n * @description Interface for field properties including validation\n * @summary Defines common properties and validation rules for UI fields\n * This interface defines the standard properties that can be applied to\n * UI fields, including basic attributes and validation rules.\n *\n * @interface FieldProperties\n * @memberOf module:ui-decorators\n *\n * @property {string} name - The name of the field\n * @property {string} path - The full hierarchical path of the field\n * @property {string} childOf - The parent path of the immediate parent field, if nested\n * @property {string} type - The type of the field (e.g., 'text', 'number')\n * @property {string|number|Date} value - The current value of the field\n * @property {boolean} [hidden] - Whether the field is hidden\n * @property {boolean} [disabled] - Whether the field is disabled\n * @property {boolean} [required] - Whether the field is required\n * @property {boolean} [readonly] - Whether the field is read-only\n * @property {number} [maxLength] - Maximum length for text fields\n * @property {number} [minLength] - Minimum length for text fields\n * @property {number|Date} [max] - Maximum value for numeric or date fields\n * @property {number|Date} [min] - Minimum value for numeric or date fields\n * @property {string} [pattern] - Regex pattern for validation\n * @property {number} [step] - Step value for numeric fields\n * @property {string} [format] - Format string for date fields\n * @property {string} [equals] - Field must equal the value of this field\n * @property {string} [diff] - Field must differ from the value of this field\n * @property {string} [lessThan] - Field must be less than this field\n * @property {string} [lessThanOrEqual] - Field must be less than or equal to this field\n * @property {string} [greaterThan] - Field must be greater than this field\n * @property {string} [greaterThanOrEqual] - Field must be greater than or equal to this field\n */\nexport interface FieldProperties {\n  name: string;\n  path: string;\n  childOf?: string;\n  type: string;\n  value: string | number | Date | string[] | number[] | Date[];\n  hidden?: boolean | CrudOperationKeys[];\n  disabled?: boolean;\n  // Validation\n  required?: boolean;\n  readonly?: boolean;\n  maxLength?: number;\n  minLength?: number;\n  max?: number | Date;\n  min?: number | Date;\n  pattern?: string;\n  step?: number;\n  format?: string;\n  pk?: string;\n  multiple?: boolean;\n  customTypes?: string | string[];\n  options?: Record<string, unknown>[];\n  activeFormGroup?: number;\n  pages?: number; \n  page?: number; \n  [UIKeys.EQUALS]?: string;\n  [UIKeys.DIFF]?: string;\n  [UIKeys.LESS_THAN]?: string;\n  [UIKeys.LESS_THAN_OR_EQUAL]?: string;\n  [UIKeys.GREATER_THAN]?: string;\n  [UIKeys.GREATER_THAN_OR_EQUAL]?: string;\n}\n\n/**\n * @typedef UIElementMetadata\n * @memberOf module:ui-decorators\n */\nexport type UIElementMetadata = {\n  tag: string;\n  props?: Record<string, any>;\n  serialize?: boolean;\n};\n\n/**\n * @typedef UIElementMetadata\n * @memberOf ui-decorators.ui.decorators\n */\nexport type UIModelMetadata = Omit<UIElementMetadata, \"serialize\">;\n\n/**\n * @typedef UIPropMetadata\n * @memberOf module:ui-decorators\n */\nexport type UIPropMetadata = {\n  name: string;\n  stringify: boolean;\n};\n\n/**\n * @typedef CrudOperationKeys\n * @memberOf module:ui-decorators\n */\nexport type CrudOperationKeys =\n  | OperationKeys.CREATE\n  | OperationKeys.READ\n  | OperationKeys.UPDATE\n  | OperationKeys.DELETE;\n\n/**\n * @typedef UIListPropMetadata\n * @memberOf module:ui-decorators\n */\nexport type UIListPropMetadata = {\n  name: string;\n  props: Record<string, any>;\n};\n\n/**\n * @typedef UIListItemModelMetadata\n * @memberOf module:ui-decorators\n */\nexport type UIListItemModelMetadata = {\n  item: UIListItemElementMetadata;\n};\n\n/**\n * @typedef UIListItemElementMetadata\n * @memberOf module:ui-decorators\n */\nexport type UIListItemElementMetadata = {\n  tag: string;\n  props?: Record<string, any>;\n  mapper?: Record<string, string>;\n};\n\n\n/**\n * @typedef UILayoutMetadata\n * @memberOf module:ui-decorators\n */\nexport type UILayoutMetadata = {\n  props: {\n    cols?: number | string[];\n    rows?: number;\n    props?: Record<string, any>;\n  }\n}\n\n/**\n * @typedef UIClassMetadata\n * @memberOf module:ui-decorators\n */\nexport type UIClassMetadata = UILayoutMetadata | UIModelMetadata | UIListItemModelMetadata;\n\n/**\n * @typedef UILayoutItemMetadata\n * @memberOf ui-decorators.ui.decorators\n */\nexport type UILayoutItemMetadata ={\n  name: string;\n  props: Record<string, any>;\n  col: number | string[];\n  row: number | string[];\n};\n\n/**\n * @typedef UIMediaBreakPoints\n * @memberOf module:ui-decorators\n */\nexport type UIMediaBreakPoints = 'xl' | 'l' | 'm' | 's' | 'medium' | 'large' | 'small' | 'xlarge';"]}
|
package/lib/ui/types.d.ts
CHANGED
|
@@ -70,7 +70,7 @@ export interface FieldProperties {
|
|
|
70
70
|
path: string;
|
|
71
71
|
childOf?: string;
|
|
72
72
|
type: string;
|
|
73
|
-
value: string | number | Date;
|
|
73
|
+
value: string | number | Date | string[] | number[] | Date[];
|
|
74
74
|
hidden?: boolean | CrudOperationKeys[];
|
|
75
75
|
disabled?: boolean;
|
|
76
76
|
required?: boolean;
|
|
@@ -85,6 +85,7 @@ export interface FieldProperties {
|
|
|
85
85
|
pk?: string;
|
|
86
86
|
multiple?: boolean;
|
|
87
87
|
customTypes?: string | string[];
|
|
88
|
+
options?: Record<string, unknown>[];
|
|
88
89
|
activeFormGroup?: number;
|
|
89
90
|
pages?: number;
|
|
90
91
|
page?: number;
|
package/lib/ui/utils.cjs
CHANGED
|
@@ -26,6 +26,11 @@ function formatByType(type, value, ...args) {
|
|
|
26
26
|
function parseValueByType(type, value, fieldProps) {
|
|
27
27
|
let result = undefined;
|
|
28
28
|
switch (type) {
|
|
29
|
+
case Array.name: {
|
|
30
|
+
const parsed = Array.isArray(value) ? value.map(v => parseValueByType(decorator_validation_1.ReservedModels.STRING, v, fieldProps)) : [value];
|
|
31
|
+
result = parsed.join(",");
|
|
32
|
+
break;
|
|
33
|
+
}
|
|
29
34
|
case constants_1.HTML5InputTypes.NUMBER:
|
|
30
35
|
result = parseToNumber(value);
|
|
31
36
|
break;
|
|
@@ -45,9 +50,11 @@ function parseValueByType(type, value, fieldProps) {
|
|
|
45
50
|
}
|
|
46
51
|
default:
|
|
47
52
|
result =
|
|
48
|
-
typeof value === decorator_validation_1.ReservedModels.
|
|
49
|
-
value
|
|
50
|
-
|
|
53
|
+
typeof value === decorator_validation_1.ReservedModels.OBJECT ?
|
|
54
|
+
(Array.isArray(value) ? value.join(",") : JSON.stringify(value)) :
|
|
55
|
+
typeof value === decorator_validation_1.ReservedModels.BOOLEAN ?
|
|
56
|
+
value : typeof value === decorator_validation_1.ReservedModels.STRING ?
|
|
57
|
+
escapeHtml(value) : result;
|
|
51
58
|
}
|
|
52
59
|
if (typeof result === "undefined") {
|
|
53
60
|
throw new db_decorators_1.InternalError(`Failed to parse value of type ${type} from ${typeof value} - ${value}`);
|
|
@@ -96,4 +103,4 @@ function generateUIModelID(model) {
|
|
|
96
103
|
const name = model.constructor.name;
|
|
97
104
|
return `${name}-${id}`;
|
|
98
105
|
}
|
|
99
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../src/ui/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  formatDate,\n  Model,\n  parseDate,\n  ReservedModels,\n} from \"@decaf-ts/decorator-validation\";\nimport { HTML5DateFormat, HTML5InputTypes, UIKeys } from \"./constants\";\nimport { findModelId, InternalError } from \"@decaf-ts/db-decorators\";\nimport { FieldProperties } from \"./types\";\n\n/**\n * @function formatByType\n *\n * @memberOf module:ui-decorators\n */\nexport function formatByType(\n  type: any,\n  value: any,\n  ...args: unknown[]\n): string | number {\n  if (type === UIKeys.DATE) {\n    if(!value)\n        return \"\";\n    const format: string = (args.shift() as string) || HTML5DateFormat;\n    return formatDate(new Date(value), format);\n  }\n  return value;\n}\n\nexport function parseValueByType(\n  type: string,\n  value: string | number,\n  fieldProps: FieldProperties\n): string | number | Date {\n  let result: string | number | Date | undefined = undefined;\n  switch (type) {\n    case Array.name: {\n      const parsed = Array.isArray(value) ?  value.map(v => parseValueByType(\n        ReservedModels.STRING, v, fieldProps\n      )) : [value];\n      result = parsed.join(\",\");\n      break;\n    }\n    case HTML5InputTypes.NUMBER:\n      result = parseToNumber(value);\n      break;\n    case HTML5InputTypes.DATE: {\n      const format: string | undefined = fieldProps.format;\n      if(value && `${value}`.trim().length) {\n        result =\n          typeof value === ReservedModels.NUMBER\n            ? new Date(value)\n            : value\n              ? format\n                ? parseDate(format, value)\n                : new Date(value)\n              : undefined;\n      }\n      break;\n    }\n    default:\n      result = \n        typeof value === ReservedModels.OBJECT ? \n          (Array.isArray(value) ? value.join(\",\") : JSON.stringify(value)) :\n            typeof value === ReservedModels.BOOLEAN ?\n              value : typeof value === ReservedModels.STRING ? \n                escapeHtml(value as string) : result;\n  }\n  if (typeof result === \"undefined\") {\n    throw new InternalError(\n      `Failed to parse value of type ${type} from ${typeof value} - ${value}`\n    );\n  }\n  return result;\n}\n\nexport function parseToNumber(value: string | number) {\n  if (typeof value === \"number\" && !isNaN(value)) return value;\n\n  const parsed = Number(value);\n  if (!isNaN(parsed)) return parsed;\n\n  return undefined;\n}\n\nexport function escapeHtml(value: string) {\n  if (!value) return value;\n\n  const tagsToReplace: Record<string, string> = {\n    \"&\": \"&amp;\",\n    \"<\": \"&lt;\",\n    \">\": \"&gt;\",\n  };\n  return `${value}`.replace(/[&<>]/g, (tag) => {\n    return tagsToReplace[tag] || tag;\n  });\n}\n\nexport function revertHtml(value: string) {\n  const tagsToReplace: Record<string, string> = {\n    \"&amp;\": \"&\",\n    \"&lt;\": \"<\",\n    \"&gt;\": \">\",\n  };\n\n  return `${value}`.replace(/&lt;|&gt;|&amp;/g, (tag) => {\n    return tagsToReplace[tag] || tag;\n  });\n}\n\nexport function generateUIModelID<M extends Model>(model: M) {\n  let id: string | number | bigint;\n  try {\n    id = findModelId(model) as string | number;\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    id = Date.now();\n  }\n  const name = model.constructor.name;\n  return `${name}-${id}`;\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@decaf-ts/ui-decorators",
|
|
3
|
-
"version": "0.5.
|
|
3
|
+
"version": "0.5.23",
|
|
4
4
|
"description": "Extension of decorator validation to ui elements to allow for web integration",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"exports": {
|
|
@@ -73,7 +73,7 @@
|
|
|
73
73
|
"ts"
|
|
74
74
|
],
|
|
75
75
|
"author": "Tiago Venceslau and Contributors",
|
|
76
|
-
"license": "MIT
|
|
76
|
+
"license": "MIT",
|
|
77
77
|
"bugs": {
|
|
78
78
|
"url": "https://github.com/decaf-ts/ui-decorators/issues"
|
|
79
79
|
},
|