@quadrel-enterprise-ui/framework 18.21.5 → 18.21.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/page/model/page-config.interface.mjs +1 -1
- package/esm2022/lib/page/page.component.mjs +7 -5
- package/esm2022/lib/page-tabs/page-tabs.component.mjs +75 -3
- package/fesm2022/quadrel-enterprise-ui-framework.mjs +164 -90
- package/fesm2022/quadrel-enterprise-ui-framework.mjs.map +1 -1
- package/lib/page/model/page-config.interface.d.ts +18 -1
- package/lib/page/page.component.d.ts +6 -4
- package/lib/page-tabs/page-tabs.component.d.ts +71 -1
- package/package.json +1 -1
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-config.interface.js","sourceRoot":"","sources":["../../../../../../libs/qd-ui/src/lib/page/model/page-config.interface.ts"],"names":[],"mappings":"","sourcesContent":["import { QdMenuButtonActionConfig } from '../../button/model/button-config';\nimport { QdPageHeaderFacetConfig } from './page-header-facet.interface';\n\n/**\n * @description Configuration of QdPage.\n *\n * @template T - The type of the page object data\n */\nexport type QdPageConfig<T extends object = object> =\n  | QdPageConfigOverview\n  | QdPageConfigCreate<T>\n  | QdPageConfigInspect<T>\n  | QdPageConfigCustom<T>;\n\nexport interface QdPageSelectedContext {\n  value: string;\n  label: {\n    i18n: string;\n  };\n}\n\ninterface QdPageContextConfigBase {\n  /**\n   * @description Description for the specific header facet\n   */\n  label: {\n    /**\n     * @description i18n string for the description, used for translation.\n     */\n    i18n: string;\n  };\n\n  /**\n   * @description Allows to identify the context\n   */\n  id: string;\n\n  /**\n   * @description Register some handler that is executed when the selection changes.\n   */\n  changed?: (selection: QdPageSelectedContext | QdPageSelectedContext[] | null) => void;\n}\n\n/**\n * @description Configuration for a context with single selection\n */\nexport interface QdPageContextConfigSingle extends QdPageContextConfigBase {\n  /**\n   * @description Configures the type on how the context selection happens.\n   */\n  type: 'single';\n\n  /**\n   * @description Available options for selection.\n   */\n  options: QdPageSelectedContext[];\n\n  /**\n   * @description Configures whether the context options are searchable for the user.\n   * @default false\n   */\n  hasSearch?: boolean;\n\n  /**\n   * @description Placeholder for search\n   */\n  searchPlaceholder?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n\n  /**\n   * @description Title for the dialog that will be opened\n   */\n  title?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n}\n\n/**\n * @description Configuration for a context with multi selection\n */\nexport interface QdPageContextConfigMulti extends QdPageContextConfigBase {\n  /**\n   * @description Configures the type on how the context selection happens.\n   */\n  type: 'multi';\n\n  /**\n   * @description Available options for selection.\n   */\n  options: QdPageSelectedContext[];\n\n  /**\n   * @description Configures whether the context options are searchable for the user.\n   * @default false\n   */\n  hasSearch?: boolean;\n\n  /**\n   * @description Placeholder for search\n\n   */\n  searchPlaceholder?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n\n  /**\n   * @description Title for the dialog that will be opened\n   */\n  title?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n}\n\n/**\n * @description Configuration for a context with custom selection (e. g. your own dialog)\n */\nexport interface QdPageContextConfigCustom extends QdPageContextConfigBase {\n  /**\n   * @description Configures the type on how the context selection happens.\n   */\n  type: 'custom';\n\n  /**\n   * @description Handler to open a custom dialog for selection\n   */\n  open: () => void;\n}\n\n/**\n * @description Configures the context with one of the available types\n */\nexport type QdPageContextConfig = QdPageContextConfigSingle | QdPageContextConfigMulti | QdPageContextConfigCustom;\n\n/**\n * @description Base configuration for QdPage.\n */\ninterface QdPageConfigBase {\n  /**\n   * @description Title of the page.\n   */\n  title: QdPageTitle;\n\n  /**\n   * @description Hides contexts from being shown. The ContextService will still work in the background.\n   *\n   * @default false\n   */\n  hideContexts?: boolean;\n\n  /**\n   * @description Label for the menu button of custom actions.\n   */\n  customActionsLabel?: QdPageCustomActionsLabel;\n}\n\n/**\n * @description Configuration for the overview page type.\n *\n * @template T - The type of the page object data\n */\nexport interface QdPageConfigOverview extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'overview'\n   */\n  pageType?: 'overview'; // Optional to make it default\n\n  /**\n   * @description Configuration specific to the overview page type.\n   */\n  pageTypeConfig?: QdPageTypeOverviewConfig;\n}\n\n/**\n * @description Configuration for the create page type.\n *\n * @template T - The type of the page object data\n */\nexport interface QdPageConfigCreate<T extends object = object> extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'create'\n   */\n  pageType: 'create';\n\n  /**\n   * @description Configuration specific to the create page type.\n   */\n  pageTypeConfig?: QdPageTypeCreateConfig;\n\n  /**\n   * @description Information placed in the header of the page. Concerns current shown object.\n   */\n  headerFacets?: QdPageHeaderFacetConfig<T>[];\n}\n\nexport interface QdPageConfigInspect<T extends object = object> extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'inspect'\n   */\n  pageType: 'inspect';\n\n  /**\n   * @description Configuration specific to the inspect page type.\n   */\n  pageTypeConfig?: QdPageTypeInspectConfig;\n\n  /**\n   * @description Information placed in the header of the page. Concerns current shown object.\n   */\n  headerFacets?: QdPageHeaderFacetConfig<T>[];\n\n  /**\n   * @description Determines whether the facets in header should be collapsed and custom actions are in a menu. This mode is designed for touch devices and mobile phones only.\n   *\n   * @default false\n   */\n  adaptiveMode?: boolean;\n}\n\nexport interface QdPageConfigCustom<T extends object = object> extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'custom'\n   */\n  pageType: 'custom';\n\n  /**\n   * @description Configuration specific to the custom page type.\n   */\n  pageTypeConfig?: QdPageTypeCustomConfig;\n\n  /**\n   * @description Information placed in the header of the page. Concerns current shown object.\n   */\n  headerFacets?: QdPageHeaderFacetConfig<T>[];\n\n  /**\n   * @description Determines whether the facets in header should be collapsed and custom actions are in a menu. This mode is designed for touch devices and mobile phones only.\n   *\n   * @default false\n   */\n  adaptiveMode?: boolean;\n}\n\n/**\n * Configures the overview page\n */\nexport interface QdPageTypeOverviewConfig {\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: QdMenuButtonActionConfig[];\n}\n\n/**\n * @description Configures the create page.\n */\nexport interface QdPageTypeCreateConfig {\n  /**\n   * @description Configuration for the save draft action.\n   */\n  saveDraft?: QdPageSaveDraftAction;\n\n  /**\n   * @description Configuration for the cancel button.\n   */\n  cancel: QdPageCancelAction;\n\n  /**\n   * @description Configuration for the submit button.\n   */\n  submit: QdPageSubmitAction;\n\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: QdMenuButtonActionConfig[];\n}\n\n/**\n * @description Configures the inspect page.\n */\nexport interface QdPageTypeInspectConfig {\n  /**\n   * @description Operation mode for the page.\n   *\n   * * This property determines the mode in which the page operates. It can be either 'view' or 'edit'.\n   *\n   * * **view**: The page is in read-only mode.\n   * * **edit**: The page allows modifications.\n   */\n  operationMode?: QdInspectOperationMode;\n\n  /**\n   * @description Specifies whether the user can edit the object or not.\n   */\n  hideEdit?: boolean;\n\n  /**\n   * @description Configuration for the edit action.\n   */\n  edit?: QdPageEditAction;\n\n  /**\n   * @description Configuration for the save action.\n   */\n  save?: QdPageSaveAction;\n\n  /**\n   * @description Configuration for the cancel action.\n   */\n  cancel: QdPageCancelAction;\n\n  /**\n   * @description Configuration for the delete action.\n   */\n  delete?: QdPageDeleteAction;\n\n  /**\n   * @description Configuration for the archive action.\n   */\n  archive?: QdPageArchiveAction;\n\n  /**\n   * @description Configuration for the submit action.\n   */\n  submit?: QdPageSubmitAction;\n\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * In the \"Inspect\" page type, you can assign custom actions to the operation modes \"view\" or \"edit.\" By default, the action is active in both modes.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: (QdMenuButtonActionConfig & { operationMode?: QdInspectOperationMode })[];\n}\n\nexport interface QdPageTypeCustomConfig {\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: QdMenuButtonActionConfig[];\n}\n\nexport interface QdPageTitle {\n  /**\n   * @description Defines the translation key\n   *\n   * * @example { i18n: \"i18n.overview.title\"}\n   */\n  i18n: string;\n}\n\n/**\n * @description Configuration for the custom actions label, used when multiple custom actions exist.\n */\nexport interface QdPageCustomActionsLabel {\n  /**\n   * @description Label for the menu button of custom actions, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   *\n   * * This label becomes visible if more than one custom action is defined, as the actions will be grouped under a QdMenuButton that displays this label.\n   */\n  i18n: string;\n}\n\n/**\n * @description Operation modes for the inspect page.\n *\n * @default 'view'\n */\nexport type QdInspectOperationMode = 'view' | 'edit';\n\n/**\n * @description Interface for the save action, including optional validation configuration.\n */\nexport interface QdPageSaveAction {\n  /**\n   * @description Label for the save action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the save action is executed.\n   */\n  handler: (formValues?: any) => void;\n\n  /**\n   * @description Optional validation flag, defaults to true.\n   * Determines whether form validation is required before saving and ensures that changes must be present in the form before saving.\n   *\n   * * By default, this flag is set to true, meaning that validation is enforced, and the form will only save if there are changes.\n   *\n   * * If set to `false`, the form will be saved without validation checks, but changes must still be present to proceed with the save operation.\n   *\n   * * This flag is useful in scenarios where form validation is not needed or should be bypassed, but it ensures that only modified forms can be saved.\n   */\n  hasValidation?: boolean;\n}\n\n/**\n * @description Interface for the safe draft action.\n */\nexport interface QdPageSaveDraftAction {\n  /**\n   * @description Label for the save draft action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the save draft action is executed.\n   */\n  handler: () => void;\n}\n\n/**\n * @description Interface for the cancel action, including an optional confirmation message.\n */\nexport interface QdPageCancelAction {\n  /**\n   * @description Label for the cancel action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the cancel action is executed.\n   */\n  handler: (formValues?: any) => void;\n\n  /**\n   * @description An optional confirmation message displayed in a dialog when the user attempts to cancel the operation, warning that all changes will be discarded.\n   * If no custom message is provided, a standard message will be shown by default.\n   *\n   * * The cancellation process checks whether there are unsaved changes in the form. If there are changes, the dialog will appear.\n   *\n   * * If no changes are detected, the cancellation proceeds without showing the confirmation dialog.\n   */\n  confirmationMessage?: { i18n: string };\n}\n\n/**\n * @description Interface for the submit action.\n */\nexport interface QdPageSubmitAction {\n  /**\n   * @description Label for the submit action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the submit action is executed.\n   */\n  handler: (formValues?: any) => void;\n\n  /**\n   * @description By default, the visibility of the submit button depends on the page type and the operation mode.\n   * To completely hide the submit button, you can set this isHidden flag.\n   *\n   * @default false\n   */\n  isHidden?: boolean;\n}\n\n/**\n * @description Interface for the edit action.\n */\nexport interface QdPageEditAction {\n  /**\n   * @description Label for the edit action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the edit action is executed.\n   */\n  handler: (formValues?: any) => void;\n}\n\n/**\n * @description Interface for the delete action.\n *\n * * This action is represented as an icon, so it does not require a label.\n */\nexport interface QdPageDeleteAction {\n  /**\n   * @description Handler function that is triggered when the delete action is executed.\n   */\n  handler: (formValues?: any) => void;\n}\n\n/**\n * @description Interface for the archive action.\n *\n * * This action is represented as an icon, so it does not require a label.\n */\nexport interface QdPageArchiveAction {\n  /**\n   * @description Handler function that is triggered when the archive action is executed.\n   */\n  handler: (formValues?: any) => void;\n}\n"]}
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-config.interface.js","sourceRoot":"","sources":["../../../../../../libs/qd-ui/src/lib/page/model/page-config.interface.ts"],"names":[],"mappings":"","sourcesContent":["import { QdMenuButtonActionConfig } from '../../button/model/button-config';\nimport { QdPageHeaderFacetConfig } from './page-header-facet.interface';\n\n/**\n * @description Configuration of QdPage.\n *\n * @template T - The type of the page object data\n */\nexport type QdPageConfig<T extends object = object> =\n  | QdPageConfigOverview\n  | QdPageConfigCreate<T>\n  | QdPageConfigInspect<T>\n  | QdPageConfigCustom<T>;\n\nexport interface QdPageSelectedContext {\n  value: string;\n  label: {\n    i18n: string;\n  };\n}\n\ninterface QdPageContextConfigBase {\n  /**\n   * @description Description for the specific header facet\n   */\n  label: {\n    /**\n     * @description i18n string for the description, used for translation.\n     */\n    i18n: string;\n  };\n\n  /**\n   * @description Allows to identify the context\n   */\n  id: string;\n\n  /**\n   * @description Register some handler that is executed when the selection changes.\n   */\n  changed?: (selection: QdPageSelectedContext | QdPageSelectedContext[] | null) => void;\n}\n\n/**\n * @description Configuration for a context with single selection\n */\nexport interface QdPageContextConfigSingle extends QdPageContextConfigBase {\n  /**\n   * @description Configures the type on how the context selection happens.\n   */\n  type: 'single';\n\n  /**\n   * @description Available options for selection.\n   */\n  options: QdPageSelectedContext[];\n\n  /**\n   * @description Configures whether the context options are searchable for the user.\n   * @default false\n   */\n  hasSearch?: boolean;\n\n  /**\n   * @description Placeholder for search\n   */\n  searchPlaceholder?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n\n  /**\n   * @description Title for the dialog that will be opened\n   */\n  title?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n}\n\n/**\n * @description Configuration for a context with multi selection\n */\nexport interface QdPageContextConfigMulti extends QdPageContextConfigBase {\n  /**\n   * @description Configures the type on how the context selection happens.\n   */\n  type: 'multi';\n\n  /**\n   * @description Available options for selection.\n   */\n  options: QdPageSelectedContext[];\n\n  /**\n   * @description Configures whether the context options are searchable for the user.\n   * @default false\n   */\n  hasSearch?: boolean;\n\n  /**\n   * @description Placeholder for search\n\n   */\n  searchPlaceholder?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n\n  /**\n   * @description Title for the dialog that will be opened\n   */\n  title?: {\n    /**\n     * @description i18n string\n     */\n    i18n: string;\n  };\n}\n\n/**\n * @description Configuration for a context with custom selection (e. g. your own dialog)\n */\nexport interface QdPageContextConfigCustom extends QdPageContextConfigBase {\n  /**\n   * @description Configures the type on how the context selection happens.\n   */\n  type: 'custom';\n\n  /**\n   * @description Handler to open a custom dialog for selection\n   */\n  open: () => void;\n}\n\n/**\n * @description Configures the context with one of the available types\n */\nexport type QdPageContextConfig = QdPageContextConfigSingle | QdPageContextConfigMulti | QdPageContextConfigCustom;\n\n/**\n * @description Base configuration for QdPage.\n */\ninterface QdPageConfigBase {\n  /**\n   * @description Title of the page.\n   */\n  title: QdPageTitle;\n\n  /**\n   * @description Hides contexts from being shown. The ContextService will still work in the background.\n   *\n   * @default false\n   */\n  hideContexts?: boolean;\n\n  /**\n   * @description Label for the menu button of custom actions.\n   */\n  customActionsLabel?: QdPageCustomActionsLabel;\n}\n\n/**\n * @description Configuration for the overview page type.\n *\n * @template T - The type of the page object data\n */\nexport interface QdPageConfigOverview extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'overview'\n   */\n  pageType?: 'overview'; // Optional to make it default\n\n  /**\n   * @description Configuration specific to the overview page type.\n   */\n  pageTypeConfig?: QdPageTypeOverviewConfig;\n}\n\n/**\n * @description Configuration for the create page type.\n *\n * @template T - The type of the page object data\n */\nexport interface QdPageConfigCreate<T extends object = object> extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'create'\n   */\n  pageType: 'create';\n\n  /**\n   * @description Configuration specific to the create page type.\n   */\n  pageTypeConfig?: QdPageTypeCreateConfig;\n\n  /**\n   * @description Information placed in the header of the page. Concerns current shown object.\n   */\n  headerFacets?: QdPageHeaderFacetConfig<T>[];\n}\n\nexport interface QdPageConfigInspect<T extends object = object> extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'inspect'\n   */\n  pageType: 'inspect';\n\n  /**\n   * @description Configuration specific to the inspect page type.\n   */\n  pageTypeConfig?: QdPageTypeInspectConfig;\n\n  /**\n   * @description Information placed in the header of the page. Concerns current shown object.\n   */\n  headerFacets?: QdPageHeaderFacetConfig<T>[];\n\n  /**\n   * @description Determines whether the facets in header should be collapsed and custom actions are in a menu. This mode is designed for touch devices and mobile phones only.\n   *\n   * @default false\n   */\n  adaptiveMode?: boolean;\n}\n\nexport interface QdPageConfigCustom<T extends object = object> extends QdPageConfigBase {\n  /**\n   * @description Page Type\n   *\n   * @default 'custom'\n   */\n  pageType: 'custom';\n\n  /**\n   * @description Configuration specific to the custom page type.\n   */\n  pageTypeConfig?: QdPageTypeCustomConfig;\n\n  /**\n   * @description Information placed in the header of the page. Concerns current shown object.\n   */\n  headerFacets?: QdPageHeaderFacetConfig<T>[];\n\n  /**\n   * @description Determines whether the facets in header should be collapsed and custom actions are in a menu. This mode is designed for touch devices and mobile phones only.\n   *\n   * @default false\n   */\n  adaptiveMode?: boolean;\n}\n\n/**\n * Configures the overview page\n */\nexport interface QdPageTypeOverviewConfig {\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: QdMenuButtonActionConfig[];\n}\n\n/**\n * @description Configures the create page.\n */\nexport interface QdPageTypeCreateConfig {\n  /**\n   * @description Configuration for the save draft action.\n   */\n  saveDraft?: QdPageSaveDraftAction;\n\n  /**\n   * @description Configuration for the cancel button.\n   */\n  cancel: QdPageCancelAction;\n\n  /**\n   * @description Configuration for the submit button.\n   */\n  submit: QdPageCreateSubmitAction;\n\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: QdMenuButtonActionConfig[];\n}\n\n/**\n * @description Configures the inspect page.\n */\nexport interface QdPageTypeInspectConfig {\n  /**\n   * @description Operation mode for the page.\n   *\n   * * This property determines the mode in which the page operates. It can be either 'view' or 'edit'.\n   *\n   * * **view**: The page is in read-only mode.\n   * * **edit**: The page allows modifications.\n   */\n  operationMode?: QdInspectOperationMode;\n\n  /**\n   * @description Specifies whether the user can edit the object or not.\n   */\n  hideEdit?: boolean;\n\n  /**\n   * @description Configuration for the edit action.\n   */\n  edit?: QdPageEditAction;\n\n  /**\n   * @description Configuration for the save action.\n   */\n  save?: QdPageSaveAction;\n\n  /**\n   * @description Configuration for the cancel action.\n   */\n  cancel: QdPageCancelAction;\n\n  /**\n   * @description Configuration for the delete action.\n   */\n  delete?: QdPageDeleteAction;\n\n  /**\n   * @description Configuration for the archive action.\n   */\n  archive?: QdPageArchiveAction;\n\n  /**\n   * @description Configuration for the submit action.\n   */\n  submit?: QdPageSubmitAction;\n\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * In the \"Inspect\" page type, you can assign custom actions to the operation modes \"view\" or \"edit.\" By default, the action is active in both modes.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: (QdMenuButtonActionConfig & { operationMode?: QdInspectOperationMode })[];\n}\n\nexport interface QdPageTypeCustomConfig {\n  /**\n   * @description Custom actions for the header.\n   * You can define a label, a handler, and you can disable or hide the action.\n   *\n   * * If there exists more than 1 Custom Action, a QdMenuButton will be displayed.\n   *\n   * * These actions are not intended to influence the core process flow or manipulate the object directly. Instead, they should be used for side effects, such as triggering notifications, updating external services, or executing supplementary tasks that do not alter the main state of the object.\n   */\n  customActions?: QdMenuButtonActionConfig[];\n}\n\nexport interface QdPageTitle {\n  /**\n   * @description Defines the translation key\n   *\n   * * @example { i18n: \"i18n.overview.title\"}\n   */\n  i18n: string;\n}\n\n/**\n * @description Configuration for the custom actions label, used when multiple custom actions exist.\n */\nexport interface QdPageCustomActionsLabel {\n  /**\n   * @description Label for the menu button of custom actions, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   *\n   * * This label becomes visible if more than one custom action is defined, as the actions will be grouped under a QdMenuButton that displays this label.\n   */\n  i18n: string;\n}\n\n/**\n * @description Operation modes for the inspect page.\n *\n * @default 'view'\n */\nexport type QdInspectOperationMode = 'view' | 'edit';\n\n/**\n * @description Interface for the save action, including optional validation configuration.\n */\nexport interface QdPageSaveAction {\n  /**\n   * @description Label for the save action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the save action is executed.\n   */\n  handler: (formValues?: any) => void;\n\n  /**\n   * @description Optional validation flag, defaults to true.\n   * Determines whether form validation is required before saving and ensures that changes must be present in the form before saving.\n   *\n   * * By default, this flag is set to true, meaning that validation is enforced, and the form will only save if there are changes.\n   *\n   * * If set to `false`, the form will be saved without validation checks, but changes must still be present to proceed with the save operation.\n   *\n   * * This flag is useful in scenarios where form validation is not needed or should be bypassed, but it ensures that only modified forms can be saved.\n   */\n  hasValidation?: boolean;\n}\n\n/**\n * @description Interface for the safe draft action.\n */\nexport interface QdPageSaveDraftAction {\n  /**\n   * @description Label for the save draft action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the save draft action is executed.\n   */\n  handler: () => void;\n}\n\n/**\n * @description Interface for the cancel action, including an optional confirmation message.\n */\nexport interface QdPageCancelAction {\n  /**\n   * @description Label for the cancel action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the cancel action is executed.\n   */\n  handler: (formValues?: any) => void;\n\n  /**\n   * @description An optional confirmation message displayed in a dialog when the user attempts to cancel the operation, warning that all changes will be discarded.\n   * If no custom message is provided, a standard message will be shown by default.\n   *\n   * * The cancellation process checks whether there are unsaved changes in the form. If there are changes, the dialog will appear.\n   *\n   * * If no changes are detected, the cancellation proceeds without showing the confirmation dialog.\n   */\n  confirmationMessage?: { i18n: string };\n}\n\n/**\n * @description Interface for the submit action on create pages.\n */\nexport interface QdPageCreateSubmitAction {\n  /**\n   * @description Label for the submit action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the submit action is executed.\n   */\n  handler: (formValues?: any) => void;\n}\n\n/**\n * @description Interface for the submit action.\n */\nexport interface QdPageSubmitAction {\n  /**\n   * @description Label for the submit action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the submit action is executed.\n   */\n  handler: (formValues?: any) => void;\n\n  /**\n   * @description By default, the visibility of the submit button depends on the page type and the operation mode.\n   * To completely hide the submit button, you can set this isHidden flag.\n   *\n   * @default false\n   */\n  isHidden?: boolean;\n}\n\n/**\n * @description Interface for the edit action.\n */\nexport interface QdPageEditAction {\n  /**\n   * @description Label for the edit action, used for translation.\n   *\n   * * If no custom translation key is provided, a standard label will be used by default.\n   */\n  label?: { i18n: string };\n\n  /**\n   * @description Handler function that is triggered when the edit action is executed.\n   */\n  handler: (formValues?: any) => void;\n}\n\n/**\n * @description Interface for the delete action.\n *\n * * This action is represented as an icon, so it does not require a label.\n */\nexport interface QdPageDeleteAction {\n  /**\n   * @description Handler function that is triggered when the delete action is executed.\n   */\n  handler: (formValues?: any) => void;\n}\n\n/**\n * @description Interface for the archive action.\n *\n * * This action is represented as an icon, so it does not require a label.\n */\nexport interface QdPageArchiveAction {\n  /**\n   * @description Handler function that is triggered when the archive action is executed.\n   */\n  handler: (formValues?: any) => void;\n}\n"]}
|
|
@@ -2,10 +2,10 @@ import { Component, ContentChild, ContentChildren, EventEmitter, HostBinding, In
|
|
|
2
2
|
import { Subject } from 'rxjs';
|
|
3
3
|
import { takeUntil, tap } from 'rxjs/operators';
|
|
4
4
|
import { QdDialogService, QdDialogSize } from '../dialog/dialog.module';
|
|
5
|
-
import { QdPageStepperComponent } from '../page-stepper/page-stepper.component';
|
|
6
5
|
import { QdPageStepperAdapterDirective } from '../page-stepper/adaptors/page-stepper-adapter.directive';
|
|
7
|
-
import {
|
|
6
|
+
import { QdPageStepperComponent } from '../page-stepper/page-stepper.component';
|
|
8
7
|
import { QdPageTabsAdapterDirective } from '../page-tabs/adaptors/page-tabs-adapter.directive';
|
|
8
|
+
import { QdPageTabsComponent } from '../page-tabs/page-tabs.component';
|
|
9
9
|
import { QdSectionComponent } from '../section/section.component';
|
|
10
10
|
import { QdPageCancelConfirmationDialogComponent } from './cancel-confirmation-dialog/page-cancel-confirmation-dialog.component';
|
|
11
11
|
import { QdPageControlPanelComponent } from './control-panel/page-control-panel.component';
|
|
@@ -241,7 +241,9 @@ import * as i9 from "./object-header/page-object-header.component";
|
|
|
241
241
|
*
|
|
242
242
|
* #### **Create Page**
|
|
243
243
|
*
|
|
244
|
-
* On the create page, a new business case can be created. This page typically includes forms for entering data. The submit button is parameterized with the form data and is validated through the QdPageStepper.
|
|
244
|
+
* On the create page, a new business case can be created. This page typically includes forms for entering data. The submit button is parameterized with the form data and is validated through the QdPageStepper.
|
|
245
|
+
* The click on the submit button should submit the creation of the business case or object. You can define a specific handler in the submit button's configuration to create the object.
|
|
246
|
+
* If only one step is needed, the single-step mode can be used. In single-step mode, the stepper is visually hidden and behaves like a **QdSection**.
|
|
245
247
|
*
|
|
246
248
|
* ```ts
|
|
247
249
|
* const createPageConfig: QdPageConfig<MyObjectType> = {
|
|
@@ -316,7 +318,7 @@ import * as i9 from "./object-header/page-object-header.component";
|
|
|
316
318
|
*
|
|
317
319
|
* #### **Inspect Page**
|
|
318
320
|
*
|
|
319
|
-
* On an inspect page, a specific business case is displayed and may be edited or submitted.
|
|
321
|
+
* On an inspect page, a specific business case is displayed and may be edited or submitted. Depending on your needs you can hide the submit button.
|
|
320
322
|
* In contrast to a create page, an existing object is shown and potentially updated.
|
|
321
323
|
*
|
|
322
324
|
* The page operates in one of two modes - `view` or `edit` - as defined by the `operationMode` configuration key.
|
|
@@ -629,4 +631,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
|
|
|
629
631
|
type: HostBinding,
|
|
630
632
|
args: ['class.has-info-banners']
|
|
631
633
|
}] } });
|
|
632
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/page/page.component.ts","../../../../../libs/qd-ui/src/lib/page/page.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEL,SAAS,EACT,YAAY,EACZ,eAAe,EACf,YAAY,EACZ,WAAW,EACX,KAAK,EAIL,MAAM,EACN,SAAS,EAEV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAc,OAAO,EAAE,MAAM,MAAM,CAAC;AAC3C,OAAO,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAEhD,OAAO,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,yBAAyB,CAAC;AACxE,OAAO,EAAE,sBAAsB,EAAE,MAAM,wCAAwC,CAAC;AAChF,OAAO,EAAE,6BAA6B,EAAE,MAAM,yDAAyD,CAAC;AACxG,OAAO,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACvE,OAAO,EAAE,0BAA0B,EAAE,MAAM,mDAAmD,CAAC;AAC/F,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAE,uCAAuC,EAAE,MAAM,wEAAwE,CAAC;AACjI,OAAO,EAAE,2BAA2B,EAAE,MAAM,8CAA8C,CAAC;AAC3F,OAAO,EAAE,yBAAyB,EAAE,MAAM,0CAA0C,CAAC;AAOrF,OAAO,EAA2B,kBAAkB,EAAsB,MAAM,sCAAsC,CAAC;AACvH,OAAO,EAAE,yBAAyB,EAAE,MAAM,8CAA8C,CAAC;AACzF,OAAO,EAAE,mBAAmB,EAAE,MAAM,uCAAuC,CAAC;AAC5E,OAAO,EAAE,kBAAkB,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,EAAE,yBAAyB,EAAE,MAAM,8CAA8C,CAAC;AACzF,OAAO,EAAE,wBAAwB,EAAE,MAAM,4CAA4C,CAAC;;;;;;;;;;;AAItF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuXG;AAQH,MAAM,OAAO,eAAe;IAwEP;IACA;IACA;IACA;IACA;IA3EnB;;;OAGG;IAEH,MAAM,CAAmB;IAEzB;;;OAGG;IACoB,MAAM,GAAI,MAAM,CAAC;IAExC;;OAEG;IAEH,oBAAoB,GAAG,IAAI,YAAY,EAA0B,CAAC;IAGlE,QAAQ,CAAiC;IAGzC,YAAY,CAA+B;IAG3C,gBAAgB,CAA0B;IAG1C,aAAa,CAAuB;IAGpC,uBAAuB,CAAiC;IAGxD,oBAAoB,CAA8B;IAGlD,WAAW,CAAwC;IAEnD,iBAAiB,CAAsB;IACvC,eAAe,CAAsB;IACrC,sBAAsB,GACpB,qHAAqH,CAAC;IAEhH,cAAc,GAAG,KAAK,CAAC;IACvB,WAAW,GAAG,IAAI,OAAO,EAAQ,CAAC;IAE1C,IAAI,qBAAqB;QACvB,OAAO,IAAI,CAAC,YAAY,KAAK,SAAS,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,QAAQ,KAAK,IAAI,CAAC;IACxF,CAAC;IAED,IAAI,mBAAmB;QACrB,OAAO,IAAI,CAAC,YAAY,EAAE,MAAM,EAAE,KAAK,KAAK,OAAO,CAAC;IACtD,CAAC;IAED,IAAI,aAAa;QACf,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,uBAAuB,IAAI,IAAI,CAAC,oBAAoB,CAAC,CAAC;IACtH,CAAC;IAED,IACI,aAAa;QACf,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED,IACI,cAAc;QAChB,OAAO,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC;IACrC,CAAC;IAED,YACmB,gBAAuC,EACvC,aAAkC,EAClC,uBAAkD,EAClD,mBAA8C,EAC9C,MAAuB;QAJvB,qBAAgB,GAAhB,gBAAgB,CAAuB;QACvC,kBAAa,GAAb,aAAa,CAAqB;QAClC,4BAAuB,GAAvB,uBAAuB,CAA2B;QAClD,wBAAmB,GAAnB,mBAAmB,CAA2B;QAC9C,WAAM,GAAN,MAAM,CAAiB;QAExC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,aAAa,CAAC,iBAAiB,CAAC;QAC9D,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,eAAe,GAAG,SAAS,CAAC,CAAC,CAAC;IACtH,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAE3B,IAAI,IAAI,CAAC,MAAM;YAAE,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IACpE,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;QAEvC,IAAI,CAAC,YAAY;YAAE,OAAO;QAE1B,MAAM,EAAE,YAAY,EAAE,IAAI,EAAE,aAAa,EAAE,IAAI,EAAE,GAAG,YAAY,CAAC;QAEjE,IAAI,IAAI,EAAE,cAAc,EAAE,MAAM,EAAE,QAAQ,KAAK,IAAI,EAAE,cAAc,EAAE,MAAM,EAAE,QAAQ;YACnF,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC;gBAC/B,EAAE,SAAS,EAAE,QAAQ,EAAE,aAAa,EAAE,EAAE,SAAS,EAAE,CAAC,IAAI,EAAE,cAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,EAAE;aAC/F,CAAC,CAAC;IACP,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,KAAK,QAAQ;YAAE,IAAI,CAAC,4BAA4B,EAAE,CAAC;QAE3E,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,EAAE,cAAc,EAAE,MAAM,KAAK,SAAS;YACxF,IAAI,CAAC,iCAAiC,EAAE,CAAC;QAE3C,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,EAAE,cAAc,EAAE,SAAS,KAAK,SAAS;YAC3F,IAAI,CAAC,yBAAyB,EAAE,CAAC;QAEnC,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,KAAK,SAAS;YACpC,IAAI,CAAC,gBAAgB,CAAC,WAAW;iBAC9B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;iBACjC,SAAS,CAAC,UAAU,CAAC,EAAE,CACtB,IAAI,CAAC,8BAA8B,CACjC,IAAI,CAAC,MAAM,CAAC,cAAyC,EACrD,UAAU,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAC7B,CACF,CAAC;QAEN,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;IAC7B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC;QACxB,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC;IAC9B,CAAC;IAEO,mBAAmB;QACzB,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,OAAO,CAAC,IAAI,CAAC,mFAAmF,CAAC,CAAC;IACtH,CAAC;IAEO,4BAA4B;QAClC,MAAM,cAAc,GAAG,IAAI,CAAC,MAAM,EAAE,cAAwC,CAAC;QAC7E,MAAM,UAAU,GAA8B,CAAC,WAAW,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC;QAChF,MAAM,OAAO,GAA0B,EAAE,CAAC;QAE1C,KAAK,MAAM,SAAS,IAAI,UAAU,EAAE,CAAC;YACnC,MAAM,MAAM,GAAG,cAAc,CAAC,SAAS,CAAC,CAAC;YAEzC,IAAI,CAAC,MAAM;gBAAE,SAAS;YAEtB,OAAO,CAAC,IAAI,CAAC;gBACX,SAAS;gBACT,aAAa,EAAE;oBACb,GAAG,CAAC,MAAM,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,SAAS,EAAE,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;oBAChE,OAAO,EAAE,IAAI,CAAC,2BAA2B,CAAC,MAAM,EAAE,OAAO,CAAC;iBAC3D;aACF,CAAC,CAAC;QACL,CAAC;QAED,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;IAC5C,CAAC;IAEO,iCAAiC;QACvC,IAAI,CAAC,uBAAuB;aACzB,iBAAiB,EAAE;aACnB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;aACjC,SAAS,CAAC,UAAU,CAAC,EAAE;YACtB,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC;gBAC/B;oBACE,SAAS,EAAE,QAAQ;oBACnB,aAAa,EAAE;wBACb,OAAO,EAAE,UAAU;4BACjB,CAAC,CAAC,GAAS,EAAE,CAAC,IAAI,CAAC,2BAA2B,EAAE;4BAChD,CAAC,CAAC,GAAS,EAAE,CAAE,IAAI,CAAC,MAAM,EAAE,cAAyC,EAAE,MAAM,EAAE,OAAO,EAAE;qBAC3F;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,yBAAyB;QAC/B,MAAM,cAAc,GAAG,IAAI,CAAC,MAAM,CAAC,cAAwC,CAAC;QAE5E,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC;YAC5B;gBACE,GAAG,EAAE,WAAW;gBAChB,MAAM,EAAE;oBACN,SAAS,EAAE,+BAA+B;oBAC1C,IAAI,EAAE,kBAAkB,CAAC,SAAS;oBAClC,OAAO,EAAE,GAAS,EAAE,CAAC,cAAc,EAAE,SAAS,EAAE,OAAO,EAAE;oBACzD,SAAS,EAAE,IAAI;oBACf,UAAU,EAAE,KAAK;iBAClB;aACF;SACF,CAAC,CAAC;IACL,CAAC;IAEO,8BAA8B,CAAC,cAAuC,EAAE,IAA4B;QAC1G,IAAI,CAAC,mBAAmB,CAAC,yBAAyB,CAAC,cAAc,EAAE,IAAI,KAAK,MAAM,CAAC,CAAC;QAEpF,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC;YAC/B,EAAE,SAAS,EAAE,QAAQ,EAAE,aAAa,EAAE,EAAE,SAAS,EAAE,IAAI,KAAK,MAAM,IAAI,CAAC,cAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,EAAE;SAC5G,CAAC,CAAC;QAEH,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAE5B,IAAI,IAAI,CAAC,cAAc;YAAE,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAChE,CAAC;IAEO,2BAA2B,CAAC,OAAmC;QACrE,OAAO,CAAC,GAAG,IAAI,EAAE,EAAE;YACjB,IAAI,CAAC,OAAO;gBAAE,OAAO;YAErB,OAAO,CAAC,IAAI,CAAC,uBAAuB,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,uBAAuB,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC7G,CAAC,CAAC;IACJ,CAAC;IAEO,2BAA2B;QACjC,IAAI,CAAC,MAAM;aACR,IAAI,CAAC,uCAAuC,EAAE;YAC7C,KAAK,EAAE,EAAE,IAAI,EAAE,+CAA+C,EAAE;YAChE,UAAU,EAAE,YAAY,CAAC,KAAK;YAC9B,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,cAAc;SACjC,CAAC;aACD,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;aACxC,SAAS,EAAE,CAAC;IACjB,CAAC;IAEO,oBAAoB;QAC1B,IAAI,CAAC,uBAAuB;aACzB,mBAAmB,EAAE;aACrB,IAAI,CACH,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,EAC3B,GAAG,CAAC,OAAO,CAAC,EAAE;YACZ,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC;gBAC/B;oBACE,SAAS,EAAE,QAAQ;oBACnB,aAAa,EAAE;wBACb,cAAc,EAAE,OAAO;4BACrB,CAAC,CAAC,SAAS;4BACX,CAAC,CAAC,EAAE,QAAQ,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE,IAAI,EAAE,iCAAiC,EAAE,EAAE;qBAClF;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CACH;aACA,SAAS,EAAE,CAAC;IACjB,CAAC;uGA/OU,eAAe;2FAAf,eAAe,4XAHf,CAAC,mBAAmB,EAAE,yBAAyB,EAAE,yBAAyB,EAAE,wBAAwB,CAAC,oEA0BlG,2BAA2B,mFAG3B,sBAAsB,gFAGtB,mBAAmB,0FAGnB,6BAA6B,uFAG7B,0BAA0B,8DAfvB,kBAAkB,8CAkBlB,yBAAyB,kDC/c5C,irDA8CA;;2FD2Xa,eAAe;kBAP3B,SAAS;+BACE,SAAS,aAGR,CAAC,mBAAmB,EAAE,yBAAyB,EAAE,yBAAyB,EAAE,wBAAwB,CAAC,QAC1G,EAAE,2BAA2B,EAAE,uBAAuB,EAAE,6BAA6B,EAAE,qBAAqB,EAAE;uOAQpH,MAAM;sBADL,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAOF,MAAM;sBAA5B,KAAK;uBAAC,cAAc;gBAMrB,oBAAoB;sBADnB,MAAM;gBAIP,QAAQ;sBADP,eAAe;uBAAC,kBAAkB;gBAInC,YAAY;sBADX,YAAY;uBAAC,2BAA2B;gBAIzC,gBAAgB;sBADf,YAAY;uBAAC,sBAAsB,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE;gBAI3D,aAAa;sBADZ,YAAY;uBAAC,mBAAmB,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE;gBAIxD,uBAAuB;sBADtB,YAAY;uBAAC,6BAA6B,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE;gBAIlE,oBAAoB;sBADnB,YAAY;uBAAC,0BAA0B,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE;gBAI/D,WAAW;sBADV,eAAe;uBAAC,yBAAyB;gBAwBtC,aAAa;sBADhB,WAAW;uBAAC,kBAAkB;gBAM3B,cAAc;sBADjB,WAAW;uBAAC,wBAAwB","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ContentChild,\n  ContentChildren,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  QueryList,\n  SimpleChanges\n} from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\nimport { takeUntil, tap } from 'rxjs/operators';\n\nimport { QdDialogService, QdDialogSize } from '../dialog/dialog.module';\nimport { QdPageStepperComponent } from '../page-stepper/page-stepper.component';\nimport { QdPageStepperAdapterDirective } from '../page-stepper/adaptors/page-stepper-adapter.directive';\nimport { QdPageTabsComponent } from '../page-tabs/page-tabs.component';\nimport { QdPageTabsAdapterDirective } from '../page-tabs/adaptors/page-tabs-adapter.directive';\nimport { QdSectionComponent } from '../section/section.component';\nimport { QdPageCancelConfirmationDialogComponent } from './cancel-confirmation-dialog/page-cancel-confirmation-dialog.component';\nimport { QdPageControlPanelComponent } from './control-panel/page-control-panel.component';\nimport { QdPageInfoBannerComponent } from './info-banner/page-info-banner.component';\nimport {\n  QdInspectOperationMode,\n  QdPageConfig,\n  QdPageTypeCreateConfig,\n  QdPageTypeInspectConfig\n} from './model/page-config.interface';\nimport { QdCreateFooterActionKey, QdFooterActionType, QdPageFooterAction } from './model/page-footer-action.interface';\nimport { QdFormGroupManagerService } from './shared/services/form-group-manager.service';\nimport { QdPageFooterService } from './shared/services/page-footer.service';\nimport { QdPageStoreService } from './shared/services/page-store.service';\nimport { QdPageSubmitActionService } from './shared/services/page-submit-action.service';\nimport { QdResolverTriggerService } from './shared/services/resolver-trigger.service';\n\ntype PartialFooterAction = { actionKey: QdCreateFooterActionKey; partialAction: Partial<QdPageFooterAction> };\n\n/**\n * **QdPage** serves as the central structural element within the hierarchy under the **QdShell**.\n * Integration is done through the Angular Routing Module, and QdPage is designed to be responsive across various screen sizes right out of the box.\n *\n * #### **Page Types**\n *\n * QdPage uses page types to decide how the page looks and works. Each page type has its own settings and actions.\n *\n * - **overview**: Displays a summary or a list of objects, providing an overview of multiple business cases.\n * - **create**: Allows the creation of a new object, representing a business case.\n * - **inspect**: Enables viewing, editing, and executing additional actions related to the object, which also represents a business case.\n * - **custom**: Provides a way to implement unique functionality that doesn’t fit the other predefined page types.\n *\n * **Note that the 'custom' page type should be used only as a fallback for very specific cases. If you encounter such a situation, please contact the Quadrel team.**\n *\n * #### **Facets**\n *\n * These are extra pieces of information shown in the page header. They give context specific to the current object model being viewed or edited. There are several types of facets, including `status`, `value`, `criticality`, `references`, `progress`, `date`, `singleSelect`, and `multiSelect`. Each type has its own options. For more details, see the `QdPageHeaderFacet` interface.\n *\n * #### **Context**\n *\n * The `context` acts as a filter that can be selected on the overview page and provides the business context, which remains consistent throughout the entire application. On create and inspect pages, the context is displayed as read-only. For more details, check the dedicated chapter in Storybook.\n *\n *\n * #### **Page Info Banners**\n *\n * To insert a page info banner under the page header, just put a `qd-page-info-banner` component inside the `qd-page`. Multiple consecutive info banners of the same type will be merged into one banner container. Page info banners have the same appearance like notifications.\n *\n * #### **Control Panel**\n *\n * The control panel is located on the right side of the page and is used to display additional information and functions relevant to a specific business case, which is linked to a specific object model. For further information, please consult the dedicated chapter in Storybook.\n *\n * #### **Actions**\n *\n * These are interactive elements that perform specific functions. Functions for these actions can be set up through configurations. Custom actions can be added to any page type. Additionally, all actions can be configured and customized with your own text, which will override the default texts.\n *\n * Please check the relevant interfaces for each page type: `QdPageConfigOverview`, `QdPageConfigCreate`, `QdPageConfigInspect`, and `QdPageConfigCustom`.\n *\n * #### **Validation/Parameterization**\n *\n * Validation and parameterization are covered in a dedicated chapter in the Storybook. Please check the \"Validation\" section for more information.\n *\n * #### **Resolver**\n *\n * Using a resolver to fetch page object data is essential to the page concept. Add the resolver to the component's providers and use the `QD_PAGE_OBJECT_RESOLVER_TOKEN` token. This approach enables type safety and user feedback by effectively managing the necessary data for the page.\n *\n * ```ts\n * interface MyObjectModel {\n *   // Metadata displayed in the ObjectHeader\n *   state: MyState;\n *   countryOfDestination: MyCountryOfDestination;\n *   // Appears not in the ObjectHeader\n *   payload: MyPayload;\n * }\n *\n * type MyObjectMetadata = 'state' | 'countryOfDestination';\n *\n * @Injectable()\n * class MyObjectModelResolver implements QdPageObjectResolver<MyObjectModel> {\n *   constructor(private http: HttpClient) {}\n *\n *   resolve(): Observable<MyObjectModel> {\n *     return this.http\n *       .get<MyObjectModel>('my-api')\n *       .pipe(tap((data: MyObjectModel) => doSomethingWith(data)));\n *   }\n * }\n *\n * @Component({\n *   selector: 'my-page',\n *   templateUrl: './my-page.component.html',\n *   providers: [\n *     {\n *       provide: QD_PAGE_OBJECT_RESOLVER_TOKEN,\n *       useClass: MyObjectModelResolver\n *     }\n *   ]\n * })\n * class MyPageComponent {\n *   pageConfig: QdPageConfig<Pick<MyObjectModel, MyObjectMetadata>> = {\n *     title: { i18n: 'i18n.page.title' },\n *     pageType: 'inspect',\n *     headerFacets: [\n *       {\n *         name: 'status',\n *         label: {\n *           i18n: 'i18n.status.label'\n *         },\n *         options: {\n *           type: 'status'\n *         }\n *       },\n *       {\n *         name: 'countryOfDestination',\n *         label: {\n *           i18n: 'i18n.countryOfDestination.label'\n *         },\n *         options: {\n *           type: 'value'\n *         }\n *       }\n *     ],\n *     pageTypeConfig: {\n *       cancel: {\n *         handler: () => handleCancel()\n *       },\n *       save: {\n *         handler: (formValues) => handleSave(formValues)\n *       }\n *     }\n *   };\n * }\n * ```\n *\n * #### **Updating Facets**\n *\n * Typically, the values of the facets on a create or inspect page are set to read-only. However, there may be cases where, for example, a status change is needed, such as through a dialog. For this purpose, we have provided an update method. In the resolver service, define an empty method called `updateMetadata`. Inject this service using its type, and then call the `updateMetadata` method with the required parameter.\n *\n * **Please note: These values should not be modified directly within a QdPage.**\n *\n * ```ts\n * @Injectable()\n * class MyObjectModelResolver implements QdPageObjectResolver<MyObjectModel> {\n *   config: QdPageObjectResolverConfig = {\n *     // your configuration options here\n *   }\n *\n *   constructor(private http: HttpClient) {}\n *\n *   resolve(): Observable<MyObjectModel> {\n *     // your implementation here\n *   }\n *\n *   updateMetadata(metadata: Partial<MyObjectModel>) {\n *     // leave empty here & call in your component\n *   }\n * }\n *\n * @Component({\n *   // ...\n *   providers: [\n *     {\n *       provide: QD_PAGE_OBJECT_RESOLVER_TOKEN,\n *       useClass: MyObjectModelResolver\n *     }\n *   ]\n * })\n * class MyPageComponent {\n *   constructor(@Inject(QD_PAGE_OBJECT_RESOLVER_TOKEN) private objectResolver: MyObjectModelResolver) {}\n *\n *   updateStatus() {\n *     this.objectResolver.updateMetadata({ state: 'Updated' });\n *   }\n * }\n * ```\n *\n * #### **Overview Page**\n *\n * On an overview page, a list of business cases can be displayed. Typically, this is done in a **QdTable**, including filters and/or search options. There are actions that handle routing to the related detail pages, where the business cases, represented by a business object model, can be viewed, edited, or newly created.\n *\n * ```ts\n * const overviewPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.my-page.title' },\n *   pageType: 'overview',\n *   pageTypeConfig: {\n *     customActions: [\n *       {\n *         i18n: 'i18n.custom.action',\n *         handler: () => doSomething()\n *       }\n *     ]\n *   }\n * };\n * ```\n *\n * ```html\n * <qd-page [config]=\"pageConfig\">\n *   <qd-section [config]=\"sectionConfig\">\n *     <qd-table [config]=\"tableConfig\"></qd-table>\n *   </qd-section>\n * </qd-page>\n * ```\n *\n * To configure contexts for a page, you can easily set contexts in the configuration for the overview page. They are then selectable on the overview page:\n *\n * ```ts\n * const overviewPageConfig: QdPageConfig<MyObjectType> = {\n *   // ...\n *   contexts: [\n *     {\n *       id: 'typeOfTraffic',\n *       label: { i18n: 'Verkehrsart' },\n *       type: 'single',\n *       selected: {\n *         label: { i18n: 'Schiene' },\n *         value: 'rail'\n *       },\n *       options: [\n *         {\n *           value: 'rail',\n *           label: { i18n: 'Schiene' }\n *         },\n *         {\n *           value: 'air',\n *           label: { i18n: 'Luft' }\n *         }\n *       ]\n *     }\n *   ]\n *   // ...\n * };\n * ```\n *\n * #### **Create Page**\n *\n * On the create page, a new business case can be created. This page typically includes forms for entering data. The submit button is parameterized with the form data and is validated through the QdPageStepper. If only one step is needed, the single-step mode can be used. In single-step mode, the stepper is visually hidden and behaves like a **QdSection**.\n *\n * ```ts\n * const createPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.page.title' },\n *   pageType: 'create',\n *   pageTypeConfig: {\n *     submit: {\n *       handler: (formValues) => handleSubmit(formValues)\n *     }\n *   }\n * };\n * ```\n *\n * ```html\n * <!-- For \"Quick Add\" use PageStepper in single-step mode -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-page-stepper [config]=\"stepperConfig\">\n *     <qd-page-step [config]=\"stepConfigOne\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myForms\" qdConnectFormStateToPage=\"myForms\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-step>\n *   </qd-page-stepper>\n * </qd-page>\n *\n * <!-- With PageStepper using multiple steps -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-page-stepper [config]=\"stepperConfig\">\n *     <qd-page-step [config]=\"stepConfigOne\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsOne\" qdConnectFormStateToPage=\"myFormsOne\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-step>\n *     <qd-page-step [config]=\"stepConfigTwo\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsTwo\" qdConnectFormStateToPage=\"myFormsTwo\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-step>\n *   </qd-page-stepper>\n * </qd-page>\n * ```\n *\n * To allow resolving asynchronous operations when going to next step, you can provide a resolver of type QdPageStepResolver:\n *\n * ```ts\n * class MyResolver implements QdPageStepResolver {\n *   ...\n *\n *   resolve(index: number): Observable<void> {\n *     return this.http.post(\"next-step.com\");\n *   }\n * }\n *\n * @Component({\n *   selector: 'qd-ui-page-create',\n *   templateUrl: './create.component.html',\n *   providers: [\n *     {\n *       provide: QD_PAGE_STEP_RESOLVER_TOKEN,\n *       useClass: PageStepResolver\n *     }\n *   ]\n * })\n * export class CreateComponent {\n * ```\n *\n * #### **Inspect Page**\n *\n * On an inspect page, a specific business case is displayed and may be edited or submitted.\n * In contrast to a create page, an existing object is shown and potentially updated.\n *\n * The page operates in one of two modes - `view` or `edit` - as defined by the `operationMode` configuration key.\n *\n * The `viewonly` behavior defined by the operation mode is automatically propagated to all nested QdSection components that are part of a QLS,\n * as well as to functional patterns such as QdTable, QdTree, QdFileCollector, and QdQuickEdit.\n * All forms are rendered either in viewonly or editable mode accordingly.\n *\n * This behavior can be explicitly overridden in each component’s configuration - for example, to allow certain fields to remain editable.\n *\n * - **view**: A submit action can optionally be added for process-related operations such as status updates.\n *   The action receives the current form values and supports validation and parameterization.\n *\n * - **edit**: Typically includes a save button to persist changes made in editable fields.\n *\n * ```ts\n * const inspectPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.page.title' },\n *   pageType: 'inspect',\n *   pageTypeConfig: {\n *     operationMode: 'edit',\n *     cancel: {\n *       handler: () => handleCancel()\n *     },\n *     save: {\n *       handler: (formValues) => handleSave(formValues)\n *     }\n *   }\n * };\n * ```\n *\n * ```html\n * <!-- with Section -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-section [config]=\"sectionConfig\">\n *     <qd-grid [config]=\"gridConfig\" [formGroup]=\"myForms\" qdConnectFormStateToPage=\"myForms\">\n *       <!-- your qd-form-components here -->\n *     </qd-grid>\n *   </qd-section>\n * </qd-page>\n *\n * <!-- with PageTabs -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-page-tabs [config]=\"stepperConfig\">\n *     <qd-page-tab [config]=\"tabConfigOne\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsOne\" qdConnectFormStateToPage=\"myFormsOne\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-tab>\n *     <qd-page-tab [config]=\"tabConfigTwo\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsTwo\" qdConnectFormStateToPage=\"myFormsTwo\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-tab>\n *   </qd-page-tabs>\n * </qd-page>\n * ```\n *\n * #### **Custom Page**\n *\n * ```ts\n * const customPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.page.title' },\n *   pageType: 'custom',\n *   pageTypeConfig: {\n *     customActions: [\n *       {\n *         i18n: 'i18n.custom.action',\n *         handler: () => doSomething()\n *       }\n *     ]\n *   }\n * };\n * ```\n *\n * ```html\n * <qd-page [config]=\"pageConfig\">\n *   <!-- Place content here that doesn’t fit the other page types -->\n * </qd-page>\n * ```\n */\n@Component({\n  selector: 'qd-page',\n  templateUrl: './page.component.html',\n  styleUrls: ['./page.component.scss'],\n  providers: [QdPageFooterService, QdFormGroupManagerService, QdPageSubmitActionService, QdResolverTriggerService],\n  host: { '[class.has-control-panel]': 'isControlPanelVisible', '[class.control-panel-broad]': 'isControlPanelBroad' }\n})\nexport class QdPageComponent<T extends object> implements OnInit, OnChanges, AfterViewInit, OnDestroy {\n  /**\n   * This property defines the configuration for the QdPage component, including the page type,\n   * title, and specific configurations for each type of page.\n   */\n  @Input({ required: true })\n  config!: QdPageConfig<T>;\n\n  /**\n   * A static test ID for integration tests can be set. <br />\n   * The value for the HTML attribute [data-test-id].\n   */\n  @Input('data-test-id') testId? = 'page';\n\n  /**\n   * This event is emitted when the operation mode of the inspect page is changed.\n   */\n  @Output()\n  operationModeChanged = new EventEmitter<QdInspectOperationMode>();\n\n  @ContentChildren(QdSectionComponent)\n  sections!: QueryList<QdSectionComponent>;\n\n  @ContentChild(QdPageControlPanelComponent)\n  controlPanel!: QdPageControlPanelComponent;\n\n  @ContentChild(QdPageStepperComponent, { descendants: true })\n  stepperComponent?: QdPageStepperComponent;\n\n  @ContentChild(QdPageTabsComponent, { descendants: true })\n  tabsComponent?: QdPageTabsComponent;\n\n  @ContentChild(QdPageStepperAdapterDirective, { descendants: true })\n  stepperAdapterDirective?: QdPageStepperAdapterDirective;\n\n  @ContentChild(QdPageTabsAdapterDirective, { descendants: true })\n  tabsAdapterDirective?: QdPageTabsAdapterDirective;\n\n  @ContentChildren(QdPageInfoBannerComponent)\n  infoBanners!: QueryList<QdPageInfoBannerComponent>;\n\n  footerHasContent$: Observable<boolean>;\n  isFooterVisible: boolean | undefined;\n  projectionGuardMessage =\n    'QD-UI | QdPage - This content is not supported. Please use QdSectionAdapterDirective for custom content projection.';\n\n  private _isInitialized = false;\n  private _destroyed$ = new Subject<void>();\n\n  get isControlPanelVisible(): boolean {\n    return this.controlPanel !== undefined && this.controlPanel.config?.isHidden !== true;\n  }\n\n  get isControlPanelBroad(): boolean {\n    return this.controlPanel?.config?.width === 'broad';\n  }\n\n  get hasNavigation(): boolean {\n    return !!(this.stepperComponent || this.tabsComponent || this.stepperAdapterDirective || this.tabsAdapterDirective);\n  }\n\n  @HostBinding('class.has-footer')\n  get footerVisible(): boolean | undefined {\n    return this.isFooterVisible;\n  }\n\n  @HostBinding('class.has-info-banners')\n  get hasInfoBanners(): boolean {\n    return this.infoBanners.length > 0;\n  }\n\n  constructor(\n    private readonly pageStoreService: QdPageStoreService<T>,\n    private readonly footerService: QdPageFooterService,\n    private readonly formGroupManagerService: QdFormGroupManagerService,\n    private readonly submitActionService: QdPageSubmitActionService,\n    private readonly dialog: QdDialogService\n  ) {\n    this.footerHasContent$ = this.footerService.footerHasContent$;\n    this.footerHasContent$.pipe(takeUntil(this._destroyed$)).subscribe(isVisible => (this.isFooterVisible = isVisible));\n  }\n\n  ngOnInit(): void {\n    this.checkConfigValidity();\n\n    if (this.config) this.pageStoreService.initPageState(this.config);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const configChange = changes['config'];\n\n    if (!configChange) return;\n\n    const { currentValue: curr, previousValue: prev } = configChange;\n\n    if (curr?.pageTypeConfig?.submit?.isHidden !== prev?.pageTypeConfig?.submit?.isHidden)\n      this.footerService.updateActions([\n        { actionKey: 'submit', partialAction: { isVisible: !curr?.pageTypeConfig?.submit?.isHidden } }\n      ]);\n  }\n\n  ngAfterViewInit(): void {\n    if (this.config.pageType === 'create') this.setupCreatePageFooterActions();\n\n    if (this.config.pageType === 'create' && this.config?.pageTypeConfig?.cancel !== undefined)\n      this.handleCancelActionWithFormChanges();\n\n    if (this.config.pageType === 'create' && this.config?.pageTypeConfig?.saveDraft !== undefined)\n      this.initSaveDraftFooterAction();\n\n    if (this.config.pageType === 'inspect')\n      this.pageStoreService.isViewonly$\n        .pipe(takeUntil(this._destroyed$))\n        .subscribe(isViewonly =>\n          this.updateInspectPageOperationMode(\n            this.config.pageTypeConfig as QdPageTypeInspectConfig,\n            isViewonly ? 'view' : 'edit'\n          )\n        );\n\n    this._isInitialized = true;\n  }\n\n  ngOnDestroy(): void {\n    this._destroyed$.next();\n    this._destroyed$.complete();\n  }\n\n  private checkConfigValidity(): void {\n    if (!this.config) console.warn('QdUi | QdPageComponent - To configure the page you should provide a valid config.');\n  }\n\n  private setupCreatePageFooterActions(): void {\n    const pageTypeConfig = this.config?.pageTypeConfig as QdPageTypeCreateConfig;\n    const actionKeys: QdCreateFooterActionKey[] = ['saveDraft', 'cancel', 'submit'];\n    const actions: PartialFooterAction[] = [];\n\n    for (const actionKey of actionKeys) {\n      const action = pageTypeConfig[actionKey];\n\n      if (!action) continue;\n\n      actions.push({\n        actionKey,\n        partialAction: {\n          ...(action?.label?.i18n ? { titleI18n: action.label.i18n } : {}),\n          handler: this.generateFooterActionHandler(action?.handler)\n        }\n      });\n    }\n\n    this.footerService.updateActions(actions);\n  }\n\n  private handleCancelActionWithFormChanges(): void {\n    this.formGroupManagerService\n      .$hasValuesChanged()\n      .pipe(takeUntil(this._destroyed$))\n      .subscribe(hasChanged => {\n        this.footerService.updateActions([\n          {\n            actionKey: 'cancel',\n            partialAction: {\n              handler: hasChanged\n                ? (): void => this.setupSubmitActionValidation()\n                : (): void => (this.config?.pageTypeConfig as QdPageTypeCreateConfig)?.cancel?.handler()\n            }\n          }\n        ]);\n      });\n  }\n\n  private initSaveDraftFooterAction(): void {\n    const pageTypeConfig = this.config.pageTypeConfig as QdPageTypeCreateConfig;\n\n    this.footerService.setActions([\n      {\n        key: 'saveDraft',\n        action: {\n          titleI18n: 'i18n.qd.page.footer.saveDraft',\n          type: QdFooterActionType.Secondary,\n          handler: (): void => pageTypeConfig?.saveDraft?.handler(),\n          isVisible: true,\n          isDisabled: false\n        }\n      }\n    ]);\n  }\n\n  private updateInspectPageOperationMode(pageTypeConfig: QdPageTypeInspectConfig, mode: QdInspectOperationMode): void {\n    this.submitActionService.setSubmitActionForInspect(pageTypeConfig, mode === 'view');\n\n    this.footerService.updateActions([\n      { actionKey: 'submit', partialAction: { isVisible: mode === 'view' && !pageTypeConfig?.submit?.isHidden } }\n    ]);\n\n    this.initSubmitValidation();\n\n    if (this._isInitialized) this.operationModeChanged.emit(mode);\n  }\n\n  private generateFooterActionHandler(handler?: (values: unknown) => void): (...args: unknown[]) => void {\n    return (...args) => {\n      if (!handler) return;\n\n      handler(this.formGroupManagerService.hasFormGroups() ? this.formGroupManagerService.getAllValues() : args);\n    };\n  }\n\n  private setupSubmitActionValidation(): void {\n    this.dialog\n      .open(QdPageCancelConfirmationDialogComponent, {\n        title: { i18n: 'i18n.qd.page.cancel.confirmation.dialog.title' },\n        dialogSize: QdDialogSize.Small,\n        data: this.config.pageTypeConfig\n      })\n      .closed.pipe(takeUntil(this._destroyed$))\n      .subscribe();\n  }\n\n  private initSubmitValidation(): void {\n    this.formGroupManagerService\n      .$areFormGroupsValid()\n      .pipe(\n        takeUntil(this._destroyed$),\n        tap(isValid => {\n          this.footerService.updateActions([\n            {\n              actionKey: 'submit',\n              partialAction: {\n                additionalInfo: isValid\n                  ? undefined\n                  : { infoType: 'warning', message: { i18n: 'i18n.qd.page.submit.invalidForm' } }\n              }\n            }\n          ]);\n        })\n      )\n      .subscribe();\n  }\n}\n","<main qdSnackbarListener>\n  <qd-page-object-header\n    [data-test-id]=\"testId\"\n    [attr.data-test-id]=\"testId + 'object-header'\"\n    [config]=\"config\"\n    [hasNavigation]=\"hasNavigation\"\n  ></qd-page-object-header>\n\n  <div class=\"page-info-banners\">\n    <ng-content select=\"qd-page-info-banner\"></ng-content>\n  </div>\n\n  <ng-container *ngIf=\"config.pageType === 'create' || config.pageType === 'custom'\">\n    <ng-content select=\"qd-page-stepper\"></ng-content>\n    <ng-content select=\"[qdPageStepperAdapter]\"></ng-content>\n  </ng-container>\n\n  <ng-container *ngIf=\"config.pageType === 'overview' || config.pageType === 'inspect' || config.pageType === 'custom'\">\n    <ng-content select=\"qd-page-tabs\"></ng-content>\n    <ng-content select=\"[qdPageTabsAdapter]\"></ng-content>\n    <ng-content select=\"qd-section\"></ng-content>\n    <ng-content select=\"[qdSectionAdapter]\"></ng-content>\n  </ng-container>\n\n  <ng-container *ngIf=\"config.pageType === 'custom'\">\n    <ng-container *ngTemplateOutlet=\"projectedContent\"></ng-container>\n  </ng-container>\n</main>\n\n<footer *ngIf=\"footerHasContent$ | async\">\n  <qd-page-footer [attr.data-test-id]=\"testId + '-footer'\">\n    <ng-content select=\"[qdPageFooter]\"></ng-content>\n  </qd-page-footer>\n</footer>\n\n<aside *ngIf=\"isControlPanelVisible\">\n  <ng-content select=\"qd-page-control-panel\"></ng-content>\n</aside>\n\n<qd-projection-guard *ngIf=\"config.pageType !== 'custom'\" [warningMessage]=\"projectionGuardMessage\">\n  <ng-container *ngTemplateOutlet=\"projectedContent\"></ng-container>\n</qd-projection-guard>\n\n<ng-template #projectedContent>\n  <ng-content></ng-content>\n</ng-template>\n"]}
|
|
634
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/page/page.component.ts","../../../../../libs/qd-ui/src/lib/page/page.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ContentChild,\n  ContentChildren,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  QueryList,\n  SimpleChanges\n} from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\nimport { takeUntil, tap } from 'rxjs/operators';\n\nimport { QdDialogService, QdDialogSize } from '../dialog/dialog.module';\nimport { QdPageStepperAdapterDirective } from '../page-stepper/adaptors/page-stepper-adapter.directive';\nimport { QdPageStepperComponent } from '../page-stepper/page-stepper.component';\nimport { QdPageTabsAdapterDirective } from '../page-tabs/adaptors/page-tabs-adapter.directive';\nimport { QdPageTabsComponent } from '../page-tabs/page-tabs.component';\nimport { QdSectionComponent } from '../section/section.component';\nimport { QdPageCancelConfirmationDialogComponent } from './cancel-confirmation-dialog/page-cancel-confirmation-dialog.component';\nimport { QdPageControlPanelComponent } from './control-panel/page-control-panel.component';\nimport { QdPageInfoBannerComponent } from './info-banner/page-info-banner.component';\nimport {\n  QdInspectOperationMode,\n  QdPageConfig,\n  QdPageTypeCreateConfig,\n  QdPageTypeInspectConfig\n} from './model/page-config.interface';\nimport { QdCreateFooterActionKey, QdFooterActionType, QdPageFooterAction } from './model/page-footer-action.interface';\nimport { QdFormGroupManagerService } from './shared/services/form-group-manager.service';\nimport { QdPageFooterService } from './shared/services/page-footer.service';\nimport { QdPageStoreService } from './shared/services/page-store.service';\nimport { QdPageSubmitActionService } from './shared/services/page-submit-action.service';\nimport { QdResolverTriggerService } from './shared/services/resolver-trigger.service';\n\ntype PartialFooterAction = { actionKey: QdCreateFooterActionKey; partialAction: Partial<QdPageFooterAction> };\n\n/**\n * **QdPage** serves as the central structural element within the hierarchy under the **QdShell**.\n * Integration is done through the Angular Routing Module, and QdPage is designed to be responsive across various screen sizes right out of the box.\n *\n * #### **Page Types**\n *\n * QdPage uses page types to decide how the page looks and works. Each page type has its own settings and actions.\n *\n * - **overview**: Displays a summary or a list of objects, providing an overview of multiple business cases.\n * - **create**: Allows the creation of a new object, representing a business case.\n * - **inspect**: Enables viewing, editing, and executing additional actions related to the object, which also represents a business case.\n * - **custom**: Provides a way to implement unique functionality that doesn’t fit the other predefined page types.\n *\n * **Note that the 'custom' page type should be used only as a fallback for very specific cases. If you encounter such a situation, please contact the Quadrel team.**\n *\n * #### **Facets**\n *\n * These are extra pieces of information shown in the page header. They give context specific to the current object model being viewed or edited. There are several types of facets, including `status`, `value`, `criticality`, `references`, `progress`, `date`, `singleSelect`, and `multiSelect`. Each type has its own options. For more details, see the `QdPageHeaderFacet` interface.\n *\n * #### **Context**\n *\n * The `context` acts as a filter that can be selected on the overview page and provides the business context, which remains consistent throughout the entire application. On create and inspect pages, the context is displayed as read-only. For more details, check the dedicated chapter in Storybook.\n *\n *\n * #### **Page Info Banners**\n *\n * To insert a page info banner under the page header, just put a `qd-page-info-banner` component inside the `qd-page`. Multiple consecutive info banners of the same type will be merged into one banner container. Page info banners have the same appearance like notifications.\n *\n * #### **Control Panel**\n *\n * The control panel is located on the right side of the page and is used to display additional information and functions relevant to a specific business case, which is linked to a specific object model. For further information, please consult the dedicated chapter in Storybook.\n *\n * #### **Actions**\n *\n * These are interactive elements that perform specific functions. Functions for these actions can be set up through configurations. Custom actions can be added to any page type. Additionally, all actions can be configured and customized with your own text, which will override the default texts.\n *\n * Please check the relevant interfaces for each page type: `QdPageConfigOverview`, `QdPageConfigCreate`, `QdPageConfigInspect`, and `QdPageConfigCustom`.\n *\n * #### **Validation/Parameterization**\n *\n * Validation and parameterization are covered in a dedicated chapter in the Storybook. Please check the \"Validation\" section for more information.\n *\n * #### **Resolver**\n *\n * Using a resolver to fetch page object data is essential to the page concept. Add the resolver to the component's providers and use the `QD_PAGE_OBJECT_RESOLVER_TOKEN` token. This approach enables type safety and user feedback by effectively managing the necessary data for the page.\n *\n * ```ts\n * interface MyObjectModel {\n *   // Metadata displayed in the ObjectHeader\n *   state: MyState;\n *   countryOfDestination: MyCountryOfDestination;\n *   // Appears not in the ObjectHeader\n *   payload: MyPayload;\n * }\n *\n * type MyObjectMetadata = 'state' | 'countryOfDestination';\n *\n * @Injectable()\n * class MyObjectModelResolver implements QdPageObjectResolver<MyObjectModel> {\n *   constructor(private http: HttpClient) {}\n *\n *   resolve(): Observable<MyObjectModel> {\n *     return this.http\n *       .get<MyObjectModel>('my-api')\n *       .pipe(tap((data: MyObjectModel) => doSomethingWith(data)));\n *   }\n * }\n *\n * @Component({\n *   selector: 'my-page',\n *   templateUrl: './my-page.component.html',\n *   providers: [\n *     {\n *       provide: QD_PAGE_OBJECT_RESOLVER_TOKEN,\n *       useClass: MyObjectModelResolver\n *     }\n *   ]\n * })\n * class MyPageComponent {\n *   pageConfig: QdPageConfig<Pick<MyObjectModel, MyObjectMetadata>> = {\n *     title: { i18n: 'i18n.page.title' },\n *     pageType: 'inspect',\n *     headerFacets: [\n *       {\n *         name: 'status',\n *         label: {\n *           i18n: 'i18n.status.label'\n *         },\n *         options: {\n *           type: 'status'\n *         }\n *       },\n *       {\n *         name: 'countryOfDestination',\n *         label: {\n *           i18n: 'i18n.countryOfDestination.label'\n *         },\n *         options: {\n *           type: 'value'\n *         }\n *       }\n *     ],\n *     pageTypeConfig: {\n *       cancel: {\n *         handler: () => handleCancel()\n *       },\n *       save: {\n *         handler: (formValues) => handleSave(formValues)\n *       }\n *     }\n *   };\n * }\n * ```\n *\n * #### **Updating Facets**\n *\n * Typically, the values of the facets on a create or inspect page are set to read-only. However, there may be cases where, for example, a status change is needed, such as through a dialog. For this purpose, we have provided an update method. In the resolver service, define an empty method called `updateMetadata`. Inject this service using its type, and then call the `updateMetadata` method with the required parameter.\n *\n * **Please note: These values should not be modified directly within a QdPage.**\n *\n * ```ts\n * @Injectable()\n * class MyObjectModelResolver implements QdPageObjectResolver<MyObjectModel> {\n *   config: QdPageObjectResolverConfig = {\n *     // your configuration options here\n *   }\n *\n *   constructor(private http: HttpClient) {}\n *\n *   resolve(): Observable<MyObjectModel> {\n *     // your implementation here\n *   }\n *\n *   updateMetadata(metadata: Partial<MyObjectModel>) {\n *     // leave empty here & call in your component\n *   }\n * }\n *\n * @Component({\n *   // ...\n *   providers: [\n *     {\n *       provide: QD_PAGE_OBJECT_RESOLVER_TOKEN,\n *       useClass: MyObjectModelResolver\n *     }\n *   ]\n * })\n * class MyPageComponent {\n *   constructor(@Inject(QD_PAGE_OBJECT_RESOLVER_TOKEN) private objectResolver: MyObjectModelResolver) {}\n *\n *   updateStatus() {\n *     this.objectResolver.updateMetadata({ state: 'Updated' });\n *   }\n * }\n * ```\n *\n * #### **Overview Page**\n *\n * On an overview page, a list of business cases can be displayed. Typically, this is done in a **QdTable**, including filters and/or search options. There are actions that handle routing to the related detail pages, where the business cases, represented by a business object model, can be viewed, edited, or newly created.\n *\n * ```ts\n * const overviewPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.my-page.title' },\n *   pageType: 'overview',\n *   pageTypeConfig: {\n *     customActions: [\n *       {\n *         i18n: 'i18n.custom.action',\n *         handler: () => doSomething()\n *       }\n *     ]\n *   }\n * };\n * ```\n *\n * ```html\n * <qd-page [config]=\"pageConfig\">\n *   <qd-section [config]=\"sectionConfig\">\n *     <qd-table [config]=\"tableConfig\"></qd-table>\n *   </qd-section>\n * </qd-page>\n * ```\n *\n * To configure contexts for a page, you can easily set contexts in the configuration for the overview page. They are then selectable on the overview page:\n *\n * ```ts\n * const overviewPageConfig: QdPageConfig<MyObjectType> = {\n *   // ...\n *   contexts: [\n *     {\n *       id: 'typeOfTraffic',\n *       label: { i18n: 'Verkehrsart' },\n *       type: 'single',\n *       selected: {\n *         label: { i18n: 'Schiene' },\n *         value: 'rail'\n *       },\n *       options: [\n *         {\n *           value: 'rail',\n *           label: { i18n: 'Schiene' }\n *         },\n *         {\n *           value: 'air',\n *           label: { i18n: 'Luft' }\n *         }\n *       ]\n *     }\n *   ]\n *   // ...\n * };\n * ```\n *\n * #### **Create Page**\n *\n * On the create page, a new business case can be created. This page typically includes forms for entering data. The submit button is parameterized with the form data and is validated through the QdPageStepper.\n * The click on the submit button should submit the creation of the business case or object. You can define a specific handler in the submit button's configuration to create the object.\n * If only one step is needed, the single-step mode can be used. In single-step mode, the stepper is visually hidden and behaves like a **QdSection**.\n *\n * ```ts\n * const createPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.page.title' },\n *   pageType: 'create',\n *   pageTypeConfig: {\n *     submit: {\n *       handler: (formValues) => handleSubmit(formValues)\n *     }\n *   }\n * };\n * ```\n *\n * ```html\n * <!-- For \"Quick Add\" use PageStepper in single-step mode -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-page-stepper [config]=\"stepperConfig\">\n *     <qd-page-step [config]=\"stepConfigOne\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myForms\" qdConnectFormStateToPage=\"myForms\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-step>\n *   </qd-page-stepper>\n * </qd-page>\n *\n * <!-- With PageStepper using multiple steps -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-page-stepper [config]=\"stepperConfig\">\n *     <qd-page-step [config]=\"stepConfigOne\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsOne\" qdConnectFormStateToPage=\"myFormsOne\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-step>\n *     <qd-page-step [config]=\"stepConfigTwo\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsTwo\" qdConnectFormStateToPage=\"myFormsTwo\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-step>\n *   </qd-page-stepper>\n * </qd-page>\n * ```\n *\n * To allow resolving asynchronous operations when going to next step, you can provide a resolver of type QdPageStepResolver:\n *\n * ```ts\n * class MyResolver implements QdPageStepResolver {\n *   ...\n *\n *   resolve(index: number): Observable<void> {\n *     return this.http.post(\"next-step.com\");\n *   }\n * }\n *\n * @Component({\n *   selector: 'qd-ui-page-create',\n *   templateUrl: './create.component.html',\n *   providers: [\n *     {\n *       provide: QD_PAGE_STEP_RESOLVER_TOKEN,\n *       useClass: PageStepResolver\n *     }\n *   ]\n * })\n * export class CreateComponent {\n * ```\n *\n * #### **Inspect Page**\n *\n * On an inspect page, a specific business case is displayed and may be edited or submitted. Depending on your needs you can hide the submit button.\n * In contrast to a create page, an existing object is shown and potentially updated.\n *\n * The page operates in one of two modes - `view` or `edit` - as defined by the `operationMode` configuration key.\n *\n * The `viewonly` behavior defined by the operation mode is automatically propagated to all nested QdSection components that are part of a QLS,\n * as well as to functional patterns such as QdTable, QdTree, QdFileCollector, and QdQuickEdit.\n * All forms are rendered either in viewonly or editable mode accordingly.\n *\n * This behavior can be explicitly overridden in each component’s configuration - for example, to allow certain fields to remain editable.\n *\n * - **view**: A submit action can optionally be added for process-related operations such as status updates.\n *   The action receives the current form values and supports validation and parameterization.\n *\n * - **edit**: Typically includes a save button to persist changes made in editable fields.\n *\n * ```ts\n * const inspectPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.page.title' },\n *   pageType: 'inspect',\n *   pageTypeConfig: {\n *     operationMode: 'edit',\n *     cancel: {\n *       handler: () => handleCancel()\n *     },\n *     save: {\n *       handler: (formValues) => handleSave(formValues)\n *     }\n *   }\n * };\n * ```\n *\n * ```html\n * <!-- with Section -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-section [config]=\"sectionConfig\">\n *     <qd-grid [config]=\"gridConfig\" [formGroup]=\"myForms\" qdConnectFormStateToPage=\"myForms\">\n *       <!-- your qd-form-components here -->\n *     </qd-grid>\n *   </qd-section>\n * </qd-page>\n *\n * <!-- with PageTabs -->\n * <qd-page [config]=\"pageConfig\">\n *   <qd-page-tabs [config]=\"stepperConfig\">\n *     <qd-page-tab [config]=\"tabConfigOne\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsOne\" qdConnectFormStateToPage=\"myFormsOne\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-tab>\n *     <qd-page-tab [config]=\"tabConfigTwo\">\n *       <qd-section [config]=\"sectionConfig\">\n *         <qd-grid [config]=\"gridConfig\" [formGroup]=\"myFormsTwo\" qdConnectFormStateToPage=\"myFormsTwo\">\n *           <!-- your qd-form-components here -->\n *         </qd-grid>\n *       </qd-section>\n *     </qd-page-tab>\n *   </qd-page-tabs>\n * </qd-page>\n * ```\n *\n * #### **Custom Page**\n *\n * ```ts\n * const customPageConfig: QdPageConfig<MyObjectType> = {\n *   title: { i18n: 'i18n.page.title' },\n *   pageType: 'custom',\n *   pageTypeConfig: {\n *     customActions: [\n *       {\n *         i18n: 'i18n.custom.action',\n *         handler: () => doSomething()\n *       }\n *     ]\n *   }\n * };\n * ```\n *\n * ```html\n * <qd-page [config]=\"pageConfig\">\n *   <!-- Place content here that doesn’t fit the other page types -->\n * </qd-page>\n * ```\n */\n@Component({\n  selector: 'qd-page',\n  templateUrl: './page.component.html',\n  styleUrls: ['./page.component.scss'],\n  providers: [QdPageFooterService, QdFormGroupManagerService, QdPageSubmitActionService, QdResolverTriggerService],\n  host: { '[class.has-control-panel]': 'isControlPanelVisible', '[class.control-panel-broad]': 'isControlPanelBroad' }\n})\nexport class QdPageComponent<T extends object> implements OnInit, OnChanges, AfterViewInit, OnDestroy {\n  /**\n   * This property defines the configuration for the QdPage component, including the page type,\n   * title, and specific configurations for each type of page.\n   */\n  @Input({ required: true })\n  config!: QdPageConfig<T>;\n\n  /**\n   * A static test ID for integration tests can be set. <br />\n   * The value for the HTML attribute [data-test-id].\n   */\n  @Input('data-test-id') testId? = 'page';\n\n  /**\n   * This event is emitted when the operation mode of the inspect page is changed.\n   */\n  @Output()\n  operationModeChanged = new EventEmitter<QdInspectOperationMode>();\n\n  @ContentChildren(QdSectionComponent)\n  sections!: QueryList<QdSectionComponent>;\n\n  @ContentChild(QdPageControlPanelComponent)\n  controlPanel!: QdPageControlPanelComponent;\n\n  @ContentChild(QdPageStepperComponent, { descendants: true })\n  stepperComponent?: QdPageStepperComponent;\n\n  @ContentChild(QdPageTabsComponent, { descendants: true })\n  tabsComponent?: QdPageTabsComponent;\n\n  @ContentChild(QdPageStepperAdapterDirective, { descendants: true })\n  stepperAdapterDirective?: QdPageStepperAdapterDirective;\n\n  @ContentChild(QdPageTabsAdapterDirective, { descendants: true })\n  tabsAdapterDirective?: QdPageTabsAdapterDirective;\n\n  @ContentChildren(QdPageInfoBannerComponent)\n  infoBanners!: QueryList<QdPageInfoBannerComponent>;\n\n  footerHasContent$: Observable<boolean>;\n  isFooterVisible: boolean | undefined;\n  projectionGuardMessage =\n    'QD-UI | QdPage - This content is not supported. Please use QdSectionAdapterDirective for custom content projection.';\n\n  private _isInitialized = false;\n  private _destroyed$ = new Subject<void>();\n\n  get isControlPanelVisible(): boolean {\n    return this.controlPanel !== undefined && this.controlPanel.config?.isHidden !== true;\n  }\n\n  get isControlPanelBroad(): boolean {\n    return this.controlPanel?.config?.width === 'broad';\n  }\n\n  get hasNavigation(): boolean {\n    return !!(this.stepperComponent || this.tabsComponent || this.stepperAdapterDirective || this.tabsAdapterDirective);\n  }\n\n  @HostBinding('class.has-footer')\n  get footerVisible(): boolean | undefined {\n    return this.isFooterVisible;\n  }\n\n  @HostBinding('class.has-info-banners')\n  get hasInfoBanners(): boolean {\n    return this.infoBanners.length > 0;\n  }\n\n  constructor(\n    private readonly pageStoreService: QdPageStoreService<T>,\n    private readonly footerService: QdPageFooterService,\n    private readonly formGroupManagerService: QdFormGroupManagerService,\n    private readonly submitActionService: QdPageSubmitActionService,\n    private readonly dialog: QdDialogService\n  ) {\n    this.footerHasContent$ = this.footerService.footerHasContent$;\n    this.footerHasContent$.pipe(takeUntil(this._destroyed$)).subscribe(isVisible => (this.isFooterVisible = isVisible));\n  }\n\n  ngOnInit(): void {\n    this.checkConfigValidity();\n\n    if (this.config) this.pageStoreService.initPageState(this.config);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const configChange = changes['config'];\n\n    if (!configChange) return;\n\n    const { currentValue: curr, previousValue: prev } = configChange;\n\n    if (curr?.pageTypeConfig?.submit?.isHidden !== prev?.pageTypeConfig?.submit?.isHidden)\n      this.footerService.updateActions([\n        { actionKey: 'submit', partialAction: { isVisible: !curr?.pageTypeConfig?.submit?.isHidden } }\n      ]);\n  }\n\n  ngAfterViewInit(): void {\n    if (this.config.pageType === 'create') this.setupCreatePageFooterActions();\n\n    if (this.config.pageType === 'create' && this.config?.pageTypeConfig?.cancel !== undefined)\n      this.handleCancelActionWithFormChanges();\n\n    if (this.config.pageType === 'create' && this.config?.pageTypeConfig?.saveDraft !== undefined)\n      this.initSaveDraftFooterAction();\n\n    if (this.config.pageType === 'inspect')\n      this.pageStoreService.isViewonly$\n        .pipe(takeUntil(this._destroyed$))\n        .subscribe(isViewonly =>\n          this.updateInspectPageOperationMode(\n            this.config.pageTypeConfig as QdPageTypeInspectConfig,\n            isViewonly ? 'view' : 'edit'\n          )\n        );\n\n    this._isInitialized = true;\n  }\n\n  ngOnDestroy(): void {\n    this._destroyed$.next();\n    this._destroyed$.complete();\n  }\n\n  private checkConfigValidity(): void {\n    if (!this.config) console.warn('QdUi | QdPageComponent - To configure the page you should provide a valid config.');\n  }\n\n  private setupCreatePageFooterActions(): void {\n    const pageTypeConfig = this.config?.pageTypeConfig as QdPageTypeCreateConfig;\n    const actionKeys: QdCreateFooterActionKey[] = ['saveDraft', 'cancel', 'submit'];\n    const actions: PartialFooterAction[] = [];\n\n    for (const actionKey of actionKeys) {\n      const action = pageTypeConfig[actionKey];\n\n      if (!action) continue;\n\n      actions.push({\n        actionKey,\n        partialAction: {\n          ...(action?.label?.i18n ? { titleI18n: action.label.i18n } : {}),\n          handler: this.generateFooterActionHandler(action?.handler)\n        }\n      });\n    }\n\n    this.footerService.updateActions(actions);\n  }\n\n  private handleCancelActionWithFormChanges(): void {\n    this.formGroupManagerService\n      .$hasValuesChanged()\n      .pipe(takeUntil(this._destroyed$))\n      .subscribe(hasChanged => {\n        this.footerService.updateActions([\n          {\n            actionKey: 'cancel',\n            partialAction: {\n              handler: hasChanged\n                ? (): void => this.setupSubmitActionValidation()\n                : (): void => (this.config?.pageTypeConfig as QdPageTypeCreateConfig)?.cancel?.handler()\n            }\n          }\n        ]);\n      });\n  }\n\n  private initSaveDraftFooterAction(): void {\n    const pageTypeConfig = this.config.pageTypeConfig as QdPageTypeCreateConfig;\n\n    this.footerService.setActions([\n      {\n        key: 'saveDraft',\n        action: {\n          titleI18n: 'i18n.qd.page.footer.saveDraft',\n          type: QdFooterActionType.Secondary,\n          handler: (): void => pageTypeConfig?.saveDraft?.handler(),\n          isVisible: true,\n          isDisabled: false\n        }\n      }\n    ]);\n  }\n\n  private updateInspectPageOperationMode(pageTypeConfig: QdPageTypeInspectConfig, mode: QdInspectOperationMode): void {\n    this.submitActionService.setSubmitActionForInspect(pageTypeConfig, mode === 'view');\n\n    this.footerService.updateActions([\n      { actionKey: 'submit', partialAction: { isVisible: mode === 'view' && !pageTypeConfig?.submit?.isHidden } }\n    ]);\n\n    this.initSubmitValidation();\n\n    if (this._isInitialized) this.operationModeChanged.emit(mode);\n  }\n\n  private generateFooterActionHandler(handler?: (values: unknown) => void): (...args: unknown[]) => void {\n    return (...args) => {\n      if (!handler) return;\n\n      handler(this.formGroupManagerService.hasFormGroups() ? this.formGroupManagerService.getAllValues() : args);\n    };\n  }\n\n  private setupSubmitActionValidation(): void {\n    this.dialog\n      .open(QdPageCancelConfirmationDialogComponent, {\n        title: { i18n: 'i18n.qd.page.cancel.confirmation.dialog.title' },\n        dialogSize: QdDialogSize.Small,\n        data: this.config.pageTypeConfig\n      })\n      .closed.pipe(takeUntil(this._destroyed$))\n      .subscribe();\n  }\n\n  private initSubmitValidation(): void {\n    this.formGroupManagerService\n      .$areFormGroupsValid()\n      .pipe(\n        takeUntil(this._destroyed$),\n        tap(isValid => {\n          this.footerService.updateActions([\n            {\n              actionKey: 'submit',\n              partialAction: {\n                additionalInfo: isValid\n                  ? undefined\n                  : { infoType: 'warning', message: { i18n: 'i18n.qd.page.submit.invalidForm' } }\n              }\n            }\n          ]);\n        })\n      )\n      .subscribe();\n  }\n}\n","<main qdSnackbarListener>\n  <qd-page-object-header\n    [data-test-id]=\"testId\"\n    [attr.data-test-id]=\"testId + 'object-header'\"\n    [config]=\"config\"\n    [hasNavigation]=\"hasNavigation\"\n  ></qd-page-object-header>\n\n  <div class=\"page-info-banners\">\n    <ng-content select=\"qd-page-info-banner\"></ng-content>\n  </div>\n\n  <ng-container *ngIf=\"config.pageType === 'create' || config.pageType === 'custom'\">\n    <ng-content select=\"qd-page-stepper\"></ng-content>\n    <ng-content select=\"[qdPageStepperAdapter]\"></ng-content>\n  </ng-container>\n\n  <ng-container *ngIf=\"config.pageType === 'overview' || config.pageType === 'inspect' || config.pageType === 'custom'\">\n    <ng-content select=\"qd-page-tabs\"></ng-content>\n    <ng-content select=\"[qdPageTabsAdapter]\"></ng-content>\n    <ng-content select=\"qd-section\"></ng-content>\n    <ng-content select=\"[qdSectionAdapter]\"></ng-content>\n  </ng-container>\n\n  <ng-container *ngIf=\"config.pageType === 'custom'\">\n    <ng-container *ngTemplateOutlet=\"projectedContent\"></ng-container>\n  </ng-container>\n</main>\n\n<footer *ngIf=\"footerHasContent$ | async\">\n  <qd-page-footer [attr.data-test-id]=\"testId + '-footer'\">\n    <ng-content select=\"[qdPageFooter]\"></ng-content>\n  </qd-page-footer>\n</footer>\n\n<aside *ngIf=\"isControlPanelVisible\">\n  <ng-content select=\"qd-page-control-panel\"></ng-content>\n</aside>\n\n<qd-projection-guard *ngIf=\"config.pageType !== 'custom'\" [warningMessage]=\"projectionGuardMessage\">\n  <ng-container *ngTemplateOutlet=\"projectedContent\"></ng-container>\n</qd-projection-guard>\n\n<ng-template #projectedContent>\n  <ng-content></ng-content>\n</ng-template>\n"]}
|
|
@@ -64,6 +64,8 @@ import * as i7 from "@angular/common";
|
|
|
64
64
|
*
|
|
65
65
|
* #### **Usage**
|
|
66
66
|
*
|
|
67
|
+
* **Basic Example:**
|
|
68
|
+
*
|
|
67
69
|
* ```typescript
|
|
68
70
|
* @Component({ ... })
|
|
69
71
|
* export class MyPageTabsComponent {
|
|
@@ -105,6 +107,74 @@ import * as i7 from "@angular/common";
|
|
|
105
107
|
* </qd-page-tab>
|
|
106
108
|
* </qd-page-tabs>
|
|
107
109
|
* ```
|
|
110
|
+
*
|
|
111
|
+
* **Bookmarkable Tabs Example: **
|
|
112
|
+
*
|
|
113
|
+
* To enable bookmarkable tabs that sync with the browser URL, set `connectWithRouter: true`
|
|
114
|
+
* and provide a unique `name` for each tab:
|
|
115
|
+
*
|
|
116
|
+
* ```typescript
|
|
117
|
+
* import { Component } from '@angular/core';
|
|
118
|
+
* import { Router } from '@angular/router';
|
|
119
|
+
* import { QdPageTabsConfig, QdPageTabConfig } from '@qd-ui/qd-ui';
|
|
120
|
+
*
|
|
121
|
+
* @Component({
|
|
122
|
+
* selector: 'app-bookmarkable-tabs',
|
|
123
|
+
* template: `
|
|
124
|
+
* <qd-page-tabs [config]="pageTabsConfig">
|
|
125
|
+
* <qd-page-tab [config]="overviewTabConfig">
|
|
126
|
+
* <qd-section>
|
|
127
|
+
* This is the overview tab content.
|
|
128
|
+
* </qd-section>
|
|
129
|
+
* </qd-page-tab>
|
|
130
|
+
*
|
|
131
|
+
* <qd-page-tab [config]="detailTabConfig">
|
|
132
|
+
* <qd-section>
|
|
133
|
+
* This is the detail tab content.
|
|
134
|
+
* </qd-section>
|
|
135
|
+
* </qd-page-tab>
|
|
136
|
+
*
|
|
137
|
+
* <qd-page-tab [config]="controlResultsTabConfig">
|
|
138
|
+
* <qd-section>
|
|
139
|
+
* This is the control results tab content.
|
|
140
|
+
* </qd-section>
|
|
141
|
+
* </qd-page-tab>
|
|
142
|
+
* </qd-page-tabs>
|
|
143
|
+
* `
|
|
144
|
+
* })
|
|
145
|
+
* export class BookmarkableTabsComponent {
|
|
146
|
+
* // Enable router connection for bookmarkable tabs
|
|
147
|
+
* pageTabsConfig: QdPageTabsConfig = {
|
|
148
|
+
* selectedIndex: 0,
|
|
149
|
+
* connectWithRouter: true
|
|
150
|
+
* };
|
|
151
|
+
*
|
|
152
|
+
* // Each tab must have a unique 'name' when connectWithRouter is enabled
|
|
153
|
+
* overviewTabConfig: QdPageTabConfig = {
|
|
154
|
+
* label: { i18n: 'Overview' },
|
|
155
|
+
* name: 'overview' // Required for router connection
|
|
156
|
+
* };
|
|
157
|
+
*
|
|
158
|
+
* detailTabConfig: QdPageTabConfig = {
|
|
159
|
+
* label: { i18n: 'Details' },
|
|
160
|
+
* name: 'details' // Required for router connection
|
|
161
|
+
* };
|
|
162
|
+
*
|
|
163
|
+
* controlResultsTabConfig: QdPageTabConfig = {
|
|
164
|
+
* label: { i18n: 'Control Results' },
|
|
165
|
+
* name: 'controlResults' // Required for router connection
|
|
166
|
+
* };
|
|
167
|
+
*
|
|
168
|
+
* constructor (private router: Router) {}
|
|
169
|
+
* }
|
|
170
|
+
* ```
|
|
171
|
+
*
|
|
172
|
+
* **Bookmarkable Tabs Behavior: **
|
|
173
|
+
* - URLs will include query parameter: `?tab=overview`, `?tab=details`, `?tab=controlResults`
|
|
174
|
+
* - Users can bookmark specific tabs and return directly to them
|
|
175
|
+
* - Browser back/forward buttons work with tab navigation
|
|
176
|
+
* - If an invalid tab name is provided in URL, the first available tab is selected
|
|
177
|
+
* - If no tab parameter is present, the `selectedIndex` or first non-disabled tab is selected
|
|
108
178
|
*/
|
|
109
179
|
export class QdPageTabsComponent extends CdkStepper {
|
|
110
180
|
footerService;
|
|
@@ -293,7 +363,7 @@ export class QdPageTabsComponent extends CdkStepper {
|
|
|
293
363
|
});
|
|
294
364
|
});
|
|
295
365
|
}
|
|
296
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdPageTabsComponent, deps: [{ token: i1.QdPageFooterService, optional: true }, { token: i2.Directionality, optional: true }, { token: i3.QdPageStoreService }, { token: i4.Router }, { token: i4.ActivatedRoute }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
366
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdPageTabsComponent, deps: [{ token: i1.QdPageFooterService, optional: true }, { token: i2.Directionality, optional: true }, { token: i3.QdPageStoreService }, { token: i4.Router }, { token: i4.ActivatedRoute, optional: true }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
297
367
|
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: QdPageTabsComponent, isStandalone: true, selector: "qd-page-tabs", inputs: { config: "config", testId: ["data-test-id", "testId"] }, outputs: { tabSelection: "tabSelection" }, host: { properties: { "class.standalone": "!footerService" }, classAttribute: "qd-tabs" }, providers: [{ provide: CdkStepper, useExisting: QdPageTabsComponent }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div class=\"qd-tabs-items-container\" [ngClass]=\"{ scrollable: config?.scrollabe }\">\n <qd-page-tab-header\n *ngFor=\"let tab of tabs; let i = index\"\n (click)=\"selectTab(tab)\"\n [id]=\"_getTabLabelId(i)\"\n [index]=\"i\"\n [state]=\"_getIndicatorType(i, tab.state)\"\n [label]=\"tab.config.label.i18n | translate\"\n [counters]=\"tab.config.counters\"\n [isSelected]=\"selectedIndex === i\"\n [isDisabled]=\"tab.config?.isDisabled\"\n [data-test-id]=\"testId + '-header' + '-' + i\"\n >\n </qd-page-tab-header>\n</div>\n\n<div class=\"tabs-content\">\n <ng-container *ngIf=\"!selected?.config?.isDisabled\" [ngTemplateOutlet]=\"selected?.content\"></ng-container>\n</div>\n\n<div class=\"qd-tabs-action-area\" *ngIf=\"isSubmitButtonShown()\">\n <button qdButton (click)=\"save()\" [disabled]=\"isSubmitButtonDisabled()\" [data-test-id]=\"testId + '-submit'\">\n {{ config?.submitButton?.i18n || \"i18n.qd.tabs.button.submit\" | translate }}\n </button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host.standalone{height:calc(100% - 50px)}:host .qd-tabs-items-container{display:flex;width:100%;flex-direction:row;flex-wrap:wrap;padding:0 1.25rem;border-bottom:.125rem solid rgb(213,213,213);background-color:#fff;gap:0 .9375rem}@media (max-width: 599.98px){:host .qd-tabs-items-container{padding:0 .9375rem}}@media (max-width: 1279.98px){:host .qd-tabs-items-container.scrollable{flex-wrap:nowrap;overflow-x:scroll;overflow-y:hidden;scrollbar-width:none}}:host .tabs-content{flex:1 1 auto}:host .qd-tabs-action-area{display:flex;width:100%;height:50px;justify-content:flex-end}:host .qd-tabs-action-area button{margin-right:1.25rem}\n"], dependencies: [{ kind: "directive", type: NgFor, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: QdPageTabHeaderComponent, selector: "qd-page-tab-header", inputs: ["state", "label", "counters", "index", "isSelected", "isDisabled", "data-test-id"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: QdButtonModule }, { kind: "component", type: i6.QdButtonComponent, selector: "button[qdButton], a[qdButton], button[qd-button]", inputs: ["disabled", "color", "icon", "data-test-id", "additionalInfo"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i7.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
298
368
|
}
|
|
299
369
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdPageTabsComponent, decorators: [{
|
|
@@ -303,7 +373,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
|
|
|
303
373
|
type: Optional
|
|
304
374
|
}] }, { type: i2.Directionality, decorators: [{
|
|
305
375
|
type: Optional
|
|
306
|
-
}] }, { type: i3.QdPageStoreService }, { type: i4.Router }, { type: i4.ActivatedRoute
|
|
376
|
+
}] }, { type: i3.QdPageStoreService }, { type: i4.Router }, { type: i4.ActivatedRoute, decorators: [{
|
|
377
|
+
type: Optional
|
|
378
|
+
}] }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }], propDecorators: { config: [{
|
|
307
379
|
type: Input
|
|
308
380
|
}], testId: [{
|
|
309
381
|
type: Input,
|
|
@@ -311,4 +383,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
|
|
|
311
383
|
}], tabSelection: [{
|
|
312
384
|
type: Output
|
|
313
385
|
}] } });
|
|
314
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-tabs.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/page-tabs/page-tabs.component.ts","../../../../../libs/qd-ui/src/lib/page-tabs/page-tabs.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Directionality } from '@angular/cdk/bidi';\nimport { CdkStepper, StepState } from '@angular/cdk/stepper';\nimport { CommonModule, NgFor, NgIf, NgTemplateOutlet } from '@angular/common';\nimport {\n  AfterContentInit,\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  DestroyRef,\n  ElementRef,\n  EventEmitter,\n  inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  SimpleChanges\n} from '@angular/core';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { QdButtonModule } from '../button/button.module';\nimport { QdPageFooterService } from '../page/shared/services/page-footer.service';\nimport { QdPageStoreService } from '../page/shared/services/page-store.service';\nimport { QdPageTabsConfig } from './model/page-tabs-config';\nimport { QdTabSelectionEvent } from './model/page-tabs-selection-event';\nimport { QdPageTabHeaderComponent } from './page-tab-header/page-tab-header.component';\nimport { QdPageTabComponent } from './page-tab/page-tab.component';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\n\n/**\n * **QdPageTabsComponent** provides a non-linear tabbed navigation system within a **QdPage**.\n * It enables switching between different sections while maintaining form validation and controlled navigation.\n *\n * #### **Features**\n *\n * - **Non-linear Navigation**: Users can navigate freely between tabs, with validation checks where necessary.\n * - **Form Validation Support**: Ensures tabs display errors when necessary if form inputs are invalid.\n * - **Customizable Tab Configurations**: Each tab can have custom labels, validation, and counters for tracking states.\n * - **Dynamic Counters**: Displays numeric indicators to signal critical or positive states.\n * - **Standalone Compatibility**: Can be used independently or integrated within a larger **QdPage** structure.\n * - **Customizable Submit Button**: Allows configuration of an action handler and visibility settings.\n * - **Bookmarkable tabs**: Allows users to bookmark tabs and make them navigable.\n *\n * #### **Tab Configuration**\n *\n * Each tab inside `QdPageTabsComponent` can be configured using `QdPageTabConfig`, supporting:\n *\n * - **label**: A translatable label for the tab.\n * - **isDisabled**: Whether the tab should be interactable.\n * - **counters**: Displays numeric counters for **critical** (red) and **positive** (green) states.\n * - **name**: Unique name of this tab. Used for query param routing via `?tab=<tabName>`. Required when `connectWithRouter` in **QdPageTabsConfig** is enabled.\n *\n * #### **Form Handling**\n *\n * Tabs support form groups, which are passed using the `[tabControl]` input.\n * The system ensures that forms validate correctly before allowing navigation.\n *\n * #### **Error Handling**\n *\n * Error states appear when:\n *\n * - **A required form field is left empty**: The tab title changes to red.\n * - **Invalid data is entered**: The validation process changes the tab title to red.\n *\n * #### **Submit Button Configuration**\n *\n * The submit button at the bottom of the tab system can be configured via `QdPageTabsConfig`:\n *\n * - **i18n**: The translated label for the submit button.\n * - **handler**: A callback function that receives form data upon submission.\n * - **isDisabled**: Controls whether the button should be active or not.\n * - **isHidden**: Determines whether the button is visible.\n * - **connectWithRouter**: If set to true, the tab will be connected to the URL and will be bookmarked. It is obligatory to set the `name` attribute for each tab.\n *\n * #### **Usage**\n *\n * ```typescript\n * @Component({ ... })\n * export class MyPageTabsComponent {\n *   form = new FormGroup({\n *     firstName: new FormControl('', Validators.required),\n *     lastName: new FormControl(''),\n *     age: new FormControl('')\n *   });\n *\n *   config: QdPageTabsConfig = {\n *     submitButton: {\n *       handler: (values) => doSomethingWith(values)\n *     }\n *   };\n *\n *   firstTabConfig: QdPageTabConfig = {\n *     label: { i18n: 'Tab One' },\n *     counters: of({ positive: 3, critical: 1 })\n *   };\n *\n *   ...\n * }\n * ```\n *\n * ```html\n * <qd-page-tabs [config]=\"config\">\n *   <qd-page-tab [config]=\"firstTabConfig\" [tabControl]=\"form\">\n *     <qd-section [config]=\"sectionConfig\">\n *       <qd-grid [config]=\"gridConfig\" [formGroup]=\"form\">\n *         <qd-input formControlName=\"firstName\"></qd-input>\n *         <qd-input formControlName=\"lastName\"></qd-input>\n *         <qd-input formControlName=\"age\"></qd-input>\n *       </qd-grid>\n *     </qd-section>\n *   </qd-page-tab>\n *\n *   <qd-page-tab ... >\n *     ...\n *   </qd-page-tab>\n * </qd-page-tabs>\n * ```\n */\n@Component({\n  selector: 'qd-page-tabs',\n  templateUrl: './page-tabs.component.html',\n  styleUrls: ['./page-tabs.component.scss'],\n  providers: [{ provide: CdkStepper, useExisting: QdPageTabsComponent }],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'qd-tabs', '[class.standalone]': '!footerService' },\n  standalone: true, // inherited by CdkStepper\n  imports: [NgFor, NgIf, NgTemplateOutlet, QdPageTabHeaderComponent, TranslateModule, QdButtonModule, CommonModule]\n})\nexport class QdPageTabsComponent\n  extends CdkStepper\n  implements OnInit, OnChanges, AfterContentInit, AfterViewInit, OnDestroy\n{\n  /**\n   * Configuration of QdPageTabs.\n   */\n  @Input()\n  config!: QdPageTabsConfig;\n\n  /**\n   * A static test ID for integration tests can be set.\n   */\n  @Input('data-test-id') testId = 'page-tabs';\n\n  @Output() readonly tabSelection = new EventEmitter<QdTabSelectionEvent>();\n\n  private _viewonly: boolean | undefined;\n  private readonly destroyRef = inject(DestroyRef);\n\n  get tabs(): QueryList<QdPageTabComponent> {\n    return this.steps as QueryList<QdPageTabComponent>;\n  }\n\n  override get selected(): QdPageTabComponent | undefined {\n    return super.selected as QdPageTabComponent;\n  }\n\n  override set selected(step: QdPageTabComponent | undefined) {\n    super.selected = step;\n  }\n\n  constructor(\n    @Optional() readonly footerService: QdPageFooterService,\n    @Optional() _dir: Directionality,\n    private pageStoreService: QdPageStoreService<any>,\n    private readonly router: Router,\n    private readonly route: ActivatedRoute,\n    _changeDetectorRef: ChangeDetectorRef,\n    _elementRef: ElementRef<HTMLElement>\n  ) {\n    super(_dir, _changeDetectorRef, _elementRef);\n    this.mapSelectionChangeToTabSelectionOutput();\n  }\n\n  ngOnInit(): void {\n    this.linear = false;\n\n    this.pageStoreService.isViewonly$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(v => {\n      this._viewonly = v;\n      this._stateChanged();\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    ['linear', 'selectedIndex', 'selected', 'orientation']\n      .filter(inputName => !!changes[inputName])\n      .forEach(inputName => this.blockCdkInput(inputName));\n  }\n\n  override ngAfterContentInit(): void {\n    super.ngAfterContentInit();\n\n    this.tabs.changes.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(() => this._stateChanged());\n  }\n\n  override ngAfterViewInit(): void {\n    super.ngAfterViewInit();\n    setTimeout(() => {\n      this.initializeTabSelection();\n      this.configureBookmarkableTabs();\n    });\n  }\n\n  private initializeTabSelection(): void {\n    if (this.config?.selectedIndex && this.isTabSelectable(this.config.selectedIndex)) {\n      this.selectedIndex = this.config.selectedIndex;\n      return;\n    }\n\n    this.selectFirstNotDisabledTab();\n  }\n\n  private configureBookmarkableTabs(): void {\n    if (this.config?.connectWithRouter) {\n      this.initializeTabFromUrl();\n      this.initializeFirstTabSelection();\n    }\n  }\n\n  /**\n   * Initializes the tab selection based on the URL parameter 'tab'.\n   * @private\n   */\n  private initializeTabFromUrl(): void {\n    this.route.queryParams.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(params => {\n      const tabNameFromParams = params['tab'];\n      const pageTab = this.tabs.find(tab => tab.config?.name === tabNameFromParams);\n      if (pageTab) {\n        pageTab.select();\n      } else {\n        console.warn('QD-UI | QdPageTabsComponent - No tab found with name ' + tabNameFromParams);\n        this.selectFirstNotDisabledTab(true);\n      }\n    });\n  }\n\n  /**\n   * If the user navigates to a page with a tab selected, the tab will be selected automatically in {@link initializeTabFromUrl()} method.\n   * If the user navigates to a page without a tab selected, the first active tab will be selected.\n   * @private\n   */\n  private initializeFirstTabSelection(): void {\n    this.route.queryParams.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(params => {\n      const tabNameFromParams = params['tab'];\n      if (!tabNameFromParams) {\n        const selectedIndex = this.config.selectedIndex;\n        const pageTab = this.tabs.find((tab, tabIndex) => tabIndex === selectedIndex);\n        if (pageTab && pageTab?.config.name) {\n          this.router.navigate([], {\n            relativeTo: this.route,\n            queryParams: { tab: pageTab.config.name },\n            queryParamsHandling: 'merge'\n          });\n        } else {\n          console.warn(\n            'QD-UI | QdPageTabsComponent - connectedWithRouter is active, however QdPageTab has no name attribute.'\n          );\n        }\n      }\n    });\n  }\n\n  override ngOnDestroy(): void {\n    super.ngOnDestroy();\n  }\n\n  override _getIndicatorType(index: number, state?: StepState): StepState {\n    let stepState = super._getIndicatorType(index, state);\n    if (stepState === 'error' && this._viewonly === true) stepState = 'edit';\n    return stepState;\n  }\n\n  private isTabSelectable(index: number): boolean {\n    const tab = this.tabs.get(index);\n\n    return !!tab && !tab.config.isDisabled;\n  }\n\n  /**\n   * Selects the first tab that is not disabled.\n   * @param updateUrl if true, the URL will be updated to reflect the selected tab.\n   */\n  private selectFirstNotDisabledTab(updateUrl = false): void {\n    this.tabs.some((tab, tabIndex) => {\n      if (!tab.config.isDisabled) {\n        if (this.config) {\n          this.config.selectedIndex = tabIndex;\n        }\n        this.selectedIndex = tabIndex;\n\n        if (updateUrl && this.config?.connectWithRouter && tab.config?.name) {\n          this.router.navigate([], {\n            relativeTo: this.route,\n            queryParams: { tab: tab.config.name },\n            queryParamsHandling: 'merge'\n          });\n        }\n        return true;\n      } else {\n        return false;\n      }\n    });\n  }\n\n  save(): void {\n    this.selected?.tabControl?.markAllAsTouched();\n\n    const tabControlValues = this.tabs.map(tab => tab.tabControl?.value);\n    this.config?.submitButton?.handler?.(tabControlValues);\n  }\n\n  selectTab(tab: QdPageTabComponent): void {\n    if (tab.config?.isDisabled) return;\n    tab.select();\n    if (this.config?.connectWithRouter && tab.config?.name) {\n      this.router.navigate([], {\n        relativeTo: this.route,\n        queryParams: { tab: tab.config.name },\n        queryParamsHandling: 'merge'\n      });\n    }\n  }\n\n  isSubmitButtonShown(): boolean {\n    return this.config?.submitButton?.isHidden !== true && !this.footerService;\n  }\n\n  isSubmitButtonDisabled(): boolean | undefined {\n    return this.config?.submitButton?.isDisabled;\n  }\n\n  _getTabLabelId(i: number): string {\n    return this._getStepLabelId(i);\n  }\n\n  private blockCdkInput(inputName: string): void {\n    throw new Error(\n      `QD-UI | QdPageTabsComponent - The use of the ${inputName} attribute is not permitted. Please use QdPageTabsConfig instead.`\n    );\n  }\n\n  private mapSelectionChangeToTabSelectionOutput(): void {\n    this.selectionChange.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(selectionEvent => {\n      this.tabSelection.emit({\n        selectedIndex: selectionEvent.selectedIndex,\n        previouslySelectedIndex: selectionEvent.previouslySelectedIndex,\n        selectedTab: selectionEvent.selectedStep as QdPageTabComponent,\n        previouslySelectedTab: selectionEvent.previouslySelectedStep as QdPageTabComponent\n      });\n    });\n  }\n}\n","<div class=\"qd-tabs-items-container\" [ngClass]=\"{ scrollable: config?.scrollabe }\">\n  <qd-page-tab-header\n    *ngFor=\"let tab of tabs; let i = index\"\n    (click)=\"selectTab(tab)\"\n    [id]=\"_getTabLabelId(i)\"\n    [index]=\"i\"\n    [state]=\"_getIndicatorType(i, tab.state)\"\n    [label]=\"tab.config.label.i18n | translate\"\n    [counters]=\"tab.config.counters\"\n    [isSelected]=\"selectedIndex === i\"\n    [isDisabled]=\"tab.config?.isDisabled\"\n    [data-test-id]=\"testId + '-header' + '-' + i\"\n  >\n  </qd-page-tab-header>\n</div>\n\n<div class=\"tabs-content\">\n  <ng-container *ngIf=\"!selected?.config?.isDisabled\" [ngTemplateOutlet]=\"selected?.content\"></ng-container>\n</div>\n\n<div class=\"qd-tabs-action-area\" *ngIf=\"isSubmitButtonShown()\">\n  <button qdButton (click)=\"save()\" [disabled]=\"isSubmitButtonDisabled()\" [data-test-id]=\"testId + '-submit'\">\n    {{ config?.submitButton?.i18n || \"i18n.qd.tabs.button.submit\" | translate }}\n  </button>\n</div>\n"]}
|
|
386
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-tabs.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/page-tabs/page-tabs.component.ts","../../../../../libs/qd-ui/src/lib/page-tabs/page-tabs.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Directionality } from '@angular/cdk/bidi';\nimport { CdkStepper, StepState } from '@angular/cdk/stepper';\nimport { CommonModule, NgFor, NgIf, NgTemplateOutlet } from '@angular/common';\nimport {\n  AfterContentInit,\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  DestroyRef,\n  ElementRef,\n  EventEmitter,\n  inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  SimpleChanges\n} from '@angular/core';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { QdButtonModule } from '../button/button.module';\nimport { QdPageFooterService } from '../page/shared/services/page-footer.service';\nimport { QdPageStoreService } from '../page/shared/services/page-store.service';\nimport { QdPageTabsConfig } from './model/page-tabs-config';\nimport { QdTabSelectionEvent } from './model/page-tabs-selection-event';\nimport { QdPageTabHeaderComponent } from './page-tab-header/page-tab-header.component';\nimport { QdPageTabComponent } from './page-tab/page-tab.component';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\n\n/**\n * **QdPageTabsComponent** provides a non-linear tabbed navigation system within a **QdPage**.\n * It enables switching between different sections while maintaining form validation and controlled navigation.\n *\n * #### **Features**\n *\n * - **Non-linear Navigation**: Users can navigate freely between tabs, with validation checks where necessary.\n * - **Form Validation Support**: Ensures tabs display errors when necessary if form inputs are invalid.\n * - **Customizable Tab Configurations**: Each tab can have custom labels, validation, and counters for tracking states.\n * - **Dynamic Counters**: Displays numeric indicators to signal critical or positive states.\n * - **Standalone Compatibility**: Can be used independently or integrated within a larger **QdPage** structure.\n * - **Customizable Submit Button**: Allows configuration of an action handler and visibility settings.\n * - **Bookmarkable tabs**: Allows users to bookmark tabs and make them navigable.\n *\n * #### **Tab Configuration**\n *\n * Each tab inside `QdPageTabsComponent` can be configured using `QdPageTabConfig`, supporting:\n *\n * - **label**: A translatable label for the tab.\n * - **isDisabled**: Whether the tab should be interactable.\n * - **counters**: Displays numeric counters for **critical** (red) and **positive** (green) states.\n * - **name**: Unique name of this tab. Used for query param routing via `?tab=<tabName>`. Required when `connectWithRouter` in **QdPageTabsConfig** is enabled.\n *\n * #### **Form Handling**\n *\n * Tabs support form groups, which are passed using the `[tabControl]` input.\n * The system ensures that forms validate correctly before allowing navigation.\n *\n * #### **Error Handling**\n *\n * Error states appear when:\n *\n * - **A required form field is left empty**: The tab title changes to red.\n * - **Invalid data is entered**: The validation process changes the tab title to red.\n *\n * #### **Submit Button Configuration**\n *\n * The submit button at the bottom of the tab system can be configured via `QdPageTabsConfig`:\n *\n * - **i18n**: The translated label for the submit button.\n * - **handler**: A callback function that receives form data upon submission.\n * - **isDisabled**: Controls whether the button should be active or not.\n * - **isHidden**: Determines whether the button is visible.\n * - **connectWithRouter**: If set to true, the tab will be connected to the URL and will be bookmarked. It is obligatory to set the `name` attribute for each tab.\n *\n * #### **Usage**\n *\n * **Basic Example:**\n *\n * ```typescript\n * @Component({ ... })\n * export class MyPageTabsComponent {\n *   form = new FormGroup({\n *     firstName: new FormControl('', Validators.required),\n *     lastName: new FormControl(''),\n *     age: new FormControl('')\n *   });\n *\n *   config: QdPageTabsConfig = {\n *     submitButton: {\n *       handler: (values) => doSomethingWith(values)\n *     }\n *   };\n *\n *   firstTabConfig: QdPageTabConfig = {\n *     label: { i18n: 'Tab One' },\n *     counters: of({ positive: 3, critical: 1 })\n *   };\n *\n *   ...\n * }\n * ```\n *\n * ```html\n * <qd-page-tabs [config]=\"config\">\n *   <qd-page-tab [config]=\"firstTabConfig\" [tabControl]=\"form\">\n *     <qd-section [config]=\"sectionConfig\">\n *       <qd-grid [config]=\"gridConfig\" [formGroup]=\"form\">\n *         <qd-input formControlName=\"firstName\"></qd-input>\n *         <qd-input formControlName=\"lastName\"></qd-input>\n *         <qd-input formControlName=\"age\"></qd-input>\n *       </qd-grid>\n *     </qd-section>\n *   </qd-page-tab>\n *\n *   <qd-page-tab ... >\n *     ...\n *   </qd-page-tab>\n * </qd-page-tabs>\n * ```\n *\n * **Bookmarkable Tabs Example: **\n *\n * To enable bookmarkable tabs that sync with the browser URL, set `connectWithRouter: true`\n * and provide a unique `name` for each tab:\n *\n * ```typescript\n * import { Component } from '@angular/core';\n * import { Router } from '@angular/router';\n * import { QdPageTabsConfig, QdPageTabConfig } from '@qd-ui/qd-ui';\n *\n * @Component({\n *   selector: 'app-bookmarkable-tabs',\n *   template: `\n *     <qd-page-tabs [config]=\"pageTabsConfig\">\n *       <qd-page-tab [config]=\"overviewTabConfig\">\n *         <qd-section>\n *           This is the overview tab content.\n *         </qd-section>\n *       </qd-page-tab>\n *\n *       <qd-page-tab [config]=\"detailTabConfig\">\n *         <qd-section>\n *           This is the detail tab content.\n *         </qd-section>\n *       </qd-page-tab>\n *\n *       <qd-page-tab [config]=\"controlResultsTabConfig\">\n *         <qd-section>\n *           This is the control results tab content.\n *         </qd-section>\n *       </qd-page-tab>\n *     </qd-page-tabs>\n *   `\n * })\n * export class BookmarkableTabsComponent {\n *   // Enable router connection for bookmarkable tabs\n *   pageTabsConfig: QdPageTabsConfig = {\n *     selectedIndex: 0,\n *     connectWithRouter: true\n *   };\n *\n *   // Each tab must have a unique 'name' when connectWithRouter is enabled\n *   overviewTabConfig: QdPageTabConfig = {\n *     label: { i18n: 'Overview' },\n *     name: 'overview' // Required for router connection\n *   };\n *\n *   detailTabConfig: QdPageTabConfig = {\n *     label: { i18n: 'Details' },\n *     name: 'details' // Required for router connection\n *   };\n *\n *   controlResultsTabConfig: QdPageTabConfig = {\n *     label: { i18n: 'Control Results' },\n *     name: 'controlResults' // Required for router connection\n *   };\n *\n *   constructor (private router: Router) {}\n * }\n * ```\n *\n * **Bookmarkable Tabs Behavior: **\n * - URLs will include query parameter: `?tab=overview`, `?tab=details`, `?tab=controlResults`\n * - Users can bookmark specific tabs and return directly to them\n * - Browser back/forward buttons work with tab navigation\n * - If an invalid tab name is provided in URL, the first available tab is selected\n * - If no tab parameter is present, the `selectedIndex` or first non-disabled tab is selected\n */\n@Component({\n  selector: 'qd-page-tabs',\n  templateUrl: './page-tabs.component.html',\n  styleUrls: ['./page-tabs.component.scss'],\n  providers: [{ provide: CdkStepper, useExisting: QdPageTabsComponent }],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'qd-tabs', '[class.standalone]': '!footerService' },\n  standalone: true, // inherited by CdkStepper\n  imports: [NgFor, NgIf, NgTemplateOutlet, QdPageTabHeaderComponent, TranslateModule, QdButtonModule, CommonModule]\n})\nexport class QdPageTabsComponent\n  extends CdkStepper\n  implements OnInit, OnChanges, AfterContentInit, AfterViewInit, OnDestroy\n{\n  /**\n   * Configuration of QdPageTabs.\n   */\n  @Input()\n  config!: QdPageTabsConfig;\n\n  /**\n   * A static test ID for integration tests can be set.\n   */\n  @Input('data-test-id') testId = 'page-tabs';\n\n  @Output() readonly tabSelection = new EventEmitter<QdTabSelectionEvent>();\n\n  private _viewonly: boolean | undefined;\n  private readonly destroyRef = inject(DestroyRef);\n\n  get tabs(): QueryList<QdPageTabComponent> {\n    return this.steps as QueryList<QdPageTabComponent>;\n  }\n\n  override get selected(): QdPageTabComponent | undefined {\n    return super.selected as QdPageTabComponent;\n  }\n\n  override set selected(step: QdPageTabComponent | undefined) {\n    super.selected = step;\n  }\n\n  constructor(\n    @Optional() readonly footerService: QdPageFooterService,\n    @Optional() _dir: Directionality,\n    private pageStoreService: QdPageStoreService<any>,\n    private readonly router: Router,\n    @Optional() private readonly route: ActivatedRoute,\n    _changeDetectorRef: ChangeDetectorRef,\n    _elementRef: ElementRef<HTMLElement>\n  ) {\n    super(_dir, _changeDetectorRef, _elementRef);\n    this.mapSelectionChangeToTabSelectionOutput();\n  }\n\n  ngOnInit(): void {\n    this.linear = false;\n\n    this.pageStoreService.isViewonly$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(v => {\n      this._viewonly = v;\n      this._stateChanged();\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    ['linear', 'selectedIndex', 'selected', 'orientation']\n      .filter(inputName => !!changes[inputName])\n      .forEach(inputName => this.blockCdkInput(inputName));\n  }\n\n  override ngAfterContentInit(): void {\n    super.ngAfterContentInit();\n\n    this.tabs.changes.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(() => this._stateChanged());\n  }\n\n  override ngAfterViewInit(): void {\n    super.ngAfterViewInit();\n    setTimeout(() => {\n      this.initializeTabSelection();\n      this.configureBookmarkableTabs();\n    });\n  }\n\n  private initializeTabSelection(): void {\n    if (this.config?.selectedIndex && this.isTabSelectable(this.config.selectedIndex)) {\n      this.selectedIndex = this.config.selectedIndex;\n      return;\n    }\n\n    this.selectFirstNotDisabledTab();\n  }\n\n  private configureBookmarkableTabs(): void {\n    if (this.config?.connectWithRouter) {\n      this.initializeTabFromUrl();\n      this.initializeFirstTabSelection();\n    }\n  }\n\n  /**\n   * Initializes the tab selection based on the URL parameter 'tab'.\n   * @private\n   */\n  private initializeTabFromUrl(): void {\n    this.route.queryParams.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(params => {\n      const tabNameFromParams = params['tab'];\n      const pageTab = this.tabs.find(tab => tab.config?.name === tabNameFromParams);\n      if (pageTab) {\n        pageTab.select();\n      } else {\n        console.warn('QD-UI | QdPageTabsComponent - No tab found with name ' + tabNameFromParams);\n        this.selectFirstNotDisabledTab(true);\n      }\n    });\n  }\n\n  /**\n   * If the user navigates to a page with a tab selected, the tab will be selected automatically in {@link initializeTabFromUrl()} method.\n   * If the user navigates to a page without a tab selected, the first active tab will be selected.\n   * @private\n   */\n  private initializeFirstTabSelection(): void {\n    this.route.queryParams.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(params => {\n      const tabNameFromParams = params['tab'];\n      if (!tabNameFromParams) {\n        const selectedIndex = this.config.selectedIndex;\n        const pageTab = this.tabs.find((tab, tabIndex) => tabIndex === selectedIndex);\n        if (pageTab && pageTab?.config.name) {\n          this.router.navigate([], {\n            relativeTo: this.route,\n            queryParams: { tab: pageTab.config.name },\n            queryParamsHandling: 'merge'\n          });\n        } else {\n          console.warn(\n            'QD-UI | QdPageTabsComponent - connectedWithRouter is active, however QdPageTab has no name attribute.'\n          );\n        }\n      }\n    });\n  }\n\n  override ngOnDestroy(): void {\n    super.ngOnDestroy();\n  }\n\n  override _getIndicatorType(index: number, state?: StepState): StepState {\n    let stepState = super._getIndicatorType(index, state);\n    if (stepState === 'error' && this._viewonly === true) stepState = 'edit';\n    return stepState;\n  }\n\n  private isTabSelectable(index: number): boolean {\n    const tab = this.tabs.get(index);\n\n    return !!tab && !tab.config.isDisabled;\n  }\n\n  /**\n   * Selects the first tab that is not disabled.\n   * @param updateUrl if true, the URL will be updated to reflect the selected tab.\n   */\n  private selectFirstNotDisabledTab(updateUrl = false): void {\n    this.tabs.some((tab, tabIndex) => {\n      if (!tab.config.isDisabled) {\n        if (this.config) {\n          this.config.selectedIndex = tabIndex;\n        }\n        this.selectedIndex = tabIndex;\n\n        if (updateUrl && this.config?.connectWithRouter && tab.config?.name) {\n          this.router.navigate([], {\n            relativeTo: this.route,\n            queryParams: { tab: tab.config.name },\n            queryParamsHandling: 'merge'\n          });\n        }\n        return true;\n      } else {\n        return false;\n      }\n    });\n  }\n\n  save(): void {\n    this.selected?.tabControl?.markAllAsTouched();\n\n    const tabControlValues = this.tabs.map(tab => tab.tabControl?.value);\n    this.config?.submitButton?.handler?.(tabControlValues);\n  }\n\n  selectTab(tab: QdPageTabComponent): void {\n    if (tab.config?.isDisabled) return;\n    tab.select();\n    if (this.config?.connectWithRouter && tab.config?.name) {\n      this.router.navigate([], {\n        relativeTo: this.route,\n        queryParams: { tab: tab.config.name },\n        queryParamsHandling: 'merge'\n      });\n    }\n  }\n\n  isSubmitButtonShown(): boolean {\n    return this.config?.submitButton?.isHidden !== true && !this.footerService;\n  }\n\n  isSubmitButtonDisabled(): boolean | undefined {\n    return this.config?.submitButton?.isDisabled;\n  }\n\n  _getTabLabelId(i: number): string {\n    return this._getStepLabelId(i);\n  }\n\n  private blockCdkInput(inputName: string): void {\n    throw new Error(\n      `QD-UI | QdPageTabsComponent - The use of the ${inputName} attribute is not permitted. Please use QdPageTabsConfig instead.`\n    );\n  }\n\n  private mapSelectionChangeToTabSelectionOutput(): void {\n    this.selectionChange.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(selectionEvent => {\n      this.tabSelection.emit({\n        selectedIndex: selectionEvent.selectedIndex,\n        previouslySelectedIndex: selectionEvent.previouslySelectedIndex,\n        selectedTab: selectionEvent.selectedStep as QdPageTabComponent,\n        previouslySelectedTab: selectionEvent.previouslySelectedStep as QdPageTabComponent\n      });\n    });\n  }\n}\n","<div class=\"qd-tabs-items-container\" [ngClass]=\"{ scrollable: config?.scrollabe }\">\n  <qd-page-tab-header\n    *ngFor=\"let tab of tabs; let i = index\"\n    (click)=\"selectTab(tab)\"\n    [id]=\"_getTabLabelId(i)\"\n    [index]=\"i\"\n    [state]=\"_getIndicatorType(i, tab.state)\"\n    [label]=\"tab.config.label.i18n | translate\"\n    [counters]=\"tab.config.counters\"\n    [isSelected]=\"selectedIndex === i\"\n    [isDisabled]=\"tab.config?.isDisabled\"\n    [data-test-id]=\"testId + '-header' + '-' + i\"\n  >\n  </qd-page-tab-header>\n</div>\n\n<div class=\"tabs-content\">\n  <ng-container *ngIf=\"!selected?.config?.isDisabled\" [ngTemplateOutlet]=\"selected?.content\"></ng-container>\n</div>\n\n<div class=\"qd-tabs-action-area\" *ngIf=\"isSubmitButtonShown()\">\n  <button qdButton (click)=\"save()\" [disabled]=\"isSubmitButtonDisabled()\" [data-test-id]=\"testId + '-submit'\">\n    {{ config?.submitButton?.i18n || \"i18n.qd.tabs.button.submit\" | translate }}\n  </button>\n</div>\n"]}
|