@cqa-lib/cqa-ui 1.1.273 → 1.1.274

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.
@@ -15,6 +15,8 @@ export class TableActionToolbarComponent {
15
15
  this.actionClick = new EventEmitter();
16
16
  this.selectAllChange = new EventEmitter();
17
17
  this.dismiss = new EventEmitter();
18
+ /** Cooldown ms to prevent double-click from firing the same action twice */
19
+ this.lastActionEmit = null;
18
20
  }
19
21
  get hasSelection() {
20
22
  return (this.selectedItems?.length || 0) > 0;
@@ -38,6 +40,13 @@ export class TableActionToolbarComponent {
38
40
  if (this.isDisabled(action)) {
39
41
  return;
40
42
  }
43
+ const now = Date.now();
44
+ if (this.lastActionEmit &&
45
+ this.lastActionEmit.id === action.id &&
46
+ now - this.lastActionEmit.time < TableActionToolbarComponent.ACTION_COOLDOWN_MS) {
47
+ return;
48
+ }
49
+ this.lastActionEmit = { id: action.id, time: now };
41
50
  const context = { id: action.id, selected: this.selectedItems || [] };
42
51
  if (action.onClick) {
43
52
  action.onClick(context);
@@ -54,6 +63,7 @@ export class TableActionToolbarComponent {
54
63
  this.dismiss.emit();
55
64
  }
56
65
  }
66
+ TableActionToolbarComponent.ACTION_COOLDOWN_MS = 500;
57
67
  TableActionToolbarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TableActionToolbarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
58
68
  TableActionToolbarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TableActionToolbarComponent, selector: "cqa-table-action-toolbar", inputs: { selectedItems: "selectedItems", actions: "actions", showSelectAll: "showSelectAll", allSelected: "allSelected", showDismiss: "showDismiss" }, outputs: { actionClick: "actionClick", selectAllChange: "selectAllChange", dismiss: "dismiss" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-ui-root\">\n <div\n class=\"action-toolbar cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-py-[15.5px] sm:cqa-px-[34px] cqa-px-[20px] cqa-bg-primary cqa-text-white cqa-rounded-[7px] cqa-shadow-md\"\n *ngIf=\"hasSelection\">\n <div class=\"action-toolbar-left cqa-flex cqa-items-center cqa-gap-2 cqa-text-[14px] cqa-leading-[21px] cqa-font-medium\">\n <ng-container *ngIf=\"showSelectAll\">\n <label class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-cursor-pointer cqa-select-none\">\n <input type=\"checkbox\" [checked]=\"allSelected\" (change)=\"onSelectAllChange($any($event.target).checked)\"\n class=\"cqa-w-4 cqa-h-4 cqa-rounded cqa-border cqa-border-white cqa-bg-transparent cqa-accent-primary\" />\n <span>Select all</span>\n </label>\n <span class=\"cqa-border-l cqa-border-white/50 cqa-pl-2 cqa-border-solid cqa-self-stretch cqa-flex cqa-items-center\">{{ selectionLabel }}</span>\n </ng-container>\n <span *ngIf=\"!showSelectAll\">{{ selectionLabel }}</span>\n </div>\n <div class=\"action-toolbar-right cqa-flex cqa-items-center cqa-gap-[7px]\">\n <ng-container *ngFor=\"let action of visibleActions()\">\n <div [attr.aria-disabled]=\"isDisabled(action)\" [class.cqa-opacity-50]=\"isDisabled(action)\" [class.cqa-cursor-not-allowed]=\"isDisabled(action)\" (click)=\"!isDisabled(action) && onAction(action)\"\n class=\"cqa-flex cqa-items-center cqa-gap-[8.75px] cqa-py-[5px] cqa-px-[8.75px] cqa-cursor-pointer cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium hover:cqa-bg-white/10 cqa-rounded\">\n <mat-icon class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ action.icon }}</mat-icon>\n <span class=\"md:cqa-flex cqa-hidden\">{{ action.label }}</span>\n </div>\n </ng-container>\n <button *ngIf=\"showDismiss\" type=\"button\" (click)=\"onDismiss()\" class=\"cqa-p-1 cqa-rounded hover:cqa-bg-white/10 cqa-flex cqa-items-center cqa-justify-center\"\n title=\"Dismiss\" aria-label=\"Dismiss\">\n <mat-icon class=\"!cqa-w-[20px] !cqa-h-[20px] !cqa-text-[20px]\">close</mat-icon>\n </button>\n </div>\n </div>\n</div>", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
59
69
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TableActionToolbarComponent, decorators: [{
@@ -76,4 +86,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
76
86
  }], dismiss: [{
77
87
  type: Output
78
88
  }] } });
79
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-action-toolbar.component.js","sourceRoot":"","sources":["../../../../../src/lib/table-action-toolbar/table-action-toolbar.component.ts","../../../../../src/lib/table-action-toolbar/table-action-toolbar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';\n\nexport interface TableAction {\n  id: string;\n  label: string;\n  icon: string;\n  tooltip?: string;\n  show?: (context: { selected: any[] }) => boolean;\n  disabled?: (context: { selected: any[] }) => boolean;\n  onClick?: (context: { id: string; selected: any[] }) => void;\n  action?: (context: { id: string; selected: any[] }) => void; // Support 'action' as an alias for onClick\n}\n\n@Component({\n  selector: 'cqa-table-action-toolbar',\n  templateUrl: './table-action-toolbar.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root' }\n})\nexport class TableActionToolbarComponent {\n  @Input() selectedItems: any[] = [];\n  @Input() actions: TableAction[] = [];\n\n  /** When true, show a \"Select all\" checkbox next to the selection count (e.g. for test case details). */\n  @Input() showSelectAll: boolean = false;\n  /** Bound to the Select all checkbox when showSelectAll is true. */\n  @Input() allSelected: boolean = false;\n  /** When true, show a dismiss (X) button at the end of the toolbar. */\n  @Input() showDismiss: boolean = false;\n\n  @Output() actionClick = new EventEmitter<{ id: string; selected: any[] }>();\n  @Output() selectAllChange = new EventEmitter<boolean>();\n  @Output() dismiss = new EventEmitter<void>();\n\n  get hasSelection(): boolean {\n    return (this.selectedItems?.length || 0) > 0;\n  }\n\n  get isSingleSelection(): boolean {\n    return this.selectedItems?.length === 1;\n  }\n\n  get selectionLabel(): string {\n    const n = this.selectedItems?.length || 0;\n    return n === 1 ? '1 selected' : `${n} selected`;\n  }\n\n  visibleActions(): TableAction[] {\n    const ctx = { selected: this.selectedItems || [] };\n    return (this.actions || []).filter(a => (a.show ? a.show(ctx) : true));\n  }\n\n  isDisabled(action: TableAction): boolean {\n    const ctx = { selected: this.selectedItems || [] };\n    return action.disabled ? !!action.disabled(ctx) : false;\n  }\n\n  onAction(action: TableAction): void {\n    if (this.isDisabled(action)) {\n      return;\n    }\n    const context = { id: action.id, selected: this.selectedItems || [] };\n    if (action.onClick) {\n      action.onClick(context);\n    } else if (action.action) {\n      action.action(context);\n    }\n    this.actionClick.emit(context);\n  }\n\n  onSelectAllChange(checked: boolean): void {\n    this.selectAllChange.emit(checked);\n  }\n\n  onDismiss(): void {\n    this.dismiss.emit();\n  }\n}\n\n\n","<div class=\"cqa-ui-root\">\n  <div\n    class=\"action-toolbar cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-py-[15.5px] sm:cqa-px-[34px] cqa-px-[20px] cqa-bg-primary cqa-text-white cqa-rounded-[7px] cqa-shadow-md\"\n    *ngIf=\"hasSelection\">\n    <div class=\"action-toolbar-left cqa-flex cqa-items-center cqa-gap-2 cqa-text-[14px] cqa-leading-[21px] cqa-font-medium\">\n      <ng-container *ngIf=\"showSelectAll\">\n        <label class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-cursor-pointer cqa-select-none\">\n          <input type=\"checkbox\" [checked]=\"allSelected\" (change)=\"onSelectAllChange($any($event.target).checked)\"\n            class=\"cqa-w-4 cqa-h-4 cqa-rounded cqa-border cqa-border-white cqa-bg-transparent cqa-accent-primary\" />\n          <span>Select all</span>\n        </label>\n        <span class=\"cqa-border-l cqa-border-white/50 cqa-pl-2 cqa-border-solid cqa-self-stretch cqa-flex cqa-items-center\">{{ selectionLabel }}</span>\n      </ng-container>\n      <span *ngIf=\"!showSelectAll\">{{ selectionLabel }}</span>\n    </div>\n    <div class=\"action-toolbar-right cqa-flex cqa-items-center cqa-gap-[7px]\">\n      <ng-container *ngFor=\"let action of visibleActions()\">\n        <div [attr.aria-disabled]=\"isDisabled(action)\" [class.cqa-opacity-50]=\"isDisabled(action)\" [class.cqa-cursor-not-allowed]=\"isDisabled(action)\" (click)=\"!isDisabled(action) && onAction(action)\"\n          class=\"cqa-flex cqa-items-center cqa-gap-[8.75px] cqa-py-[5px] cqa-px-[8.75px] cqa-cursor-pointer cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium hover:cqa-bg-white/10 cqa-rounded\">\n          <mat-icon class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ action.icon }}</mat-icon>\n          <span class=\"md:cqa-flex cqa-hidden\">{{ action.label }}</span>\n        </div>\n      </ng-container>\n      <button *ngIf=\"showDismiss\" type=\"button\" (click)=\"onDismiss()\" class=\"cqa-p-1 cqa-rounded hover:cqa-bg-white/10 cqa-flex cqa-items-center cqa-justify-center\"\n        title=\"Dismiss\" aria-label=\"Dismiss\">\n        <mat-icon class=\"!cqa-w-[20px] !cqa-h-[20px] !cqa-text-[20px]\">close</mat-icon>\n      </button>\n    </div>\n  </div>\n</div>"]}
89
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-action-toolbar.component.js","sourceRoot":"","sources":["../../../../../src/lib/table-action-toolbar/table-action-toolbar.component.ts","../../../../../src/lib/table-action-toolbar/table-action-toolbar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';\n\nexport interface TableAction {\n  id: string;\n  label: string;\n  icon: string;\n  tooltip?: string;\n  show?: (context: { selected: any[] }) => boolean;\n  disabled?: (context: { selected: any[] }) => boolean;\n  onClick?: (context: { id: string; selected: any[] }) => void;\n  action?: (context: { id: string; selected: any[] }) => void; // Support 'action' as an alias for onClick\n}\n\n@Component({\n  selector: 'cqa-table-action-toolbar',\n  templateUrl: './table-action-toolbar.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root' }\n})\nexport class TableActionToolbarComponent {\n  @Input() selectedItems: any[] = [];\n  @Input() actions: TableAction[] = [];\n\n  /** When true, show a \"Select all\" checkbox next to the selection count (e.g. for test case details). */\n  @Input() showSelectAll: boolean = false;\n  /** Bound to the Select all checkbox when showSelectAll is true. */\n  @Input() allSelected: boolean = false;\n  /** When true, show a dismiss (X) button at the end of the toolbar. */\n  @Input() showDismiss: boolean = false;\n\n  @Output() actionClick = new EventEmitter<{ id: string; selected: any[] }>();\n  @Output() selectAllChange = new EventEmitter<boolean>();\n  @Output() dismiss = new EventEmitter<void>();\n\n  /** Cooldown ms to prevent double-click from firing the same action twice */\n  private lastActionEmit: { id: string; time: number } | null = null;\n  private static readonly ACTION_COOLDOWN_MS = 500;\n\n  get hasSelection(): boolean {\n    return (this.selectedItems?.length || 0) > 0;\n  }\n\n  get isSingleSelection(): boolean {\n    return this.selectedItems?.length === 1;\n  }\n\n  get selectionLabel(): string {\n    const n = this.selectedItems?.length || 0;\n    return n === 1 ? '1 selected' : `${n} selected`;\n  }\n\n  visibleActions(): TableAction[] {\n    const ctx = { selected: this.selectedItems || [] };\n    return (this.actions || []).filter(a => (a.show ? a.show(ctx) : true));\n  }\n\n  isDisabled(action: TableAction): boolean {\n    const ctx = { selected: this.selectedItems || [] };\n    return action.disabled ? !!action.disabled(ctx) : false;\n  }\n\n  onAction(action: TableAction): void {\n    if (this.isDisabled(action)) {\n      return;\n    }\n    const now = Date.now();\n    if (\n      this.lastActionEmit &&\n      this.lastActionEmit.id === action.id &&\n      now - this.lastActionEmit.time < TableActionToolbarComponent.ACTION_COOLDOWN_MS\n    ) {\n      return;\n    }\n    this.lastActionEmit = { id: action.id, time: now };\n    const context = { id: action.id, selected: this.selectedItems || [] };\n    if (action.onClick) {\n      action.onClick(context);\n    } else if (action.action) {\n      action.action(context);\n    }\n    this.actionClick.emit(context);\n  }\n\n  onSelectAllChange(checked: boolean): void {\n    this.selectAllChange.emit(checked);\n  }\n\n  onDismiss(): void {\n    this.dismiss.emit();\n  }\n}\n\n\n","<div class=\"cqa-ui-root\">\n  <div\n    class=\"action-toolbar cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-py-[15.5px] sm:cqa-px-[34px] cqa-px-[20px] cqa-bg-primary cqa-text-white cqa-rounded-[7px] cqa-shadow-md\"\n    *ngIf=\"hasSelection\">\n    <div class=\"action-toolbar-left cqa-flex cqa-items-center cqa-gap-2 cqa-text-[14px] cqa-leading-[21px] cqa-font-medium\">\n      <ng-container *ngIf=\"showSelectAll\">\n        <label class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-cursor-pointer cqa-select-none\">\n          <input type=\"checkbox\" [checked]=\"allSelected\" (change)=\"onSelectAllChange($any($event.target).checked)\"\n            class=\"cqa-w-4 cqa-h-4 cqa-rounded cqa-border cqa-border-white cqa-bg-transparent cqa-accent-primary\" />\n          <span>Select all</span>\n        </label>\n        <span class=\"cqa-border-l cqa-border-white/50 cqa-pl-2 cqa-border-solid cqa-self-stretch cqa-flex cqa-items-center\">{{ selectionLabel }}</span>\n      </ng-container>\n      <span *ngIf=\"!showSelectAll\">{{ selectionLabel }}</span>\n    </div>\n    <div class=\"action-toolbar-right cqa-flex cqa-items-center cqa-gap-[7px]\">\n      <ng-container *ngFor=\"let action of visibleActions()\">\n        <div [attr.aria-disabled]=\"isDisabled(action)\" [class.cqa-opacity-50]=\"isDisabled(action)\" [class.cqa-cursor-not-allowed]=\"isDisabled(action)\" (click)=\"!isDisabled(action) && onAction(action)\"\n          class=\"cqa-flex cqa-items-center cqa-gap-[8.75px] cqa-py-[5px] cqa-px-[8.75px] cqa-cursor-pointer cqa-text-[12.3px] cqa-leading-[17.5px] cqa-font-medium hover:cqa-bg-white/10 cqa-rounded\">\n          <mat-icon class=\"!cqa-w-[16px] !cqa-h-[16px] !cqa-text-[16px]\">{{ action.icon }}</mat-icon>\n          <span class=\"md:cqa-flex cqa-hidden\">{{ action.label }}</span>\n        </div>\n      </ng-container>\n      <button *ngIf=\"showDismiss\" type=\"button\" (click)=\"onDismiss()\" class=\"cqa-p-1 cqa-rounded hover:cqa-bg-white/10 cqa-flex cqa-items-center cqa-justify-center\"\n        title=\"Dismiss\" aria-label=\"Dismiss\">\n        <mat-icon class=\"!cqa-w-[20px] !cqa-h-[20px] !cqa-text-[20px]\">close</mat-icon>\n      </button>\n    </div>\n  </div>\n</div>"]}