@cqa-lib/cqa-ui 1.1.520 → 1.1.521

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.
@@ -0,0 +1,204 @@
1
+ import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "../custom-input/custom-input.component";
4
+ import * as i2 from "../dropdown-button/dropdown-button.component";
5
+ import * as i3 from "../permission-toggle/permission-toggle.component";
6
+ import * as i4 from "../custom-textarea/custom-textarea.component";
7
+ import * as i5 from "@angular/common";
8
+ export class NewGlobalVariableDialogComponent {
9
+ constructor(cdr) {
10
+ this.cdr = cdr;
11
+ this.mode = 'create';
12
+ this.existingNames = [];
13
+ this.typeOptions = ['String', 'Number', 'Boolean', 'Password'];
14
+ this.typeDropdownOptions = [
15
+ { label: 'String', value: 'String' },
16
+ { label: 'Number', value: 'Number' },
17
+ { label: 'Boolean', value: 'Boolean' },
18
+ { label: 'Password', value: 'Password' },
19
+ ];
20
+ this.permissionSegments = [
21
+ { label: 'RO', value: 'RO', icon: 'lock' },
22
+ { label: 'RW', value: 'RW', icon: 'edit' },
23
+ ];
24
+ this.name = '';
25
+ this.type = 'String';
26
+ this.value = '';
27
+ this.readWriteMode = 'RW';
28
+ this.description = '';
29
+ this.passwordTouched = false;
30
+ this.nameError = null;
31
+ this.valueError = null;
32
+ }
33
+ ngOnInit() {
34
+ if (this.initialValue) {
35
+ this.name = this.initialValue.name ?? '';
36
+ this.type = this.initialValue.type ?? 'String';
37
+ this.value = this.initialValue.value ?? '';
38
+ this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';
39
+ this.description = this.initialValue.description ?? '';
40
+ }
41
+ if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {
42
+ this.value = 'false';
43
+ }
44
+ if (this.mode === 'edit' && this.type === 'Password') {
45
+ this.value = '';
46
+ }
47
+ }
48
+ get title() {
49
+ return this.mode === 'edit' ? 'Edit global variable' : 'New global variable';
50
+ }
51
+ get subtitle() {
52
+ return 'Workspace-wide — accessible from any test case, any environment.';
53
+ }
54
+ get primaryButtonLabel() {
55
+ return this.mode === 'edit' ? 'Save changes' : 'Create variable';
56
+ }
57
+ get isBoolean() {
58
+ return this.type === 'Boolean';
59
+ }
60
+ get isPassword() {
61
+ return this.type === 'Password';
62
+ }
63
+ get isNumber() {
64
+ return this.type === 'Number';
65
+ }
66
+ get valuePlaceholder() {
67
+ if (this.isPassword && this.mode === 'edit') {
68
+ return '•••••• (leave blank to keep existing)';
69
+ }
70
+ return '';
71
+ }
72
+ get nameHelperText() {
73
+ if (this.nameError) {
74
+ return this.nameError;
75
+ }
76
+ return 'Names must be unique across the entire workspace (primary key — no duplicates).';
77
+ }
78
+ get nameErrorsArray() {
79
+ return this.nameError ? [this.nameError] : [];
80
+ }
81
+ get valueErrorsArray() {
82
+ return this.valueError ? [this.valueError] : [];
83
+ }
84
+ get booleanOptions() {
85
+ return [
86
+ { label: 'true', value: 'true' },
87
+ { label: 'false', value: 'false' },
88
+ ];
89
+ }
90
+ get booleanDropdownOptions() {
91
+ return this.booleanOptions;
92
+ }
93
+ onNameChange(next) {
94
+ this.name = next;
95
+ this.nameError = null;
96
+ this.cdr.markForCheck();
97
+ }
98
+ onTypeChange(next) {
99
+ if (!next) {
100
+ return;
101
+ }
102
+ this.type = next;
103
+ if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {
104
+ this.value = 'false';
105
+ }
106
+ if (this.type === 'Password') {
107
+ this.passwordTouched = false;
108
+ if (this.mode === 'edit') {
109
+ this.value = '';
110
+ }
111
+ }
112
+ this.valueError = null;
113
+ this.cdr.markForCheck();
114
+ }
115
+ onPermissionChange(next) {
116
+ this.readWriteMode = next === 'RO' ? 'RO' : 'RW';
117
+ this.cdr.markForCheck();
118
+ }
119
+ onValueChange(next) {
120
+ this.value = next;
121
+ if (this.isPassword) {
122
+ this.passwordTouched = true;
123
+ }
124
+ this.valueError = null;
125
+ this.cdr.markForCheck();
126
+ }
127
+ onBooleanValueChange(next) {
128
+ this.value = next;
129
+ this.cdr.markForCheck();
130
+ }
131
+ onDescriptionChange(next) {
132
+ this.description = next;
133
+ this.cdr.markForCheck();
134
+ }
135
+ /**
136
+ * Called by the host (via DialogRef.getComponentInstance()) when the primary
137
+ * button is clicked. Returns the captured value when valid, or null when the
138
+ * form has errors — in which case the host should leave the dialog open.
139
+ */
140
+ getValue() {
141
+ const trimmedName = this.name.trim();
142
+ if (!trimmedName) {
143
+ this.nameError = 'Name is required.';
144
+ }
145
+ else if (this.isDuplicateName(trimmedName)) {
146
+ this.nameError = 'A global variable with this name already exists — names must be unique.';
147
+ }
148
+ else {
149
+ this.nameError = null;
150
+ }
151
+ if (this.isNumber && this.value && !this.isValidNumber(this.value)) {
152
+ this.valueError = 'Value must be a number.';
153
+ }
154
+ else {
155
+ this.valueError = null;
156
+ }
157
+ if (this.nameError || this.valueError) {
158
+ this.cdr.markForCheck();
159
+ return null;
160
+ }
161
+ const omitPasswordValue = this.isPassword && this.mode === 'edit' && !this.passwordTouched;
162
+ return {
163
+ name: trimmedName,
164
+ type: this.type,
165
+ value: omitPasswordValue ? null : this.serializeValue(),
166
+ readWriteMode: this.readWriteMode,
167
+ description: this.description ? this.description : null,
168
+ };
169
+ }
170
+ serializeValue() {
171
+ if (this.isBoolean) {
172
+ return this.value === 'true' ? 'true' : 'false';
173
+ }
174
+ return this.value ?? '';
175
+ }
176
+ isDuplicateName(candidate) {
177
+ const existing = (this.existingNames ?? []).map(n => (n ?? '').trim());
178
+ if (this.mode === 'edit') {
179
+ const original = (this.initialValue?.name ?? '').trim();
180
+ return candidate !== original && existing.includes(candidate);
181
+ }
182
+ return existing.includes(candidate);
183
+ }
184
+ isValidNumber(raw) {
185
+ const trimmed = raw.trim();
186
+ if (!trimmed) {
187
+ return true;
188
+ }
189
+ return !Number.isNaN(Number(trimmed));
190
+ }
191
+ }
192
+ NewGlobalVariableDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewGlobalVariableDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
193
+ NewGlobalVariableDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NewGlobalVariableDialogComponent, selector: "cqa-new-global-variable-dialog", inputs: { mode: "mode", initialValue: "initialValue", existingNames: "existingNames" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Name\"\n placeholder=\"e.g. ihg_api_key\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n <span *ngIf=\"!nameError\"\n class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Names must be unique across the entire workspace (primary key \u2014 no duplicates).\n </span>\n </div>\n\n <!-- Type + Permission row -->\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Type</label>\n <cqa-dropdown-button\n [options]=\"typeDropdownOptions\"\n [selectedValue]=\"type\"\n (selectionChange)=\"onTypeChange($event)\">\n </cqa-dropdown-button>\n </div>\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n <div class=\"cqa-pt-0.5\">\n <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n </div>\n </div>\n\n <!-- Value -->\n <div class=\"cqa-flex cqa-flex-col\">\n <ng-container *ngIf=\"isBoolean; else nonBooleanValue\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Value</label>\n <cqa-dropdown-button\n [options]=\"booleanDropdownOptions\"\n [selectedValue]=\"value\"\n (selectionChange)=\"onBooleanValueChange($event)\">\n </cqa-dropdown-button>\n </ng-container>\n <ng-template #nonBooleanValue>\n <cqa-custom-input\n label=\"Value\"\n [type]=\"isPassword ? 'password' : 'text'\"\n [value]=\"value\"\n [placeholder]=\"valuePlaceholder\"\n [fullWidth]=\"true\"\n [errors]=\"valueErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onValueChange($event)\">\n </cqa-custom-input>\n </ng-template>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-textarea\n label=\"Description\"\n placeholder=\"What is this variable used for?\"\n [value]=\"description\"\n [rows]=\"2\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-textarea>\n </div>\n\n</div>\n", components: [{ type: i1.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: i2.DropdownButtonComponent, selector: "cqa-dropdown-button", inputs: ["label", "options", "selectedValue", "disabled", "placeholder"], outputs: ["selectionChange", "opened", "closed"] }, { type: i3.PermissionToggleComponent, selector: "cqa-permission-toggle", inputs: ["value", "disabled", "roTooltip", "rwTooltip"], outputs: ["valueChange"] }, { type: i4.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "enableMarkdown", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
194
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewGlobalVariableDialogComponent, decorators: [{
195
+ type: Component,
196
+ args: [{ selector: 'cqa-new-global-variable-dialog', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Name\"\n placeholder=\"e.g. ihg_api_key\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n <span *ngIf=\"!nameError\"\n class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Names must be unique across the entire workspace (primary key \u2014 no duplicates).\n </span>\n </div>\n\n <!-- Type + Permission row -->\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Type</label>\n <cqa-dropdown-button\n [options]=\"typeDropdownOptions\"\n [selectedValue]=\"type\"\n (selectionChange)=\"onTypeChange($event)\">\n </cqa-dropdown-button>\n </div>\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n <div class=\"cqa-pt-0.5\">\n <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n </div>\n </div>\n\n <!-- Value -->\n <div class=\"cqa-flex cqa-flex-col\">\n <ng-container *ngIf=\"isBoolean; else nonBooleanValue\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Value</label>\n <cqa-dropdown-button\n [options]=\"booleanDropdownOptions\"\n [selectedValue]=\"value\"\n (selectionChange)=\"onBooleanValueChange($event)\">\n </cqa-dropdown-button>\n </ng-container>\n <ng-template #nonBooleanValue>\n <cqa-custom-input\n label=\"Value\"\n [type]=\"isPassword ? 'password' : 'text'\"\n [value]=\"value\"\n [placeholder]=\"valuePlaceholder\"\n [fullWidth]=\"true\"\n [errors]=\"valueErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onValueChange($event)\">\n </cqa-custom-input>\n </ng-template>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-textarea\n label=\"Description\"\n placeholder=\"What is this variable used for?\"\n [value]=\"description\"\n [rows]=\"2\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-textarea>\n </div>\n\n</div>\n" }]
197
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { mode: [{
198
+ type: Input
199
+ }], initialValue: [{
200
+ type: Input
201
+ }], existingNames: [{
202
+ type: Input
203
+ }] } });
204
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new-global-variable-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/new-global-variable-dialog/new-global-variable-dialog.component.ts","../../../../../src/lib/new-global-variable-dialog/new-global-variable-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport {\n  GlobalVariableDialogValue,\n  GlobalVariableUIPermission,\n  GlobalVariableUIType,\n} from './new-global-variable-dialog.models';\n\n@Component({\n  selector: 'cqa-new-global-variable-dialog',\n  templateUrl: './new-global-variable-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class NewGlobalVariableDialogComponent implements OnInit {\n  @Input() mode: 'create' | 'edit' = 'create';\n  @Input() initialValue?: Partial<GlobalVariableDialogValue>;\n  @Input() existingNames: string[] = [];\n\n  readonly typeOptions: GlobalVariableUIType[] = ['String', 'Number', 'Boolean', 'Password'];\n\n  readonly typeDropdownOptions: { label: string; value: GlobalVariableUIType }[] = [\n    { label: 'String', value: 'String' },\n    { label: 'Number', value: 'Number' },\n    { label: 'Boolean', value: 'Boolean' },\n    { label: 'Password', value: 'Password' },\n  ];\n\n  readonly permissionSegments = [\n    { label: 'RO', value: 'RO' as GlobalVariableUIPermission, icon: 'lock' },\n    { label: 'RW', value: 'RW' as GlobalVariableUIPermission, icon: 'edit' },\n  ];\n\n  name = '';\n  type: GlobalVariableUIType = 'String';\n  value = '';\n  readWriteMode: GlobalVariableUIPermission = 'RW';\n  description = '';\n  passwordTouched = false;\n\n  nameError: string | null = null;\n  valueError: string | null = null;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    if (this.initialValue) {\n      this.name = this.initialValue.name ?? '';\n      this.type = this.initialValue.type ?? 'String';\n      this.value = this.initialValue.value ?? '';\n      this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';\n      this.description = this.initialValue.description ?? '';\n    }\n    if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {\n      this.value = 'false';\n    }\n    if (this.mode === 'edit' && this.type === 'Password') {\n      this.value = '';\n    }\n  }\n\n  get title(): string {\n    return this.mode === 'edit' ? 'Edit global variable' : 'New global variable';\n  }\n\n  get subtitle(): string {\n    return 'Workspace-wide — accessible from any test case, any environment.';\n  }\n\n  get primaryButtonLabel(): string {\n    return this.mode === 'edit' ? 'Save changes' : 'Create variable';\n  }\n\n  get isBoolean(): boolean {\n    return this.type === 'Boolean';\n  }\n\n  get isPassword(): boolean {\n    return this.type === 'Password';\n  }\n\n  get isNumber(): boolean {\n    return this.type === 'Number';\n  }\n\n  get valuePlaceholder(): string {\n    if (this.isPassword && this.mode === 'edit') {\n      return '•••••• (leave blank to keep existing)';\n    }\n    return '';\n  }\n\n  get nameHelperText(): string {\n    if (this.nameError) {\n      return this.nameError;\n    }\n    return 'Names must be unique across the entire workspace (primary key — no duplicates).';\n  }\n\n  get nameErrorsArray(): string[] {\n    return this.nameError ? [this.nameError] : [];\n  }\n\n  get valueErrorsArray(): string[] {\n    return this.valueError ? [this.valueError] : [];\n  }\n\n  get booleanOptions(): { label: string; value: string }[] {\n    return [\n      { label: 'true', value: 'true' },\n      { label: 'false', value: 'false' },\n    ];\n  }\n\n  get booleanDropdownOptions(): { label: string; value: string }[] {\n    return this.booleanOptions;\n  }\n\n  onNameChange(next: string): void {\n    this.name = next;\n    this.nameError = null;\n    this.cdr.markForCheck();\n  }\n\n  onTypeChange(next: GlobalVariableUIType): void {\n    if (!next) { return; }\n    this.type = next;\n    if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {\n      this.value = 'false';\n    }\n    if (this.type === 'Password') {\n      this.passwordTouched = false;\n      if (this.mode === 'edit') {\n        this.value = '';\n      }\n    }\n    this.valueError = null;\n    this.cdr.markForCheck();\n  }\n\n  onPermissionChange(next: string): void {\n    this.readWriteMode = next === 'RO' ? 'RO' : 'RW';\n    this.cdr.markForCheck();\n  }\n\n  onValueChange(next: string): void {\n    this.value = next;\n    if (this.isPassword) {\n      this.passwordTouched = true;\n    }\n    this.valueError = null;\n    this.cdr.markForCheck();\n  }\n\n  onBooleanValueChange(next: string): void {\n    this.value = next;\n    this.cdr.markForCheck();\n  }\n\n  onDescriptionChange(next: string): void {\n    this.description = next;\n    this.cdr.markForCheck();\n  }\n\n  /**\n   * Called by the host (via DialogRef.getComponentInstance()) when the primary\n   * button is clicked. Returns the captured value when valid, or null when the\n   * form has errors — in which case the host should leave the dialog open.\n   */\n  getValue(): GlobalVariableDialogValue | null {\n    const trimmedName = this.name.trim();\n\n    if (!trimmedName) {\n      this.nameError = 'Name is required.';\n    } else if (this.isDuplicateName(trimmedName)) {\n      this.nameError = 'A global variable with this name already exists — names must be unique.';\n    } else {\n      this.nameError = null;\n    }\n\n    if (this.isNumber && this.value && !this.isValidNumber(this.value)) {\n      this.valueError = 'Value must be a number.';\n    } else {\n      this.valueError = null;\n    }\n\n    if (this.nameError || this.valueError) {\n      this.cdr.markForCheck();\n      return null;\n    }\n\n    const omitPasswordValue = this.isPassword && this.mode === 'edit' && !this.passwordTouched;\n\n    return {\n      name: trimmedName,\n      type: this.type,\n      value: omitPasswordValue ? null : this.serializeValue(),\n      readWriteMode: this.readWriteMode,\n      description: this.description ? this.description : null,\n    };\n  }\n\n  private serializeValue(): string | null {\n    if (this.isBoolean) {\n      return this.value === 'true' ? 'true' : 'false';\n    }\n    return this.value ?? '';\n  }\n\n  private isDuplicateName(candidate: string): boolean {\n    const existing = (this.existingNames ?? []).map(n => (n ?? '').trim());\n    if (this.mode === 'edit') {\n      const original = (this.initialValue?.name ?? '').trim();\n      return candidate !== original && existing.includes(candidate);\n    }\n    return existing.includes(candidate);\n  }\n\n  private isValidNumber(raw: string): boolean {\n    const trimmed = raw.trim();\n    if (!trimmed) { return true; }\n    return !Number.isNaN(Number(trimmed));\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n  <!-- Name -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-input\n      label=\"Name\"\n      placeholder=\"e.g. ihg_api_key\"\n      type=\"text\"\n      [value]=\"name\"\n      [required]=\"true\"\n      [fullWidth]=\"true\"\n      [errors]=\"nameErrorsArray\"\n      inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n      (valueChange)=\"onNameChange($event)\">\n    </cqa-custom-input>\n    <span *ngIf=\"!nameError\"\n      class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n      Names must be unique across the entire workspace (primary key — no duplicates).\n    </span>\n  </div>\n\n  <!-- Type + Permission row -->\n  <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n    <div class=\"cqa-flex cqa-flex-col\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Type</label>\n      <cqa-dropdown-button\n        [options]=\"typeDropdownOptions\"\n        [selectedValue]=\"type\"\n        (selectionChange)=\"onTypeChange($event)\">\n      </cqa-dropdown-button>\n    </div>\n    <div class=\"cqa-flex cqa-flex-col\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n      <div class=\"cqa-pt-0.5\">\n        <cqa-permission-toggle\n          [value]=\"readWriteMode\"\n          (valueChange)=\"onPermissionChange($event)\">\n        </cqa-permission-toggle>\n      </div>\n    </div>\n  </div>\n\n  <!-- Value -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <ng-container *ngIf=\"isBoolean; else nonBooleanValue\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Value</label>\n      <cqa-dropdown-button\n        [options]=\"booleanDropdownOptions\"\n        [selectedValue]=\"value\"\n        (selectionChange)=\"onBooleanValueChange($event)\">\n      </cqa-dropdown-button>\n    </ng-container>\n    <ng-template #nonBooleanValue>\n      <cqa-custom-input\n        label=\"Value\"\n        [type]=\"isPassword ? 'password' : 'text'\"\n        [value]=\"value\"\n        [placeholder]=\"valuePlaceholder\"\n        [fullWidth]=\"true\"\n        [errors]=\"valueErrorsArray\"\n        inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n        (valueChange)=\"onValueChange($event)\">\n      </cqa-custom-input>\n    </ng-template>\n  </div>\n\n  <!-- Description -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-textarea\n      label=\"Description\"\n      placeholder=\"What is this variable used for?\"\n      [value]=\"description\"\n      [rows]=\"2\"\n      [fullWidth]=\"true\"\n      (valueChange)=\"onDescriptionChange($event)\">\n    </cqa-custom-textarea>\n  </div>\n\n</div>\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV3LWdsb2JhbC12YXJpYWJsZS1kaWFsb2cubW9kZWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9uZXctZ2xvYmFsLXZhcmlhYmxlLWRpYWxvZy9uZXctZ2xvYmFsLXZhcmlhYmxlLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIEdsb2JhbFZhcmlhYmxlVUlUeXBlID0gJ1N0cmluZycgfCAnTnVtYmVyJyB8ICdCb29sZWFuJyB8ICdQYXNzd29yZCc7XG5leHBvcnQgdHlwZSBHbG9iYWxWYXJpYWJsZVVJUGVybWlzc2lvbiA9ICdSTycgfCAnUlcnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEdsb2JhbFZhcmlhYmxlRGlhbG9nVmFsdWUge1xuICBuYW1lOiBzdHJpbmc7XG4gIHR5cGU6IEdsb2JhbFZhcmlhYmxlVUlUeXBlO1xuICB2YWx1ZTogc3RyaW5nIHwgbnVsbDtcbiAgcmVhZFdyaXRlTW9kZTogR2xvYmFsVmFyaWFibGVVSVBlcm1pc3Npb247XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmcgfCBudWxsO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE5ld0dsb2JhbFZhcmlhYmxlRGlhbG9nSW5wdXRzIHtcbiAgbW9kZT86ICdjcmVhdGUnIHwgJ2VkaXQnO1xuICBpbml0aWFsVmFsdWU/OiBQYXJ0aWFsPEdsb2JhbFZhcmlhYmxlRGlhbG9nVmFsdWU+O1xuICBleGlzdGluZ05hbWVzPzogc3RyaW5nW107XG59XG4iXX0=
@@ -0,0 +1,86 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/material/icon";
4
+ export class PermissionToggleComponent {
5
+ constructor() {
6
+ this.value = 'RW';
7
+ this.disabled = false;
8
+ this.roTooltip = 'Read only — click to allow writes';
9
+ this.rwTooltip = 'Read / write — click to lock';
10
+ this.valueChange = new EventEmitter();
11
+ }
12
+ onClick(next, event) {
13
+ event.stopPropagation();
14
+ if (this.disabled || next === this.value) {
15
+ return;
16
+ }
17
+ this.value = next;
18
+ this.valueChange.emit(next);
19
+ }
20
+ }
21
+ PermissionToggleComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: PermissionToggleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
22
+ PermissionToggleComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: PermissionToggleComponent, selector: "cqa-permission-toggle", inputs: { value: "value", disabled: "disabled", roTooltip: "roTooltip", rwTooltip: "rwTooltip" }, outputs: { valueChange: "valueChange" }, ngImport: i0, template: `
23
+ <div class="cqa-ui-root" style="display:inline-flex;">
24
+ <div class="cqa-permission-toggle" [class.cqa-permission-toggle--disabled]="disabled">
25
+ <button
26
+ type="button"
27
+ class="cqa-perm-pill cqa-perm-pill--ro"
28
+ [class.active]="value === 'RO'"
29
+ [disabled]="disabled"
30
+ [title]="roTooltip"
31
+ (click)="onClick('RO', $event)">
32
+ <mat-icon class="cqa-perm-icon">lock</mat-icon>RO
33
+ </button>
34
+ <button
35
+ type="button"
36
+ class="cqa-perm-pill cqa-perm-pill--rw"
37
+ [class.active]="value === 'RW'"
38
+ [disabled]="disabled"
39
+ [title]="rwTooltip"
40
+ (click)="onClick('RW', $event)">
41
+ <mat-icon class="cqa-perm-icon">edit</mat-icon>RW
42
+ </button>
43
+ </div>
44
+ </div>
45
+ `, isInline: true, components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }] });
46
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: PermissionToggleComponent, decorators: [{
47
+ type: Component,
48
+ args: [{
49
+ selector: 'cqa-permission-toggle',
50
+ template: `
51
+ <div class="cqa-ui-root" style="display:inline-flex;">
52
+ <div class="cqa-permission-toggle" [class.cqa-permission-toggle--disabled]="disabled">
53
+ <button
54
+ type="button"
55
+ class="cqa-perm-pill cqa-perm-pill--ro"
56
+ [class.active]="value === 'RO'"
57
+ [disabled]="disabled"
58
+ [title]="roTooltip"
59
+ (click)="onClick('RO', $event)">
60
+ <mat-icon class="cqa-perm-icon">lock</mat-icon>RO
61
+ </button>
62
+ <button
63
+ type="button"
64
+ class="cqa-perm-pill cqa-perm-pill--rw"
65
+ [class.active]="value === 'RW'"
66
+ [disabled]="disabled"
67
+ [title]="rwTooltip"
68
+ (click)="onClick('RW', $event)">
69
+ <mat-icon class="cqa-perm-icon">edit</mat-icon>RW
70
+ </button>
71
+ </div>
72
+ </div>
73
+ `,
74
+ }]
75
+ }], propDecorators: { value: [{
76
+ type: Input
77
+ }], disabled: [{
78
+ type: Input
79
+ }], roTooltip: [{
80
+ type: Input
81
+ }], rwTooltip: [{
82
+ type: Input
83
+ }], valueChange: [{
84
+ type: Output
85
+ }] } });
86
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGVybWlzc2lvbi10b2dnbGUuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9wZXJtaXNzaW9uLXRvZ2dsZS9wZXJtaXNzaW9uLXRvZ2dsZS5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBK0J2RSxNQUFNLE9BQU8seUJBQXlCO0lBM0J0QztRQTRCVyxVQUFLLEdBQTBCLElBQUksQ0FBQztRQUNwQyxhQUFRLEdBQUcsS0FBSyxDQUFDO1FBQ2pCLGNBQVMsR0FBRyxtQ0FBbUMsQ0FBQztRQUNoRCxjQUFTLEdBQUcsOEJBQThCLENBQUM7UUFFMUMsZ0JBQVcsR0FBRyxJQUFJLFlBQVksRUFBeUIsQ0FBQztLQVFuRTtJQU5DLE9BQU8sQ0FBQyxJQUEyQixFQUFFLEtBQWlCO1FBQ3BELEtBQUssQ0FBQyxlQUFlLEVBQUUsQ0FBQztRQUN4QixJQUFJLElBQUksQ0FBQyxRQUFRLElBQUksSUFBSSxLQUFLLElBQUksQ0FBQyxLQUFLLEVBQUU7WUFBRSxPQUFPO1NBQUU7UUFDckQsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUM7UUFDbEIsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDOUIsQ0FBQzs7c0hBYlUseUJBQXlCOzBHQUF6Qix5QkFBeUIsd01BekIxQjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F1QlQ7MkZBRVUseUJBQXlCO2tCQTNCckMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsdUJBQXVCO29CQUNqQyxRQUFRLEVBQUU7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBdUJUO2lCQUNGOzhCQUVVLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxRQUFRO3NCQUFoQixLQUFLO2dCQUNHLFNBQVM7c0JBQWpCLEtBQUs7Z0JBQ0csU0FBUztzQkFBakIsS0FBSztnQkFFSSxXQUFXO3NCQUFwQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBFdmVudEVtaXR0ZXIsIElucHV0LCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IHR5cGUgUGVybWlzc2lvblRvZ2dsZVZhbHVlID0gJ1JPJyB8ICdSVyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NxYS1wZXJtaXNzaW9uLXRvZ2dsZScsXG4gIHRlbXBsYXRlOiBgXG4gICAgPGRpdiBjbGFzcz1cImNxYS11aS1yb290XCIgc3R5bGU9XCJkaXNwbGF5OmlubGluZS1mbGV4O1wiPlxuICAgICAgPGRpdiBjbGFzcz1cImNxYS1wZXJtaXNzaW9uLXRvZ2dsZVwiIFtjbGFzcy5jcWEtcGVybWlzc2lvbi10b2dnbGUtLWRpc2FibGVkXT1cImRpc2FibGVkXCI+XG4gICAgICAgIDxidXR0b25cbiAgICAgICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgICAgICBjbGFzcz1cImNxYS1wZXJtLXBpbGwgY3FhLXBlcm0tcGlsbC0tcm9cIlxuICAgICAgICAgIFtjbGFzcy5hY3RpdmVdPVwidmFsdWUgPT09ICdSTydcIlxuICAgICAgICAgIFtkaXNhYmxlZF09XCJkaXNhYmxlZFwiXG4gICAgICAgICAgW3RpdGxlXT1cInJvVG9vbHRpcFwiXG4gICAgICAgICAgKGNsaWNrKT1cIm9uQ2xpY2soJ1JPJywgJGV2ZW50KVwiPlxuICAgICAgICAgIDxtYXQtaWNvbiBjbGFzcz1cImNxYS1wZXJtLWljb25cIj5sb2NrPC9tYXQtaWNvbj5ST1xuICAgICAgICA8L2J1dHRvbj5cbiAgICAgICAgPGJ1dHRvblxuICAgICAgICAgIHR5cGU9XCJidXR0b25cIlxuICAgICAgICAgIGNsYXNzPVwiY3FhLXBlcm0tcGlsbCBjcWEtcGVybS1waWxsLS1yd1wiXG4gICAgICAgICAgW2NsYXNzLmFjdGl2ZV09XCJ2YWx1ZSA9PT0gJ1JXJ1wiXG4gICAgICAgICAgW2Rpc2FibGVkXT1cImRpc2FibGVkXCJcbiAgICAgICAgICBbdGl0bGVdPVwicndUb29sdGlwXCJcbiAgICAgICAgICAoY2xpY2spPVwib25DbGljaygnUlcnLCAkZXZlbnQpXCI+XG4gICAgICAgICAgPG1hdC1pY29uIGNsYXNzPVwiY3FhLXBlcm0taWNvblwiPmVkaXQ8L21hdC1pY29uPlJXXG4gICAgICAgIDwvYnV0dG9uPlxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIGAsXG59KVxuZXhwb3J0IGNsYXNzIFBlcm1pc3Npb25Ub2dnbGVDb21wb25lbnQge1xuICBASW5wdXQoKSB2YWx1ZTogUGVybWlzc2lvblRvZ2dsZVZhbHVlID0gJ1JXJztcbiAgQElucHV0KCkgZGlzYWJsZWQgPSBmYWxzZTtcbiAgQElucHV0KCkgcm9Ub29sdGlwID0gJ1JlYWQgb25seSDigJQgY2xpY2sgdG8gYWxsb3cgd3JpdGVzJztcbiAgQElucHV0KCkgcndUb29sdGlwID0gJ1JlYWQgLyB3cml0ZSDigJQgY2xpY2sgdG8gbG9jayc7XG5cbiAgQE91dHB1dCgpIHZhbHVlQ2hhbmdlID0gbmV3IEV2ZW50RW1pdHRlcjxQZXJtaXNzaW9uVG9nZ2xlVmFsdWU+KCk7XG5cbiAgb25DbGljayhuZXh0OiBQZXJtaXNzaW9uVG9nZ2xlVmFsdWUsIGV2ZW50OiBNb3VzZUV2ZW50KTogdm9pZCB7XG4gICAgZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgaWYgKHRoaXMuZGlzYWJsZWQgfHwgbmV4dCA9PT0gdGhpcy52YWx1ZSkgeyByZXR1cm47IH1cbiAgICB0aGlzLnZhbHVlID0gbmV4dDtcbiAgICB0aGlzLnZhbHVlQ2hhbmdlLmVtaXQobmV4dCk7XG4gIH1cbn1cbiJdfQ==