@decaf-ts/for-angular 0.0.9 → 0.0.11

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 (28) hide show
  1. package/dist/lib/components/empty-state/empty-state.component.d.ts +3 -3
  2. package/dist/lib/components/fieldset/fieldset.component.d.ts +1 -1
  3. package/dist/lib/components/filter/filter.component.d.ts +2 -2
  4. package/dist/lib/components/layout/layout.component.d.ts +2 -2
  5. package/dist/lib/components/list/list.component.d.ts +3 -3
  6. package/dist/lib/components/list-item/list-item.component.d.ts +3 -3
  7. package/dist/lib/components/model-renderer/model-renderer.component.d.ts +1 -1
  8. package/dist/lib/components/pagination/constants.d.ts +1 -1
  9. package/dist/lib/components/pagination/pagination.component.d.ts +2 -2
  10. package/dist/lib/components/searchbar/searchbar.component.d.ts +2 -2
  11. package/dist/lib/engine/NgxBaseComponent.d.ts +1 -1
  12. package/dist/lib/esm2022/components/component-renderer/component-renderer.component.mjs +3 -3
  13. package/dist/lib/esm2022/components/crud-field/crud-field.component.mjs +2 -2
  14. package/dist/lib/esm2022/components/crud-form/crud-form.component.mjs +2 -2
  15. package/dist/lib/esm2022/components/empty-state/empty-state.component.mjs +5 -5
  16. package/dist/lib/esm2022/components/fieldset/fieldset.component.mjs +4 -4
  17. package/dist/lib/esm2022/components/filter/filter.component.mjs +5 -5
  18. package/dist/lib/esm2022/components/layout/layout.component.mjs +3 -3
  19. package/dist/lib/esm2022/components/list/list.component.mjs +5 -5
  20. package/dist/lib/esm2022/components/list-item/list-item.component.mjs +6 -6
  21. package/dist/lib/esm2022/components/model-renderer/model-renderer.component.mjs +2 -2
  22. package/dist/lib/esm2022/components/pagination/constants.mjs +1 -1
  23. package/dist/lib/esm2022/components/pagination/pagination.component.mjs +4 -4
  24. package/dist/lib/esm2022/components/searchbar/searchbar.component.mjs +5 -5
  25. package/dist/lib/esm2022/engine/NgxBaseComponent.mjs +2 -3
  26. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs +3029 -3036
  27. package/dist/lib/fesm2022/decaf-ts-for-angular.mjs.map +1 -1
  28. package/package.json +4 -2
@@ -3,10 +3,10 @@ import { Component, inject, Input } from '@angular/core';
3
3
  import { IonCard, IonCardContent, IonIcon, IonTitle, NavController } from '@ionic/angular/standalone';
4
4
  import * as allIcons from 'ionicons/icons';
5
5
  import { addIcons } from 'ionicons';
6
- import { ForAngularModule } from 'src/lib/for-angular.module';
7
- import { Dynamic } from 'src/lib/engine';
8
- import { stringToBoolean } from 'src/lib/helpers';
9
- import { NgxBaseComponent } from 'src/lib/engine/NgxBaseComponent';
6
+ import { ForAngularModule } from '../../for-angular.module';
7
+ import { Dynamic } from '../../engine';
8
+ import { stringToBoolean } from '../../helpers';
9
+ import { NgxBaseComponent } from '../../engine/NgxBaseComponent';
10
10
  import { DomSanitizer } from '@angular/platform-browser';
11
11
  import { TranslateService } from '@ngx-translate/core';
12
12
  import * as i0 from "@angular/core";
@@ -345,4 +345,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
345
345
  }], searchValue: [{
346
346
  type: Input
347
347
  }] } });
348
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"empty-state.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/empty-state/empty-state.component.ts","../../../../../src/lib/components/empty-state/empty-state.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnInit  } from '@angular/core';\nimport { Color, PredefinedColors } from '@ionic/core';\nimport {\n  IonCard,\n  IonCardContent,\n  IonIcon,\n  IonTitle,\n  NavController\n}\nfrom '@ionic/angular/standalone';\nimport * as allIcons from 'ionicons/icons';\nimport { addIcons } from 'ionicons';\nimport { ForAngularModule } from 'src/lib/for-angular.module';\nimport { Dynamic, StringOrBoolean } from 'src/lib/engine';\nimport { stringToBoolean } from 'src/lib/helpers';\nimport { NgxBaseComponent } from 'src/lib/engine/NgxBaseComponent';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\nimport { TranslateService } from '@ngx-translate/core';\nimport { FunctionLike } from 'src/lib/engine/types';\n\n\n/**\n * @description Component for displaying empty state messages with optional actions.\n * @summary This component provides a standardized way to display empty state messages\n * when no data is available or when a user needs to take an action to populate content.\n * It includes customizable title, subtitle, icon, and action button elements that can be\n * styled and configured through input properties. The component supports localization\n * and can trigger navigation or custom actions when the button is clicked.\n *\n * @mermaid\n * classDiagram\n *   class EmptyStateComponent {\n *     +string title\n *     +string titleColor\n *     +string subtitle\n *     +string subtitleColor\n *     +StringOrBoolean showIcon\n *     +string icon\n *     +string iconSize\n *     +PredefinedColors iconColor\n *     +string|Function buttonLink\n *     +string buttonText\n *     +string buttonFill\n *     +Color buttonColor\n *     +string buttonSize\n *     +string searchValue\n *     -NavController navController\n *     +ngOnInit()\n *     +handleClick()\n *   }\n *   EmptyStateComponent --|> NgxBaseComponent\n *   EmptyStateComponent --|> OnInit\n *\n * @extends {NgxBaseComponent}\n * @implements {OnInit}\n */\n@Dynamic()\n@Component({\n  selector: 'ngx-decaf-empty-state',\n  templateUrl: './empty-state.component.html',\n  styleUrls: ['./empty-state.component.scss'],\n  standalone: true,\n  imports: [\n    ForAngularModule,\n    IonCard,\n    IonCardContent,\n    IonTitle,\n    IonIcon\n  ]\n\n})\nexport class EmptyStateComponent extends NgxBaseComponent implements OnInit {\n\n  /**\n   * @description The main title displayed in the empty state.\n   * @summary Specifies the primary message to show in the empty state component.\n   * This text is typically used to inform the user about why they're seeing an empty view.\n   * If translatable is true, this will be processed through the localization system.\n   *\n   * @type {string}\n   * @default \"title\"\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  title: string = \"title\";\n\n  /**\n   * @description The color of the title text.\n   * @summary Specifies the color for the title text using the application's color system.\n   * The value should correspond to a color variable defined in the application's theme.\n   * The component will automatically prefix this with \"color-\" to create the CSS class.\n   *\n   * @type {string}\n   * @default 'gray-6'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  titleColor = 'gray-6';\n\n  /**\n   * @description The secondary message displayed in the empty state.\n   * @summary Provides additional context or instructions below the main title.\n   * This text is typically used to guide the user on what actions they can take.\n   * If translatable is true, this will be processed through the localization system.\n   *\n   * @type {string | undefined}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  subtitle: string = \"\";\n\n  /**\n   * @description The color of the subtitle text.\n   * @summary Specifies the color for the subtitle text using the application's color system.\n   * The value should correspond to a color variable defined in the application's theme.\n   * The component will automatically prefix this with \"color-\" to create the CSS class.\n   *\n   * @type {string}\n   * @default 'gray-4'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  subtitleColor: string = 'gray-4';\n\n  /**\n   * @description Controls whether the icon is displayed.\n   * @summary Determines if the visual icon should be shown in the empty state.\n   * This can be provided as a boolean or a string that will be converted to a boolean.\n   * Icons help visually communicate the empty state context to users.\n   *\n   * @type {StringOrBoolean}\n   * @default true\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  showIcon: StringOrBoolean = true;\n\n  /**\n   * @description The name of the icon to display.\n   * @summary Specifies which icon to show when showIcon is true.\n   * The component uses the icon system defined in the application,\n   * and this value should correspond to an available icon name.\n   *\n   * @type {string}\n   * @default \"ti-info-square-rounded\"\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  icon: string = \"ti-info-square-rounded\";\n\n  /**\n   * @description The size of the displayed icon.\n   * @summary Controls the size of the icon shown in the empty state.\n   * Can be either 'large' or 'small' to accommodate different layout needs.\n   *\n   * @type {'large' | 'small' | undefined}\n   * @default 'large'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  iconSize?: 'large' | 'small' = 'large';\n\n  /**\n   * @description The color of the displayed icon.\n   * @summary Specifies the color for the icon using Ionic's predefined color system.\n   * This allows the icon to match the application's color scheme.\n   *\n   * @type {PredefinedColors | undefined}\n   * @default 'medium'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  iconColor?: PredefinedColors = 'medium';\n\n  /**\n   * @description The navigation target or action for the button.\n   * @summary Specifies where the button should navigate to when clicked or what function\n   * it should execute. This can be either a URL string or a function that handles navigation.\n   * When not provided, the button will not perform any action.\n   *\n   * @type {string | FunctionLike | undefined}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonLink?: string | FunctionLike;\n\n  /**\n   * @description The text displayed on the action button.\n   * @summary Specifies the label for the action button in the empty state.\n   * If translatable is true, this will be processed through the localization system.\n   * If not provided, the button will not display any text.\n   *\n   * @type {string | undefined}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonText?: string;\n\n  /**\n   * @description The fill style of the action button.\n   * @summary Controls the visual style of the button using Ionic's button fill options.\n   * 'solid' creates a button with a solid background, 'outline' creates a button with\n   * just a border, and 'clear' creates a button with no background or border.\n   *\n   * @type {'clear' | 'solid' | 'outline'}\n   * @default 'solid'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonFill: 'clear' | 'solid' | 'outline' =  'solid';\n\n  /**\n   * @description The color of the action button.\n   * @summary Specifies the color for the button using Ionic's color system.\n   * This allows the button to match the application's color scheme.\n   *\n   * @type {Color}\n   * @default 'primary'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonColor: Color =  'primary';\n\n  /**\n   * @description The size of the action button.\n   * @summary Controls the size of the button shown in the empty state.\n   * Can be 'large', 'small', or 'default' to accommodate different layout needs.\n   *\n   * @type {'large' | 'small' | 'default'}\n   * @default 'default'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonSize: 'large' | 'small' | 'default' =  'default';\n\n  /**\n   * @description The search value that resulted in no results.\n   * @summary When the empty state is shown due to a search with no results,\n   * this property can hold the search term that was used. This can be displayed\n   * in the empty state message to provide context to the user.\n   *\n   * @type {string}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  searchValue!: string;\n\n  /**\n   * @description Service for handling navigation operations.\n   * @summary Injected service that provides methods for navigating between routes.\n   * This service is used when the buttonLink is a string URL to navigate to that location.\n   *\n   * @private\n   * @type {NavController}\n   * @memberOf EmptyStateComponent\n   */\n  private navController: NavController = inject(NavController);\n\n  private sanitizer: DomSanitizer = inject(DomSanitizer);\n\n  private translate: TranslateService = inject(TranslateService);\n\n  searchSubtitle!: SafeHtml\n\n\n  /**\n   * @description Creates an instance of EmptyStateComponent.\n   * @summary Initializes a new EmptyStateComponent by calling the parent class constructor\n   * with the component name for logging and identification purposes. This component provides\n   * a standardized way to display empty state messages with optional icons and action buttons.\n   *\n   * @memberOf EmptyStateComponent\n   */\n  constructor() {\n    super(\"EmptyStateComponent\");\n    addIcons(allIcons);\n  }\n\n  /**\n   * @description Initializes the component after Angular first displays the data-bound properties.\n   * @summary Sets up the component by processing boolean inputs, applying localization to text\n   * elements if translation is enabled, and formatting CSS classes for title and subtitle colors.\n   * This method prepares the component for user interaction by ensuring all properties are\n   * properly initialized and localized.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant E as EmptyStateComponent\n   *\n   *   A->>E: ngOnInit()\n   *   E->>E: Process translatable flag\n   *   E->>E: Process showIcon flag\n   *   E->>E: Get locale settings\n   *   alt translatable is true\n   *     E->>E: Localize title\n   *     E->>E: Localize subtitle\n   *     E->>E: Localize buttonText\n   *   end\n   *   E->>E: Format title CSS class\n   *   E->>E: Format subtitle CSS class\n   *\n   * @return {Promise<void>}\n   * @memberOf EmptyStateComponent\n   */\n  async ngOnInit(): Promise<void> {\n    this.parseProps(this);\n    this.translatable = stringToBoolean(this.translatable);\n    this.showIcon = stringToBoolean(this.showIcon);\n    this.locale = this.getLocale(this.translatable);\n\n    // if(this.translatable) {\n    //   this.title = generateLocaleFromString(this.locale, this.title);\n    //   this.subtitle = generateLocaleFromString(this.locale, this.subtitle);\n    //   this.buttonText = generateLocaleFromString(this.locale, this.buttonText);\n    // }\n\n    this.titleColor = `dcf-title color-${this.titleColor}`;\n    this.subtitleColor = `dcf-subtitle color-${this.titleColor}`;\n\n    if(this.searchValue && this.translatable)\n      this.searchSubtitle = await this.getSearchSubtitle(this.subtitle as string);\n  }\n\n  /**\n   * @description Handles click events on the action button.\n   * @summary This method is triggered when the user clicks the action button in the empty state\n   * component. It supports three navigation patterns: 1) no action when buttonLink is not provided,\n   * 2) custom function execution when buttonLink is a function, and 3) navigation to a specific URL\n   * when buttonLink is a string. This flexibility allows the empty state to trigger various actions\n   * based on the context in which it's used.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant E as EmptyStateComponent\n   *   participant N as NavController\n   *\n   *   U->>E: Click action button\n   *   E->>E: handleClick()\n   *   alt buttonLink is not provided\n   *     E-->>U: Return false (no action)\n   *   else buttonLink is a function\n   *     E->>E: Execute buttonLink function\n   *     E-->>U: Return function result\n   *   else buttonLink is a URL string\n   *     E->>N: navigateForward(buttonLink)\n   *     N-->>E: Return navigation result\n   *     E-->>U: Return navigation result\n   *   end\n   *\n   * @return {boolean | void | Promise<boolean>}\n   *   - false if no action is taken\n   *   - The result of the buttonLink function if it's a function\n   *   - A Promise resolving to the navigation result if buttonLink is a URL\n   * @memberOf EmptyStateComponent\n   */\n  handleClick(): boolean | void | Promise<boolean> {\n    const fn = this.buttonLink;\n    if(!fn)\n      return false;\n    if(fn instanceof Function)\n      return fn();\n    return this.navController.navigateForward(fn as string);\n  }\n\n\n /**\n   * @description Generates a localized and sanitized subtitle for search results.\n   * @summary This method takes a content string, typically the subtitle, and processes it\n   * through the translation service. It replaces a placeholder ('value0') with the actual\n   * search value, then sanitizes the result to safely use as HTML. This is particularly\n   * useful for displaying dynamic, localized messages in the empty state when a search\n   * yields no results.\n   *\n   * @param {string} content - The content string to be translated and processed\n   * @return {Promise<SafeHtml>} A promise that resolves to a sanitized HTML string\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant E as EmptyStateComponent\n   *   participant T as TranslateService\n   *   participant S as DomSanitizer\n   *\n   *   E->>T: instant(content, {'value0': searchValue})\n   *   T-->>E: Return translated string\n   *   E->>S: bypassSecurityTrustHtml(translatedString)\n   *   S-->>E: Return sanitized SafeHtml\n   *\n   * @memberOf EmptyStateComponent\n   */\n  async getSearchSubtitle(content: string): Promise<SafeHtml> {\n    const result = await this.translate.instant(content, {'value0': this.searchValue});\n    return this.sanitizer.bypassSecurityTrustHtml(result);\n  }\n}\n","\n<ion-card id=\"dcf-card-empty-state\" [ngClass]=\"className\">\n  <ion-card-content>\n    @if(icon && showIcon) {\n      <div class=\"dcf-icon-container\">\n        <ion-icon\n          name=\"alert-circle-outline\"\n          size=\"large\"\n          color=\"danger\"\n          />\n      </div>\n    }\n    @if(title) {\n      <h5 [class]=\"titleColor\" [innerHTML]=\"title\"></h5>\n    }\n    @if(subtitle) {\n      @if(!searchValue) {\n        <p [class]=\"subtitleColor\" [innerHTML]=\"subtitle\"></p>\n      } @else {\n        <p [class]=\"subtitleColor\" [innerHTML]=\"searchSubtitle\"></p>\n      }\n    }\n    @if(buttonLink && buttonText) {\n      <div>\n        <ion-button\n          [size]=\"buttonSize\"\n          [fill]=\"buttonFill\"\n          [color]=\"buttonColor\"\n          (click)=\"handleClick()\">\n          {{  buttonText }}\n        </ion-button>\n      </div>\n    }\n  </ion-card-content>\n</ion-card>\n"]}
348
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"empty-state.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/empty-state/empty-state.component.ts","../../../../../src/lib/components/empty-state/empty-state.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnInit  } from '@angular/core';\nimport { Color, PredefinedColors } from '@ionic/core';\nimport {\n  IonCard,\n  IonCardContent,\n  IonIcon,\n  IonTitle,\n  NavController\n}\nfrom '@ionic/angular/standalone';\nimport * as allIcons from 'ionicons/icons';\nimport { addIcons } from 'ionicons';\nimport { ForAngularModule } from '../../for-angular.module';\nimport { Dynamic, StringOrBoolean } from '../../engine';\nimport { stringToBoolean } from '../../helpers';\nimport { NgxBaseComponent } from '../../engine/NgxBaseComponent';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\nimport { TranslateService } from '@ngx-translate/core';\nimport { FunctionLike } from '../../engine/types';\n\n\n/**\n * @description Component for displaying empty state messages with optional actions.\n * @summary This component provides a standardized way to display empty state messages\n * when no data is available or when a user needs to take an action to populate content.\n * It includes customizable title, subtitle, icon, and action button elements that can be\n * styled and configured through input properties. The component supports localization\n * and can trigger navigation or custom actions when the button is clicked.\n *\n * @mermaid\n * classDiagram\n *   class EmptyStateComponent {\n *     +string title\n *     +string titleColor\n *     +string subtitle\n *     +string subtitleColor\n *     +StringOrBoolean showIcon\n *     +string icon\n *     +string iconSize\n *     +PredefinedColors iconColor\n *     +string|Function buttonLink\n *     +string buttonText\n *     +string buttonFill\n *     +Color buttonColor\n *     +string buttonSize\n *     +string searchValue\n *     -NavController navController\n *     +ngOnInit()\n *     +handleClick()\n *   }\n *   EmptyStateComponent --|> NgxBaseComponent\n *   EmptyStateComponent --|> OnInit\n *\n * @extends {NgxBaseComponent}\n * @implements {OnInit}\n */\n@Dynamic()\n@Component({\n  selector: 'ngx-decaf-empty-state',\n  templateUrl: './empty-state.component.html',\n  styleUrls: ['./empty-state.component.scss'],\n  standalone: true,\n  imports: [\n    ForAngularModule,\n    IonCard,\n    IonCardContent,\n    IonTitle,\n    IonIcon\n  ]\n\n})\nexport class EmptyStateComponent extends NgxBaseComponent implements OnInit {\n\n  /**\n   * @description The main title displayed in the empty state.\n   * @summary Specifies the primary message to show in the empty state component.\n   * This text is typically used to inform the user about why they're seeing an empty view.\n   * If translatable is true, this will be processed through the localization system.\n   *\n   * @type {string}\n   * @default \"title\"\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  title: string = \"title\";\n\n  /**\n   * @description The color of the title text.\n   * @summary Specifies the color for the title text using the application's color system.\n   * The value should correspond to a color variable defined in the application's theme.\n   * The component will automatically prefix this with \"color-\" to create the CSS class.\n   *\n   * @type {string}\n   * @default 'gray-6'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  titleColor = 'gray-6';\n\n  /**\n   * @description The secondary message displayed in the empty state.\n   * @summary Provides additional context or instructions below the main title.\n   * This text is typically used to guide the user on what actions they can take.\n   * If translatable is true, this will be processed through the localization system.\n   *\n   * @type {string | undefined}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  subtitle: string = \"\";\n\n  /**\n   * @description The color of the subtitle text.\n   * @summary Specifies the color for the subtitle text using the application's color system.\n   * The value should correspond to a color variable defined in the application's theme.\n   * The component will automatically prefix this with \"color-\" to create the CSS class.\n   *\n   * @type {string}\n   * @default 'gray-4'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  subtitleColor: string = 'gray-4';\n\n  /**\n   * @description Controls whether the icon is displayed.\n   * @summary Determines if the visual icon should be shown in the empty state.\n   * This can be provided as a boolean or a string that will be converted to a boolean.\n   * Icons help visually communicate the empty state context to users.\n   *\n   * @type {StringOrBoolean}\n   * @default true\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  showIcon: StringOrBoolean = true;\n\n  /**\n   * @description The name of the icon to display.\n   * @summary Specifies which icon to show when showIcon is true.\n   * The component uses the icon system defined in the application,\n   * and this value should correspond to an available icon name.\n   *\n   * @type {string}\n   * @default \"ti-info-square-rounded\"\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  icon: string = \"ti-info-square-rounded\";\n\n  /**\n   * @description The size of the displayed icon.\n   * @summary Controls the size of the icon shown in the empty state.\n   * Can be either 'large' or 'small' to accommodate different layout needs.\n   *\n   * @type {'large' | 'small' | undefined}\n   * @default 'large'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  iconSize?: 'large' | 'small' = 'large';\n\n  /**\n   * @description The color of the displayed icon.\n   * @summary Specifies the color for the icon using Ionic's predefined color system.\n   * This allows the icon to match the application's color scheme.\n   *\n   * @type {PredefinedColors | undefined}\n   * @default 'medium'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  iconColor?: PredefinedColors = 'medium';\n\n  /**\n   * @description The navigation target or action for the button.\n   * @summary Specifies where the button should navigate to when clicked or what function\n   * it should execute. This can be either a URL string or a function that handles navigation.\n   * When not provided, the button will not perform any action.\n   *\n   * @type {string | FunctionLike | undefined}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonLink?: string | FunctionLike;\n\n  /**\n   * @description The text displayed on the action button.\n   * @summary Specifies the label for the action button in the empty state.\n   * If translatable is true, this will be processed through the localization system.\n   * If not provided, the button will not display any text.\n   *\n   * @type {string | undefined}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonText?: string;\n\n  /**\n   * @description The fill style of the action button.\n   * @summary Controls the visual style of the button using Ionic's button fill options.\n   * 'solid' creates a button with a solid background, 'outline' creates a button with\n   * just a border, and 'clear' creates a button with no background or border.\n   *\n   * @type {'clear' | 'solid' | 'outline'}\n   * @default 'solid'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonFill: 'clear' | 'solid' | 'outline' =  'solid';\n\n  /**\n   * @description The color of the action button.\n   * @summary Specifies the color for the button using Ionic's color system.\n   * This allows the button to match the application's color scheme.\n   *\n   * @type {Color}\n   * @default 'primary'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonColor: Color =  'primary';\n\n  /**\n   * @description The size of the action button.\n   * @summary Controls the size of the button shown in the empty state.\n   * Can be 'large', 'small', or 'default' to accommodate different layout needs.\n   *\n   * @type {'large' | 'small' | 'default'}\n   * @default 'default'\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  buttonSize: 'large' | 'small' | 'default' =  'default';\n\n  /**\n   * @description The search value that resulted in no results.\n   * @summary When the empty state is shown due to a search with no results,\n   * this property can hold the search term that was used. This can be displayed\n   * in the empty state message to provide context to the user.\n   *\n   * @type {string}\n   * @memberOf EmptyStateComponent\n   */\n  @Input()\n  searchValue!: string;\n\n  /**\n   * @description Service for handling navigation operations.\n   * @summary Injected service that provides methods for navigating between routes.\n   * This service is used when the buttonLink is a string URL to navigate to that location.\n   *\n   * @private\n   * @type {NavController}\n   * @memberOf EmptyStateComponent\n   */\n  private navController: NavController = inject(NavController);\n\n  private sanitizer: DomSanitizer = inject(DomSanitizer);\n\n  private translate: TranslateService = inject(TranslateService);\n\n  searchSubtitle!: SafeHtml\n\n\n  /**\n   * @description Creates an instance of EmptyStateComponent.\n   * @summary Initializes a new EmptyStateComponent by calling the parent class constructor\n   * with the component name for logging and identification purposes. This component provides\n   * a standardized way to display empty state messages with optional icons and action buttons.\n   *\n   * @memberOf EmptyStateComponent\n   */\n  constructor() {\n    super(\"EmptyStateComponent\");\n    addIcons(allIcons);\n  }\n\n  /**\n   * @description Initializes the component after Angular first displays the data-bound properties.\n   * @summary Sets up the component by processing boolean inputs, applying localization to text\n   * elements if translation is enabled, and formatting CSS classes for title and subtitle colors.\n   * This method prepares the component for user interaction by ensuring all properties are\n   * properly initialized and localized.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant E as EmptyStateComponent\n   *\n   *   A->>E: ngOnInit()\n   *   E->>E: Process translatable flag\n   *   E->>E: Process showIcon flag\n   *   E->>E: Get locale settings\n   *   alt translatable is true\n   *     E->>E: Localize title\n   *     E->>E: Localize subtitle\n   *     E->>E: Localize buttonText\n   *   end\n   *   E->>E: Format title CSS class\n   *   E->>E: Format subtitle CSS class\n   *\n   * @return {Promise<void>}\n   * @memberOf EmptyStateComponent\n   */\n  async ngOnInit(): Promise<void> {\n    this.parseProps(this);\n    this.translatable = stringToBoolean(this.translatable);\n    this.showIcon = stringToBoolean(this.showIcon);\n    this.locale = this.getLocale(this.translatable);\n\n    // if(this.translatable) {\n    //   this.title = generateLocaleFromString(this.locale, this.title);\n    //   this.subtitle = generateLocaleFromString(this.locale, this.subtitle);\n    //   this.buttonText = generateLocaleFromString(this.locale, this.buttonText);\n    // }\n\n    this.titleColor = `dcf-title color-${this.titleColor}`;\n    this.subtitleColor = `dcf-subtitle color-${this.titleColor}`;\n\n    if(this.searchValue && this.translatable)\n      this.searchSubtitle = await this.getSearchSubtitle(this.subtitle as string);\n  }\n\n  /**\n   * @description Handles click events on the action button.\n   * @summary This method is triggered when the user clicks the action button in the empty state\n   * component. It supports three navigation patterns: 1) no action when buttonLink is not provided,\n   * 2) custom function execution when buttonLink is a function, and 3) navigation to a specific URL\n   * when buttonLink is a string. This flexibility allows the empty state to trigger various actions\n   * based on the context in which it's used.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant E as EmptyStateComponent\n   *   participant N as NavController\n   *\n   *   U->>E: Click action button\n   *   E->>E: handleClick()\n   *   alt buttonLink is not provided\n   *     E-->>U: Return false (no action)\n   *   else buttonLink is a function\n   *     E->>E: Execute buttonLink function\n   *     E-->>U: Return function result\n   *   else buttonLink is a URL string\n   *     E->>N: navigateForward(buttonLink)\n   *     N-->>E: Return navigation result\n   *     E-->>U: Return navigation result\n   *   end\n   *\n   * @return {boolean | void | Promise<boolean>}\n   *   - false if no action is taken\n   *   - The result of the buttonLink function if it's a function\n   *   - A Promise resolving to the navigation result if buttonLink is a URL\n   * @memberOf EmptyStateComponent\n   */\n  handleClick(): boolean | void | Promise<boolean> {\n    const fn = this.buttonLink;\n    if(!fn)\n      return false;\n    if(fn instanceof Function)\n      return fn();\n    return this.navController.navigateForward(fn as string);\n  }\n\n\n /**\n   * @description Generates a localized and sanitized subtitle for search results.\n   * @summary This method takes a content string, typically the subtitle, and processes it\n   * through the translation service. It replaces a placeholder ('value0') with the actual\n   * search value, then sanitizes the result to safely use as HTML. This is particularly\n   * useful for displaying dynamic, localized messages in the empty state when a search\n   * yields no results.\n   *\n   * @param {string} content - The content string to be translated and processed\n   * @return {Promise<SafeHtml>} A promise that resolves to a sanitized HTML string\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant E as EmptyStateComponent\n   *   participant T as TranslateService\n   *   participant S as DomSanitizer\n   *\n   *   E->>T: instant(content, {'value0': searchValue})\n   *   T-->>E: Return translated string\n   *   E->>S: bypassSecurityTrustHtml(translatedString)\n   *   S-->>E: Return sanitized SafeHtml\n   *\n   * @memberOf EmptyStateComponent\n   */\n  async getSearchSubtitle(content: string): Promise<SafeHtml> {\n    const result = await this.translate.instant(content, {'value0': this.searchValue});\n    return this.sanitizer.bypassSecurityTrustHtml(result);\n  }\n}\n","\n<ion-card id=\"dcf-card-empty-state\" [ngClass]=\"className\">\n  <ion-card-content>\n    @if(icon && showIcon) {\n      <div class=\"dcf-icon-container\">\n        <ion-icon\n          name=\"alert-circle-outline\"\n          size=\"large\"\n          color=\"danger\"\n          />\n      </div>\n    }\n    @if(title) {\n      <h5 [class]=\"titleColor\" [innerHTML]=\"title\"></h5>\n    }\n    @if(subtitle) {\n      @if(!searchValue) {\n        <p [class]=\"subtitleColor\" [innerHTML]=\"subtitle\"></p>\n      } @else {\n        <p [class]=\"subtitleColor\" [innerHTML]=\"searchSubtitle\"></p>\n      }\n    }\n    @if(buttonLink && buttonText) {\n      <div>\n        <ion-button\n          [size]=\"buttonSize\"\n          [fill]=\"buttonFill\"\n          [color]=\"buttonColor\"\n          (click)=\"handleClick()\">\n          {{  buttonText }}\n        </ion-button>\n      </div>\n    }\n  </ion-card-content>\n</ion-card>\n"]}
@@ -1,9 +1,9 @@
1
1
  import { __decorate } from "tslib";
2
2
  import { ChangeDetectorRef, Component, CUSTOM_ELEMENTS_SCHEMA, ElementRef, inject, Input, ViewChild } from '@angular/core';
3
- import { Dynamic } from 'src/lib/engine';
3
+ import { Dynamic } from '../../engine';
4
4
  import { OperationKeys } from '@decaf-ts/db-decorators';
5
- import { ForAngularModule } from 'src/lib/for-angular.module';
6
- import { CollapsableDirective } from 'src/lib/directives/collapsable.directive';
5
+ import { ForAngularModule } from '../../for-angular.module';
6
+ import { CollapsableDirective } from '../../directives/collapsable.directive';
7
7
  import { IonAccordion, IonAccordionGroup, IonItem } from '@ionic/angular/standalone';
8
8
  import * as i0 from "@angular/core";
9
9
  /**
@@ -222,4 +222,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
222
222
  }], target: [{
223
223
  type: Input
224
224
  }] } });
225
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fieldset.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/fieldset/fieldset.component.ts","../../../../../src/lib/components/fieldset/fieldset.component.html"],"names":[],"mappings":";AACA,OAAO,EAAiB,iBAAiB,EAAE,SAAS,EAAE,sBAAsB,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1I,OAAO,EAAE,OAAO,EAAkB,MAAM,gBAAgB,CAAC;AACzD,OAAO,EAAkB,aAAa,EAAE,MAAM,yBAAyB,CAAC;AACxE,OAAO,EAAE,gBAAgB,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,EAAE,oBAAoB,EAAE,MAAM,0CAA0C,CAAC;AAChF,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;;AAErF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiDG;AAUI,IAAM,iBAAiB,GAAvB,MAAM,iBAAiB;IAAvB;QAeL;;;;;;;;;;WAUG;QAEH,SAAI,GAAW,OAAO,CAAC;QAEvB;;;;;;;;;;WAUG;QAEH,cAAS,GAAkB,aAAa,CAAC,IAAI,CAAC;QAE9C;;;;;;;;;;WAUG;QAEH,WAAM,GAAmB,OAAO,CAAC;QAEjC;;;;;;;;;;WAUG;QACH,WAAM,GAAY,KAAK,CAAC;QAExB;;;;;;;;;;WAUG;QACgB,kBAAa,GAAmB,aAAa,CAAC,MAAM,CAAC;QAExE;;;;;;;;;;WAUG;QACK,sBAAiB,GAAsB,MAAM,CAAC,iBAAiB,CAAC,CAAC;KA0E1E;IAxEC;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA4BG;IACH,eAAe;QACb,IAAI,IAAI,CAAC,SAAS,KAAK,aAAa,CAAC,IAAI,IAAI,IAAI,CAAC,SAAS,KAAK,aAAa,CAAC,MAAM,EAAE,CAAC;YACrF,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;YACnB,MAAM,gBAAgB,GAAG,IAAI,CAAC,SAAS,EAAE,aAAa,CAAC,aAAa,CAAC,qBAAqB,CAAC,CAAC;YAC5F,IAAG,gBAAgB;gBACjB,gBAAgB,CAAC,YAAY,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QACnD,CAAC;QACD,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;IACzC,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;OA0BG;IACH,YAAY,CAAC,KAAkB;QAC7B,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,KAAK,CAAC;QACjC,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,CAAC;QACzB,IAAK,MAAuC,CAAC,OAAO,KAAK,qBAAqB;YAC5E,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,KAAK,CAAC;IAC1B,CAAC;+GAvKU,iBAAiB;mGAAjB,iBAAiB,qOAYmB,UAAU,6BC/E3D,0bAYA,owBDqDY,gBAAgB,+BAAE,iBAAiB,qJAAE,YAAY,6IAAE,OAAO,0NAAE,oBAAoB;;AAE/E,iBAAiB;IAT7B,OAAO,EAAE;GASG,iBAAiB,CAwK7B;;4FAxKY,iBAAiB;kBAR7B,SAAS;iCACI,IAAI,YACN,oBAAoB,WAGrB,CAAC,sBAAsB,CAAC,WACxB,CAAC,gBAAgB,EAAE,iBAAiB,EAAE,YAAY,EAAE,OAAO,EAAE,oBAAoB,CAAC;8BAe3F,SAAS;sBADR,SAAS;uBAAC,WAAW,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE;gBAe3D,IAAI;sBADH,KAAK;gBAeN,SAAS;sBADR,KAAK;gBAeN,MAAM;sBADL,KAAK","sourcesContent":["\nimport { AfterViewInit, ChangeDetectorRef, Component, CUSTOM_ELEMENTS_SCHEMA, ElementRef, inject, Input, ViewChild } from '@angular/core';\nimport { Dynamic, HTMLFormTarget } from 'src/lib/engine';\nimport { CrudOperations, OperationKeys } from '@decaf-ts/db-decorators';\nimport { ForAngularModule } from 'src/lib/for-angular.module';\nimport { CollapsableDirective } from 'src/lib/directives/collapsable.directive';\nimport { IonAccordion, IonAccordionGroup, IonItem } from '@ionic/angular/standalone';\n\n/**\n * @description Dynamic fieldset component with collapsible accordion functionality.\n * @summary This component provides a sophisticated fieldset container that automatically\n * adapts its behavior based on CRUD operations. It integrates seamlessly with Ionic's\n * accordion components to create expandable/collapsible sections for organizing form\n * content and related information. The component intelligently determines its initial\n * state based on the operation type, opening automatically for READ and DELETE operations\n * while remaining closed for CREATE and UPDATE operations.\n *\n * @example\n * ```html\n * <!-- Basic usage with automatic state management -->\n * <ngx-decaf-fieldset\n *   name=\"Personal Information\"\n *   [operation]=\"OperationKeys.READ\"\n *   target=\"_self\">\n *   <ion-input label=\"Name\" placeholder=\"Enter name\"></ion-input>\n *   <ion-input label=\"Email\" type=\"email\" placeholder=\"Enter email\"></ion-input>\n * </ngx-decaf-fieldset>\n *\n * <!-- Advanced usage with custom operation -->\n * <ngx-decaf-fieldset\n *   name=\"Contact Details\"\n *   [operation]=\"currentOperation\"\n *   target=\"_blank\">\n *   <!-- Complex form fields -->\n * </ngx-decaf-fieldset>\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant U as User\n *   participant F as FieldsetComponent\n *   participant I as Ionic Accordion\n *   participant D as DOM\n *\n *   F->>F: ngAfterViewInit()\n *   alt operation is READ or DELETE\n *     F->>F: Set isOpen = true\n *     F->>D: Query accordion element\n *     F->>I: Set value attribute to 'open'\n *     F->>F: Trigger change detection\n *   end\n *   U->>I: Click accordion header\n *   I->>F: handleChange(event)\n *   F->>F: Update isOpen state\n *   F->>I: Reflect new state\n *\n * @memberOf ForAngularModule\n */\n@Dynamic()\n@Component({\n  standalone: true,\n  selector: 'ngx-decaf-fieldset',\n  templateUrl: './fieldset.component.html',\n  styleUrls: ['./fieldset.component.scss'],\n  schemas: [CUSTOM_ELEMENTS_SCHEMA],\n  imports: [ForAngularModule, IonAccordionGroup, IonAccordion, IonItem, CollapsableDirective],\n})\nexport class FieldsetComponent implements AfterViewInit {\n\n  /**\n   * @description Reference to the component's native DOM element.\n   * @summary ViewChild reference that provides direct access to the component's root DOM element.\n   * This is essential for manipulating the Ionic accordion group after view initialization,\n   * particularly for setting the initial open state programmatically. The reference is used\n   * to query and modify accordion attributes that control the component's expanded state.\n   *\n   * @type {ElementRef}\n   * @memberOf FieldsetComponent\n   */\n  @ViewChild('component', { static: false, read: ElementRef })\n  component!: ElementRef;\n\n  /**\n   * @description The display name or title of the fieldset section.\n   * @summary Sets the legend or header text that appears in the accordion header. This text\n   * provides a clear label for the collapsible section, helping users understand what content\n   * is contained within. The name is displayed prominently and serves as the clickable area\n   * for expanding/collapsing the fieldset.\n   *\n   * @type {string}\n   * @default 'Child'\n   * @memberOf FieldsetComponent\n   */\n  @Input()\n  name: string = 'Child';\n\n  /**\n   * @description The current CRUD operation context.\n   * @summary Determines the component's initial behavior and state based on the current operation.\n   * This input is crucial for auto-state management: READ and DELETE operations automatically\n   * open the fieldset to show content, while CREATE and UPDATE operations keep it closed\n   * initially. This provides an intuitive user experience aligned with operation semantics.\n   *\n   * @type {OperationKeys}\n   * @default OperationKeys.READ\n   * @memberOf FieldsetComponent\n   */\n  @Input()\n  operation: OperationKeys = OperationKeys.READ;\n\n  /**\n   * @description Form target attribute for nested form submissions.\n   * @summary Specifies where to display the response after submitting forms contained within\n   * the fieldset. This attribute mirrors the HTML form target behavior, allowing control over\n   * whether form submissions open in the same window, new window, or specific frame. Useful\n   * for complex form workflows and multi-step processes.\n   *\n   * @type {HTMLFormTarget}\n   * @default '_self'\n   * @memberOf FieldsetComponent\n   */\n  @Input()\n  target: HTMLFormTarget = '_self';\n\n  /**\n   * @description Current state of the accordion (expanded or collapsed).\n   * @summary Boolean flag that tracks whether the fieldset accordion is currently open or closed.\n   * This property is automatically managed based on user interactions and initial operation state.\n   * It serves as the single source of truth for the component's visibility state and is used\n   * to coordinate between user actions and programmatic state changes.\n   *\n   * @type {boolean}\n   * @default false\n   * @memberOf FieldsetComponent\n   */\n  isOpen: boolean = false;\n\n  /**\n   * @description Reference to CRUD operation constants for template usage.\n   * @summary Exposes the OperationKeys enum to the component template, enabling conditional\n   * rendering and behavior based on operation types. This protected readonly property ensures\n   * that template logic can access operation constants while maintaining encapsulation and\n   * preventing accidental modification of the enum values.\n   *\n   * @type {CrudOperations}\n   * @default OperationKeys.CREATE\n   * @memberOf FieldsetComponent\n   */\n  protected readonly OperationKeys: CrudOperations = OperationKeys.CREATE;\n\n  /**\n   * @description Angular change detection service.\n   * @summary Injected service that provides manual control over change detection cycles.\n   * This is essential for ensuring that programmatic DOM changes (like setting accordion\n   * attributes) are properly reflected in the component's state and trigger appropriate\n   * view updates when modifications occur outside the normal Angular change detection flow.\n   *\n   * @private\n   * @type {ChangeDetectorRef}\n   * @memberOf FieldsetComponent\n   */\n  private changeDetectorRef: ChangeDetectorRef = inject(ChangeDetectorRef);\n\n  /**\n   * @description Initializes the component state after view and child components are rendered.\n   * @summary This lifecycle hook implements intelligent auto-state management based on the current\n   * CRUD operation. For READ and DELETE operations, the fieldset automatically opens to provide\n   * immediate access to information, while CREATE and UPDATE operations keep it closed to maintain\n   * a clean initial interface. The method directly manipulates the DOM to ensure proper accordion\n   * synchronization and triggers change detection to reflect the programmatic state changes.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant F as FieldsetComponent\n   *   participant D as DOM\n   *   participant C as ChangeDetector\n   *\n   *   A->>F: ngAfterViewInit()\n   *   alt operation is READ or DELETE\n   *     F->>F: Set isOpen = true\n   *     F->>D: Query ion-accordion-group element\n   *     alt accordion element exists\n   *       F->>D: Set value attribute to 'open'\n   *     end\n   *   end\n   *   F->>C: detectChanges()\n   *   C->>F: Update view with new state\n   *\n   * @returns {void}\n   * @memberOf FieldsetComponent\n   */\n  ngAfterViewInit(): void {\n    if (this.operation === OperationKeys.READ || this.operation === OperationKeys.DELETE) {\n      this.isOpen = true;\n      const accordionElement = this.component?.nativeElement.querySelector('ion-accordion-group');\n      if(accordionElement)\n        accordionElement.setAttribute('value', 'open');\n    }\n    this.changeDetectorRef.detectChanges();\n  }\n\n  /**\n   * @description Handles accordion state change events from user interactions.\n   * @summary Processes CustomEvent objects triggered when users expand or collapse the accordion.\n   * This method extracts the new state from the event details and updates the component's\n   * internal state accordingly. It specifically listens for ION-ACCORDION-GROUP events to\n   * ensure proper event source validation and prevent handling of unrelated events.\n   *\n   * @param {CustomEvent} event - The event object containing accordion state change details\n   * @returns {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant I as Ion-Accordion\n   *   participant F as FieldsetComponent\n   *\n   *   U->>I: Click accordion header\n   *   I->>F: handleChange(CustomEvent)\n   *   F->>F: Extract target and detail from event\n   *   F->>F: Validate target is ION-ACCORDION-GROUP\n   *   alt valid target\n   *     F->>F: Update isOpen = !!value\n   *   end\n   *   F->>I: Reflect updated state\n   *\n   * @memberOf FieldsetComponent\n   */\n  handleChange(event: CustomEvent): void {\n    const { target, detail } = event;\n    const { value } = detail;\n    if ((target as HTMLIonAccordionGroupElement).tagName === 'ION-ACCORDION-GROUP')\n      this.isOpen = !!value;\n  }\n}\n","<fieldset [class]=\"'dcf-fieldset ' + operation\" #component>\n  <ion-accordion-group (ionChange)=\"handleChange($event)\">\n    <ion-accordion value=\"open\">\n      <ion-item slot=\"header\" >\n        <legend>{{ name }}</legend>\n      </ion-item>\n      <div slot=\"content\" [attr.aria-hidden]=\"!isOpen\" decafCollapsable>\n        <ng-content></ng-content>\n      </div>\n    </ion-accordion>\n  </ion-accordion-group>\n</fieldset>\n"]}
225
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fieldset.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/fieldset/fieldset.component.ts","../../../../../src/lib/components/fieldset/fieldset.component.html"],"names":[],"mappings":";AACA,OAAO,EAAiB,iBAAiB,EAAE,SAAS,EAAE,sBAAsB,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1I,OAAO,EAAE,OAAO,EAAkB,MAAM,cAAc,CAAC;AACvD,OAAO,EAAkB,aAAa,EAAE,MAAM,yBAAyB,CAAC;AACxE,OAAO,EAAE,gBAAgB,EAAE,MAAM,0BAA0B,CAAC;AAC5D,OAAO,EAAE,oBAAoB,EAAE,MAAM,wCAAwC,CAAC;AAC9E,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;;AAErF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiDG;AAUI,IAAM,iBAAiB,GAAvB,MAAM,iBAAiB;IAAvB;QAeL;;;;;;;;;;WAUG;QAEH,SAAI,GAAW,OAAO,CAAC;QAEvB;;;;;;;;;;WAUG;QAEH,cAAS,GAAkB,aAAa,CAAC,IAAI,CAAC;QAE9C;;;;;;;;;;WAUG;QAEH,WAAM,GAAmB,OAAO,CAAC;QAEjC;;;;;;;;;;WAUG;QACH,WAAM,GAAY,KAAK,CAAC;QAExB;;;;;;;;;;WAUG;QACgB,kBAAa,GAAmB,aAAa,CAAC,MAAM,CAAC;QAExE;;;;;;;;;;WAUG;QACK,sBAAiB,GAAsB,MAAM,CAAC,iBAAiB,CAAC,CAAC;KA0E1E;IAxEC;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA4BG;IACH,eAAe;QACb,IAAI,IAAI,CAAC,SAAS,KAAK,aAAa,CAAC,IAAI,IAAI,IAAI,CAAC,SAAS,KAAK,aAAa,CAAC,MAAM,EAAE,CAAC;YACrF,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;YACnB,MAAM,gBAAgB,GAAG,IAAI,CAAC,SAAS,EAAE,aAAa,CAAC,aAAa,CAAC,qBAAqB,CAAC,CAAC;YAC5F,IAAG,gBAAgB;gBACjB,gBAAgB,CAAC,YAAY,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QACnD,CAAC;QACD,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;IACzC,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;OA0BG;IACH,YAAY,CAAC,KAAkB;QAC7B,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,KAAK,CAAC;QACjC,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,CAAC;QACzB,IAAK,MAAuC,CAAC,OAAO,KAAK,qBAAqB;YAC5E,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,KAAK,CAAC;IAC1B,CAAC;+GAvKU,iBAAiB;mGAAjB,iBAAiB,qOAYmB,UAAU,6BC/E3D,0bAYA,owBDqDY,gBAAgB,+BAAE,iBAAiB,qJAAE,YAAY,6IAAE,OAAO,0NAAE,oBAAoB;;AAE/E,iBAAiB;IAT7B,OAAO,EAAE;GASG,iBAAiB,CAwK7B;;4FAxKY,iBAAiB;kBAR7B,SAAS;iCACI,IAAI,YACN,oBAAoB,WAGrB,CAAC,sBAAsB,CAAC,WACxB,CAAC,gBAAgB,EAAE,iBAAiB,EAAE,YAAY,EAAE,OAAO,EAAE,oBAAoB,CAAC;8BAe3F,SAAS;sBADR,SAAS;uBAAC,WAAW,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE;gBAe3D,IAAI;sBADH,KAAK;gBAeN,SAAS;sBADR,KAAK;gBAeN,MAAM;sBADL,KAAK","sourcesContent":["\nimport { AfterViewInit, ChangeDetectorRef, Component, CUSTOM_ELEMENTS_SCHEMA, ElementRef, inject, Input, ViewChild } from '@angular/core';\nimport { Dynamic, HTMLFormTarget } from '../../engine';\nimport { CrudOperations, OperationKeys } from '@decaf-ts/db-decorators';\nimport { ForAngularModule } from '../../for-angular.module';\nimport { CollapsableDirective } from '../../directives/collapsable.directive';\nimport { IonAccordion, IonAccordionGroup, IonItem } from '@ionic/angular/standalone';\n\n/**\n * @description Dynamic fieldset component with collapsible accordion functionality.\n * @summary This component provides a sophisticated fieldset container that automatically\n * adapts its behavior based on CRUD operations. It integrates seamlessly with Ionic's\n * accordion components to create expandable/collapsible sections for organizing form\n * content and related information. The component intelligently determines its initial\n * state based on the operation type, opening automatically for READ and DELETE operations\n * while remaining closed for CREATE and UPDATE operations.\n *\n * @example\n * ```html\n * <!-- Basic usage with automatic state management -->\n * <ngx-decaf-fieldset\n *   name=\"Personal Information\"\n *   [operation]=\"OperationKeys.READ\"\n *   target=\"_self\">\n *   <ion-input label=\"Name\" placeholder=\"Enter name\"></ion-input>\n *   <ion-input label=\"Email\" type=\"email\" placeholder=\"Enter email\"></ion-input>\n * </ngx-decaf-fieldset>\n *\n * <!-- Advanced usage with custom operation -->\n * <ngx-decaf-fieldset\n *   name=\"Contact Details\"\n *   [operation]=\"currentOperation\"\n *   target=\"_blank\">\n *   <!-- Complex form fields -->\n * </ngx-decaf-fieldset>\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant U as User\n *   participant F as FieldsetComponent\n *   participant I as Ionic Accordion\n *   participant D as DOM\n *\n *   F->>F: ngAfterViewInit()\n *   alt operation is READ or DELETE\n *     F->>F: Set isOpen = true\n *     F->>D: Query accordion element\n *     F->>I: Set value attribute to 'open'\n *     F->>F: Trigger change detection\n *   end\n *   U->>I: Click accordion header\n *   I->>F: handleChange(event)\n *   F->>F: Update isOpen state\n *   F->>I: Reflect new state\n *\n * @memberOf ForAngularModule\n */\n@Dynamic()\n@Component({\n  standalone: true,\n  selector: 'ngx-decaf-fieldset',\n  templateUrl: './fieldset.component.html',\n  styleUrls: ['./fieldset.component.scss'],\n  schemas: [CUSTOM_ELEMENTS_SCHEMA],\n  imports: [ForAngularModule, IonAccordionGroup, IonAccordion, IonItem, CollapsableDirective],\n})\nexport class FieldsetComponent implements AfterViewInit {\n\n  /**\n   * @description Reference to the component's native DOM element.\n   * @summary ViewChild reference that provides direct access to the component's root DOM element.\n   * This is essential for manipulating the Ionic accordion group after view initialization,\n   * particularly for setting the initial open state programmatically. The reference is used\n   * to query and modify accordion attributes that control the component's expanded state.\n   *\n   * @type {ElementRef}\n   * @memberOf FieldsetComponent\n   */\n  @ViewChild('component', { static: false, read: ElementRef })\n  component!: ElementRef;\n\n  /**\n   * @description The display name or title of the fieldset section.\n   * @summary Sets the legend or header text that appears in the accordion header. This text\n   * provides a clear label for the collapsible section, helping users understand what content\n   * is contained within. The name is displayed prominently and serves as the clickable area\n   * for expanding/collapsing the fieldset.\n   *\n   * @type {string}\n   * @default 'Child'\n   * @memberOf FieldsetComponent\n   */\n  @Input()\n  name: string = 'Child';\n\n  /**\n   * @description The current CRUD operation context.\n   * @summary Determines the component's initial behavior and state based on the current operation.\n   * This input is crucial for auto-state management: READ and DELETE operations automatically\n   * open the fieldset to show content, while CREATE and UPDATE operations keep it closed\n   * initially. This provides an intuitive user experience aligned with operation semantics.\n   *\n   * @type {OperationKeys}\n   * @default OperationKeys.READ\n   * @memberOf FieldsetComponent\n   */\n  @Input()\n  operation: OperationKeys = OperationKeys.READ;\n\n  /**\n   * @description Form target attribute for nested form submissions.\n   * @summary Specifies where to display the response after submitting forms contained within\n   * the fieldset. This attribute mirrors the HTML form target behavior, allowing control over\n   * whether form submissions open in the same window, new window, or specific frame. Useful\n   * for complex form workflows and multi-step processes.\n   *\n   * @type {HTMLFormTarget}\n   * @default '_self'\n   * @memberOf FieldsetComponent\n   */\n  @Input()\n  target: HTMLFormTarget = '_self';\n\n  /**\n   * @description Current state of the accordion (expanded or collapsed).\n   * @summary Boolean flag that tracks whether the fieldset accordion is currently open or closed.\n   * This property is automatically managed based on user interactions and initial operation state.\n   * It serves as the single source of truth for the component's visibility state and is used\n   * to coordinate between user actions and programmatic state changes.\n   *\n   * @type {boolean}\n   * @default false\n   * @memberOf FieldsetComponent\n   */\n  isOpen: boolean = false;\n\n  /**\n   * @description Reference to CRUD operation constants for template usage.\n   * @summary Exposes the OperationKeys enum to the component template, enabling conditional\n   * rendering and behavior based on operation types. This protected readonly property ensures\n   * that template logic can access operation constants while maintaining encapsulation and\n   * preventing accidental modification of the enum values.\n   *\n   * @type {CrudOperations}\n   * @default OperationKeys.CREATE\n   * @memberOf FieldsetComponent\n   */\n  protected readonly OperationKeys: CrudOperations = OperationKeys.CREATE;\n\n  /**\n   * @description Angular change detection service.\n   * @summary Injected service that provides manual control over change detection cycles.\n   * This is essential for ensuring that programmatic DOM changes (like setting accordion\n   * attributes) are properly reflected in the component's state and trigger appropriate\n   * view updates when modifications occur outside the normal Angular change detection flow.\n   *\n   * @private\n   * @type {ChangeDetectorRef}\n   * @memberOf FieldsetComponent\n   */\n  private changeDetectorRef: ChangeDetectorRef = inject(ChangeDetectorRef);\n\n  /**\n   * @description Initializes the component state after view and child components are rendered.\n   * @summary This lifecycle hook implements intelligent auto-state management based on the current\n   * CRUD operation. For READ and DELETE operations, the fieldset automatically opens to provide\n   * immediate access to information, while CREATE and UPDATE operations keep it closed to maintain\n   * a clean initial interface. The method directly manipulates the DOM to ensure proper accordion\n   * synchronization and triggers change detection to reflect the programmatic state changes.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant F as FieldsetComponent\n   *   participant D as DOM\n   *   participant C as ChangeDetector\n   *\n   *   A->>F: ngAfterViewInit()\n   *   alt operation is READ or DELETE\n   *     F->>F: Set isOpen = true\n   *     F->>D: Query ion-accordion-group element\n   *     alt accordion element exists\n   *       F->>D: Set value attribute to 'open'\n   *     end\n   *   end\n   *   F->>C: detectChanges()\n   *   C->>F: Update view with new state\n   *\n   * @returns {void}\n   * @memberOf FieldsetComponent\n   */\n  ngAfterViewInit(): void {\n    if (this.operation === OperationKeys.READ || this.operation === OperationKeys.DELETE) {\n      this.isOpen = true;\n      const accordionElement = this.component?.nativeElement.querySelector('ion-accordion-group');\n      if(accordionElement)\n        accordionElement.setAttribute('value', 'open');\n    }\n    this.changeDetectorRef.detectChanges();\n  }\n\n  /**\n   * @description Handles accordion state change events from user interactions.\n   * @summary Processes CustomEvent objects triggered when users expand or collapse the accordion.\n   * This method extracts the new state from the event details and updates the component's\n   * internal state accordingly. It specifically listens for ION-ACCORDION-GROUP events to\n   * ensure proper event source validation and prevent handling of unrelated events.\n   *\n   * @param {CustomEvent} event - The event object containing accordion state change details\n   * @returns {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant I as Ion-Accordion\n   *   participant F as FieldsetComponent\n   *\n   *   U->>I: Click accordion header\n   *   I->>F: handleChange(CustomEvent)\n   *   F->>F: Extract target and detail from event\n   *   F->>F: Validate target is ION-ACCORDION-GROUP\n   *   alt valid target\n   *     F->>F: Update isOpen = !!value\n   *   end\n   *   F->>I: Reflect updated state\n   *\n   * @memberOf FieldsetComponent\n   */\n  handleChange(event: CustomEvent): void {\n    const { target, detail } = event;\n    const { value } = detail;\n    if ((target as HTMLIonAccordionGroupElement).tagName === 'ION-ACCORDION-GROUP')\n      this.isOpen = !!value;\n  }\n}\n","<fieldset [class]=\"'dcf-fieldset ' + operation\" #component>\n  <ion-accordion-group (ionChange)=\"handleChange($event)\">\n    <ion-accordion value=\"open\">\n      <ion-item slot=\"header\" >\n        <legend>{{ name }}</legend>\n      </ion-item>\n      <div slot=\"content\" [attr.aria-hidden]=\"!isOpen\" decafCollapsable>\n        <ng-content></ng-content>\n      </div>\n    </ion-accordion>\n  </ion-accordion-group>\n</fieldset>\n"]}
@@ -1,10 +1,10 @@
1
1
  import { __decorate, __metadata } from "tslib";
2
2
  import { Component, ElementRef, EventEmitter, Input, Output, ViewChild } from '@angular/core';
3
- import { ForAngularModule } from 'src/lib/for-angular.module';
4
- import { NgxBaseComponent } from 'src/lib/engine/NgxBaseComponent';
3
+ import { ForAngularModule } from '../../for-angular.module';
4
+ import { NgxBaseComponent } from '../../engine/NgxBaseComponent';
5
5
  import { IonChip, IonIcon, IonItem, IonLabel, IonSelect } from '@ionic/angular/standalone';
6
- import { Dynamic } from 'src/lib/engine';
7
- import { getWindowWidth } from 'src/lib/helpers/utils';
6
+ import { Dynamic } from '../../engine';
7
+ import { getWindowWidth } from '../../helpers/utils';
8
8
  import { debounceTime, fromEvent } from 'rxjs';
9
9
  import { OrderDirection, Repository } from '@decaf-ts/core';
10
10
  import { SearchbarComponent } from '../searchbar/searchbar.component';
@@ -686,4 +686,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
686
686
  }], searchEvent: [{
687
687
  type: Output
688
688
  }] } });
689
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/filter/filter.component.ts","../../../../../src/lib/components/filter/filter.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, Input, OnDestroy, OnInit, Output, ViewChild  } from '@angular/core';\nimport { ForAngularModule } from 'src/lib/for-angular.module';\nimport { NgxBaseComponent } from 'src/lib/engine/NgxBaseComponent';\nimport { IonChip, IonIcon, IonItem, IonLabel, IonSelect} from '@ionic/angular/standalone';\nimport { Dynamic, IFilterQuery, IFilterQueryItem } from 'src/lib/engine';\nimport { getWindowWidth } from 'src/lib/helpers/utils';\nimport { debounceTime, fromEvent, Subscription } from 'rxjs';\nimport { OrderDirection, Repository } from '@decaf-ts/core';\nimport { Model } from '@decaf-ts/decorator-validation';\nimport { SearchbarComponent } from '../searchbar/searchbar.component';\nimport { addIcons } from 'ionicons';\nimport { chevronDownOutline, chevronUpOutline } from 'ionicons/icons';\n\n/**\n * @description Advanced filter component for creating dynamic search filters with step-by-step construction.\n * @summary This component provides a comprehensive filtering interface that allows users to build\n * complex search criteria using a three-step approach: select index → select condition → enter value.\n * It supports filtering by multiple field indexes, comparison conditions, and values, displaying\n * selected filters as removable chips. The component is responsive and includes auto-suggestions\n * with keyboard navigation support.\n *\n * @example\n * ```html\n * <ngx-decaf-filter\n *   [indexes]=\"['name', 'email', 'department', 'status']\"\n *   [conditions]=\"['Equal', 'Contains', 'Greater Than', 'Less Than']\"\n *   [sort]=\"['createdAt', 'updatedAt']\"\n *   [disableSort]=\"false\"\n *   (filterEvent)=\"onFiltersChanged($event)\">\n * </ngx-decaf-filter>\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant U as User\n *   participant F as FilterComponent\n *   participant P as Parent Component\n *\n *   U->>F: Focus input field\n *   F->>F: handleFocus() - Show available indexes\n *   U->>F: Select index (e.g., \"name\")\n *   F->>F: addFilter() - Step 1 completed\n *   F->>F: Show available conditions\n *   U->>F: Select condition (e.g., \"Contains\")\n *   F->>F: addFilter() - Step 2 completed\n *   F->>F: Show value input prompt\n *   U->>F: Enter value and press Enter\n *   F->>F: addFilter() - Step 3 completed\n *   F->>F: Create complete filter object\n *   F->>P: Emit filterEvent with new filter array\n *   F->>F: Reset to step 1 for next filter\n *\n * @memberOf ForAngularModule\n */\n@Dynamic()\n@Component({\n  selector: 'ngx-decaf-filter',\n  templateUrl: './filter.component.html',\n  styleUrls: ['./filter.component.scss'],\n  imports: [\n    ForAngularModule,\n    IonLabel,\n    IonItem,\n    IonChip,\n    IonIcon,\n    IonSelect,\n    IonIcon,\n    SearchbarComponent\n  ],\n  standalone: true,\n})\nexport class FilterComponent extends NgxBaseComponent implements OnInit, OnDestroy {\n\n  /**\n   * @description Reference to the dropdown options container element.\n   * @summary ViewChild reference used to access and manipulate the dropdown options element\n   * for highlighting filtered items and managing visual feedback during option selection.\n   * This element contains the filterable suggestions that users can interact with.\n   *\n   * @type {ElementRef}\n   * @memberOf FilterComponent\n   */\n  @ViewChild('optionsFilterElement', { read: ElementRef, static: false })\n  optionsFilterElement!: ElementRef;\n\n  /**\n   * @description Available field indexes for filtering operations.\n   * @summary Defines the list of field names that users can filter by. These represent\n   * the data properties available for filtering operations. Each index corresponds to\n   * a field in the data model that supports comparison operations.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  @Input()\n  indexes: string[] = [];\n\n  /**\n   * @description Available comparison conditions for filters.\n   * @summary Defines the list of comparison operators that can be used when creating filters.\n   * These conditions determine how the filter value is compared against the field value.\n   * Common conditions include equality, containment, and numerical comparison operations.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  @Input()\n  conditions: string[] = ['Equal', 'Contains', 'Not Contains', 'Greater Than', 'Less Than', 'Not Equal'];\n\n  /**\n   * @description Available sorting options for the filtered data.\n   * @summary Defines the list of field names that can be used for sorting the filtered results.\n   * When disableSort is false, this array is automatically merged with the indexes array\n   * to provide comprehensive sorting capabilities.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  @Input()\n  sortBy: string[] = [];\n\n  /**\n   * @description Controls whether sorting functionality is disabled.\n   * @summary When set to true, prevents the automatic merging of sort and indexes arrays,\n   * effectively disabling sorting capabilities. This is useful when you want to provide\n   * filtering without sorting options.\n   *\n   * @type {boolean}\n   * @default false\n   * @memberOf FilterComponent\n   */\n  @Input()\n  disableSort: boolean = false;\n\n  /**\n   * @description Current window width for responsive behavior.\n   * @summary Stores the current browser window width in pixels. This value is updated\n   * on window resize events to enable responsive filtering behavior and layout adjustments\n   * based on available screen space.\n   *\n   * @type {number}\n   * @memberOf FilterComponent\n   */\n  windowWidth!: number;\n\n  /**\n   * @description Available options for the current filter step.\n   * @summary Contains the list of options available for selection in the current step.\n   * This array changes dynamically based on the current step: indexes → conditions → empty for value input.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  options: string[] = [];\n\n  /**\n   * @description Filtered options based on user input.\n   * @summary Contains the subset of options that match the current user input for real-time\n   * filtering. This array is updated as the user types to show only relevant suggestions\n   * in the dropdown menu.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  filteredOptions: string[] = [];\n\n  /**\n   * @description Complete filter objects created by the user.\n   * @summary Array of complete filter objects, each containing index, condition, and value properties.\n   * These represent the active filters that can be applied to data operations.\n   *\n   * @type {KeyValue[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  filterValue: IFilterQueryItem[] = [];\n\n  /**\n   * @description Current filter being constructed.\n   * @summary Temporary object that accumulates filter properties (index, condition, value)\n   * during the three-step filter creation process. Gets added to filterValue when complete.\n   *\n   * @type {KeyValue}\n   * @default {}\n   * @memberOf FilterComponent\n   */\n  lastFilter: IFilterQueryItem = {};\n\n  /**\n   * @description Current step in the filter creation process.\n   * @summary Tracks the current step of filter creation: 1 = index selection, 2 = condition selection,\n   * 3 = value input. Automatically resets to 1 after completing a filter.\n   *\n   * @type {number}\n   * @default 1\n   * @memberOf FilterComponent\n   */\n  step: number = 1;\n\n  /**\n   * @description Controls dropdown visibility state.\n   * @summary Boolean flag that determines whether the options dropdown is currently visible.\n   * Used to manage the dropdown's open/close state and coordinate with focus/blur events.\n   *\n   * @type {boolean}\n   * @default false\n   * @memberOf FilterComponent\n   */\n  dropdownOpen: boolean = false;\n\n  /**\n   * @description Current input field value.\n   * @summary Stores the current text input value that the user is typing. This value is\n   * bound to the input field and is cleared after each successful filter step completion.\n   *\n   * @type {string}\n   * @default ''\n   * @memberOf FilterComponent\n   */\n  value: string = '';\n\n  /**\n   * @description Current sorting field value.\n   * @summary Stores the field name currently selected for sorting operations.\n   * This value determines which field is used to order the filtered results.\n   * Defaults to 'id' and can be changed through the sort dropdown selection.\n   *\n   * @type {string}\n   * @default 'id'\n   * @memberOf FilterComponent\n   */\n  sortValue: string = 'id';\n\n  /**\n   * @description Current sorting direction.\n   * @summary Defines the direction of the sort operation - ascending or descending.\n   * This value works in conjunction with sortValue to determine the complete\n   * sorting configuration for filtered results.\n   *\n   * @type {OrderDirection}\n   * @default OrderDirection.DSC\n   * @memberOf FilterComponent\n   */\n  sortDirection: OrderDirection = OrderDirection.DSC;\n\n  /**\n   * @description Subscription for window resize events.\n   * @summary RxJS subscription that listens for window resize events with debouncing\n   * to update the windowWidth property. This enables responsive behavior and prevents\n   * excessive updates during resize operations.\n   *\n   * @type {Subscription}\n   * @memberOf FilterComponent\n   */\n  windowResizeSubscription!: Subscription;\n\n  /**\n   * @description Event emitter for filter changes.\n   * @summary Emits filter events when the user creates, modifies, or clears filters.\n   * The emitted value contains an array of complete filter objects or undefined when\n   * filters are cleared. Parent components listen to this event to update their data display.\n   *\n   * @type {EventEmitter<KeyValue[] | undefined>}\n   * @memberOf FilterComponent\n   */\n  @Output()\n  filterEvent: EventEmitter<IFilterQuery | undefined> = new EventEmitter<IFilterQuery | undefined>();\n\n  /**\n   * @description Event emitter for search events.\n   * @summary Emits search events when the user interacts with the searchbar.\n   * @type {EventEmitter<string>}\n   * @memberOf FilterComponent\n   */\n  @Output()\n  searchEvent: EventEmitter<string> = new EventEmitter<string>();\n\n\n  /**\n   * @description Constructor for FilterComponent.\n   * @summary Initializes a new instance of the FilterComponent.\n   * Calls the parent constructor with the component name to establish base locale string generation\n   * and internationalization support.\n   *\n   * @memberOf FilterComponent\n   */\n  constructor() {\n    super(\"FilterComponent\");\n    addIcons({chevronDownOutline, chevronUpOutline});\n  }\n\n  /**\n   * @description Initializes the component after Angular first displays the data-bound properties.\n   * @summary Sets up the component by initializing window width tracking, setting up resize event\n   * subscriptions with debouncing, configuring sorting options, and calling the base initialization.\n   * This method prepares the component for user interaction and responsive behavior.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant F as FilterComponent\n   *   participant W as Window\n   *   participant R as RxJS\n   *\n   *   A->>F: ngOnInit()\n   *   F->>W: getWindowWidth()\n   *   W-->>F: Return current width\n   *   F->>R: Setup resize subscription with debounce\n   *   R-->>F: Subscription created\n   *   alt disableSort is false\n   *     F->>F: Merge sort and indexes arrays\n   *   end\n   *   F->>F: Call initialize()\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  ngOnInit(): void {\n    this.windowWidth = getWindowWidth() as number;\n    this.windowResizeSubscription = fromEvent(window, 'resize')\n    .pipe(debounceTime(300))\n    .subscribe(() => {\n     this.windowWidth = getWindowWidth() as number;\n    });\n\n    this.getIndexes();\n    this.initialize();\n  }\n\n  /**\n   * @description Retrieves and configures available indexes for filtering and sorting.\n   * @summary Extracts field indexes from the model if available and merges them with\n   * sorting options when sorting is enabled. This method sets up the available field\n   * options for both filtering and sorting operations based on the model structure.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  getIndexes(): void {\n    if(this.model)\n      this.indexes = Object.keys(Repository.indexes(this.model as Model) || {});\n    if(!this.disableSort)\n      this.sortBy = [... this.sortBy, ...this.indexes];\n  }\n\n\n  /**\n   * @description Cleanup method called when the component is destroyed.\n   * @summary Unsubscribes from window resize events to prevent memory leaks.\n   * This is essential for proper cleanup of RxJS subscriptions when the component\n   * is removed from the DOM.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  ngOnDestroy(): void {\n    this.windowResizeSubscription.unsubscribe();\n    this.clear();\n  }\n\n  /**\n   * @description Handles input events from the text field.\n   * @summary Processes user input and filters the available options based on the typed value.\n   * This method provides real-time filtering of suggestions as the user types in the input field.\n   *\n   * @param {InputEvent} event - The input event containing the new value\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleInput(event: InputEvent): void {\n    const {value} = event.target as HTMLInputElement;\n      this.filteredOptions = this.filterOptions(value);\n  }\n\n  /**\n   * @description Handles focus events on the input field.\n   * @summary Sets up the available options when the input field receives focus and opens the dropdown.\n   * If no options are provided, automatically determines the appropriate options based on current step.\n   * This method initializes the dropdown with contextually relevant suggestions.\n   *\n   * @param {string[]} options - Optional array of options to display\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleFocus(options: string[]  = []): void {\n    if(!options.length)\n     options = this.getOptions();\n    this.filteredOptions = this.options = options;\n    this.dropdownOpen = true;\n  }\n\n  /**\n   * @description Handles blur events on the input field with delayed closing.\n   * @summary Manages the dropdown closing behavior with a delay to allow for option selection.\n   * Uses a two-phase approach to prevent premature closing when users click on dropdown options.\n   *\n   * @param {boolean} close - Internal flag to control the closing phase\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleBlur(close: boolean = false): void {\n    if(!close) {\n      this.dropdownOpen = false;\n      setTimeout(() => {\n        this.handleBlur(true);\n      }, 100);\n    } else {\n      if(!this.dropdownOpen && this.options.length) {\n        setTimeout(() => {\n          this.options = [];\n          this.dropdownOpen = false;\n        }, 50);\n      }\n    }\n  }\n\n  /**\n   * @description Determines the appropriate options based on the current filter step.\n   * @summary Returns the contextually relevant options for the current step in the filter creation process.\n   * Step 1 shows indexes, Step 2 shows conditions, Step 3 shows no options (value input).\n   *\n   * @returns {string[]} Array of options appropriate for the current step\n   * @memberOf FilterComponent\n   */\n  getOptions(): string[] {\n   switch (this.step) {\n      case 1:\n        this.options = this.indexes;\n        break;\n      case 2:\n        this.options = this.conditions;\n        break;\n      case 3:\n        this.options = [];\n        break;\n    }\n    return this.options\n  }\n\n  /**\n   * @description Adds a filter step or completes filter creation through a three-step process.\n   * @summary Core method for building filters step by step: Step 1 (Index) → Step 2 (Condition) → Step 3 (Value).\n   * When all steps are complete, creates a complete filter object and adds it to the filter collection.\n   * Handles both keyboard events (Enter to submit) and programmatic calls.\n   *\n   * @param {string} value - The value to add for the current step\n   * @param {CustomEvent} event - Optional event (KeyboardEvent triggers submission when value is empty)\n   * @returns {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant F as FilterComponent\n   *\n   *   U->>F: addFilter(value, event)\n   *   F->>F: Trim and validate value\n   *   alt KeyboardEvent && empty value\n   *     F->>F: submit() - Send current filters\n   *   else Valid value or step 3\n   *     alt Step 1 (Index)\n   *       F->>F: lastFilter.index = value\n   *       F->>F: options = conditions\n   *     else Step 2 (Condition)\n   *       F->>F: lastFilter.condition = value\n   *       F->>F: options = []\n   *     else Step 3 (Value)\n   *       F->>F: lastFilter.value = value\n   *       F->>F: Add complete filter to filterValue\n   *       F->>F: Reset step to 1\n   *     end\n   *     F->>F: Increment step\n   *     F->>F: Clear input & focus\n   *     F->>F: Show next options\n   *   end\n   *\n   * @memberOf FilterComponent\n   */\n  addFilter(value: string, event?: CustomEvent): void {\n    value = value.trim();\n    if(event instanceof KeyboardEvent && !value) {\n      this.submit();\n    } else {\n       if((value && (!(event instanceof KeyboardEvent)) || this.step === 3)) {\n        const filter = this.lastFilter;\n        switch (this.step) {\n          case 1:\n            filter['index'] = value;\n            this.options = this.conditions;\n            break;\n          case 2:\n            filter['condition'] = value;\n            this.options = [];\n            break;\n          case 3:\n            filter['value'] = value;\n            this.options = this.indexes;\n            break;\n        }\n        if(!this.filterValue.length) {\n          this.filterValue.push(filter);\n        } else {\n          if(this.step === 1)\n            this.filterValue.push(filter);\n        }\n        if(this.step === 3) {\n          this.step = 0;\n          this.filterValue[this.filterValue.length - 1] = filter;\n          this.lastFilter = {};\n        }\n        this.step++;\n        this.value = '';\n        if(this.options.length)\n          this.handleFocus(this.options);\n        this.component.nativeElement.focus();\n      }\n    }\n  }\n\n  /**\n   * @description Selects an option from the dropdown suggestions.\n   * @summary Handles option selection when a user clicks on a suggestion in the dropdown.\n   * This method acts as a bridge between dropdown clicks and the main addFilter logic.\n   *\n   * @param {CustomEvent} event - The click event from the dropdown option\n   * @param {string} value - The selected option value\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  selectOption(value: string): void {\n    this.addFilter(value);\n  }\n\n  /**\n   * @description Determines if a filter option can be individually removed.\n   * @summary Checks whether a filter component should display a close icon for removal.\n   * Only value options can be removed individually; index and condition options are part\n   * of the complete filter structure and cannot be removed separately.\n   *\n   * @param {string} option - The filter option text to check\n   * @returns {boolean} True if the option can be cleared individually, false otherwise\n   * @memberOf FilterComponent\n   */\n  allowClear(option: string): boolean {\n    return this.indexes.indexOf(option) === -1 && this.conditions.indexOf(option) === -1;\n  }\n\n  /**\n   * @description Removes a complete filter from the collection based on filter value.\n   * @summary Removes a complete filter by matching the provided value against filter values\n   * in the collection. Uses string normalization to handle accents and case differences.\n   * After removal, resets the interface to show available indexes for new filter creation.\n   *\n   * @param {string} filter - The filter value to remove (matches against filter.value property)\n   * @returns {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant F as FilterComponent\n   *\n   *   U->>F: removeFilter(filterValue)\n   *   F->>F: cleanString(filterValue)\n   *   F->>F: Filter out matching filter objects\n   *   F->>F: Clear input value\n   *   F->>F: handleFocus(indexes) - Reset to index selection\n   *   Note over F: Filter removed and UI reset\n   *\n   * @memberOf FilterComponent\n   */\n  removeFilter(filter: string): void {\n    function cleanString(filter: string): string {\n      return filter\n        .toLowerCase()                 // convert all characters to lowercase\n        .normalize(\"NFD\")              // separate accent marks from characters\n        .replace(/[\\u0300-\\u036f]/g, \"\") // remove accent marks\n        .replace(/\\s+/g, \"\");          // remove all whitespace\n    }\n    this.value = \"\";\n    this.filterValue = this.filterValue.filter((item) => item?.['value'] && cleanString(item?.['value']) !== cleanString(filter));\n    if(this.filterValue.length === 0) {\n      this.step = 1;\n      this.lastFilter = {};\n    }\n    this.handleFocus(this.indexes);\n  }\n\n  /**\n   * @description Resets the component to its initial state.\n   * @summary Clears all filter data, options, and resets the step counter to 1.\n   * This method provides a clean slate for new filter creation without emitting events.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  reset(): void {\n    this.options = this.filteredOptions = this.filterValue = [];\n    this.step = 1;\n    this.lastFilter = {};\n    this.value = '';\n    setTimeout(() => {\n       this.submit();\n    }, 100);\n  }\n\n  /**\n   * @description Clears all filters and notifies parent components.\n   * @summary Resets the component state and emits undefined to notify parent components\n   * that all filters have been cleared. This triggers any connected data refresh logic.\n   *\n   * @param {string} value - Optional parameter (currently unused)\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  clear(value?: string): void {\n    if(!value)\n      this.reset();\n  }\n\n  /**\n   * @description Submits the current filter collection to parent components.\n   * @summary Emits the current filter array to parent components when filters are ready\n   * to be applied. Only emits if there are active filters. Clears options after submission.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  submit(): void {\n    this.filterEvent.emit({\n      query: this.filterValue.length > 0 ? this.filterValue : undefined,\n      sort: {\n        value: this.sortValue,\n        direction: this.sortDirection\n      }\n    } as IFilterQuery);\n    if(this.filterValue.length === 0)\n      this.options = [];\n  }\n\n  /**\n   * @description Toggles the sort direction between ascending and descending.\n   * @summary Handles sort direction changes by toggling between ASC and DSC values.\n   * When the direction changes, automatically triggers a submit to apply the new\n   * sorting configuration to the filtered results.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n   handleSortDirectionChange(): void {\n    const direction = this.sortDirection ===  OrderDirection.ASC ? OrderDirection.DSC : OrderDirection.ASC;\n    if(direction !== this.sortDirection) {\n      this.sortDirection = direction;\n      this.submit();\n    }\n  }\n\n  /**\n   * @description Handles sort field selection changes from the dropdown.\n   * @summary Processes sort field changes when users select a different field\n   * from the sort dropdown. Updates the sortValue property and triggers\n   * a submit to apply the new sorting configuration if the value has changed.\n   *\n   * @param {CustomEvent} event - The select change event containing the new sort field value\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleSortChange(event: CustomEvent): void {\n    const target = event.target as HTMLIonSelectElement;\n    const value = target.value;\n    if(value !== this.sortValue) {\n      this.sortValue = value as string;\n      this.submit();\n    }\n  }\n\n  /**\n   * @description Filters available options based on user input with visual highlighting.\n   * @summary Performs real-time filtering of available options based on user input.\n   * Also handles visual highlighting of matching options in the dropdown. Returns all\n   * options if input is less than 2 characters for performance optimization.\n   *\n   * @param {string | null | undefined} value - The search value to filter by\n   * @returns {string[]} Array of filtered options that match the input\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant F as FilterComponent\n   *   participant D as DOM\n   *\n   *   U->>F: filterOptions(inputValue)\n   *   alt inputValue < 2 characters\n   *     F->>D: Remove existing highlights\n   *     F-->>U: Return all options\n   *   else inputValue >= 2 characters\n   *     F->>D: Query all option elements\n   *     F->>D: Add highlight to first matching option\n   *     F->>F: Filter options by substring match\n   *     F-->>U: Return filtered options\n   *   end\n   *\n   * @memberOf FilterComponent\n   */\n  filterOptions(value: string | null |  undefined): string[] {\n    const optionsElement = this.optionsFilterElement.nativeElement;\n    if(!value?.length || !value || value.length < 2) {\n      const filteredOption = optionsElement.querySelector('.dcf-filtering-item');\n      if(filteredOption)\n        filteredOption.classList.remove('dcf-filtering-item');\n      return this.options;\n    }\n    const options = optionsElement.querySelectorAll('.dcf-item');\n    for (const option of options) {\n      const isActive = option.textContent?.toLowerCase().includes(value.toLowerCase());\n      if(isActive) {\n        option.classList.add('dcf-filtering-item');\n        break;\n      }\n    }\n    return this.options.filter((option: string) => option.toLowerCase().includes(value.toLowerCase() as string));\n  }\n\n  /**\n   * @description Handles search events from the integrated searchbar component.\n   * @summary Processes search input from the searchbar and emits search events\n   * to parent components. This method acts as a bridge between the internal\n   * searchbar component and external search event listeners.\n   *\n   * @param {string | undefined} value - The search value entered by the user\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleSearch(value: string | undefined): void {\n    this.searchEvent.emit(value);\n  }\n\n}\n","\n@if(!indexes.length) {\n  <ngx-decaf-searchbar [emitEventToWindow]=\"false\" [debounce]=\"500\" (searchEvent)=\"handleSearch($event)\" />\n}\n\n<div class=\"dcf-grid dcf-grid-small dcf-grid-match dcf-filter-grid\" [ngClass]=\"{'dcf-hidden': !indexes.length}\">\n  <div class=\"dcf-width-expand\">\n    <div class=\"dcf-filter\">\n      <div class=\"dcf-input\">\n        @for(filter of filterValue; track trackItemFn($index, filter?.['index'])) {\n          @if(filter?.['index']) {\n            <ion-chip [outline]=\"true\">{{ filter?.['index'] }}</ion-chip>\n          }\n          @if(filter?.['condition']) {\n            <ion-chip [outline]=\"true\">{{ filter?.['condition'] }}</ion-chip>\n          }\n          @if(filter?.['value']) {\n            <ion-chip [outline]=\"true\" class=\"dcf-filter-value\">\n              {{ filter?.['value'] }}\n              <ion-icon name=\"close\" (click)=\"removeFilter(filter?.['value'])\" size=\"small\"></ion-icon>\n            </ion-chip>\n          }\n        }\n        <div class=\"dcf-width-1-1\">\n          <input\n            fill=\"none\"\n            [(ngModel)]=\"value\"\n            (keydown.enter)=\"addFilter(value, $event)\"\n            (keydown.backspace)=\"clear(value)\"\n            (input)=\"handleInput($event)\"\n            (click)=\"handleFocus()\"\n            (blur)=\"handleBlur()\"\n            type=\"text\"\n            [readonly]=\"step !== 3\"\n            placeholder=\"{{ locale + (step === 3 ? '.type' : '.select') | translate }}\"\n            #component\n          />\n          @if(windowWidth >= 768) {\n            <div [class]=\"'dcf-dropdown ' + (options.length > 0 ? ' dcf-active' : '')\" #optionsFilterElement>\n              <div>\n                @if(filteredOptions.length > 0) {\n                  @for(key of filteredOptions; track key) {\n                    <div\n                      class=\"dcf-item\"\n                      tabindex=\"0\"\n                      (keydown.enter)=\"selectOption(key)\"\n                      (click)=\"selectOption(key)\">\n                      {{ key }}\n                    </div>\n                  }\n                } @else {\n                  <div class=\"dcf-empty\"\n                    (click)=\"filteredOptions = options; value = ''\"\n                    tabindex=\"0\"\n                    (keydown.enter)=\"filteredOptions = options; value = ''\"\n                  >\n                    {{ 'no_suggestions' | translate }}\n                  </div>\n                }\n              </div>\n            </div>\n          }\n        </div>\n      </div>\n      @if(filterValue.length > 0) {\n        <div class=\"dcf-icon-clear\">\n          <ion-button fill=\"clear\" size=\"small\" (click)=\"clear()\">\n            <ion-icon name=\"trash-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n          </ion-button>\n        </div>\n      }\n      <div class=\"dcf-icon-search\">\n        <ion-button fill=\"clear\" size=\"small\" (click)=\"submit()\">\n          <ion-icon name=\"search-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n        </ion-button>\n      </div>\n    </div>\n    @if(windowWidth < 768) {\n      <div [class]=\"'dcf-dropdown ' + (options.length > 0 ? ' dcf-active' : '')\" #optionsFilterElement>\n        <div>\n          @if(filteredOptions.length > 0) {\n            @for(key of filteredOptions; track key) {\n              <div\n                class=\"dcf-item\"\n                tabindex=\"0\"\n                (keydown.enter)=\"selectOption(key)\"\n                (click)=\"selectOption(key)\">\n                {{ key }}\n              </div>\n            }\n          } @else {\n            <div class=\"dcf-empty\"\n            (click)=\"filteredOptions = options; value = ''\"\n            tabindex=\"0\"\n            (keydown.enter)=\"filteredOptions = options; value = ''\"\n            >\n              {{ 'no_suggestions' | translate }}\n            </div>\n          }\n        </div>\n      </div>\n    }\n  </div>\n  @if(!disableSort) {\n    <div class=\"dcf-width-1-5@m dcf-width-1-1 dcf-sort-container\">\n      <div class=\"dcf-grid dcf-grid-collapse dcf-flex dcf-flex-middle dcf-grid-match\">\n        <div class=\"dcf-width-expand\">\n          <ion-select\n              toggleIcon=\"chevron-down-outline\"\n              expandedIcon=\"chevron-up-outline\"\n              class=\"dcf-sort-select\"\n              (ionChange)=\"handleSortChange($event)\"\n              interface=\"popover\"\n              [value]=\"sortValue\"\n              label-placement=\"floating\"\n              fill=\"outline\"\n            >\n            @for(sort of sortBy; track sort) {\n              aa\n              <ion-select-option [value]=\"sort\">{{ sort | translate }}</ion-select-option>\n            }\n          </ion-select>\n        </div>\n        <div class=\"dcf-width-auto\">\n          <ion-button (click)=\"handleSortDirectionChange()\" fill=\"clear\">\n            <ion-icon slot=\"icon-only\" [name]=\"sortDirection === 'desc' ? 'arrow-down-outline' : 'arrow-up-outline'\"></ion-icon>\n          </ion-button>\n        </div>\n      </div>\n    </div>\n  }\n</div>\n\n\n"]}
689
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/filter/filter.component.ts","../../../../../src/lib/components/filter/filter.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, Input, OnDestroy, OnInit, Output, ViewChild  } from '@angular/core';\nimport { ForAngularModule } from '../../for-angular.module';\nimport { NgxBaseComponent } from '../../engine/NgxBaseComponent';\nimport { IonChip, IonIcon, IonItem, IonLabel, IonSelect} from '@ionic/angular/standalone';\nimport { Dynamic, IFilterQuery, IFilterQueryItem } from '../../engine';\nimport { getWindowWidth } from '../../helpers/utils';\nimport { debounceTime, fromEvent, Subscription } from 'rxjs';\nimport { OrderDirection, Repository } from '@decaf-ts/core';\nimport { Model } from '@decaf-ts/decorator-validation';\nimport { SearchbarComponent } from '../searchbar/searchbar.component';\nimport { addIcons } from 'ionicons';\nimport { chevronDownOutline, chevronUpOutline } from 'ionicons/icons';\n\n/**\n * @description Advanced filter component for creating dynamic search filters with step-by-step construction.\n * @summary This component provides a comprehensive filtering interface that allows users to build\n * complex search criteria using a three-step approach: select index → select condition → enter value.\n * It supports filtering by multiple field indexes, comparison conditions, and values, displaying\n * selected filters as removable chips. The component is responsive and includes auto-suggestions\n * with keyboard navigation support.\n *\n * @example\n * ```html\n * <ngx-decaf-filter\n *   [indexes]=\"['name', 'email', 'department', 'status']\"\n *   [conditions]=\"['Equal', 'Contains', 'Greater Than', 'Less Than']\"\n *   [sort]=\"['createdAt', 'updatedAt']\"\n *   [disableSort]=\"false\"\n *   (filterEvent)=\"onFiltersChanged($event)\">\n * </ngx-decaf-filter>\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant U as User\n *   participant F as FilterComponent\n *   participant P as Parent Component\n *\n *   U->>F: Focus input field\n *   F->>F: handleFocus() - Show available indexes\n *   U->>F: Select index (e.g., \"name\")\n *   F->>F: addFilter() - Step 1 completed\n *   F->>F: Show available conditions\n *   U->>F: Select condition (e.g., \"Contains\")\n *   F->>F: addFilter() - Step 2 completed\n *   F->>F: Show value input prompt\n *   U->>F: Enter value and press Enter\n *   F->>F: addFilter() - Step 3 completed\n *   F->>F: Create complete filter object\n *   F->>P: Emit filterEvent with new filter array\n *   F->>F: Reset to step 1 for next filter\n *\n * @memberOf ForAngularModule\n */\n@Dynamic()\n@Component({\n  selector: 'ngx-decaf-filter',\n  templateUrl: './filter.component.html',\n  styleUrls: ['./filter.component.scss'],\n  imports: [\n    ForAngularModule,\n    IonLabel,\n    IonItem,\n    IonChip,\n    IonIcon,\n    IonSelect,\n    IonIcon,\n    SearchbarComponent\n  ],\n  standalone: true,\n})\nexport class FilterComponent extends NgxBaseComponent implements OnInit, OnDestroy {\n\n  /**\n   * @description Reference to the dropdown options container element.\n   * @summary ViewChild reference used to access and manipulate the dropdown options element\n   * for highlighting filtered items and managing visual feedback during option selection.\n   * This element contains the filterable suggestions that users can interact with.\n   *\n   * @type {ElementRef}\n   * @memberOf FilterComponent\n   */\n  @ViewChild('optionsFilterElement', { read: ElementRef, static: false })\n  optionsFilterElement!: ElementRef;\n\n  /**\n   * @description Available field indexes for filtering operations.\n   * @summary Defines the list of field names that users can filter by. These represent\n   * the data properties available for filtering operations. Each index corresponds to\n   * a field in the data model that supports comparison operations.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  @Input()\n  indexes: string[] = [];\n\n  /**\n   * @description Available comparison conditions for filters.\n   * @summary Defines the list of comparison operators that can be used when creating filters.\n   * These conditions determine how the filter value is compared against the field value.\n   * Common conditions include equality, containment, and numerical comparison operations.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  @Input()\n  conditions: string[] = ['Equal', 'Contains', 'Not Contains', 'Greater Than', 'Less Than', 'Not Equal'];\n\n  /**\n   * @description Available sorting options for the filtered data.\n   * @summary Defines the list of field names that can be used for sorting the filtered results.\n   * When disableSort is false, this array is automatically merged with the indexes array\n   * to provide comprehensive sorting capabilities.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  @Input()\n  sortBy: string[] = [];\n\n  /**\n   * @description Controls whether sorting functionality is disabled.\n   * @summary When set to true, prevents the automatic merging of sort and indexes arrays,\n   * effectively disabling sorting capabilities. This is useful when you want to provide\n   * filtering without sorting options.\n   *\n   * @type {boolean}\n   * @default false\n   * @memberOf FilterComponent\n   */\n  @Input()\n  disableSort: boolean = false;\n\n  /**\n   * @description Current window width for responsive behavior.\n   * @summary Stores the current browser window width in pixels. This value is updated\n   * on window resize events to enable responsive filtering behavior and layout adjustments\n   * based on available screen space.\n   *\n   * @type {number}\n   * @memberOf FilterComponent\n   */\n  windowWidth!: number;\n\n  /**\n   * @description Available options for the current filter step.\n   * @summary Contains the list of options available for selection in the current step.\n   * This array changes dynamically based on the current step: indexes → conditions → empty for value input.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  options: string[] = [];\n\n  /**\n   * @description Filtered options based on user input.\n   * @summary Contains the subset of options that match the current user input for real-time\n   * filtering. This array is updated as the user types to show only relevant suggestions\n   * in the dropdown menu.\n   *\n   * @type {string[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  filteredOptions: string[] = [];\n\n  /**\n   * @description Complete filter objects created by the user.\n   * @summary Array of complete filter objects, each containing index, condition, and value properties.\n   * These represent the active filters that can be applied to data operations.\n   *\n   * @type {KeyValue[]}\n   * @default []\n   * @memberOf FilterComponent\n   */\n  filterValue: IFilterQueryItem[] = [];\n\n  /**\n   * @description Current filter being constructed.\n   * @summary Temporary object that accumulates filter properties (index, condition, value)\n   * during the three-step filter creation process. Gets added to filterValue when complete.\n   *\n   * @type {KeyValue}\n   * @default {}\n   * @memberOf FilterComponent\n   */\n  lastFilter: IFilterQueryItem = {};\n\n  /**\n   * @description Current step in the filter creation process.\n   * @summary Tracks the current step of filter creation: 1 = index selection, 2 = condition selection,\n   * 3 = value input. Automatically resets to 1 after completing a filter.\n   *\n   * @type {number}\n   * @default 1\n   * @memberOf FilterComponent\n   */\n  step: number = 1;\n\n  /**\n   * @description Controls dropdown visibility state.\n   * @summary Boolean flag that determines whether the options dropdown is currently visible.\n   * Used to manage the dropdown's open/close state and coordinate with focus/blur events.\n   *\n   * @type {boolean}\n   * @default false\n   * @memberOf FilterComponent\n   */\n  dropdownOpen: boolean = false;\n\n  /**\n   * @description Current input field value.\n   * @summary Stores the current text input value that the user is typing. This value is\n   * bound to the input field and is cleared after each successful filter step completion.\n   *\n   * @type {string}\n   * @default ''\n   * @memberOf FilterComponent\n   */\n  value: string = '';\n\n  /**\n   * @description Current sorting field value.\n   * @summary Stores the field name currently selected for sorting operations.\n   * This value determines which field is used to order the filtered results.\n   * Defaults to 'id' and can be changed through the sort dropdown selection.\n   *\n   * @type {string}\n   * @default 'id'\n   * @memberOf FilterComponent\n   */\n  sortValue: string = 'id';\n\n  /**\n   * @description Current sorting direction.\n   * @summary Defines the direction of the sort operation - ascending or descending.\n   * This value works in conjunction with sortValue to determine the complete\n   * sorting configuration for filtered results.\n   *\n   * @type {OrderDirection}\n   * @default OrderDirection.DSC\n   * @memberOf FilterComponent\n   */\n  sortDirection: OrderDirection = OrderDirection.DSC;\n\n  /**\n   * @description Subscription for window resize events.\n   * @summary RxJS subscription that listens for window resize events with debouncing\n   * to update the windowWidth property. This enables responsive behavior and prevents\n   * excessive updates during resize operations.\n   *\n   * @type {Subscription}\n   * @memberOf FilterComponent\n   */\n  windowResizeSubscription!: Subscription;\n\n  /**\n   * @description Event emitter for filter changes.\n   * @summary Emits filter events when the user creates, modifies, or clears filters.\n   * The emitted value contains an array of complete filter objects or undefined when\n   * filters are cleared. Parent components listen to this event to update their data display.\n   *\n   * @type {EventEmitter<KeyValue[] | undefined>}\n   * @memberOf FilterComponent\n   */\n  @Output()\n  filterEvent: EventEmitter<IFilterQuery | undefined> = new EventEmitter<IFilterQuery | undefined>();\n\n  /**\n   * @description Event emitter for search events.\n   * @summary Emits search events when the user interacts with the searchbar.\n   * @type {EventEmitter<string>}\n   * @memberOf FilterComponent\n   */\n  @Output()\n  searchEvent: EventEmitter<string> = new EventEmitter<string>();\n\n\n  /**\n   * @description Constructor for FilterComponent.\n   * @summary Initializes a new instance of the FilterComponent.\n   * Calls the parent constructor with the component name to establish base locale string generation\n   * and internationalization support.\n   *\n   * @memberOf FilterComponent\n   */\n  constructor() {\n    super(\"FilterComponent\");\n    addIcons({chevronDownOutline, chevronUpOutline});\n  }\n\n  /**\n   * @description Initializes the component after Angular first displays the data-bound properties.\n   * @summary Sets up the component by initializing window width tracking, setting up resize event\n   * subscriptions with debouncing, configuring sorting options, and calling the base initialization.\n   * This method prepares the component for user interaction and responsive behavior.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant A as Angular Lifecycle\n   *   participant F as FilterComponent\n   *   participant W as Window\n   *   participant R as RxJS\n   *\n   *   A->>F: ngOnInit()\n   *   F->>W: getWindowWidth()\n   *   W-->>F: Return current width\n   *   F->>R: Setup resize subscription with debounce\n   *   R-->>F: Subscription created\n   *   alt disableSort is false\n   *     F->>F: Merge sort and indexes arrays\n   *   end\n   *   F->>F: Call initialize()\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  ngOnInit(): void {\n    this.windowWidth = getWindowWidth() as number;\n    this.windowResizeSubscription = fromEvent(window, 'resize')\n    .pipe(debounceTime(300))\n    .subscribe(() => {\n     this.windowWidth = getWindowWidth() as number;\n    });\n\n    this.getIndexes();\n    this.initialize();\n  }\n\n  /**\n   * @description Retrieves and configures available indexes for filtering and sorting.\n   * @summary Extracts field indexes from the model if available and merges them with\n   * sorting options when sorting is enabled. This method sets up the available field\n   * options for both filtering and sorting operations based on the model structure.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  getIndexes(): void {\n    if(this.model)\n      this.indexes = Object.keys(Repository.indexes(this.model as Model) || {});\n    if(!this.disableSort)\n      this.sortBy = [... this.sortBy, ...this.indexes];\n  }\n\n\n  /**\n   * @description Cleanup method called when the component is destroyed.\n   * @summary Unsubscribes from window resize events to prevent memory leaks.\n   * This is essential for proper cleanup of RxJS subscriptions when the component\n   * is removed from the DOM.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  ngOnDestroy(): void {\n    this.windowResizeSubscription.unsubscribe();\n    this.clear();\n  }\n\n  /**\n   * @description Handles input events from the text field.\n   * @summary Processes user input and filters the available options based on the typed value.\n   * This method provides real-time filtering of suggestions as the user types in the input field.\n   *\n   * @param {InputEvent} event - The input event containing the new value\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleInput(event: InputEvent): void {\n    const {value} = event.target as HTMLInputElement;\n      this.filteredOptions = this.filterOptions(value);\n  }\n\n  /**\n   * @description Handles focus events on the input field.\n   * @summary Sets up the available options when the input field receives focus and opens the dropdown.\n   * If no options are provided, automatically determines the appropriate options based on current step.\n   * This method initializes the dropdown with contextually relevant suggestions.\n   *\n   * @param {string[]} options - Optional array of options to display\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleFocus(options: string[]  = []): void {\n    if(!options.length)\n     options = this.getOptions();\n    this.filteredOptions = this.options = options;\n    this.dropdownOpen = true;\n  }\n\n  /**\n   * @description Handles blur events on the input field with delayed closing.\n   * @summary Manages the dropdown closing behavior with a delay to allow for option selection.\n   * Uses a two-phase approach to prevent premature closing when users click on dropdown options.\n   *\n   * @param {boolean} close - Internal flag to control the closing phase\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleBlur(close: boolean = false): void {\n    if(!close) {\n      this.dropdownOpen = false;\n      setTimeout(() => {\n        this.handleBlur(true);\n      }, 100);\n    } else {\n      if(!this.dropdownOpen && this.options.length) {\n        setTimeout(() => {\n          this.options = [];\n          this.dropdownOpen = false;\n        }, 50);\n      }\n    }\n  }\n\n  /**\n   * @description Determines the appropriate options based on the current filter step.\n   * @summary Returns the contextually relevant options for the current step in the filter creation process.\n   * Step 1 shows indexes, Step 2 shows conditions, Step 3 shows no options (value input).\n   *\n   * @returns {string[]} Array of options appropriate for the current step\n   * @memberOf FilterComponent\n   */\n  getOptions(): string[] {\n   switch (this.step) {\n      case 1:\n        this.options = this.indexes;\n        break;\n      case 2:\n        this.options = this.conditions;\n        break;\n      case 3:\n        this.options = [];\n        break;\n    }\n    return this.options\n  }\n\n  /**\n   * @description Adds a filter step or completes filter creation through a three-step process.\n   * @summary Core method for building filters step by step: Step 1 (Index) → Step 2 (Condition) → Step 3 (Value).\n   * When all steps are complete, creates a complete filter object and adds it to the filter collection.\n   * Handles both keyboard events (Enter to submit) and programmatic calls.\n   *\n   * @param {string} value - The value to add for the current step\n   * @param {CustomEvent} event - Optional event (KeyboardEvent triggers submission when value is empty)\n   * @returns {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant F as FilterComponent\n   *\n   *   U->>F: addFilter(value, event)\n   *   F->>F: Trim and validate value\n   *   alt KeyboardEvent && empty value\n   *     F->>F: submit() - Send current filters\n   *   else Valid value or step 3\n   *     alt Step 1 (Index)\n   *       F->>F: lastFilter.index = value\n   *       F->>F: options = conditions\n   *     else Step 2 (Condition)\n   *       F->>F: lastFilter.condition = value\n   *       F->>F: options = []\n   *     else Step 3 (Value)\n   *       F->>F: lastFilter.value = value\n   *       F->>F: Add complete filter to filterValue\n   *       F->>F: Reset step to 1\n   *     end\n   *     F->>F: Increment step\n   *     F->>F: Clear input & focus\n   *     F->>F: Show next options\n   *   end\n   *\n   * @memberOf FilterComponent\n   */\n  addFilter(value: string, event?: CustomEvent): void {\n    value = value.trim();\n    if(event instanceof KeyboardEvent && !value) {\n      this.submit();\n    } else {\n       if((value && (!(event instanceof KeyboardEvent)) || this.step === 3)) {\n        const filter = this.lastFilter;\n        switch (this.step) {\n          case 1:\n            filter['index'] = value;\n            this.options = this.conditions;\n            break;\n          case 2:\n            filter['condition'] = value;\n            this.options = [];\n            break;\n          case 3:\n            filter['value'] = value;\n            this.options = this.indexes;\n            break;\n        }\n        if(!this.filterValue.length) {\n          this.filterValue.push(filter);\n        } else {\n          if(this.step === 1)\n            this.filterValue.push(filter);\n        }\n        if(this.step === 3) {\n          this.step = 0;\n          this.filterValue[this.filterValue.length - 1] = filter;\n          this.lastFilter = {};\n        }\n        this.step++;\n        this.value = '';\n        if(this.options.length)\n          this.handleFocus(this.options);\n        this.component.nativeElement.focus();\n      }\n    }\n  }\n\n  /**\n   * @description Selects an option from the dropdown suggestions.\n   * @summary Handles option selection when a user clicks on a suggestion in the dropdown.\n   * This method acts as a bridge between dropdown clicks and the main addFilter logic.\n   *\n   * @param {CustomEvent} event - The click event from the dropdown option\n   * @param {string} value - The selected option value\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  selectOption(value: string): void {\n    this.addFilter(value);\n  }\n\n  /**\n   * @description Determines if a filter option can be individually removed.\n   * @summary Checks whether a filter component should display a close icon for removal.\n   * Only value options can be removed individually; index and condition options are part\n   * of the complete filter structure and cannot be removed separately.\n   *\n   * @param {string} option - The filter option text to check\n   * @returns {boolean} True if the option can be cleared individually, false otherwise\n   * @memberOf FilterComponent\n   */\n  allowClear(option: string): boolean {\n    return this.indexes.indexOf(option) === -1 && this.conditions.indexOf(option) === -1;\n  }\n\n  /**\n   * @description Removes a complete filter from the collection based on filter value.\n   * @summary Removes a complete filter by matching the provided value against filter values\n   * in the collection. Uses string normalization to handle accents and case differences.\n   * After removal, resets the interface to show available indexes for new filter creation.\n   *\n   * @param {string} filter - The filter value to remove (matches against filter.value property)\n   * @returns {void}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant F as FilterComponent\n   *\n   *   U->>F: removeFilter(filterValue)\n   *   F->>F: cleanString(filterValue)\n   *   F->>F: Filter out matching filter objects\n   *   F->>F: Clear input value\n   *   F->>F: handleFocus(indexes) - Reset to index selection\n   *   Note over F: Filter removed and UI reset\n   *\n   * @memberOf FilterComponent\n   */\n  removeFilter(filter: string): void {\n    function cleanString(filter: string): string {\n      return filter\n        .toLowerCase()                 // convert all characters to lowercase\n        .normalize(\"NFD\")              // separate accent marks from characters\n        .replace(/[\\u0300-\\u036f]/g, \"\") // remove accent marks\n        .replace(/\\s+/g, \"\");          // remove all whitespace\n    }\n    this.value = \"\";\n    this.filterValue = this.filterValue.filter((item) => item?.['value'] && cleanString(item?.['value']) !== cleanString(filter));\n    if(this.filterValue.length === 0) {\n      this.step = 1;\n      this.lastFilter = {};\n    }\n    this.handleFocus(this.indexes);\n  }\n\n  /**\n   * @description Resets the component to its initial state.\n   * @summary Clears all filter data, options, and resets the step counter to 1.\n   * This method provides a clean slate for new filter creation without emitting events.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  reset(): void {\n    this.options = this.filteredOptions = this.filterValue = [];\n    this.step = 1;\n    this.lastFilter = {};\n    this.value = '';\n    setTimeout(() => {\n       this.submit();\n    }, 100);\n  }\n\n  /**\n   * @description Clears all filters and notifies parent components.\n   * @summary Resets the component state and emits undefined to notify parent components\n   * that all filters have been cleared. This triggers any connected data refresh logic.\n   *\n   * @param {string} value - Optional parameter (currently unused)\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  clear(value?: string): void {\n    if(!value)\n      this.reset();\n  }\n\n  /**\n   * @description Submits the current filter collection to parent components.\n   * @summary Emits the current filter array to parent components when filters are ready\n   * to be applied. Only emits if there are active filters. Clears options after submission.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  submit(): void {\n    this.filterEvent.emit({\n      query: this.filterValue.length > 0 ? this.filterValue : undefined,\n      sort: {\n        value: this.sortValue,\n        direction: this.sortDirection\n      }\n    } as IFilterQuery);\n    if(this.filterValue.length === 0)\n      this.options = [];\n  }\n\n  /**\n   * @description Toggles the sort direction between ascending and descending.\n   * @summary Handles sort direction changes by toggling between ASC and DSC values.\n   * When the direction changes, automatically triggers a submit to apply the new\n   * sorting configuration to the filtered results.\n   *\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n   handleSortDirectionChange(): void {\n    const direction = this.sortDirection ===  OrderDirection.ASC ? OrderDirection.DSC : OrderDirection.ASC;\n    if(direction !== this.sortDirection) {\n      this.sortDirection = direction;\n      this.submit();\n    }\n  }\n\n  /**\n   * @description Handles sort field selection changes from the dropdown.\n   * @summary Processes sort field changes when users select a different field\n   * from the sort dropdown. Updates the sortValue property and triggers\n   * a submit to apply the new sorting configuration if the value has changed.\n   *\n   * @param {CustomEvent} event - The select change event containing the new sort field value\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleSortChange(event: CustomEvent): void {\n    const target = event.target as HTMLIonSelectElement;\n    const value = target.value;\n    if(value !== this.sortValue) {\n      this.sortValue = value as string;\n      this.submit();\n    }\n  }\n\n  /**\n   * @description Filters available options based on user input with visual highlighting.\n   * @summary Performs real-time filtering of available options based on user input.\n   * Also handles visual highlighting of matching options in the dropdown. Returns all\n   * options if input is less than 2 characters for performance optimization.\n   *\n   * @param {string | null | undefined} value - The search value to filter by\n   * @returns {string[]} Array of filtered options that match the input\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant F as FilterComponent\n   *   participant D as DOM\n   *\n   *   U->>F: filterOptions(inputValue)\n   *   alt inputValue < 2 characters\n   *     F->>D: Remove existing highlights\n   *     F-->>U: Return all options\n   *   else inputValue >= 2 characters\n   *     F->>D: Query all option elements\n   *     F->>D: Add highlight to first matching option\n   *     F->>F: Filter options by substring match\n   *     F-->>U: Return filtered options\n   *   end\n   *\n   * @memberOf FilterComponent\n   */\n  filterOptions(value: string | null |  undefined): string[] {\n    const optionsElement = this.optionsFilterElement.nativeElement;\n    if(!value?.length || !value || value.length < 2) {\n      const filteredOption = optionsElement.querySelector('.dcf-filtering-item');\n      if(filteredOption)\n        filteredOption.classList.remove('dcf-filtering-item');\n      return this.options;\n    }\n    const options = optionsElement.querySelectorAll('.dcf-item');\n    for (const option of options) {\n      const isActive = option.textContent?.toLowerCase().includes(value.toLowerCase());\n      if(isActive) {\n        option.classList.add('dcf-filtering-item');\n        break;\n      }\n    }\n    return this.options.filter((option: string) => option.toLowerCase().includes(value.toLowerCase() as string));\n  }\n\n  /**\n   * @description Handles search events from the integrated searchbar component.\n   * @summary Processes search input from the searchbar and emits search events\n   * to parent components. This method acts as a bridge between the internal\n   * searchbar component and external search event listeners.\n   *\n   * @param {string | undefined} value - The search value entered by the user\n   * @returns {void}\n   * @memberOf FilterComponent\n   */\n  handleSearch(value: string | undefined): void {\n    this.searchEvent.emit(value);\n  }\n\n}\n","\n@if(!indexes.length) {\n  <ngx-decaf-searchbar [emitEventToWindow]=\"false\" [debounce]=\"500\" (searchEvent)=\"handleSearch($event)\" />\n}\n\n<div class=\"dcf-grid dcf-grid-small dcf-grid-match dcf-filter-grid\" [ngClass]=\"{'dcf-hidden': !indexes.length}\">\n  <div class=\"dcf-width-expand\">\n    <div class=\"dcf-filter\">\n      <div class=\"dcf-input\">\n        @for(filter of filterValue; track trackItemFn($index, filter?.['index'])) {\n          @if(filter?.['index']) {\n            <ion-chip [outline]=\"true\">{{ filter?.['index'] }}</ion-chip>\n          }\n          @if(filter?.['condition']) {\n            <ion-chip [outline]=\"true\">{{ filter?.['condition'] }}</ion-chip>\n          }\n          @if(filter?.['value']) {\n            <ion-chip [outline]=\"true\" class=\"dcf-filter-value\">\n              {{ filter?.['value'] }}\n              <ion-icon name=\"close\" (click)=\"removeFilter(filter?.['value'])\" size=\"small\"></ion-icon>\n            </ion-chip>\n          }\n        }\n        <div class=\"dcf-width-1-1\">\n          <input\n            fill=\"none\"\n            [(ngModel)]=\"value\"\n            (keydown.enter)=\"addFilter(value, $event)\"\n            (keydown.backspace)=\"clear(value)\"\n            (input)=\"handleInput($event)\"\n            (click)=\"handleFocus()\"\n            (blur)=\"handleBlur()\"\n            type=\"text\"\n            [readonly]=\"step !== 3\"\n            placeholder=\"{{ locale + (step === 3 ? '.type' : '.select') | translate }}\"\n            #component\n          />\n          @if(windowWidth >= 768) {\n            <div [class]=\"'dcf-dropdown ' + (options.length > 0 ? ' dcf-active' : '')\" #optionsFilterElement>\n              <div>\n                @if(filteredOptions.length > 0) {\n                  @for(key of filteredOptions; track key) {\n                    <div\n                      class=\"dcf-item\"\n                      tabindex=\"0\"\n                      (keydown.enter)=\"selectOption(key)\"\n                      (click)=\"selectOption(key)\">\n                      {{ key }}\n                    </div>\n                  }\n                } @else {\n                  <div class=\"dcf-empty\"\n                    (click)=\"filteredOptions = options; value = ''\"\n                    tabindex=\"0\"\n                    (keydown.enter)=\"filteredOptions = options; value = ''\"\n                  >\n                    {{ 'no_suggestions' | translate }}\n                  </div>\n                }\n              </div>\n            </div>\n          }\n        </div>\n      </div>\n      @if(filterValue.length > 0) {\n        <div class=\"dcf-icon-clear\">\n          <ion-button fill=\"clear\" size=\"small\" (click)=\"clear()\">\n            <ion-icon name=\"trash-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n          </ion-button>\n        </div>\n      }\n      <div class=\"dcf-icon-search\">\n        <ion-button fill=\"clear\" size=\"small\" (click)=\"submit()\">\n          <ion-icon name=\"search-outline\" color=\"dark\" slot=\"icon-only\"></ion-icon>\n        </ion-button>\n      </div>\n    </div>\n    @if(windowWidth < 768) {\n      <div [class]=\"'dcf-dropdown ' + (options.length > 0 ? ' dcf-active' : '')\" #optionsFilterElement>\n        <div>\n          @if(filteredOptions.length > 0) {\n            @for(key of filteredOptions; track key) {\n              <div\n                class=\"dcf-item\"\n                tabindex=\"0\"\n                (keydown.enter)=\"selectOption(key)\"\n                (click)=\"selectOption(key)\">\n                {{ key }}\n              </div>\n            }\n          } @else {\n            <div class=\"dcf-empty\"\n            (click)=\"filteredOptions = options; value = ''\"\n            tabindex=\"0\"\n            (keydown.enter)=\"filteredOptions = options; value = ''\"\n            >\n              {{ 'no_suggestions' | translate }}\n            </div>\n          }\n        </div>\n      </div>\n    }\n  </div>\n  @if(!disableSort) {\n    <div class=\"dcf-width-1-5@m dcf-width-1-1 dcf-sort-container\">\n      <div class=\"dcf-grid dcf-grid-collapse dcf-flex dcf-flex-middle dcf-grid-match\">\n        <div class=\"dcf-width-expand\">\n          <ion-select\n              toggleIcon=\"chevron-down-outline\"\n              expandedIcon=\"chevron-up-outline\"\n              class=\"dcf-sort-select\"\n              (ionChange)=\"handleSortChange($event)\"\n              interface=\"popover\"\n              [value]=\"sortValue\"\n              label-placement=\"floating\"\n              fill=\"outline\"\n            >\n            @for(sort of sortBy; track sort) {\n              aa\n              <ion-select-option [value]=\"sort\">{{ sort | translate }}</ion-select-option>\n            }\n          </ion-select>\n        </div>\n        <div class=\"dcf-width-auto\">\n          <ion-button (click)=\"handleSortDirectionChange()\" fill=\"clear\">\n            <ion-icon slot=\"icon-only\" [name]=\"sortDirection === 'desc' ? 'arrow-down-outline' : 'arrow-up-outline'\"></ion-icon>\n          </ion-button>\n        </div>\n      </div>\n    </div>\n  }\n</div>\n\n\n"]}
@@ -1,6 +1,6 @@
1
1
  import { Component, Input } from '@angular/core';
2
- import { ForAngularModule } from 'src/lib/for-angular.module';
3
- import { NgxBaseComponent } from 'src/lib/engine/NgxBaseComponent';
2
+ import { ForAngularModule } from '../../for-angular.module';
3
+ import { NgxBaseComponent } from '../../engine/NgxBaseComponent';
4
4
  import { ComponentRendererComponent } from '../component-renderer/component-renderer.component';
5
5
  import { ModelRendererComponent } from '../model-renderer/model-renderer.component';
6
6
  import * as i0 from "@angular/core";
@@ -173,4 +173,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
173
173
  }], children: [{
174
174
  type: Input
175
175
  }] } });
176
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"layout.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/layout/layout.component.ts","../../../../../src/lib/components/layout/layout.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAS,MAAM,eAAe,CAAC;AACxD,OAAO,EAAE,gBAAgB,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,EAAE,gBAAgB,EAAE,MAAM,iCAAiC,CAAC;AAEnE,OAAO,EAAE,0BAA0B,EAAE,MAAM,oDAAoD,CAAC;AAChG,OAAO,EAAE,sBAAsB,EAAE,MAAM,4CAA4C,CAAC;;;AAGpF;;;;;;;;;;;GAWG;AASH,MAAM,OAAO,eAAgB,SAAQ,gBAAgB;IAyDnD;;;;;;;OAOG;IACH;QACE,KAAK,CAAC,iBAAiB,CAAC,CAAA;QAhE1B;;;;;;;;;;WAUG;QAEH,SAAI,GAAsB,CAAC,CAAC;QAE5B;;;;;;;;;WASG;QAEH,SAAI,GAAmC,CAAC,CAAC;QAEzC;;;;;;;;;;WAUG;QAEH,eAAU,GAAuB,QAAQ,CAAC;QAE1C;;;;;;;;;;WAUG;QAEH,aAAQ,GAAe,EAAE,CAAC;IAY1B,CAAC;IAED;;;;;;;;;;OAUG;IACH,IAAI,KAAK;QACP,IAAI,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QACrB,IAAG,OAAO,IAAI,KAAK,QAAQ;YACzB,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,EAAC,EAAE,GAAG,EAAE,CAAE,EAAE,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;;;;;OAUG;IACH,IAAI,KAAK;QACP,IAAI,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QACrB,IAAG,OAAO,IAAI,KAAK,QAAQ;YACzB,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,EAAC,EAAE,GAAG,EAAE,CAAC,CAAC,EAAC,KAAK,EAAE,EAAE,EAAC,CAAC,CAAC,CAAC;QACjE,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;YAC7B,OAAO;gBACL,KAAK,EAAG,GAAgB,EAAE,CAAC,OAAO,CAAC;gBACnC,IAAI,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE;oBACjC,IAAG,KAAK,CAAC,KAAK,CAAC,KAAK,KAAK,GAAG,CAAC;wBAC3B,OAAO,KAAK,CAAC;gBACjB,CAAC,CAAC;aACH,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;OAOG;IACH,QAAQ;QACN,IAAI,CAAC,UAAU,EAAE,CAAC;IACpB,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;OAqBG;IACM,UAAU;QACjB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACtB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAwB,CAAC;QAClF,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC;QACvB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC;QACvB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;IAC1B,CAAC;+GAzJU,eAAe;mGAAf,eAAe,2LC5B5B,43CAoCA,4pBDZY,gBAAgB,4FAAE,sBAAsB,2IAAE,0BAA0B;;4FAInE,eAAe;kBAR3B,SAAS;+BACE,kBAAkB,WAGnB,CAAC,gBAAgB,EAAE,sBAAsB,EAAE,0BAA0B,CAAC,cACnE,IAAI;wDAiBhB,IAAI;sBADH,KAAK;gBAcN,IAAI;sBADH,KAAK;gBAeN,UAAU;sBADT,KAAK;gBAeN,QAAQ;sBADP,KAAK","sourcesContent":["import { Component, Input, OnInit} from '@angular/core';\nimport { ForAngularModule } from 'src/lib/for-angular.module';\nimport { NgxBaseComponent } from 'src/lib/engine/NgxBaseComponent';\nimport { KeyValue } from 'src/lib/engine';\nimport { ComponentRendererComponent } from '../component-renderer/component-renderer.component';\nimport { ModelRendererComponent } from '../model-renderer/model-renderer.component';\nimport { UIMediaBreakPoints } from '@decaf-ts/ui-decorators';\n\n/**\n * @description Layout component for creating responsive grid layouts in Angular applications.\n * @summary This component provides a flexible grid system that can be configured with dynamic\n * rows and columns. It supports responsive breakpoints and can render child components within\n * the grid structure. The component extends NgxBaseComponent to inherit common functionality\n * and integrates with the model and component renderer systems.\n *\n * @class LayoutComponent\n * @extends {NgxBaseComponent}\n * @implements {OnInit}\n * @memberOf LayoutComponent\n */\n@Component({\n  selector: 'ngx-decaf-layout',\n  templateUrl: './layout.component.html',\n  styleUrls: ['./layout.component.scss'],\n  imports: [ForAngularModule, ModelRendererComponent, ComponentRendererComponent],\n  standalone: true,\n\n})\nexport class LayoutComponent extends NgxBaseComponent implements OnInit {\n\n  /**\n   * @description Number of columns or array of column definitions for the grid layout.\n   * @summary Defines the column structure of the grid. When a number is provided, it creates\n   * that many equal-width columns. When an array is provided, each element can define specific\n   * column properties or sizing. This allows for flexible grid layouts that can adapt to\n   * different content requirements.\n   *\n   * @type {(number | string[])}\n   * @default 1\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  cols: number | string[] = 1;\n\n  /**\n   * @description Number of rows or array of row definitions for the grid layout.\n   * @summary Defines the row structure of the grid. When a number is provided, it creates\n   * that many equal-height rows. When an array is provided, each element can define specific\n   * row properties or sizing. This provides control over vertical spacing and content organization.\n   *\n   * @type {(number | string[])}\n   * @default 1\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  rows: number | KeyValue[] | string[] = 1;\n\n  /**\n   * @description Media breakpoint for responsive behavior.\n   * @summary Determines the responsive breakpoint at which the layout should adapt.\n   * This affects how the grid behaves on different screen sizes, allowing for\n   * mobile-first or desktop-first responsive design patterns. The breakpoint\n   * is automatically processed to ensure compatibility with the UI framework.\n   *\n   * @type {UIMediaBreakPoints}\n   * @default 'medium'\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  breakpoint: UIMediaBreakPoints = 'medium';\n\n  /**\n   * @description Array of child components or data to render within the grid.\n   * @summary Contains the child elements that will be distributed across the grid layout.\n   * Each item in the array represents content that will be rendered using the appropriate\n   * renderer component (ModelRenderer or ComponentRenderer). This allows for mixed content\n   * types within a single layout structure.\n   *\n   * @type {KeyValue[]}\n   * @default []\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  children: KeyValue[] = [];\n\n  /**\n   * @description Creates an instance of LayoutComponent.\n   * @summary Initializes a new LayoutComponent with the component name \"LayoutComponent\".\n   * This constructor calls the parent NgxBaseComponent constructor to set up base\n   * functionality and component identification.\n   *\n   * @memberOf LayoutComponent\n   */\n  constructor() {\n    super(\"LayoutComponent\")\n  }\n\n  /**\n   * @description Getter that converts columns input to an array format.\n   * @summary Transforms the cols input property into a standardized string array format.\n   * When cols is a number, it creates an array with that many empty string elements.\n   * When cols is already an array, it returns the array as-is. This normalization\n   * ensures consistent handling of column definitions in the template.\n   *\n   * @type {string[]}\n   * @readonly\n   * @memberOf LayoutComponent\n   */\n  get _cols(): string[] {\n    let cols = this.cols;\n    if(typeof cols === \"number\")\n      cols = Array.from({length: Number(cols)}, () =>  ``);\n    return cols;\n  }\n\n  /**\n   * @description Getter that converts rows input to an array format.\n   * @summary Transforms the rows input property into a standardized string array format.\n   * When rows is a number, it creates an array with that many empty string elements.\n   * When rows is already an array, it returns the array as-is. This normalization\n   * ensures consistent handling of row definitions in the template.\n   *\n   * @type {KeyValue[]}\n   * @readonly\n   * @memberOf LayoutComponent\n   */\n  get _rows(): KeyValue[] {\n    let rows = this.rows;\n    if(typeof rows === \"number\")\n      rows = Array.from({length: Number(rows)}, () => ({title: ''}));\n    return rows.map((row, index) => {\n      return {\n        title: (row as KeyValue)?.['title'],\n        cols: this.children.filter(child => {\n          if(child['row'] === index + 1)\n            return child;\n        })\n      };\n    });\n  }\n\n  /**\n   * @description Angular lifecycle hook that runs after component initialization.\n   * @summary Called once, after the first ngOnChanges(). This method triggers the\n   * component's initialization process, which includes property parsing and grid\n   * setup. It ensures the component is properly configured before rendering.\n   *\n   * @memberOf LayoutComponent\n   */\n  ngOnInit(): void {\n    this.initialize();\n  }\n\n  /**\n   * @description Initializes the layout component with processed properties.\n   * @summary Overrides the base component's initialize method to set up the grid layout.\n   * This method processes input properties, normalizes the breakpoint value, converts\n   * rows and columns to their array representations, and marks the component as initialized.\n   * The initialization ensures all properties are in the correct format for rendering.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant L as LayoutComponent\n   *   participant B as NgxBaseComponent\n   *\n   *   L->>B: parseProps(this)\n   *   Note over L: Process component properties\n   *   L->>L: Normalize breakpoint to lowercase\n   *   L->>L: Convert rows to array format\n   *   L->>L: Convert cols to array format\n   *   L->>L: Set initialized = true\n   *\n   * @override\n   * @memberOf LayoutComponent\n   */\n  override initialize() {\n    this.parseProps(this);\n    this.breakpoint = this.breakpoint.slice(0, 2).toLowerCase() as UIMediaBreakPoints;\n    this.cols = this._cols;\n    this.rows = this._rows;\n    this.initialized = true;\n  }\n}\n","\n@if(initialized) {\n  @for (row of rows; track trackItemFn($index, row); let rowIndex = $index) {\n    <div class=\"dcf-grid dcf-grid-collapse dcf-grid-match\">\n      @if(row) {\n        <div class=\"dcf-width-1-1 dcf-grid-title\">\n          <ion-card class=\"dcf-grid-title\">\n            {{row.title | translate}}\n          </ion-card>\n        </div>\n      }\n      @for (child of row.cols; track trackItemFn($index, child.col); let colIndex = $index) {\n        <div [class]=\"(child.col === cols.length ? 'dcf-width-1-1' : 'dcf-width-'+child.col+'-'+cols.length+'@'+breakpoint)\">\n            <div [class]=\"'dcf-grid-child '+child.col \">\n              @if(child.tag === 'ngx-decaf-crud-form') {\n                <ion-card [class]=\"'dcf-height-1-1 ' + className\">\n                  <ion-card-content>\n                    <ngx-decaf-model-renderer\n                      [model]=\"child.props.name\"\n                      (listenEvent)=\"handleEvent($event)\"\n                      />\n                    </ion-card-content>\n                  </ion-card>\n              } @else {\n                <ngx-decaf-component-renderer\n                  [tag]=\"child.tag\"\n                  (listenEvent)=\"handleEvent($event)\"\n                  [globals]=\"{props: child.props}\"\n                />\n              }\n            </div>\n        </div>\n      }\n    </div>\n  }\n}\n"]}
176
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"layout.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/layout/layout.component.ts","../../../../../src/lib/components/layout/layout.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAS,MAAM,eAAe,CAAC;AACxD,OAAO,EAAE,gBAAgB,EAAE,MAAM,0BAA0B,CAAC;AAC5D,OAAO,EAAE,gBAAgB,EAAE,MAAM,+BAA+B,CAAC;AAEjE,OAAO,EAAE,0BAA0B,EAAE,MAAM,oDAAoD,CAAC;AAChG,OAAO,EAAE,sBAAsB,EAAE,MAAM,4CAA4C,CAAC;;;AAGpF;;;;;;;;;;;GAWG;AASH,MAAM,OAAO,eAAgB,SAAQ,gBAAgB;IAyDnD;;;;;;;OAOG;IACH;QACE,KAAK,CAAC,iBAAiB,CAAC,CAAA;QAhE1B;;;;;;;;;;WAUG;QAEH,SAAI,GAAsB,CAAC,CAAC;QAE5B;;;;;;;;;WASG;QAEH,SAAI,GAAmC,CAAC,CAAC;QAEzC;;;;;;;;;;WAUG;QAEH,eAAU,GAAuB,QAAQ,CAAC;QAE1C;;;;;;;;;;WAUG;QAEH,aAAQ,GAAe,EAAE,CAAC;IAY1B,CAAC;IAED;;;;;;;;;;OAUG;IACH,IAAI,KAAK;QACP,IAAI,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QACrB,IAAG,OAAO,IAAI,KAAK,QAAQ;YACzB,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,EAAC,EAAE,GAAG,EAAE,CAAE,EAAE,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;;;;;OAUG;IACH,IAAI,KAAK;QACP,IAAI,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QACrB,IAAG,OAAO,IAAI,KAAK,QAAQ;YACzB,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,EAAC,EAAE,GAAG,EAAE,CAAC,CAAC,EAAC,KAAK,EAAE,EAAE,EAAC,CAAC,CAAC,CAAC;QACjE,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;YAC7B,OAAO;gBACL,KAAK,EAAG,GAAgB,EAAE,CAAC,OAAO,CAAC;gBACnC,IAAI,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE;oBACjC,IAAG,KAAK,CAAC,KAAK,CAAC,KAAK,KAAK,GAAG,CAAC;wBAC3B,OAAO,KAAK,CAAC;gBACjB,CAAC,CAAC;aACH,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;OAOG;IACH,QAAQ;QACN,IAAI,CAAC,UAAU,EAAE,CAAC;IACpB,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;OAqBG;IACM,UAAU;QACjB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACtB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAwB,CAAC;QAClF,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC;QACvB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC;QACvB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;IAC1B,CAAC;+GAzJU,eAAe;mGAAf,eAAe,2LC5B5B,43CAoCA,4pBDZY,gBAAgB,4FAAE,sBAAsB,2IAAE,0BAA0B;;4FAInE,eAAe;kBAR3B,SAAS;+BACE,kBAAkB,WAGnB,CAAC,gBAAgB,EAAE,sBAAsB,EAAE,0BAA0B,CAAC,cACnE,IAAI;wDAiBhB,IAAI;sBADH,KAAK;gBAcN,IAAI;sBADH,KAAK;gBAeN,UAAU;sBADT,KAAK;gBAeN,QAAQ;sBADP,KAAK","sourcesContent":["import { Component, Input, OnInit} from '@angular/core';\nimport { ForAngularModule } from '../../for-angular.module';\nimport { NgxBaseComponent } from '../../engine/NgxBaseComponent';\nimport { KeyValue } from '../../engine';\nimport { ComponentRendererComponent } from '../component-renderer/component-renderer.component';\nimport { ModelRendererComponent } from '../model-renderer/model-renderer.component';\nimport { UIMediaBreakPoints } from '@decaf-ts/ui-decorators';\n\n/**\n * @description Layout component for creating responsive grid layouts in Angular applications.\n * @summary This component provides a flexible grid system that can be configured with dynamic\n * rows and columns. It supports responsive breakpoints and can render child components within\n * the grid structure. The component extends NgxBaseComponent to inherit common functionality\n * and integrates with the model and component renderer systems.\n *\n * @class LayoutComponent\n * @extends {NgxBaseComponent}\n * @implements {OnInit}\n * @memberOf LayoutComponent\n */\n@Component({\n  selector: 'ngx-decaf-layout',\n  templateUrl: './layout.component.html',\n  styleUrls: ['./layout.component.scss'],\n  imports: [ForAngularModule, ModelRendererComponent, ComponentRendererComponent],\n  standalone: true,\n\n})\nexport class LayoutComponent extends NgxBaseComponent implements OnInit {\n\n  /**\n   * @description Number of columns or array of column definitions for the grid layout.\n   * @summary Defines the column structure of the grid. When a number is provided, it creates\n   * that many equal-width columns. When an array is provided, each element can define specific\n   * column properties or sizing. This allows for flexible grid layouts that can adapt to\n   * different content requirements.\n   *\n   * @type {(number | string[])}\n   * @default 1\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  cols: number | string[] = 1;\n\n  /**\n   * @description Number of rows or array of row definitions for the grid layout.\n   * @summary Defines the row structure of the grid. When a number is provided, it creates\n   * that many equal-height rows. When an array is provided, each element can define specific\n   * row properties or sizing. This provides control over vertical spacing and content organization.\n   *\n   * @type {(number | string[])}\n   * @default 1\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  rows: number | KeyValue[] | string[] = 1;\n\n  /**\n   * @description Media breakpoint for responsive behavior.\n   * @summary Determines the responsive breakpoint at which the layout should adapt.\n   * This affects how the grid behaves on different screen sizes, allowing for\n   * mobile-first or desktop-first responsive design patterns. The breakpoint\n   * is automatically processed to ensure compatibility with the UI framework.\n   *\n   * @type {UIMediaBreakPoints}\n   * @default 'medium'\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  breakpoint: UIMediaBreakPoints = 'medium';\n\n  /**\n   * @description Array of child components or data to render within the grid.\n   * @summary Contains the child elements that will be distributed across the grid layout.\n   * Each item in the array represents content that will be rendered using the appropriate\n   * renderer component (ModelRenderer or ComponentRenderer). This allows for mixed content\n   * types within a single layout structure.\n   *\n   * @type {KeyValue[]}\n   * @default []\n   * @memberOf LayoutComponent\n   */\n  @Input()\n  children: KeyValue[] = [];\n\n  /**\n   * @description Creates an instance of LayoutComponent.\n   * @summary Initializes a new LayoutComponent with the component name \"LayoutComponent\".\n   * This constructor calls the parent NgxBaseComponent constructor to set up base\n   * functionality and component identification.\n   *\n   * @memberOf LayoutComponent\n   */\n  constructor() {\n    super(\"LayoutComponent\")\n  }\n\n  /**\n   * @description Getter that converts columns input to an array format.\n   * @summary Transforms the cols input property into a standardized string array format.\n   * When cols is a number, it creates an array with that many empty string elements.\n   * When cols is already an array, it returns the array as-is. This normalization\n   * ensures consistent handling of column definitions in the template.\n   *\n   * @type {string[]}\n   * @readonly\n   * @memberOf LayoutComponent\n   */\n  get _cols(): string[] {\n    let cols = this.cols;\n    if(typeof cols === \"number\")\n      cols = Array.from({length: Number(cols)}, () =>  ``);\n    return cols;\n  }\n\n  /**\n   * @description Getter that converts rows input to an array format.\n   * @summary Transforms the rows input property into a standardized string array format.\n   * When rows is a number, it creates an array with that many empty string elements.\n   * When rows is already an array, it returns the array as-is. This normalization\n   * ensures consistent handling of row definitions in the template.\n   *\n   * @type {KeyValue[]}\n   * @readonly\n   * @memberOf LayoutComponent\n   */\n  get _rows(): KeyValue[] {\n    let rows = this.rows;\n    if(typeof rows === \"number\")\n      rows = Array.from({length: Number(rows)}, () => ({title: ''}));\n    return rows.map((row, index) => {\n      return {\n        title: (row as KeyValue)?.['title'],\n        cols: this.children.filter(child => {\n          if(child['row'] === index + 1)\n            return child;\n        })\n      };\n    });\n  }\n\n  /**\n   * @description Angular lifecycle hook that runs after component initialization.\n   * @summary Called once, after the first ngOnChanges(). This method triggers the\n   * component's initialization process, which includes property parsing and grid\n   * setup. It ensures the component is properly configured before rendering.\n   *\n   * @memberOf LayoutComponent\n   */\n  ngOnInit(): void {\n    this.initialize();\n  }\n\n  /**\n   * @description Initializes the layout component with processed properties.\n   * @summary Overrides the base component's initialize method to set up the grid layout.\n   * This method processes input properties, normalizes the breakpoint value, converts\n   * rows and columns to their array representations, and marks the component as initialized.\n   * The initialization ensures all properties are in the correct format for rendering.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant L as LayoutComponent\n   *   participant B as NgxBaseComponent\n   *\n   *   L->>B: parseProps(this)\n   *   Note over L: Process component properties\n   *   L->>L: Normalize breakpoint to lowercase\n   *   L->>L: Convert rows to array format\n   *   L->>L: Convert cols to array format\n   *   L->>L: Set initialized = true\n   *\n   * @override\n   * @memberOf LayoutComponent\n   */\n  override initialize() {\n    this.parseProps(this);\n    this.breakpoint = this.breakpoint.slice(0, 2).toLowerCase() as UIMediaBreakPoints;\n    this.cols = this._cols;\n    this.rows = this._rows;\n    this.initialized = true;\n  }\n}\n","\n@if(initialized) {\n  @for (row of rows; track trackItemFn($index, row); let rowIndex = $index) {\n    <div class=\"dcf-grid dcf-grid-collapse dcf-grid-match\">\n      @if(row) {\n        <div class=\"dcf-width-1-1 dcf-grid-title\">\n          <ion-card class=\"dcf-grid-title\">\n            {{row.title | translate}}\n          </ion-card>\n        </div>\n      }\n      @for (child of row.cols; track trackItemFn($index, child.col); let colIndex = $index) {\n        <div [class]=\"(child.col === cols.length ? 'dcf-width-1-1' : 'dcf-width-'+child.col+'-'+cols.length+'@'+breakpoint)\">\n            <div [class]=\"'dcf-grid-child '+child.col \">\n              @if(child.tag === 'ngx-decaf-crud-form') {\n                <ion-card [class]=\"'dcf-height-1-1 ' + className\">\n                  <ion-card-content>\n                    <ngx-decaf-model-renderer\n                      [model]=\"child.props.name\"\n                      (listenEvent)=\"handleEvent($event)\"\n                      />\n                    </ion-card-content>\n                  </ion-card>\n              } @else {\n                <ngx-decaf-component-renderer\n                  [tag]=\"child.tag\"\n                  (listenEvent)=\"handleEvent($event)\"\n                  [globals]=\"{props: child.props}\"\n                />\n              }\n            </div>\n        </div>\n      }\n    </div>\n  }\n}\n"]}