@cqa-lib/cqa-ui 1.1.550 → 1.1.551

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.
@@ -20,6 +20,7 @@ export class DeleteFolderDialogComponent {
20
20
  this.cdr = cdr;
21
21
  this.folderName = '';
22
22
  this.testCount = 0;
23
+ this.stepGroupCount = 0;
23
24
  this.hasParent = false;
24
25
  this.labels = { ...DEFAULT_MODULAR_LABELS };
25
26
  this.strategy = 'MOVE_TO_PARENT';
@@ -39,10 +40,21 @@ export class DeleteFolderDialogComponent {
39
40
  }
40
41
  }
41
42
  get bodyText() {
42
- const template = this.testCount === 1
43
- ? this.labels.deleteFolderDialogBodySingular
44
- : this.labels.deleteFolderDialogBodyPlural;
45
- return template.replace('{n}', '' + this.testCount);
43
+ const parts = [];
44
+ if (this.testCount > 0) {
45
+ const tpl = this.testCount === 1
46
+ ? this.labels.deleteFolderDialogBodyTestCaseSingular
47
+ : this.labels.deleteFolderDialogBodyTestCasePlural;
48
+ parts.push(tpl.replace('{n}', '' + this.testCount));
49
+ }
50
+ if (this.stepGroupCount > 0) {
51
+ const tpl = this.stepGroupCount === 1
52
+ ? this.labels.deleteFolderDialogBodyStepGroupSingular
53
+ : this.labels.deleteFolderDialogBodyStepGroupPlural;
54
+ parts.push(tpl.replace('{n}', '' + this.stepGroupCount));
55
+ }
56
+ const items = parts.join(` ${this.labels.deleteFolderDialogBodyJoin} `);
57
+ return this.labels.deleteFolderDialogBodyTemplate.replace('{items}', items);
46
58
  }
47
59
  setStrategy(value) {
48
60
  if (this.strategy === value)
@@ -53,7 +65,7 @@ export class DeleteFolderDialogComponent {
53
65
  }
54
66
  }
55
67
  DeleteFolderDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DeleteFolderDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
56
- DeleteFolderDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DeleteFolderDialogComponent, selector: "cqa-delete-folder-dialog", inputs: { folderName: "folderName", testCount: "testCount", hasParent: "hasParent", labels: "labels", strategy: "strategy" }, outputs: { strategyChange: "strategyChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: `
68
+ DeleteFolderDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DeleteFolderDialogComponent, selector: "cqa-delete-folder-dialog", inputs: { folderName: "folderName", testCount: "testCount", stepGroupCount: "stepGroupCount", hasParent: "hasParent", labels: "labels", strategy: "strategy" }, outputs: { strategyChange: "strategyChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: `
57
69
  <div class="cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full">
58
70
  <p class="cqa-text-sm cqa-text-neutral-800 cqa-m-0">{{ bodyText }}</p>
59
71
 
@@ -169,6 +181,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
169
181
  type: Input
170
182
  }], testCount: [{
171
183
  type: Input
184
+ }], stepGroupCount: [{
185
+ type: Input
172
186
  }], hasParent: [{
173
187
  type: Input
174
188
  }], labels: [{
@@ -178,4 +192,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
178
192
  }], strategyChange: [{
179
193
  type: Output
180
194
  }] } });
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"delete-folder-dialog.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/templates/modular-table-template/dialogs/delete-folder-dialog.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EAEvB,SAAS,EACT,YAAY,EACZ,KAAK,EAGL,MAAM,GAEP,MAAM,eAAe,CAAC;AACvB,OAAO,EACL,sBAAsB,GAGvB,MAAM,kCAAkC,CAAC;;;AAE1C;;;;;;;;;;;;GAYG;AA8DH,MAAM,OAAO,2BAA2B;IAStC,YAAoB,GAAsB;QAAtB,QAAG,GAAH,GAAG,CAAmB;QARjC,eAAU,GAAW,EAAE,CAAC;QACxB,cAAS,GAAW,CAAC,CAAC;QACtB,cAAS,GAAY,KAAK,CAAC;QAC3B,WAAM,GAAkB,EAAE,GAAG,sBAAsB,EAAE,CAAC;QACtD,aAAQ,GAAyB,gBAAgB,CAAC;QAEjD,mBAAc,GAAG,IAAI,YAAY,EAAwB,CAAC;IAEvB,CAAC;IAE9C,QAAQ;QACN,IAAI,CAAC,iBAAiB,EAAE,CAAC;IAC3B,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,WAAW,CAAC;YAAE,IAAI,CAAC,iBAAiB,EAAE,CAAC;IACrD,CAAC;IAED,4FAA4F;IACpF,iBAAiB;QACvB,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,QAAQ,KAAK,gBAAgB,EAAE;YACzD,IAAI,CAAC,QAAQ,GAAG,qBAAqB,CAAC;SACvC;IACH,CAAC;IAED,IAAI,QAAQ;QACV,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,KAAK,CAAC;YACnC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,8BAA8B;YAC5C,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,4BAA4B,CAAC;QAC7C,OAAO,QAAQ,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC;IACtD,CAAC;IAED,WAAW,CAAC,KAA2B;QACrC,IAAI,IAAI,CAAC,QAAQ,KAAK,KAAK;YAAE,OAAO;QACpC,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QACtB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAChC,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;;wHAtCU,2BAA2B;4GAA3B,2BAA2B,2SA3D5B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT;2FAKU,2BAA2B;kBA7DvC,SAAS;+BACE,0BAA0B,YAC1B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT,QAEK,EAAE,KAAK,EAAE,aAAa,EAAE,mBACb,uBAAuB,CAAC,MAAM;wGAGtC,UAAU;sBAAlB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,MAAM;sBAAd,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBAEI,cAAc;sBAAvB,MAAM","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  DEFAULT_MODULAR_LABELS,\n  FolderDeleteStrategy,\n  ModularLabels,\n} from '../modular-table-template.models';\n\n/**\n * Body of the \"Delete folder\" dialog — a radio-card picker for what happens to\n * the folder's contained test cases.\n *\n * Designed to be usable two ways:\n *\n *  1. **From within this library** via `DialogService.open(...)`. The outer\n *     `cqa-dialog` supplies the title/Cancel/Delete-folder buttons; the Delete\n *     button reads `strategy` via `getComponentInstance()`.\n *\n *  2. **From the host/UI layer directly** — hosts render this inside their own\n *     modal and observe `(strategyChange)`.\n */\n@Component({\n  selector: 'cqa-delete-folder-dialog',\n  template: `\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n      <p class=\"cqa-text-sm cqa-text-neutral-800 cqa-m-0\">{{ bodyText }}</p>\n\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n        <!-- Option 1: Move to parent folder (hidden when folder has no parent) -->\n        <button\n          *ngIf=\"hasParent\"\n          type=\"button\"\n          class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full\"\n          [ngClass]=\"strategy === 'MOVE_TO_PARENT' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'\"\n          (click)=\"setStrategy('MOVE_TO_PARENT')\"\n        >\n          <ng-container *ngTemplateOutlet=\"radio; context: { checked: strategy === 'MOVE_TO_PARENT' }\"></ng-container>\n          <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n            <span class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n              {{ labels.deleteFolderDialogMoveToParentTitle }}\n            </span>\n            <span class=\"cqa-text-sm cqa-text-neutral-500\">\n              {{ labels.deleteFolderDialogMoveToParentDescription }}\n            </span>\n          </div>\n        </button>\n\n        <!-- Option 2: Move to Unorganized -->\n        <button\n          type=\"button\"\n          class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full\"\n          [ngClass]=\"strategy === 'MOVE_TO_UNORGANISED' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'\"\n          (click)=\"setStrategy('MOVE_TO_UNORGANISED')\"\n        >\n          <ng-container *ngTemplateOutlet=\"radio; context: { checked: strategy === 'MOVE_TO_UNORGANISED' }\"></ng-container>\n          <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n            <span class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n              {{ labels.deleteFolderDialogMoveToUnorganizedTitle }}\n            </span>\n            <span class=\"cqa-text-sm cqa-text-neutral-500\">\n              {{ labels.deleteFolderDialogMoveToUnorganizedDescription }}\n            </span>\n          </div>\n        </button>\n      </div>\n    </div>\n\n    <!-- Reusable radio bullet -->\n    <ng-template #radio let-checked=\"checked\">\n      <span\n        class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-rounded-full cqa-border-solid cqa-flex-shrink-0 cqa-mt-0.5\"\n        [ngClass]=\"checked ? 'cqa-border-indigo-600' : 'cqa-border-neutral-400'\"\n        [style.borderWidth]=\"'2px'\"\n      >\n        <span *ngIf=\"checked\" class=\"cqa-w-2.5 cqa-h-2.5 cqa-rounded-full cqa-bg-indigo-600\"></span>\n      </span>\n    </ng-template>\n  `,\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DeleteFolderDialogComponent implements OnInit, OnChanges {\n  @Input() folderName: string = '';\n  @Input() testCount: number = 0;\n  @Input() hasParent: boolean = false;\n  @Input() labels: ModularLabels = { ...DEFAULT_MODULAR_LABELS };\n  @Input() strategy: FolderDeleteStrategy = 'MOVE_TO_PARENT';\n\n  @Output() strategyChange = new EventEmitter<FolderDeleteStrategy>();\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.normaliseStrategy();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['hasParent']) this.normaliseStrategy();\n  }\n\n  /** If the folder has no parent, the \"move to parent\" option isn't available — fall back. */\n  private normaliseStrategy(): void {\n    if (!this.hasParent && this.strategy === 'MOVE_TO_PARENT') {\n      this.strategy = 'MOVE_TO_UNORGANISED';\n    }\n  }\n\n  get bodyText(): string {\n    const template = this.testCount === 1\n      ? this.labels.deleteFolderDialogBodySingular\n      : this.labels.deleteFolderDialogBodyPlural;\n    return template.replace('{n}', '' + this.testCount);\n  }\n\n  setStrategy(value: FolderDeleteStrategy): void {\n    if (this.strategy === value) return;\n    this.strategy = value;\n    this.strategyChange.emit(value);\n    this.cdr.markForCheck();\n  }\n}\n"]}
195
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"delete-folder-dialog.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/templates/modular-table-template/dialogs/delete-folder-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  DEFAULT_MODULAR_LABELS,\n  FolderDeleteStrategy,\n  ModularLabels,\n} from '../modular-table-template.models';\n\n/**\n * Body of the \"Delete folder\" dialog — a radio-card picker for what happens to\n * the folder's contained test cases.\n *\n * Designed to be usable two ways:\n *\n *  1. **From within this library** via `DialogService.open(...)`. The outer\n *     `cqa-dialog` supplies the title/Cancel/Delete-folder buttons; the Delete\n *     button reads `strategy` via `getComponentInstance()`.\n *\n *  2. **From the host/UI layer directly** — hosts render this inside their own\n *     modal and observe `(strategyChange)`.\n */\n@Component({\n  selector: 'cqa-delete-folder-dialog',\n  template: `\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n      <p class=\"cqa-text-sm cqa-text-neutral-800 cqa-m-0\">{{ bodyText }}</p>\n\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n        <!-- Option 1: Move to parent folder (hidden when folder has no parent) -->\n        <button\n          *ngIf=\"hasParent\"\n          type=\"button\"\n          class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full\"\n          [ngClass]=\"strategy === 'MOVE_TO_PARENT' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'\"\n          (click)=\"setStrategy('MOVE_TO_PARENT')\"\n        >\n          <ng-container *ngTemplateOutlet=\"radio; context: { checked: strategy === 'MOVE_TO_PARENT' }\"></ng-container>\n          <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n            <span class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n              {{ labels.deleteFolderDialogMoveToParentTitle }}\n            </span>\n            <span class=\"cqa-text-sm cqa-text-neutral-500\">\n              {{ labels.deleteFolderDialogMoveToParentDescription }}\n            </span>\n          </div>\n        </button>\n\n        <!-- Option 2: Move to Unorganized -->\n        <button\n          type=\"button\"\n          class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full\"\n          [ngClass]=\"strategy === 'MOVE_TO_UNORGANISED' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'\"\n          (click)=\"setStrategy('MOVE_TO_UNORGANISED')\"\n        >\n          <ng-container *ngTemplateOutlet=\"radio; context: { checked: strategy === 'MOVE_TO_UNORGANISED' }\"></ng-container>\n          <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n            <span class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n              {{ labels.deleteFolderDialogMoveToUnorganizedTitle }}\n            </span>\n            <span class=\"cqa-text-sm cqa-text-neutral-500\">\n              {{ labels.deleteFolderDialogMoveToUnorganizedDescription }}\n            </span>\n          </div>\n        </button>\n      </div>\n    </div>\n\n    <!-- Reusable radio bullet -->\n    <ng-template #radio let-checked=\"checked\">\n      <span\n        class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-rounded-full cqa-border-solid cqa-flex-shrink-0 cqa-mt-0.5\"\n        [ngClass]=\"checked ? 'cqa-border-indigo-600' : 'cqa-border-neutral-400'\"\n        [style.borderWidth]=\"'2px'\"\n      >\n        <span *ngIf=\"checked\" class=\"cqa-w-2.5 cqa-h-2.5 cqa-rounded-full cqa-bg-indigo-600\"></span>\n      </span>\n    </ng-template>\n  `,\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DeleteFolderDialogComponent implements OnInit, OnChanges {\n  @Input() folderName: string = '';\n  @Input() testCount: number = 0;\n  @Input() stepGroupCount: number = 0;\n  @Input() hasParent: boolean = false;\n  @Input() labels: ModularLabels = { ...DEFAULT_MODULAR_LABELS };\n  @Input() strategy: FolderDeleteStrategy = 'MOVE_TO_PARENT';\n\n  @Output() strategyChange = new EventEmitter<FolderDeleteStrategy>();\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.normaliseStrategy();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['hasParent']) this.normaliseStrategy();\n  }\n\n  /** If the folder has no parent, the \"move to parent\" option isn't available — fall back. */\n  private normaliseStrategy(): void {\n    if (!this.hasParent && this.strategy === 'MOVE_TO_PARENT') {\n      this.strategy = 'MOVE_TO_UNORGANISED';\n    }\n  }\n\n  get bodyText(): string {\n    const parts: string[] = [];\n    if (this.testCount > 0) {\n      const tpl = this.testCount === 1\n        ? this.labels.deleteFolderDialogBodyTestCaseSingular\n        : this.labels.deleteFolderDialogBodyTestCasePlural;\n      parts.push(tpl.replace('{n}', '' + this.testCount));\n    }\n    if (this.stepGroupCount > 0) {\n      const tpl = this.stepGroupCount === 1\n        ? this.labels.deleteFolderDialogBodyStepGroupSingular\n        : this.labels.deleteFolderDialogBodyStepGroupPlural;\n      parts.push(tpl.replace('{n}', '' + this.stepGroupCount));\n    }\n    const items = parts.join(` ${this.labels.deleteFolderDialogBodyJoin} `);\n    return this.labels.deleteFolderDialogBodyTemplate.replace('{items}', items);\n  }\n\n  setStrategy(value: FolderDeleteStrategy): void {\n    if (this.strategy === value) return;\n    this.strategy = value;\n    this.strategyChange.emit(value);\n    this.cdr.markForCheck();\n  }\n}\n"]}
@@ -62,12 +62,16 @@ export const DEFAULT_MODULAR_LABELS = {
62
62
  folderMenuDuplicate: 'Duplicate folder',
63
63
  folderMenuDelete: 'Delete folder',
64
64
  deleteFolderDialogTitle: 'Delete folder',
65
- deleteFolderDialogBodySingular: 'This folder contains 1 test case. What should happen to them?',
66
- deleteFolderDialogBodyPlural: 'This folder contains {n} test cases. What should happen to them?',
67
- deleteFolderDialogMoveToParentTitle: 'Move test cases to parent folder',
68
- deleteFolderDialogMoveToParentDescription: 'Test cases move one level up — nothing is lost',
69
- deleteFolderDialogMoveToUnorganizedTitle: 'Move test cases to Unorganized',
70
- deleteFolderDialogMoveToUnorganizedDescription: 'Test cases return to the Unorganized section',
65
+ deleteFolderDialogBodyTemplate: 'This folder contains {items}. What should happen to them?',
66
+ deleteFolderDialogBodyTestCaseSingular: '1 test case',
67
+ deleteFolderDialogBodyTestCasePlural: '{n} test cases',
68
+ deleteFolderDialogBodyStepGroupSingular: '1 step group',
69
+ deleteFolderDialogBodyStepGroupPlural: '{n} step groups',
70
+ deleteFolderDialogBodyJoin: 'and',
71
+ deleteFolderDialogMoveToParentTitle: 'Move to parent folder',
72
+ deleteFolderDialogMoveToParentDescription: 'Move them one level up — nothing is lost',
73
+ deleteFolderDialogMoveToUnorganizedTitle: 'Move to Unorganized',
74
+ deleteFolderDialogMoveToUnorganizedDescription: 'Move them to the Unorganized section',
71
75
  deleteFolderDialogCancel: 'Cancel',
72
76
  deleteFolderDialogConfirm: 'Delete folder',
73
77
  emptyNoTestsInFolderTitle: 'No Test Case Found',
@@ -89,4 +93,4 @@ export const DEFAULT_REORDER_LABELS = {
89
93
  export const ROW_DRAG_MIME = 'application/x-cqa-test-ids';
90
94
  /** Custom MIME for folder-row drags in the sidebar. Payload is a JSON-stringified folder id. */
91
95
  export const FOLDER_DRAG_MIME = 'application/x-cqa-folder-id';
92
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modular-table-template.models.js","sourceRoot":"","sources":["../../../../../../src/lib/templates/modular-table-template/modular-table-template.models.ts"],"names":[],"mappings":"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","sourcesContent":["export interface FolderNode {\n  id: number;\n  name: string;\n  /** Direct test-case count — cases with `folderId === this.id`. Rendered as the row badge. */\n  count?: number;\n  /** Total test-case count including all descendants. Rendered as a tooltip on the folder row. */\n  totalCount?: number;\n  color?: string;\n  children?: FolderNode[];\n  /** Backend hint: true when this node has at least one direct child. Drives\n   *  the sidebar chevron under lazy loading, so the affordance shows even\n   *  before `children` has been fetched. */\n  hasChildren?: boolean;\n  /** Backend-reported total number of direct children. The sidebar shows a\n   *  \"Load more\" row when `(children?.length ?? 0) < totalChildren`. */\n  totalChildren?: number;\n  /** Last 0-based page already fetched into `children`. `undefined` means the\n   *  subtree has not been loaded yet — the host should fetch on first expand. */\n  loadedPages?: number;\n  /** True while a children-page fetch is in flight, so the sidebar can render\n   *  a spinner row and the host can suppress duplicate requests. */\n  childrenLoading?: boolean;\n}\n\nexport type FolderIdAccessor = (row: any) => number | null | undefined;\n\nexport interface ModularConfig {\n  showSidebar: boolean;\n  showBreadcrumb: boolean;\n  showFolderGrid: boolean;\n  showUnorganizedSection: boolean;\n  showSubfolderSection: boolean;\n  showCounts: boolean;\n  allowCreateFolder: boolean;\n  allowRenameFolder: boolean;\n  allowDeleteFolder: boolean;\n  allowNestedFolders: boolean;\n  allowTestDragDrop: boolean;\n  allowBulkSelection: boolean;\n  /**\n   * When true, the table renders a folder column (icon + folder name) and lists it in the\n   * settings dropdown. Default visibility on first init follows `viewMode`: shown in list,\n   * hidden in modular. Once the user toggles it the choice persists.\n   */\n  showFolderColumn: boolean;\n}\n\nexport const DEFAULT_MODULAR_CONFIG: ModularConfig = {\n  showSidebar: true,\n  showBreadcrumb: true,\n  showFolderGrid: true,\n  showUnorganizedSection: true,\n  showSubfolderSection: true,\n  showCounts: true,\n  allowCreateFolder: true,\n  allowRenameFolder: true,\n  allowDeleteFolder: true,\n  allowNestedFolders: true,\n  allowTestDragDrop: true,\n  allowBulkSelection: true,\n  showFolderColumn: true,\n};\n\n/**\n * Reserved fieldId for the auto-injected folder column. Hosts should avoid\n * defining a column with this id of their own.\n */\nexport const FOLDER_COLUMN_FIELD_ID = 'folder';\n\nexport interface ModularLabels {\n  folders: string;\n  organized: string;\n  unorganized: string;\n  newFolder: string;\n  searchFoldersPlaceholder: string;\n  allFolders: string;\n  subfoldersIn: string;\n  moveToFolder: string;\n  addTag: string;\n  delete: string;\n  selectAll: string;\n  testsCountSingular: string;\n  testsCountPlural: string;\n  foldersCountSingular: string;\n  foldersCountPlural: string;\n  subfolderCountSingular: string;\n  subfolderCountPlural: string;\n  oneSubfolder: string;\n  subfoldersCount: string;\n  clearFilter: string;\n  moveDialogTitle: string;\n  moveDialogCancel: string;\n  moveDialogConfirm: string;\n  moveDialogRoot: string;\n  moveDialogDescriptionSingular: string;\n  moveDialogDescriptionPlural: string;\n\n  // New Folder dialog\n  newFolderDialogTitle: string;\n  newFolderDialogDescription: string;\n  newFolderDialogNameLabel: string;\n  newFolderDialogNamePlaceholder: string;\n  newFolderDialogParentLabel: string;\n  newFolderDialogParentNone: string;\n  newFolderDialogColorLabel: string;\n  newFolderDialogCancel: string;\n  newFolderDialogConfirm: string;\n  newFolderDialogNameRequiredError: string;\n  newFolderDialogNameMaxLengthError: string;\n\n  // Folder row context menu (right-click / ellipsis)\n  folderMenuCreateSubfolder: string;\n  folderMenuRename: string;\n  folderMenuMove: string;\n  folderMenuDuplicate: string;\n  folderMenuDelete: string;\n\n  // Delete Folder dialog\n  deleteFolderDialogTitle: string;\n  deleteFolderDialogBodySingular: string;\n  deleteFolderDialogBodyPlural: string;\n  deleteFolderDialogMoveToParentTitle: string;\n  deleteFolderDialogMoveToParentDescription: string;\n  deleteFolderDialogMoveToUnorganizedTitle: string;\n  deleteFolderDialogMoveToUnorganizedDescription: string;\n  deleteFolderDialogCancel: string;\n  deleteFolderDialogConfirm: string;\n\n  // Auto-derived empty states (shown when pagedRows is empty and host did not set isEmptyState)\n  emptyNoTestsInFolderTitle: string;\n  emptyNoTestsInFolderDescription: string;\n\n  /** Header label for the auto-injected folder column. */\n  folderColumnLabel: string;\n  /** Cell label for rows with no folderId (i.e. the \"Unorganized\" bucket). */\n  unorganizedRowLabel: string;\n  /** Shown in the folder sidebar when the search input filters out every folder. */\n  noFoldersFound: string;\n  /** Sidebar folder-row hover tooltip count line. `{n}` is the descendant total\n   *  (FolderNode.totalCount). The host swaps the noun for \"step group\" when in\n   *  step-group mode by overriding these two keys. The full tooltip is composed\n   *  as `${node.name}\\n${tpl.replace('{n}', total)}`. */\n  folderTooltipTotalSingular: string;\n  folderTooltipTotalPlural: string;\n}\n\nexport const DEFAULT_MODULAR_LABELS: ModularLabels = {\n  folders: 'Folders',\n  organized: 'Organized',\n  unorganized: 'Unorganized',\n  newFolder: 'New folder',\n  searchFoldersPlaceholder: 'Search folders...',\n  allFolders: 'All folders',\n  subfoldersIn: 'Subfolders in',\n  moveToFolder: 'Move to folder',\n  addTag: 'Add Tag',\n  delete: 'Delete',\n  selectAll: 'Select all',\n  testsCountSingular: '1 test',\n  testsCountPlural: '{n} tests',\n  foldersCountSingular: '1 folder',\n  foldersCountPlural: '{n} folders',\n  subfolderCountSingular: '1 subfolder',\n  subfolderCountPlural: '{n} subfolders',\n  oneSubfolder: '1 subfolder',\n  subfoldersCount: '{n} subfolders',\n  clearFilter: 'Clear filter',\n  moveDialogTitle: 'Move to Folder',\n  moveDialogCancel: 'Cancel',\n  moveDialogConfirm: 'Move here',\n  moveDialogRoot: 'Unorganized',\n  moveDialogDescriptionSingular: 'Moving 1 test case — select destination',\n  moveDialogDescriptionPlural: 'Moving {n} test cases — select destination',\n\n  newFolderDialogTitle: 'New Folder',\n  newFolderDialogDescription: 'Rename any time by double-clicking in the tree',\n  newFolderDialogNameLabel: 'Folder name',\n  newFolderDialogNamePlaceholder: 'e.g. Checkout Flow, User Auth, Payment Tests',\n  newFolderDialogParentLabel: 'Parent folder (optional)',\n  newFolderDialogParentNone: 'None',\n  newFolderDialogColorLabel: 'Folder color (optional)',\n  newFolderDialogCancel: 'Cancel',\n  newFolderDialogConfirm: 'Create folder',\n  newFolderDialogNameRequiredError: 'Folder name is required',\n  newFolderDialogNameMaxLengthError: 'Folder name cannot exceed 20 characters',\n\n  folderMenuCreateSubfolder: 'Create subfolder',\n  folderMenuRename: 'Rename',\n  folderMenuMove: 'Move folder',\n  folderMenuDuplicate: 'Duplicate folder',\n  folderMenuDelete: 'Delete folder',\n\n  deleteFolderDialogTitle: 'Delete folder',\n  deleteFolderDialogBodySingular: 'This folder contains 1 test case. What should happen to them?',\n  deleteFolderDialogBodyPlural: 'This folder contains {n} test cases. What should happen to them?',\n  deleteFolderDialogMoveToParentTitle: 'Move test cases to parent folder',\n  deleteFolderDialogMoveToParentDescription: 'Test cases move one level up — nothing is lost',\n  deleteFolderDialogMoveToUnorganizedTitle: 'Move test cases to Unorganized',\n  deleteFolderDialogMoveToUnorganizedDescription: 'Test cases return to the Unorganized section',\n  deleteFolderDialogCancel: 'Cancel',\n  deleteFolderDialogConfirm: 'Delete folder',\n\n  emptyNoTestsInFolderTitle: 'No Test Case Found',\n  emptyNoTestsInFolderDescription: 'This folder does not contain any test cases yet.',\n\n  folderColumnLabel: 'Folder',\n  unorganizedRowLabel: 'Unorganized',\n  noFoldersFound: 'No Folders Found',\n  folderTooltipTotalSingular: '1 total test case including subfolders',\n  folderTooltipTotalPlural: '{n} total test cases including subfolders',\n};\n\n/**\n * Emitted after a successful inline folder rename. Carries the full renamed node\n * (with the new name already applied) plus its parent id, so hosts can issue\n * a complete update request (e.g. `PUT /test_case_folders/{id}`) without a\n * secondary lookup against their cached tree.\n */\nexport interface FolderRenamedPayload {\n  folder: FolderNode;\n  parentId: number | null;\n  /** Previous name, useful for undo messages / audit logs. */\n  previousName: string;\n}\n\nexport interface FolderCreatedPayload {\n  parentId: number | null;\n  name: string;\n  color?: string;\n}\n\n/**\n * Delete strategies match the backend `strategy` query param values on\n * `DELETE /test_case_folders/{id}?strategy=…`. Only the two move strategies\n * are covered by the library's internal dialog; `DELETE_IF_EMPTY` is the\n * host's responsibility (precheck `count === 0 && totalCount === 0`).\n */\nexport type FolderDeleteStrategy = 'MOVE_TO_PARENT' | 'MOVE_TO_UNORGANISED';\n\nexport interface FolderDeleteRequestedPayload {\n  id: number;\n  name: string;\n  testCount: number;\n  hasParent: boolean;\n}\n\nexport interface FolderMoveRequestedPayload {\n  id: number;\n  name: string;\n  hasParent: boolean;\n}\n\nexport interface FolderMovedPayload {\n  id: number;\n  newParentId: number | null;\n}\n\nexport interface FolderDuplicateRequestedPayload {\n  id: number;\n  name: string;\n}\n\nexport interface FolderDeletedPayload {\n  id: number;\n  strategy?: FolderDeleteStrategy;\n}\n\nexport interface TestsMovedPayload {\n  testIds: Array<string | number>;\n  targetFolderId: number | null;\n}\n\nexport interface BulkActionInvokedPayload {\n  id: string;\n  selected: any[];\n}\n\nexport interface ReorderLabels {\n  reorderButton: string;\n  cancelButton: string;\n  doneButton: string;\n  savingButton: string;\n  bannerTitle: string;\n  bannerDescription: string;\n}\n\nexport const DEFAULT_REORDER_LABELS: ReorderLabels = {\n  reorderButton: 'Reorder',\n  cancelButton: 'Cancel',\n  doneButton: 'Done',\n  savingButton: 'Saving...',\n  bannerTitle: 'Reorder mode',\n  bannerDescription: 'Drag rows using the handle to rearrange.',\n};\n\nexport interface ReorderSavePayload {\n  orderedItems: any[];\n}\n\nexport const ROW_DRAG_MIME = 'application/x-cqa-test-ids';\n\n/** Custom MIME for folder-row drags in the sidebar. Payload is a JSON-stringified folder id. */\nexport const FOLDER_DRAG_MIME = 'application/x-cqa-folder-id';\n"]}
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modular-table-template.models.js","sourceRoot":"","sources":["../../../../../../src/lib/templates/modular-table-template/modular-table-template.models.ts"],"names":[],"mappings":"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","sourcesContent":["export interface FolderNode {\n  id: number;\n  name: string;\n  /** Direct test-case count — cases with `folderId === this.id`. Rendered as the row badge. */\n  count?: number;\n  /** Total test-case count including all descendants. Rendered as a tooltip on the folder row. */\n  totalCount?: number;\n  color?: string;\n  children?: FolderNode[];\n  /** Backend hint: true when this node has at least one direct child. Drives\n   *  the sidebar chevron under lazy loading, so the affordance shows even\n   *  before `children` has been fetched. */\n  hasChildren?: boolean;\n  /** Backend-reported total number of direct children. The sidebar shows a\n   *  \"Load more\" row when `(children?.length ?? 0) < totalChildren`. */\n  totalChildren?: number;\n  /** Last 0-based page already fetched into `children`. `undefined` means the\n   *  subtree has not been loaded yet — the host should fetch on first expand. */\n  loadedPages?: number;\n  /** True while a children-page fetch is in flight, so the sidebar can render\n   *  a spinner row and the host can suppress duplicate requests. */\n  childrenLoading?: boolean;\n}\n\nexport type FolderIdAccessor = (row: any) => number | null | undefined;\n\nexport interface ModularConfig {\n  showSidebar: boolean;\n  showBreadcrumb: boolean;\n  showFolderGrid: boolean;\n  showUnorganizedSection: boolean;\n  showSubfolderSection: boolean;\n  showCounts: boolean;\n  allowCreateFolder: boolean;\n  allowRenameFolder: boolean;\n  allowDeleteFolder: boolean;\n  allowNestedFolders: boolean;\n  allowTestDragDrop: boolean;\n  allowBulkSelection: boolean;\n  /**\n   * When true, the table renders a folder column (icon + folder name) and lists it in the\n   * settings dropdown. Default visibility on first init follows `viewMode`: shown in list,\n   * hidden in modular. Once the user toggles it the choice persists.\n   */\n  showFolderColumn: boolean;\n}\n\nexport const DEFAULT_MODULAR_CONFIG: ModularConfig = {\n  showSidebar: true,\n  showBreadcrumb: true,\n  showFolderGrid: true,\n  showUnorganizedSection: true,\n  showSubfolderSection: true,\n  showCounts: true,\n  allowCreateFolder: true,\n  allowRenameFolder: true,\n  allowDeleteFolder: true,\n  allowNestedFolders: true,\n  allowTestDragDrop: true,\n  allowBulkSelection: true,\n  showFolderColumn: true,\n};\n\n/**\n * Reserved fieldId for the auto-injected folder column. Hosts should avoid\n * defining a column with this id of their own.\n */\nexport const FOLDER_COLUMN_FIELD_ID = 'folder';\n\nexport interface ModularLabels {\n  folders: string;\n  organized: string;\n  unorganized: string;\n  newFolder: string;\n  searchFoldersPlaceholder: string;\n  allFolders: string;\n  subfoldersIn: string;\n  moveToFolder: string;\n  addTag: string;\n  delete: string;\n  selectAll: string;\n  testsCountSingular: string;\n  testsCountPlural: string;\n  foldersCountSingular: string;\n  foldersCountPlural: string;\n  subfolderCountSingular: string;\n  subfolderCountPlural: string;\n  oneSubfolder: string;\n  subfoldersCount: string;\n  clearFilter: string;\n  moveDialogTitle: string;\n  moveDialogCancel: string;\n  moveDialogConfirm: string;\n  moveDialogRoot: string;\n  moveDialogDescriptionSingular: string;\n  moveDialogDescriptionPlural: string;\n\n  // New Folder dialog\n  newFolderDialogTitle: string;\n  newFolderDialogDescription: string;\n  newFolderDialogNameLabel: string;\n  newFolderDialogNamePlaceholder: string;\n  newFolderDialogParentLabel: string;\n  newFolderDialogParentNone: string;\n  newFolderDialogColorLabel: string;\n  newFolderDialogCancel: string;\n  newFolderDialogConfirm: string;\n  newFolderDialogNameRequiredError: string;\n  newFolderDialogNameMaxLengthError: string;\n\n  // Folder row context menu (right-click / ellipsis)\n  folderMenuCreateSubfolder: string;\n  folderMenuRename: string;\n  folderMenuMove: string;\n  folderMenuDuplicate: string;\n  folderMenuDelete: string;\n\n  // Delete Folder dialog\n  deleteFolderDialogTitle: string;\n  deleteFolderDialogBodyTemplate: string;\n  deleteFolderDialogBodyTestCaseSingular: string;\n  deleteFolderDialogBodyTestCasePlural: string;\n  deleteFolderDialogBodyStepGroupSingular: string;\n  deleteFolderDialogBodyStepGroupPlural: string;\n  deleteFolderDialogBodyJoin: string;\n  deleteFolderDialogMoveToParentTitle: string;\n  deleteFolderDialogMoveToParentDescription: string;\n  deleteFolderDialogMoveToUnorganizedTitle: string;\n  deleteFolderDialogMoveToUnorganizedDescription: string;\n  deleteFolderDialogCancel: string;\n  deleteFolderDialogConfirm: string;\n\n  // Auto-derived empty states (shown when pagedRows is empty and host did not set isEmptyState)\n  emptyNoTestsInFolderTitle: string;\n  emptyNoTestsInFolderDescription: string;\n\n  /** Header label for the auto-injected folder column. */\n  folderColumnLabel: string;\n  /** Cell label for rows with no folderId (i.e. the \"Unorganized\" bucket). */\n  unorganizedRowLabel: string;\n  /** Shown in the folder sidebar when the search input filters out every folder. */\n  noFoldersFound: string;\n  /** Sidebar folder-row hover tooltip count line. `{n}` is the descendant total\n   *  (FolderNode.totalCount). The host swaps the noun for \"step group\" when in\n   *  step-group mode by overriding these two keys. The full tooltip is composed\n   *  as `${node.name}\\n${tpl.replace('{n}', total)}`. */\n  folderTooltipTotalSingular: string;\n  folderTooltipTotalPlural: string;\n}\n\nexport const DEFAULT_MODULAR_LABELS: ModularLabels = {\n  folders: 'Folders',\n  organized: 'Organized',\n  unorganized: 'Unorganized',\n  newFolder: 'New folder',\n  searchFoldersPlaceholder: 'Search folders...',\n  allFolders: 'All folders',\n  subfoldersIn: 'Subfolders in',\n  moveToFolder: 'Move to folder',\n  addTag: 'Add Tag',\n  delete: 'Delete',\n  selectAll: 'Select all',\n  testsCountSingular: '1 test',\n  testsCountPlural: '{n} tests',\n  foldersCountSingular: '1 folder',\n  foldersCountPlural: '{n} folders',\n  subfolderCountSingular: '1 subfolder',\n  subfolderCountPlural: '{n} subfolders',\n  oneSubfolder: '1 subfolder',\n  subfoldersCount: '{n} subfolders',\n  clearFilter: 'Clear filter',\n  moveDialogTitle: 'Move to Folder',\n  moveDialogCancel: 'Cancel',\n  moveDialogConfirm: 'Move here',\n  moveDialogRoot: 'Unorganized',\n  moveDialogDescriptionSingular: 'Moving 1 test case — select destination',\n  moveDialogDescriptionPlural: 'Moving {n} test cases — select destination',\n\n  newFolderDialogTitle: 'New Folder',\n  newFolderDialogDescription: 'Rename any time by double-clicking in the tree',\n  newFolderDialogNameLabel: 'Folder name',\n  newFolderDialogNamePlaceholder: 'e.g. Checkout Flow, User Auth, Payment Tests',\n  newFolderDialogParentLabel: 'Parent folder (optional)',\n  newFolderDialogParentNone: 'None',\n  newFolderDialogColorLabel: 'Folder color (optional)',\n  newFolderDialogCancel: 'Cancel',\n  newFolderDialogConfirm: 'Create folder',\n  newFolderDialogNameRequiredError: 'Folder name is required',\n  newFolderDialogNameMaxLengthError: 'Folder name cannot exceed 20 characters',\n\n  folderMenuCreateSubfolder: 'Create subfolder',\n  folderMenuRename: 'Rename',\n  folderMenuMove: 'Move folder',\n  folderMenuDuplicate: 'Duplicate folder',\n  folderMenuDelete: 'Delete folder',\n\n  deleteFolderDialogTitle: 'Delete folder',\n  deleteFolderDialogBodyTemplate: 'This folder contains {items}. What should happen to them?',\n  deleteFolderDialogBodyTestCaseSingular: '1 test case',\n  deleteFolderDialogBodyTestCasePlural: '{n} test cases',\n  deleteFolderDialogBodyStepGroupSingular: '1 step group',\n  deleteFolderDialogBodyStepGroupPlural: '{n} step groups',\n  deleteFolderDialogBodyJoin: 'and',\n  deleteFolderDialogMoveToParentTitle: 'Move to parent folder',\n  deleteFolderDialogMoveToParentDescription: 'Move them one level up — nothing is lost',\n  deleteFolderDialogMoveToUnorganizedTitle: 'Move to Unorganized',\n  deleteFolderDialogMoveToUnorganizedDescription: 'Move them to the Unorganized section',\n  deleteFolderDialogCancel: 'Cancel',\n  deleteFolderDialogConfirm: 'Delete folder',\n\n  emptyNoTestsInFolderTitle: 'No Test Case Found',\n  emptyNoTestsInFolderDescription: 'This folder does not contain any test cases yet.',\n\n  folderColumnLabel: 'Folder',\n  unorganizedRowLabel: 'Unorganized',\n  noFoldersFound: 'No Folders Found',\n  folderTooltipTotalSingular: '1 total test case including subfolders',\n  folderTooltipTotalPlural: '{n} total test cases including subfolders',\n};\n\n/**\n * Emitted after a successful inline folder rename. Carries the full renamed node\n * (with the new name already applied) plus its parent id, so hosts can issue\n * a complete update request (e.g. `PUT /test_case_folders/{id}`) without a\n * secondary lookup against their cached tree.\n */\nexport interface FolderRenamedPayload {\n  folder: FolderNode;\n  parentId: number | null;\n  /** Previous name, useful for undo messages / audit logs. */\n  previousName: string;\n}\n\nexport interface FolderCreatedPayload {\n  parentId: number | null;\n  name: string;\n  color?: string;\n}\n\n/**\n * Delete strategies match the backend `strategy` query param values on\n * `DELETE /test_case_folders/{id}?strategy=…`. Only the two move strategies\n * are covered by the library's internal dialog; `DELETE_IF_EMPTY` is the\n * host's responsibility (precheck `count === 0 && totalCount === 0`).\n */\nexport type FolderDeleteStrategy = 'MOVE_TO_PARENT' | 'MOVE_TO_UNORGANISED';\n\nexport interface FolderDeleteRequestedPayload {\n  id: number;\n  name: string;\n  testCount: number;\n  hasParent: boolean;\n}\n\nexport interface FolderMoveRequestedPayload {\n  id: number;\n  name: string;\n  hasParent: boolean;\n}\n\nexport interface FolderMovedPayload {\n  id: number;\n  newParentId: number | null;\n}\n\nexport interface FolderDuplicateRequestedPayload {\n  id: number;\n  name: string;\n}\n\nexport interface FolderDeletedPayload {\n  id: number;\n  strategy?: FolderDeleteStrategy;\n}\n\nexport interface TestsMovedPayload {\n  testIds: Array<string | number>;\n  targetFolderId: number | null;\n}\n\nexport interface BulkActionInvokedPayload {\n  id: string;\n  selected: any[];\n}\n\nexport interface ReorderLabels {\n  reorderButton: string;\n  cancelButton: string;\n  doneButton: string;\n  savingButton: string;\n  bannerTitle: string;\n  bannerDescription: string;\n}\n\nexport const DEFAULT_REORDER_LABELS: ReorderLabels = {\n  reorderButton: 'Reorder',\n  cancelButton: 'Cancel',\n  doneButton: 'Done',\n  savingButton: 'Saving...',\n  bannerTitle: 'Reorder mode',\n  bannerDescription: 'Drag rows using the handle to rearrange.',\n};\n\nexport interface ReorderSavePayload {\n  orderedItems: any[];\n}\n\nexport const ROW_DRAG_MIME = 'application/x-cqa-test-ids';\n\n/** Custom MIME for folder-row drags in the sidebar. Payload is a JSON-stringified folder id. */\nexport const FOLDER_DRAG_MIME = 'application/x-cqa-folder-id';\n"]}
@@ -7922,12 +7922,16 @@ const DEFAULT_MODULAR_LABELS = {
7922
7922
  folderMenuDuplicate: 'Duplicate folder',
7923
7923
  folderMenuDelete: 'Delete folder',
7924
7924
  deleteFolderDialogTitle: 'Delete folder',
7925
- deleteFolderDialogBodySingular: 'This folder contains 1 test case. What should happen to them?',
7926
- deleteFolderDialogBodyPlural: 'This folder contains {n} test cases. What should happen to them?',
7927
- deleteFolderDialogMoveToParentTitle: 'Move test cases to parent folder',
7928
- deleteFolderDialogMoveToParentDescription: 'Test cases move one level up — nothing is lost',
7929
- deleteFolderDialogMoveToUnorganizedTitle: 'Move test cases to Unorganized',
7930
- deleteFolderDialogMoveToUnorganizedDescription: 'Test cases return to the Unorganized section',
7925
+ deleteFolderDialogBodyTemplate: 'This folder contains {items}. What should happen to them?',
7926
+ deleteFolderDialogBodyTestCaseSingular: '1 test case',
7927
+ deleteFolderDialogBodyTestCasePlural: '{n} test cases',
7928
+ deleteFolderDialogBodyStepGroupSingular: '1 step group',
7929
+ deleteFolderDialogBodyStepGroupPlural: '{n} step groups',
7930
+ deleteFolderDialogBodyJoin: 'and',
7931
+ deleteFolderDialogMoveToParentTitle: 'Move to parent folder',
7932
+ deleteFolderDialogMoveToParentDescription: 'Move them one level up — nothing is lost',
7933
+ deleteFolderDialogMoveToUnorganizedTitle: 'Move to Unorganized',
7934
+ deleteFolderDialogMoveToUnorganizedDescription: 'Move them to the Unorganized section',
7931
7935
  deleteFolderDialogCancel: 'Cancel',
7932
7936
  deleteFolderDialogConfirm: 'Delete folder',
7933
7937
  emptyNoTestsInFolderTitle: 'No Test Case Found',
@@ -8600,6 +8604,7 @@ class DeleteFolderDialogComponent {
8600
8604
  this.cdr = cdr;
8601
8605
  this.folderName = '';
8602
8606
  this.testCount = 0;
8607
+ this.stepGroupCount = 0;
8603
8608
  this.hasParent = false;
8604
8609
  this.labels = Object.assign({}, DEFAULT_MODULAR_LABELS);
8605
8610
  this.strategy = 'MOVE_TO_PARENT';
@@ -8619,10 +8624,21 @@ class DeleteFolderDialogComponent {
8619
8624
  }
8620
8625
  }
8621
8626
  get bodyText() {
8622
- const template = this.testCount === 1
8623
- ? this.labels.deleteFolderDialogBodySingular
8624
- : this.labels.deleteFolderDialogBodyPlural;
8625
- return template.replace('{n}', '' + this.testCount);
8627
+ const parts = [];
8628
+ if (this.testCount > 0) {
8629
+ const tpl = this.testCount === 1
8630
+ ? this.labels.deleteFolderDialogBodyTestCaseSingular
8631
+ : this.labels.deleteFolderDialogBodyTestCasePlural;
8632
+ parts.push(tpl.replace('{n}', '' + this.testCount));
8633
+ }
8634
+ if (this.stepGroupCount > 0) {
8635
+ const tpl = this.stepGroupCount === 1
8636
+ ? this.labels.deleteFolderDialogBodyStepGroupSingular
8637
+ : this.labels.deleteFolderDialogBodyStepGroupPlural;
8638
+ parts.push(tpl.replace('{n}', '' + this.stepGroupCount));
8639
+ }
8640
+ const items = parts.join(` ${this.labels.deleteFolderDialogBodyJoin} `);
8641
+ return this.labels.deleteFolderDialogBodyTemplate.replace('{items}', items);
8626
8642
  }
8627
8643
  setStrategy(value) {
8628
8644
  if (this.strategy === value)
@@ -8633,7 +8649,7 @@ class DeleteFolderDialogComponent {
8633
8649
  }
8634
8650
  }
8635
8651
  DeleteFolderDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DeleteFolderDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
8636
- DeleteFolderDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DeleteFolderDialogComponent, selector: "cqa-delete-folder-dialog", inputs: { folderName: "folderName", testCount: "testCount", hasParent: "hasParent", labels: "labels", strategy: "strategy" }, outputs: { strategyChange: "strategyChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: `
8652
+ DeleteFolderDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DeleteFolderDialogComponent, selector: "cqa-delete-folder-dialog", inputs: { folderName: "folderName", testCount: "testCount", stepGroupCount: "stepGroupCount", hasParent: "hasParent", labels: "labels", strategy: "strategy" }, outputs: { strategyChange: "strategyChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: `
8637
8653
  <div class="cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full">
8638
8654
  <p class="cqa-text-sm cqa-text-neutral-800 cqa-m-0">{{ bodyText }}</p>
8639
8655
 
@@ -8749,6 +8765,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
8749
8765
  type: Input
8750
8766
  }], testCount: [{
8751
8767
  type: Input
8768
+ }], stepGroupCount: [{
8769
+ type: Input
8752
8770
  }], hasParent: [{
8753
8771
  type: Input
8754
8772
  }], labels: [{