@cqa-lib/cqa-ui 1.1.542 → 1.1.543
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/esm2020/lib/dialogs/name-prompt-modal.component.mjs +122 -0
- package/esm2020/lib/templates/modular-table-template/modular-table-template.component.mjs +24 -6
- package/esm2020/lib/ui-kit.module.mjs +6 -1
- package/esm2020/public-api.mjs +2 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +144 -6
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +144 -6
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/dialogs/name-prompt-modal.component.d.ts +50 -0
- package/lib/templates/modular-table-template/modular-table-template.component.d.ts +15 -2
- package/lib/ui-kit.module.d.ts +150 -149
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/icon";
|
|
4
|
+
import * as i2 from "../custom-input/custom-input.component";
|
|
5
|
+
import * as i3 from "../button/button.component";
|
|
6
|
+
import * as i4 from "@angular/common";
|
|
7
|
+
/**
|
|
8
|
+
* Generic single-input prompt modal. Used wherever a flow needs to ask the
|
|
9
|
+
* user for a single name (or short text) and confirm with a primary button.
|
|
10
|
+
*
|
|
11
|
+
* Examples: New Folder, Clone Suite, Rename, New Plan, etc.
|
|
12
|
+
*
|
|
13
|
+
* All copy is configurable through Inputs — see `title`, `description`,
|
|
14
|
+
* `label`, `placeholder`, `submitButtonText`, `cancelButtonText`. Defaults
|
|
15
|
+
* preserve the legacy "New Folder" wording for callers that haven't migrated.
|
|
16
|
+
*/
|
|
17
|
+
export class NamePromptModalComponent {
|
|
18
|
+
constructor(cdr) {
|
|
19
|
+
this.cdr = cdr;
|
|
20
|
+
this.isOpen = false;
|
|
21
|
+
this.title = 'New Folder';
|
|
22
|
+
this.description = 'Rename any time by double-clicking';
|
|
23
|
+
this.label = 'Folder name';
|
|
24
|
+
this.placeholder = 'e.g. Checkout Flow, User Auth, Payment Tests';
|
|
25
|
+
/** Submit (primary) button text. Lets consumers reuse this modal for
|
|
26
|
+
* "Create folder", "Clone Suite", "Rename", etc. */
|
|
27
|
+
this.submitButtonText = 'Create folder';
|
|
28
|
+
/** Cancel (secondary) button text. Most consumers can leave the default. */
|
|
29
|
+
this.cancelButtonText = 'Cancel';
|
|
30
|
+
/** When true, locks the modal: spinner on the submit button, all dismiss
|
|
31
|
+
* paths (cancel, close X, backdrop, ESC) become no-ops, and submit() bails
|
|
32
|
+
* out so a fast Enter press can't double-fire while the parent is mid-API. */
|
|
33
|
+
this.isSubmitting = false;
|
|
34
|
+
this._name = '';
|
|
35
|
+
this.nameChange = new EventEmitter();
|
|
36
|
+
this.submitted = new EventEmitter();
|
|
37
|
+
this.cancelled = new EventEmitter();
|
|
38
|
+
this.nameTouched = false;
|
|
39
|
+
}
|
|
40
|
+
set name(value) {
|
|
41
|
+
this._name = value || '';
|
|
42
|
+
this.nameTouched = false;
|
|
43
|
+
this.cdr.markForCheck();
|
|
44
|
+
}
|
|
45
|
+
get name() {
|
|
46
|
+
return this._name;
|
|
47
|
+
}
|
|
48
|
+
onNameChange(value) {
|
|
49
|
+
this._name = value;
|
|
50
|
+
this.nameTouched = true;
|
|
51
|
+
this.nameChange.emit(value);
|
|
52
|
+
this.cdr.markForCheck();
|
|
53
|
+
}
|
|
54
|
+
get nameErrors() {
|
|
55
|
+
if (!this.nameTouched)
|
|
56
|
+
return [];
|
|
57
|
+
const trimmed = (this._name || '').trim();
|
|
58
|
+
if (!trimmed)
|
|
59
|
+
return [`${this.label} is required`];
|
|
60
|
+
return [];
|
|
61
|
+
}
|
|
62
|
+
get isValid() {
|
|
63
|
+
return (this._name || '').trim().length > 0;
|
|
64
|
+
}
|
|
65
|
+
submit() {
|
|
66
|
+
if (!this.isValid || this.isSubmitting)
|
|
67
|
+
return;
|
|
68
|
+
this.submitted.emit({ name: this._name.trim() });
|
|
69
|
+
}
|
|
70
|
+
onCancel() {
|
|
71
|
+
if (this.isSubmitting)
|
|
72
|
+
return;
|
|
73
|
+
this.cancelled.emit();
|
|
74
|
+
}
|
|
75
|
+
onClose() {
|
|
76
|
+
if (this.isSubmitting)
|
|
77
|
+
return;
|
|
78
|
+
this.cancelled.emit();
|
|
79
|
+
}
|
|
80
|
+
onBackdropClick() {
|
|
81
|
+
if (this.isSubmitting)
|
|
82
|
+
return;
|
|
83
|
+
this.cancelled.emit();
|
|
84
|
+
}
|
|
85
|
+
onEscape(event) {
|
|
86
|
+
event.stopPropagation();
|
|
87
|
+
if (this.isSubmitting)
|
|
88
|
+
return;
|
|
89
|
+
this.cancelled.emit();
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
NamePromptModalComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NamePromptModalComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
93
|
+
NamePromptModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NamePromptModalComponent, selector: "cqa-name-prompt-modal", inputs: { isOpen: "isOpen", title: "title", description: "description", label: "label", placeholder: "placeholder", submitButtonText: "submitButtonText", cancelButtonText: "cancelButtonText", isSubmitting: "isSubmitting", name: "name" }, outputs: { nameChange: "nameChange", submitted: "submitted", cancelled: "cancelled" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div *ngIf=\"isOpen\"\n id=\"cqa-ui-root\"\n style=\"position: fixed; inset: 0; z-index: 1000;\">\n <div class=\"cqa-fixed cqa-inset-0 cqa-bg-black/40\"\n (click)=\"onBackdropClick()\"></div>\n\n <div class=\"cqa-fixed cqa-inset-0 cqa-flex cqa-items-center cqa-justify-center cqa-p-4 cqa-pointer-events-none\">\n <div class=\"cqa-relative cqa-w-full cqa-max-w-[480px] cqa-bg-white cqa-rounded-2xl cqa-shadow-md cqa-border cqa-border-border-default cqa-pointer-events-auto\"\n (keydown.escape)=\"onEscape($event)\"\n tabindex=\"-1\">\n <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-p-6 cqa-pb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n <h2 class=\"cqa-text-lg cqa-font-semibold cqa-text-title cqa-m-0\">{{ title }}</h2>\n <p *ngIf=\"description\"\n class=\"cqa-text-sm cqa-text-description cqa-m-0\">\n {{ description }}\n </p>\n </div>\n <button type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-h-6 cqa-w-6 cqa-rounded cqa-bg-transparent cqa-border-0 cqa-text-description hover:cqa-text-title\"\n [class.cqa-cursor-pointer]=\"!isSubmitting\"\n [class.cqa-cursor-not-allowed]=\"isSubmitting\"\n [class.cqa-opacity-50]=\"isSubmitting\"\n [disabled]=\"isSubmitting\"\n (click)=\"onClose()\"\n aria-label=\"Close\">\n <mat-icon class=\"cqa-text-[20px] cqa-leading-5 cqa-h-5 cqa-w-5\">close</mat-icon>\n </button>\n </div>\n\n <div class=\"cqa-px-6 cqa-pb-5\">\n <cqa-custom-input\n [label]=\"label\"\n [placeholder]=\"placeholder\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrors\"\n (valueChange)=\"onNameChange($event)\"\n (enterPressed)=\"submit()\">\n </cqa-custom-input>\n </div>\n\n <div class=\"cqa-border-t cqa-border-border-default cqa-px-6 cqa-py-4 cqa-grid cqa-grid-cols-2 cqa-gap-3\">\n <cqa-button variant=\"outlined\"\n [text]=\"cancelButtonText\"\n btnSize=\"lg\"\n [fullWidth]=\"true\"\n [disabled]=\"isSubmitting\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button variant=\"filled\"\n [text]=\"submitButtonText\"\n btnSize=\"lg\"\n [fullWidth]=\"true\"\n [loading]=\"isSubmitting\"\n [disabled]=\"!isValid || isSubmitting\"\n (clicked)=\"submit()\">\n </cqa-button>\n </div>\n </div>\n </div>\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "loading", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
94
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NamePromptModalComponent, decorators: [{
|
|
95
|
+
type: Component,
|
|
96
|
+
args: [{ selector: 'cqa-name-prompt-modal', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root' }, template: "<div *ngIf=\"isOpen\"\n id=\"cqa-ui-root\"\n style=\"position: fixed; inset: 0; z-index: 1000;\">\n <div class=\"cqa-fixed cqa-inset-0 cqa-bg-black/40\"\n (click)=\"onBackdropClick()\"></div>\n\n <div class=\"cqa-fixed cqa-inset-0 cqa-flex cqa-items-center cqa-justify-center cqa-p-4 cqa-pointer-events-none\">\n <div class=\"cqa-relative cqa-w-full cqa-max-w-[480px] cqa-bg-white cqa-rounded-2xl cqa-shadow-md cqa-border cqa-border-border-default cqa-pointer-events-auto\"\n (keydown.escape)=\"onEscape($event)\"\n tabindex=\"-1\">\n <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-p-6 cqa-pb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n <h2 class=\"cqa-text-lg cqa-font-semibold cqa-text-title cqa-m-0\">{{ title }}</h2>\n <p *ngIf=\"description\"\n class=\"cqa-text-sm cqa-text-description cqa-m-0\">\n {{ description }}\n </p>\n </div>\n <button type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-h-6 cqa-w-6 cqa-rounded cqa-bg-transparent cqa-border-0 cqa-text-description hover:cqa-text-title\"\n [class.cqa-cursor-pointer]=\"!isSubmitting\"\n [class.cqa-cursor-not-allowed]=\"isSubmitting\"\n [class.cqa-opacity-50]=\"isSubmitting\"\n [disabled]=\"isSubmitting\"\n (click)=\"onClose()\"\n aria-label=\"Close\">\n <mat-icon class=\"cqa-text-[20px] cqa-leading-5 cqa-h-5 cqa-w-5\">close</mat-icon>\n </button>\n </div>\n\n <div class=\"cqa-px-6 cqa-pb-5\">\n <cqa-custom-input\n [label]=\"label\"\n [placeholder]=\"placeholder\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrors\"\n (valueChange)=\"onNameChange($event)\"\n (enterPressed)=\"submit()\">\n </cqa-custom-input>\n </div>\n\n <div class=\"cqa-border-t cqa-border-border-default cqa-px-6 cqa-py-4 cqa-grid cqa-grid-cols-2 cqa-gap-3\">\n <cqa-button variant=\"outlined\"\n [text]=\"cancelButtonText\"\n btnSize=\"lg\"\n [fullWidth]=\"true\"\n [disabled]=\"isSubmitting\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button variant=\"filled\"\n [text]=\"submitButtonText\"\n btnSize=\"lg\"\n [fullWidth]=\"true\"\n [loading]=\"isSubmitting\"\n [disabled]=\"!isValid || isSubmitting\"\n (clicked)=\"submit()\">\n </cqa-button>\n </div>\n </div>\n </div>\n</div>\n" }]
|
|
97
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { isOpen: [{
|
|
98
|
+
type: Input
|
|
99
|
+
}], title: [{
|
|
100
|
+
type: Input
|
|
101
|
+
}], description: [{
|
|
102
|
+
type: Input
|
|
103
|
+
}], label: [{
|
|
104
|
+
type: Input
|
|
105
|
+
}], placeholder: [{
|
|
106
|
+
type: Input
|
|
107
|
+
}], submitButtonText: [{
|
|
108
|
+
type: Input
|
|
109
|
+
}], cancelButtonText: [{
|
|
110
|
+
type: Input
|
|
111
|
+
}], isSubmitting: [{
|
|
112
|
+
type: Input
|
|
113
|
+
}], name: [{
|
|
114
|
+
type: Input
|
|
115
|
+
}], nameChange: [{
|
|
116
|
+
type: Output
|
|
117
|
+
}], submitted: [{
|
|
118
|
+
type: Output
|
|
119
|
+
}], cancelled: [{
|
|
120
|
+
type: Output
|
|
121
|
+
}] } });
|
|
122
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"name-prompt-modal.component.js","sourceRoot":"","sources":["../../../../../src/lib/dialogs/name-prompt-modal.component.ts","../../../../../src/lib/dialogs/name-prompt-modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, Input, Output } from '@angular/core';\n\nexport interface NamePromptSubmitPayload {\n  name: string;\n}\n\n/**\n * Generic single-input prompt modal. Used wherever a flow needs to ask the\n * user for a single name (or short text) and confirm with a primary button.\n *\n * Examples: New Folder, Clone Suite, Rename, New Plan, etc.\n *\n * All copy is configurable through Inputs — see `title`, `description`,\n * `label`, `placeholder`, `submitButtonText`, `cancelButtonText`. Defaults\n * preserve the legacy \"New Folder\" wording for callers that haven't migrated.\n */\n@Component({\n  selector: 'cqa-name-prompt-modal',\n  templateUrl: './name-prompt-modal.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root' },\n})\nexport class NamePromptModalComponent {\n  @Input() isOpen = false;\n  @Input() title = 'New Folder';\n  @Input() description: string | null = 'Rename any time by double-clicking';\n  @Input() label = 'Folder name';\n  @Input() placeholder = 'e.g. Checkout Flow, User Auth, Payment Tests';\n  /** Submit (primary) button text. Lets consumers reuse this modal for\n   * \"Create folder\", \"Clone Suite\", \"Rename\", etc. */\n  @Input() submitButtonText = 'Create folder';\n  /** Cancel (secondary) button text. Most consumers can leave the default. */\n  @Input() cancelButtonText = 'Cancel';\n  /** When true, locks the modal: spinner on the submit button, all dismiss\n   * paths (cancel, close X, backdrop, ESC) become no-ops, and submit() bails\n   * out so a fast Enter press can't double-fire while the parent is mid-API. */\n  @Input() isSubmitting = false;\n\n  private _name = '';\n  @Input() set name(value: string) {\n    this._name = value || '';\n    this.nameTouched = false;\n    this.cdr.markForCheck();\n  }\n  get name(): string {\n    return this._name;\n  }\n\n  @Output() nameChange = new EventEmitter<string>();\n  @Output() submitted = new EventEmitter<NamePromptSubmitPayload>();\n  @Output() cancelled = new EventEmitter<void>();\n\n  private nameTouched = false;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  onNameChange(value: string): void {\n    this._name = value;\n    this.nameTouched = true;\n    this.nameChange.emit(value);\n    this.cdr.markForCheck();\n  }\n\n  get nameErrors(): string[] {\n    if (!this.nameTouched) return [];\n    const trimmed = (this._name || '').trim();\n    if (!trimmed) return [`${this.label} is required`];\n    return [];\n  }\n\n  get isValid(): boolean {\n    return (this._name || '').trim().length > 0;\n  }\n\n  submit(): void {\n    if (!this.isValid || this.isSubmitting) return;\n    this.submitted.emit({ name: this._name.trim() });\n  }\n\n  onCancel(): void {\n    if (this.isSubmitting) return;\n    this.cancelled.emit();\n  }\n\n  onClose(): void {\n    if (this.isSubmitting) return;\n    this.cancelled.emit();\n  }\n\n  onBackdropClick(): void {\n    if (this.isSubmitting) return;\n    this.cancelled.emit();\n  }\n\n  onEscape(event: Event): void {\n    event.stopPropagation();\n    if (this.isSubmitting) return;\n    this.cancelled.emit();\n  }\n}\n","<div *ngIf=\"isOpen\"\n     id=\"cqa-ui-root\"\n     style=\"position: fixed; inset: 0; z-index: 1000;\">\n  <div class=\"cqa-fixed cqa-inset-0 cqa-bg-black/40\"\n       (click)=\"onBackdropClick()\"></div>\n\n  <div class=\"cqa-fixed cqa-inset-0 cqa-flex cqa-items-center cqa-justify-center cqa-p-4 cqa-pointer-events-none\">\n    <div class=\"cqa-relative cqa-w-full cqa-max-w-[480px] cqa-bg-white cqa-rounded-2xl cqa-shadow-md cqa-border cqa-border-border-default cqa-pointer-events-auto\"\n         (keydown.escape)=\"onEscape($event)\"\n         tabindex=\"-1\">\n      <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-p-6 cqa-pb-4\">\n        <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n          <h2 class=\"cqa-text-lg cqa-font-semibold cqa-text-title cqa-m-0\">{{ title }}</h2>\n          <p *ngIf=\"description\"\n             class=\"cqa-text-sm cqa-text-description cqa-m-0\">\n            {{ description }}\n          </p>\n        </div>\n        <button type=\"button\"\n                class=\"cqa-flex cqa-items-center cqa-justify-center cqa-h-6 cqa-w-6 cqa-rounded cqa-bg-transparent cqa-border-0 cqa-text-description hover:cqa-text-title\"\n                [class.cqa-cursor-pointer]=\"!isSubmitting\"\n                [class.cqa-cursor-not-allowed]=\"isSubmitting\"\n                [class.cqa-opacity-50]=\"isSubmitting\"\n                [disabled]=\"isSubmitting\"\n                (click)=\"onClose()\"\n                aria-label=\"Close\">\n          <mat-icon class=\"cqa-text-[20px] cqa-leading-5 cqa-h-5 cqa-w-5\">close</mat-icon>\n        </button>\n      </div>\n\n      <div class=\"cqa-px-6 cqa-pb-5\">\n        <cqa-custom-input\n          [label]=\"label\"\n          [placeholder]=\"placeholder\"\n          [value]=\"name\"\n          [required]=\"true\"\n          [fullWidth]=\"true\"\n          [errors]=\"nameErrors\"\n          (valueChange)=\"onNameChange($event)\"\n          (enterPressed)=\"submit()\">\n        </cqa-custom-input>\n      </div>\n\n      <div class=\"cqa-border-t cqa-border-border-default cqa-px-6 cqa-py-4 cqa-grid cqa-grid-cols-2 cqa-gap-3\">\n        <cqa-button variant=\"outlined\"\n                    [text]=\"cancelButtonText\"\n                    btnSize=\"lg\"\n                    [fullWidth]=\"true\"\n                    [disabled]=\"isSubmitting\"\n                    (clicked)=\"onCancel()\">\n        </cqa-button>\n        <cqa-button variant=\"filled\"\n                    [text]=\"submitButtonText\"\n                    btnSize=\"lg\"\n                    [fullWidth]=\"true\"\n                    [loading]=\"isSubmitting\"\n                    [disabled]=\"!isValid || isSubmitting\"\n                    (clicked)=\"submit()\">\n        </cqa-button>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|