@cqa-lib/cqa-ui 1.1.548-gamma.17 → 1.1.548-gamma.19
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/assign-environments-dialog/assign-environments-dialog.component.mjs +269 -72
- package/esm2020/lib/assign-environments-dialog/assign-environments-dialog.models.mjs +1 -1
- package/esm2020/lib/step-builder/template-variables-form/template-variables-form.component.mjs +53 -7
- package/fesm2015/cqa-lib-cqa-ui.mjs +319 -79
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +314 -76
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/assign-environments-dialog/assign-environments-dialog.component.d.ts +53 -14
- package/lib/assign-environments-dialog/assign-environments-dialog.models.d.ts +20 -2
- package/lib/step-builder/template-variables-form/template-variables-form.component.d.ts +15 -1
- package/package.json +1 -1
- package/styles.css +1 -1
|
@@ -1,50 +1,57 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input, ViewChild, } from '@angular/core';
|
|
2
2
|
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
|
+
import { Subject, of } from 'rxjs';
|
|
4
|
+
import { catchError, debounceTime, distinctUntilChanged, switchMap, takeUntil, } from 'rxjs/operators';
|
|
3
5
|
import * as i0 from "@angular/core";
|
|
4
|
-
import * as i1 from "../
|
|
5
|
-
import * as i2 from "
|
|
6
|
+
import * as i1 from "../segment-control/segment-control.component";
|
|
7
|
+
import * as i2 from "../search-bar/search-bar.component";
|
|
8
|
+
import * as i3 from "../dynamic-select/dynamic-select-field.component";
|
|
9
|
+
import * as i4 from "@angular/material/icon";
|
|
10
|
+
import * as i5 from "@angular/common";
|
|
6
11
|
export class AssignEnvironmentsDialogComponent {
|
|
7
12
|
constructor(cdr) {
|
|
8
13
|
this.cdr = cdr;
|
|
9
14
|
this.mode = 'assign';
|
|
10
15
|
this.profileName = '';
|
|
11
|
-
this.
|
|
16
|
+
this.pageSize = 50;
|
|
17
|
+
/** Source-env candidates the per-row "Copy from which environment" picker
|
|
18
|
+
* populates from. Synchronous list — typically bounded by the profile's
|
|
19
|
+
* current attachments (small, often < 10). */
|
|
12
20
|
this.assignedEnvironments = [];
|
|
13
21
|
this.selected = new Set();
|
|
14
|
-
this.
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
this.
|
|
18
|
-
this.
|
|
22
|
+
this.selectedSnapshot = [];
|
|
23
|
+
this.searchTerm = '';
|
|
24
|
+
this.isInitialLoading = false;
|
|
25
|
+
this.isLoadingMore = false;
|
|
26
|
+
this.currentResults = [];
|
|
27
|
+
this.totalElements = 0;
|
|
28
|
+
this.viewMode = 'all';
|
|
29
|
+
/** Stable segment array — recomputed only when selection size flips zero ↔ non-zero. */
|
|
30
|
+
this.viewSegments = this.computeViewSegments();
|
|
31
|
+
/** One FormGroup, controls keyed `tgt_${targetEnvId}`. */
|
|
32
|
+
this.sourceForm = new FormGroup({});
|
|
33
|
+
this.sourceConfigById = new Map();
|
|
34
|
+
this.skeletonRows = [0, 1, 2, 3, 4];
|
|
35
|
+
this.pageIndex = 0;
|
|
36
|
+
this.searchInput$ = new Subject();
|
|
37
|
+
this.destroy$ = new Subject();
|
|
19
38
|
}
|
|
20
39
|
ngOnInit() {
|
|
21
|
-
//
|
|
22
|
-
//
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
this.sourceForm.get('sourceEnvId').setValue(initial);
|
|
31
|
-
this.sourceConfig = this.buildSourceConfig(this.assignedEnvironments ?? []);
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
get showSourcePicker() {
|
|
35
|
-
return (this.assignedEnvironments && this.assignedEnvironments.length > 0);
|
|
36
|
-
}
|
|
37
|
-
get sourceLabel() {
|
|
38
|
-
return this.mode === 'clone' ? 'Copy from which environment' : 'Copy rows from (optional)';
|
|
40
|
+
// Search-term pipeline: every distinct, debounced term resets the page and
|
|
41
|
+
// replaces the visible list.
|
|
42
|
+
this.searchInput$
|
|
43
|
+
.pipe(debounceTime(300), distinctUntilChanged(), switchMap(term => this.fetchPage(term, 0)), takeUntil(this.destroy$))
|
|
44
|
+
.subscribe(({ term, page, payload }) => {
|
|
45
|
+
this.applyPage(term, page, /* append */ false, payload);
|
|
46
|
+
});
|
|
47
|
+
// Initial page-0 fetch on dialog open.
|
|
48
|
+
this.runFetch('', 0, /* append */ false);
|
|
39
49
|
}
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
}
|
|
44
|
-
return this.mode === 'clone'
|
|
45
|
-
? "The selected environment's rows will be copied into each target environment below."
|
|
46
|
-
: 'Pick an environment to copy its rows into the new ones. Leave blank to seed with the base profile rows.';
|
|
50
|
+
ngOnDestroy() {
|
|
51
|
+
this.destroy$.next();
|
|
52
|
+
this.destroy$.complete();
|
|
47
53
|
}
|
|
54
|
+
// -- Public template state -----------------------------------------------
|
|
48
55
|
get title() {
|
|
49
56
|
return this.mode === 'clone' ? 'Clone to other environments' : 'Assign to environments';
|
|
50
57
|
}
|
|
@@ -54,6 +61,33 @@ export class AssignEnvironmentsDialogComponent {
|
|
|
54
61
|
? `Duplicate ${name} into other environments. Column structure is shared; row values are independent per environment.`
|
|
55
62
|
: `Make ${name} available in the selected environments. Columns are shared — each environment gets an independent row set.`;
|
|
56
63
|
}
|
|
64
|
+
get showPerRowSource() {
|
|
65
|
+
return (this.assignedEnvironments?.length ?? 0) > 0;
|
|
66
|
+
}
|
|
67
|
+
get isEmpty() {
|
|
68
|
+
return !this.currentResults || this.currentResults.length === 0;
|
|
69
|
+
}
|
|
70
|
+
get hasMore() {
|
|
71
|
+
return this.currentResults.length < this.totalElements;
|
|
72
|
+
}
|
|
73
|
+
get displayedRows() {
|
|
74
|
+
return this.viewMode === 'selected' ? this.selectedSnapshot : this.currentResults;
|
|
75
|
+
}
|
|
76
|
+
get unresolvedSourcesCount() {
|
|
77
|
+
if (!this.showPerRowSource) {
|
|
78
|
+
return 0;
|
|
79
|
+
}
|
|
80
|
+
let n = 0;
|
|
81
|
+
for (const env of this.selectedSnapshot) {
|
|
82
|
+
if (env.alreadyAssigned) {
|
|
83
|
+
continue;
|
|
84
|
+
}
|
|
85
|
+
if (!this.isSourceResolved(env.id)) {
|
|
86
|
+
n++;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
return n;
|
|
90
|
+
}
|
|
57
91
|
get primaryButtonLabel() {
|
|
58
92
|
const count = this.selected.size;
|
|
59
93
|
const verb = this.mode === 'clone' ? 'Clone to' : 'Assign to';
|
|
@@ -63,63 +97,164 @@ export class AssignEnvironmentsDialogComponent {
|
|
|
63
97
|
if (this.selected.size === 0) {
|
|
64
98
|
return true;
|
|
65
99
|
}
|
|
66
|
-
if (this.
|
|
100
|
+
if (this.unresolvedSourcesCount > 0) {
|
|
67
101
|
return true;
|
|
68
102
|
}
|
|
69
103
|
return false;
|
|
70
104
|
}
|
|
71
|
-
|
|
72
|
-
return this.
|
|
105
|
+
isSelected(id) {
|
|
106
|
+
return this.selected.has(id);
|
|
73
107
|
}
|
|
74
|
-
|
|
75
|
-
|
|
108
|
+
isSourceResolved(targetEnvId) {
|
|
109
|
+
if (!this.showPerRowSource) {
|
|
110
|
+
return true;
|
|
111
|
+
}
|
|
112
|
+
const ctrl = this.sourceForm.get(this.controlKey(targetEnvId));
|
|
113
|
+
return ctrl?.value != null;
|
|
76
114
|
}
|
|
77
|
-
|
|
78
|
-
return this.
|
|
115
|
+
getSourceConfig(targetEnvId) {
|
|
116
|
+
return this.sourceConfigById.get(targetEnvId);
|
|
79
117
|
}
|
|
80
|
-
|
|
81
|
-
|
|
118
|
+
pickedSourceName(targetEnvId) {
|
|
119
|
+
const value = this.sourceForm.get(this.controlKey(targetEnvId))?.value;
|
|
120
|
+
if (value == null) {
|
|
121
|
+
return '';
|
|
122
|
+
}
|
|
123
|
+
const env = (this.assignedEnvironments || []).find(e => Number(e.id) === Number(value));
|
|
124
|
+
return env ? env.name : '';
|
|
82
125
|
}
|
|
83
|
-
|
|
126
|
+
helperFor(env) {
|
|
84
127
|
if (env.alreadyAssigned) {
|
|
128
|
+
return 'Already assigned to this environment';
|
|
129
|
+
}
|
|
130
|
+
return env.description || '';
|
|
131
|
+
}
|
|
132
|
+
trackById(_index, env) {
|
|
133
|
+
return env.id;
|
|
134
|
+
}
|
|
135
|
+
// -- Public template events ---------------------------------------------
|
|
136
|
+
onSearchChange(term) {
|
|
137
|
+
this.searchTerm = term ?? '';
|
|
138
|
+
this.isInitialLoading = true;
|
|
139
|
+
this.currentResults = [];
|
|
140
|
+
this.cdr.markForCheck();
|
|
141
|
+
this.searchInput$.next(this.searchTerm);
|
|
142
|
+
}
|
|
143
|
+
onViewModeChange(value) {
|
|
144
|
+
this.viewMode = value === 'selected' ? 'selected' : 'all';
|
|
145
|
+
this.cdr.markForCheck();
|
|
146
|
+
}
|
|
147
|
+
toggle(env) {
|
|
148
|
+
if (env?.alreadyAssigned) {
|
|
85
149
|
return;
|
|
86
150
|
}
|
|
87
|
-
|
|
88
|
-
|
|
151
|
+
const id = Number(env.id);
|
|
152
|
+
const wasZero = this.selected.size === 0;
|
|
153
|
+
if (this.selected.has(id)) {
|
|
154
|
+
this.selected.delete(id);
|
|
155
|
+
this.selectedSnapshot = this.selectedSnapshot.filter(e => Number(e.id) !== id);
|
|
156
|
+
this.removeSourceState(id);
|
|
157
|
+
// If the Selected view just emptied, fall back to All so the user
|
|
158
|
+
// doesn't sit on a blank screen.
|
|
159
|
+
if (this.viewMode === 'selected' && this.selectedSnapshot.length === 0) {
|
|
160
|
+
this.viewMode = 'all';
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
else {
|
|
164
|
+
this.selected.add(id);
|
|
165
|
+
this.selectedSnapshot = [...this.selectedSnapshot, env];
|
|
166
|
+
this.ensureSourceState(id);
|
|
167
|
+
}
|
|
168
|
+
const isZero = this.selected.size === 0;
|
|
169
|
+
if (wasZero !== isZero) {
|
|
170
|
+
this.viewSegments = this.computeViewSegments();
|
|
89
171
|
}
|
|
90
172
|
else {
|
|
91
|
-
|
|
173
|
+
// Keep the count up to date even when zero ↔ non-zero didn't flip.
|
|
174
|
+
this.viewSegments = this.computeViewSegments();
|
|
92
175
|
}
|
|
93
176
|
this.cdr.markForCheck();
|
|
94
177
|
}
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
178
|
+
onListScroll() {
|
|
179
|
+
const el = this.listScrollRef?.nativeElement;
|
|
180
|
+
if (!el) {
|
|
181
|
+
return;
|
|
182
|
+
}
|
|
183
|
+
if (this.viewMode === 'selected') {
|
|
184
|
+
return;
|
|
185
|
+
}
|
|
186
|
+
if (this.isInitialLoading || this.isLoadingMore) {
|
|
187
|
+
return;
|
|
188
|
+
}
|
|
189
|
+
if (!this.hasMore) {
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
const distanceFromBottom = el.scrollHeight - (el.scrollTop + el.clientHeight);
|
|
193
|
+
if (distanceFromBottom < 80) {
|
|
194
|
+
this.runFetch(this.searchTerm, this.pageIndex + 1, /* append */ true);
|
|
98
195
|
}
|
|
99
|
-
return env.description || '';
|
|
100
196
|
}
|
|
101
197
|
getValue() {
|
|
102
198
|
if (this.selected.size === 0) {
|
|
103
199
|
return null;
|
|
104
200
|
}
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
201
|
+
if (this.unresolvedSourcesCount > 0) {
|
|
202
|
+
return null;
|
|
203
|
+
}
|
|
204
|
+
if (!this.showPerRowSource) {
|
|
205
|
+
return { selectedIds: Array.from(this.selected), sources: [] };
|
|
206
|
+
}
|
|
207
|
+
const sources = Array.from(this.selected).map(id => {
|
|
208
|
+
const value = this.sourceForm.get(this.controlKey(id))?.value;
|
|
209
|
+
return { targetEnvId: id, sourceEnvId: Number(value) };
|
|
210
|
+
});
|
|
211
|
+
return { selectedIds: Array.from(this.selected), sources };
|
|
212
|
+
}
|
|
213
|
+
// -- Internals -----------------------------------------------------------
|
|
214
|
+
controlKey(targetEnvId) {
|
|
215
|
+
return `tgt_${targetEnvId}`;
|
|
216
|
+
}
|
|
217
|
+
computeViewSegments() {
|
|
218
|
+
const count = this.selected.size;
|
|
219
|
+
return [
|
|
220
|
+
{ label: 'All', value: 'all' },
|
|
221
|
+
count > 0
|
|
222
|
+
? { label: 'Selected', value: 'selected', count }
|
|
223
|
+
: { label: 'Selected', value: 'selected' },
|
|
224
|
+
];
|
|
225
|
+
}
|
|
226
|
+
ensureSourceState(targetEnvId) {
|
|
227
|
+
if (!this.showPerRowSource) {
|
|
228
|
+
return;
|
|
229
|
+
}
|
|
230
|
+
if (!this.sourceForm.contains(this.controlKey(targetEnvId))) {
|
|
231
|
+
this.sourceForm.addControl(this.controlKey(targetEnvId), new FormControl(null));
|
|
232
|
+
}
|
|
233
|
+
if (!this.sourceConfigById.has(targetEnvId)) {
|
|
234
|
+
this.sourceConfigById.set(targetEnvId, this.buildSourceConfig(targetEnvId));
|
|
235
|
+
}
|
|
236
|
+
// Pre-select the default source if the caller suggested one (and it's a
|
|
237
|
+
// valid candidate). Saves the user a click when there's an obvious choice.
|
|
238
|
+
const ctrl = this.sourceForm.get(this.controlKey(targetEnvId));
|
|
239
|
+
if (ctrl && ctrl.value == null) {
|
|
240
|
+
const defaultId = this.resolveDefaultSourceId(targetEnvId);
|
|
241
|
+
if (defaultId != null) {
|
|
242
|
+
ctrl.setValue(defaultId);
|
|
111
243
|
}
|
|
112
|
-
this.sourceError = null;
|
|
113
|
-
return { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) };
|
|
114
244
|
}
|
|
115
|
-
// Assign mode: source is optional — include it when the user picked one.
|
|
116
|
-
this.sourceError = null;
|
|
117
|
-
return src != null
|
|
118
|
-
? { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) }
|
|
119
|
-
: { selectedIds: Array.from(this.selected) };
|
|
120
245
|
}
|
|
121
|
-
|
|
122
|
-
const
|
|
246
|
+
removeSourceState(targetEnvId) {
|
|
247
|
+
const key = this.controlKey(targetEnvId);
|
|
248
|
+
if (this.sourceForm.contains(key)) {
|
|
249
|
+
this.sourceForm.removeControl(key);
|
|
250
|
+
}
|
|
251
|
+
this.sourceConfigById.delete(targetEnvId);
|
|
252
|
+
}
|
|
253
|
+
buildSourceConfig(targetEnvId) {
|
|
254
|
+
// Filter the target out of its own source list — copying an env into
|
|
255
|
+
// itself is non-sensical.
|
|
256
|
+
const candidates = (this.assignedEnvironments || []).filter(e => Number(e.id) !== Number(targetEnvId));
|
|
257
|
+
const options = candidates.map(e => ({
|
|
123
258
|
id: e.id,
|
|
124
259
|
value: e.id,
|
|
125
260
|
name: e.name,
|
|
@@ -127,29 +262,91 @@ export class AssignEnvironmentsDialogComponent {
|
|
|
127
262
|
statusColor: e.color,
|
|
128
263
|
}));
|
|
129
264
|
return {
|
|
130
|
-
key:
|
|
265
|
+
key: this.controlKey(targetEnvId),
|
|
131
266
|
label: '',
|
|
132
267
|
placeholder: 'Select source environment…',
|
|
133
268
|
multiple: false,
|
|
134
|
-
searchable:
|
|
269
|
+
searchable: options.length > 6,
|
|
135
270
|
options,
|
|
136
271
|
};
|
|
137
272
|
}
|
|
273
|
+
resolveDefaultSourceId(targetEnvId) {
|
|
274
|
+
const candidates = (this.assignedEnvironments || []).filter(e => Number(e.id) !== Number(targetEnvId));
|
|
275
|
+
if (candidates.length === 0) {
|
|
276
|
+
return null;
|
|
277
|
+
}
|
|
278
|
+
if (this.defaultSourceEnvId != null) {
|
|
279
|
+
const match = candidates.find(e => Number(e.id) === Number(this.defaultSourceEnvId));
|
|
280
|
+
if (match) {
|
|
281
|
+
return Number(match.id);
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
if (this.mode === 'clone') {
|
|
285
|
+
// Clone implies a source was already in scope — pick the first attached env.
|
|
286
|
+
return Number(candidates[0].id);
|
|
287
|
+
}
|
|
288
|
+
return null;
|
|
289
|
+
}
|
|
290
|
+
// -- Fetch pipeline ------------------------------------------------------
|
|
291
|
+
runFetch(term, page, append) {
|
|
292
|
+
if (append) {
|
|
293
|
+
this.isLoadingMore = true;
|
|
294
|
+
}
|
|
295
|
+
else {
|
|
296
|
+
this.isInitialLoading = true;
|
|
297
|
+
}
|
|
298
|
+
this.cdr.markForCheck();
|
|
299
|
+
this.fetchPage(term, page)
|
|
300
|
+
.pipe(takeUntil(this.destroy$))
|
|
301
|
+
.subscribe(result => {
|
|
302
|
+
this.applyPage(result.term, result.page, append, result.payload);
|
|
303
|
+
});
|
|
304
|
+
}
|
|
305
|
+
fetchPage(term, page) {
|
|
306
|
+
if (!this.searchFn) {
|
|
307
|
+
return of({ term, page, payload: { items: [], total: 0 } });
|
|
308
|
+
}
|
|
309
|
+
return this.searchFn(term, page, this.pageSize).pipe(catchError(() => of({ items: [], total: 0 })), switchMap(payload => of({ term, page, payload })));
|
|
310
|
+
}
|
|
311
|
+
applyPage(term, page, append, payload) {
|
|
312
|
+
// Drop responses whose term no longer matches the active search.
|
|
313
|
+
if (term !== this.searchTerm) {
|
|
314
|
+
return;
|
|
315
|
+
}
|
|
316
|
+
const items = payload?.items ?? [];
|
|
317
|
+
const total = payload?.total ?? items.length;
|
|
318
|
+
if (append) {
|
|
319
|
+
this.currentResults = [...this.currentResults, ...items];
|
|
320
|
+
}
|
|
321
|
+
else {
|
|
322
|
+
this.currentResults = [...items];
|
|
323
|
+
}
|
|
324
|
+
this.totalElements = total;
|
|
325
|
+
this.pageIndex = page;
|
|
326
|
+
this.isInitialLoading = false;
|
|
327
|
+
this.isLoadingMore = false;
|
|
328
|
+
this.cdr.markForCheck();
|
|
329
|
+
}
|
|
138
330
|
}
|
|
139
331
|
AssignEnvironmentsDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AssignEnvironmentsDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
140
|
-
AssignEnvironmentsDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AssignEnvironmentsDialogComponent, selector: "cqa-assign-environments-dialog", inputs: { mode: "mode", profileName: "profileName",
|
|
332
|
+
AssignEnvironmentsDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AssignEnvironmentsDialogComponent, selector: "cqa-assign-environments-dialog", inputs: { mode: "mode", profileName: "profileName", searchFn: "searchFn", pageSize: "pageSize", assignedEnvironments: "assignedEnvironments", defaultSourceEnvId: "defaultSourceEnvId" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "listScrollRef", first: true, predicate: ["listScroll"], descendants: true }], ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-w-full\">\n\n <!-- All / Selected (N) toggle. Visible only when source picking is enabled\n (otherwise this is a single-purpose pick list and a toggle is noise). -->\n <div *ngIf=\"showPerRowSource\" class=\"cqa-flex cqa-items-center\">\n <cqa-segment-control\n [segments]=\"viewSegments\"\n [value]=\"viewMode\"\n (valueChange)=\"onViewModeChange($event)\">\n </cqa-segment-control>\n </div>\n\n <div *ngIf=\"viewMode === 'all'\">\n <cqa-search-bar\n placeholder=\"Search environments...\"\n [value]=\"searchTerm\"\n [showClear]=\"true\"\n [fullWidth]=\"true\"\n (valueChange)=\"onSearchChange($event)\"\n (search)=\"onSearchChange($event)\"\n (cleared)=\"onSearchChange('')\">\n </cqa-search-bar>\n </div>\n\n <!-- Loading skeleton \u2014 only relevant in the searchable \"All\" view. -->\n <div *ngIf=\"viewMode === 'all' && isInitialLoading\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <div *ngFor=\"let _ of skeletonRows\"\n class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3.5 cqa-py-3 cqa-rounded-[10px]\"\n [style.border]=\"'1px solid #E5E7EB'\"\n [style.background]=\"'#FFFFFF'\">\n <div class=\"cqa-aed-shimmer cqa-rounded-[4px]\" [style.width.px]=\"16\" [style.height.px]=\"16\"></div>\n <div class=\"cqa-aed-shimmer cqa-rounded-full\" [style.width.px]=\"10\" [style.height.px]=\"10\"></div>\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-flex-1\">\n <div class=\"cqa-aed-shimmer cqa-rounded-[3px]\" style=\"height: 10px; width: 160px;\"></div>\n <div class=\"cqa-aed-shimmer cqa-rounded-[3px]\" style=\"height: 8px; width: 220px;\"></div>\n </div>\n </div>\n </div>\n\n <div\n *ngIf=\"viewMode === 'all' && !isInitialLoading && isEmpty\"\n class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-[13px]\"\n [style.color]=\"'#64748B'\">\n <ng-container *ngIf=\"searchTerm; else noEnvsTpl\">\n No environments match \"{{ searchTerm }}\".\n </ng-container>\n <ng-template #noEnvsTpl>\n No environments available to assign.\n </ng-template>\n </div>\n\n <div\n *ngIf=\"viewMode === 'selected' && selectedSnapshot.length === 0\"\n class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-[13px]\"\n [style.color]=\"'#64748B'\">\n Nothing selected yet \u2014 switch back to \"All\" to pick environments.\n </div>\n\n <div\n *ngIf=\"(viewMode === 'all' && !isInitialLoading && !isEmpty) || (viewMode === 'selected' && selectedSnapshot.length > 0)\"\n #listScroll\n class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-overflow-y-auto cqa-overflow-x-hidden\"\n style=\"max-height: 420px;\"\n (scroll)=\"onListScroll()\">\n\n <div\n *ngFor=\"let env of displayedRows; trackBy: trackById\"\n class=\"cqa-flex cqa-flex-col cqa-rounded-[10px] cqa-transition-colors cqa-min-w-0\"\n [style.border]=\"'1px solid ' + (isSelected(env.id) ? '#8A8CF4' : '#E5E7EB')\"\n [style.background]=\"isSelected(env.id) ? '#EEF0FF' : '#FFFFFF'\"\n [style.opacity]=\"env.alreadyAssigned ? 0.5 : 1\">\n\n <!-- Click-target region: checkbox + name/desc ONLY. The source picker\n below is a SIBLING div (not nested in this region) so clicks on\n the dropdown never trigger the toggle handler. NB: don't wrap the\n whole row in a <label> \u2014 the label's native click \u2192 checkbox\n association fires even when bubbling is stopped. -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3.5 cqa-py-3\"\n [style.cursor]=\"env.alreadyAssigned ? 'not-allowed' : 'pointer'\"\n (click)=\"toggle(env)\">\n <input\n type=\"checkbox\"\n class=\"cqa-cursor-pointer cqa-flex-none\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [checked]=\"isSelected(env.id)\"\n [disabled]=\"!!env.alreadyAssigned\"\n (click)=\"$event.stopPropagation(); toggle(env)\"\n [attr.aria-label]=\"env.alreadyAssigned ? env.name + ' (already assigned)' : 'Select ' + env.name\" />\n\n <span\n class=\"cqa-inline-block cqa-rounded-full cqa-flex-none\"\n [style.width.px]=\"10\"\n [style.height.px]=\"10\"\n [style.background]=\"env.color || '#3F43EE'\"></span>\n\n <div class=\"cqa-flex cqa-flex-col cqa-min-w-0 cqa-flex-1\">\n <div class=\"cqa-text-[13px] cqa-font-medium cqa-truncate\" [style.color]=\"'#0F172A'\">{{ env.name }}</div>\n <div *ngIf=\"helperFor(env)\"\n class=\"cqa-text-[11px] cqa-leading-[1.4]\"\n style=\"color: #64748B; display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical; overflow: hidden; word-break: break-word; overflow-wrap: anywhere;\"\n [innerHTML]=\"helperFor(env)\"></div>\n </div>\n </div>\n\n <!-- Inline \"Copy from which environment\" picker \u2014 sits inside the card,\n below the head. Sibling of the head (not nested) so interacting with\n the dropdown never reaches the toggle handler. -->\n <div\n *ngIf=\"showPerRowSource && isSelected(env.id) && !env.alreadyAssigned\"\n class=\"cqa-aed-source cqa-flex cqa-flex-col cqa-gap-1.5 cqa-px-3.5 cqa-pb-3 cqa-pt-2.5\"\n style=\"border-top: 1px dashed #C7D2FE;\"\n (click)=\"$event.stopPropagation()\">\n <div class=\"cqa-text-[12px] cqa-font-semibold\" [style.color]=\"'#4338CA'\">Copy from which environment</div>\n\n <div *ngIf=\"getSourceConfig(env.id) as cfg\">\n <cqa-dynamic-select\n [form]=\"sourceForm\"\n [config]=\"cfg\">\n </cqa-dynamic-select>\n </div>\n\n <div\n *ngIf=\"pickedSourceName(env.id) as pickedName; else pickPromptTpl\"\n class=\"cqa-flex cqa-items-start cqa-gap-1.5 cqa-text-[11px] cqa-leading-[1.4]\"\n [style.color]=\"'#15803D'\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size: 14px; width: 14px; height: 14px; color: #16A34A;\">check_circle</mat-icon>\n <span>\n Rows will be copied from <strong>{{ pickedName }}</strong> into <strong>{{ env.name }}</strong>.\n </span>\n </div>\n <ng-template #pickPromptTpl>\n <div class=\"cqa-text-[11px] cqa-leading-[1.4]\" [style.color]=\"'#64748B'\">\n Pick where this environment's rows should be copied from.\n </div>\n </ng-template>\n </div>\n </div>\n\n <!-- Loading-more indicator at list bottom while paginating (All view only). -->\n <div *ngIf=\"viewMode === 'all' && isLoadingMore\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-gap-2 cqa-py-3 cqa-text-[12px]\"\n [style.color]=\"'#64748B'\"\n aria-live=\"polite\">\n <span class=\"cqa-aed-spinner\" aria-hidden=\"true\"></span>\n Loading more environments\u2026\n </div>\n\n <!-- End-of-list marker when the user has scrolled through all results. -->\n <div\n *ngIf=\"viewMode === 'all' && !isLoadingMore && !hasMore && currentResults.length > 0\"\n class=\"cqa-text-center cqa-py-2 cqa-text-[11px]\"\n [style.color]=\"'#94A3B8'\">\n End of list \u2014 {{ currentResults.length }} of {{ totalElements }} shown.\n </div>\n </div>\n\n <!-- Inline note: tells the user why the primary button stays disabled. -->\n <div\n *ngIf=\"showPerRowSource && selected.size > 0 && unresolvedSourcesCount > 0\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-px-3 cqa-py-2 cqa-rounded-[8px] cqa-text-[12px]\"\n style=\"background: #FEF3C7; color: #92400E; border: 1px solid #FDE68A;\">\n Pick a source for {{ unresolvedSourcesCount }} environment{{ unresolvedSourcesCount === 1 ? '' : 's' }} before continuing.\n </div>\n</div>\n", components: [{ type: i1.SegmentControlComponent, selector: "cqa-segment-control", inputs: ["segments", "value", "disabled", "containerBgColor", "fullWidth", "size"], outputs: ["valueChange"] }, { type: i2.SearchBarComponent, selector: "cqa-search-bar", inputs: ["placeholder", "value", "disabled", "showClear", "ariaLabel", "autoFocus", "size", "fullWidth"], outputs: ["valueChange", "search", "cleared"] }, { type: i3.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i4.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
141
333
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AssignEnvironmentsDialogComponent, decorators: [{
|
|
142
334
|
type: Component,
|
|
143
|
-
args: [{ selector: 'cqa-assign-environments-dialog', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-
|
|
335
|
+
args: [{ selector: 'cqa-assign-environments-dialog', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-w-full\">\n\n <!-- All / Selected (N) toggle. Visible only when source picking is enabled\n (otherwise this is a single-purpose pick list and a toggle is noise). -->\n <div *ngIf=\"showPerRowSource\" class=\"cqa-flex cqa-items-center\">\n <cqa-segment-control\n [segments]=\"viewSegments\"\n [value]=\"viewMode\"\n (valueChange)=\"onViewModeChange($event)\">\n </cqa-segment-control>\n </div>\n\n <div *ngIf=\"viewMode === 'all'\">\n <cqa-search-bar\n placeholder=\"Search environments...\"\n [value]=\"searchTerm\"\n [showClear]=\"true\"\n [fullWidth]=\"true\"\n (valueChange)=\"onSearchChange($event)\"\n (search)=\"onSearchChange($event)\"\n (cleared)=\"onSearchChange('')\">\n </cqa-search-bar>\n </div>\n\n <!-- Loading skeleton \u2014 only relevant in the searchable \"All\" view. -->\n <div *ngIf=\"viewMode === 'all' && isInitialLoading\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <div *ngFor=\"let _ of skeletonRows\"\n class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3.5 cqa-py-3 cqa-rounded-[10px]\"\n [style.border]=\"'1px solid #E5E7EB'\"\n [style.background]=\"'#FFFFFF'\">\n <div class=\"cqa-aed-shimmer cqa-rounded-[4px]\" [style.width.px]=\"16\" [style.height.px]=\"16\"></div>\n <div class=\"cqa-aed-shimmer cqa-rounded-full\" [style.width.px]=\"10\" [style.height.px]=\"10\"></div>\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-flex-1\">\n <div class=\"cqa-aed-shimmer cqa-rounded-[3px]\" style=\"height: 10px; width: 160px;\"></div>\n <div class=\"cqa-aed-shimmer cqa-rounded-[3px]\" style=\"height: 8px; width: 220px;\"></div>\n </div>\n </div>\n </div>\n\n <div\n *ngIf=\"viewMode === 'all' && !isInitialLoading && isEmpty\"\n class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-[13px]\"\n [style.color]=\"'#64748B'\">\n <ng-container *ngIf=\"searchTerm; else noEnvsTpl\">\n No environments match \"{{ searchTerm }}\".\n </ng-container>\n <ng-template #noEnvsTpl>\n No environments available to assign.\n </ng-template>\n </div>\n\n <div\n *ngIf=\"viewMode === 'selected' && selectedSnapshot.length === 0\"\n class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-[13px]\"\n [style.color]=\"'#64748B'\">\n Nothing selected yet \u2014 switch back to \"All\" to pick environments.\n </div>\n\n <div\n *ngIf=\"(viewMode === 'all' && !isInitialLoading && !isEmpty) || (viewMode === 'selected' && selectedSnapshot.length > 0)\"\n #listScroll\n class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-overflow-y-auto cqa-overflow-x-hidden\"\n style=\"max-height: 420px;\"\n (scroll)=\"onListScroll()\">\n\n <div\n *ngFor=\"let env of displayedRows; trackBy: trackById\"\n class=\"cqa-flex cqa-flex-col cqa-rounded-[10px] cqa-transition-colors cqa-min-w-0\"\n [style.border]=\"'1px solid ' + (isSelected(env.id) ? '#8A8CF4' : '#E5E7EB')\"\n [style.background]=\"isSelected(env.id) ? '#EEF0FF' : '#FFFFFF'\"\n [style.opacity]=\"env.alreadyAssigned ? 0.5 : 1\">\n\n <!-- Click-target region: checkbox + name/desc ONLY. The source picker\n below is a SIBLING div (not nested in this region) so clicks on\n the dropdown never trigger the toggle handler. NB: don't wrap the\n whole row in a <label> \u2014 the label's native click \u2192 checkbox\n association fires even when bubbling is stopped. -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3.5 cqa-py-3\"\n [style.cursor]=\"env.alreadyAssigned ? 'not-allowed' : 'pointer'\"\n (click)=\"toggle(env)\">\n <input\n type=\"checkbox\"\n class=\"cqa-cursor-pointer cqa-flex-none\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [checked]=\"isSelected(env.id)\"\n [disabled]=\"!!env.alreadyAssigned\"\n (click)=\"$event.stopPropagation(); toggle(env)\"\n [attr.aria-label]=\"env.alreadyAssigned ? env.name + ' (already assigned)' : 'Select ' + env.name\" />\n\n <span\n class=\"cqa-inline-block cqa-rounded-full cqa-flex-none\"\n [style.width.px]=\"10\"\n [style.height.px]=\"10\"\n [style.background]=\"env.color || '#3F43EE'\"></span>\n\n <div class=\"cqa-flex cqa-flex-col cqa-min-w-0 cqa-flex-1\">\n <div class=\"cqa-text-[13px] cqa-font-medium cqa-truncate\" [style.color]=\"'#0F172A'\">{{ env.name }}</div>\n <div *ngIf=\"helperFor(env)\"\n class=\"cqa-text-[11px] cqa-leading-[1.4]\"\n style=\"color: #64748B; display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical; overflow: hidden; word-break: break-word; overflow-wrap: anywhere;\"\n [innerHTML]=\"helperFor(env)\"></div>\n </div>\n </div>\n\n <!-- Inline \"Copy from which environment\" picker \u2014 sits inside the card,\n below the head. Sibling of the head (not nested) so interacting with\n the dropdown never reaches the toggle handler. -->\n <div\n *ngIf=\"showPerRowSource && isSelected(env.id) && !env.alreadyAssigned\"\n class=\"cqa-aed-source cqa-flex cqa-flex-col cqa-gap-1.5 cqa-px-3.5 cqa-pb-3 cqa-pt-2.5\"\n style=\"border-top: 1px dashed #C7D2FE;\"\n (click)=\"$event.stopPropagation()\">\n <div class=\"cqa-text-[12px] cqa-font-semibold\" [style.color]=\"'#4338CA'\">Copy from which environment</div>\n\n <div *ngIf=\"getSourceConfig(env.id) as cfg\">\n <cqa-dynamic-select\n [form]=\"sourceForm\"\n [config]=\"cfg\">\n </cqa-dynamic-select>\n </div>\n\n <div\n *ngIf=\"pickedSourceName(env.id) as pickedName; else pickPromptTpl\"\n class=\"cqa-flex cqa-items-start cqa-gap-1.5 cqa-text-[11px] cqa-leading-[1.4]\"\n [style.color]=\"'#15803D'\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size: 14px; width: 14px; height: 14px; color: #16A34A;\">check_circle</mat-icon>\n <span>\n Rows will be copied from <strong>{{ pickedName }}</strong> into <strong>{{ env.name }}</strong>.\n </span>\n </div>\n <ng-template #pickPromptTpl>\n <div class=\"cqa-text-[11px] cqa-leading-[1.4]\" [style.color]=\"'#64748B'\">\n Pick where this environment's rows should be copied from.\n </div>\n </ng-template>\n </div>\n </div>\n\n <!-- Loading-more indicator at list bottom while paginating (All view only). -->\n <div *ngIf=\"viewMode === 'all' && isLoadingMore\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-gap-2 cqa-py-3 cqa-text-[12px]\"\n [style.color]=\"'#64748B'\"\n aria-live=\"polite\">\n <span class=\"cqa-aed-spinner\" aria-hidden=\"true\"></span>\n Loading more environments\u2026\n </div>\n\n <!-- End-of-list marker when the user has scrolled through all results. -->\n <div\n *ngIf=\"viewMode === 'all' && !isLoadingMore && !hasMore && currentResults.length > 0\"\n class=\"cqa-text-center cqa-py-2 cqa-text-[11px]\"\n [style.color]=\"'#94A3B8'\">\n End of list \u2014 {{ currentResults.length }} of {{ totalElements }} shown.\n </div>\n </div>\n\n <!-- Inline note: tells the user why the primary button stays disabled. -->\n <div\n *ngIf=\"showPerRowSource && selected.size > 0 && unresolvedSourcesCount > 0\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-px-3 cqa-py-2 cqa-rounded-[8px] cqa-text-[12px]\"\n style=\"background: #FEF3C7; color: #92400E; border: 1px solid #FDE68A;\">\n Pick a source for {{ unresolvedSourcesCount }} environment{{ unresolvedSourcesCount === 1 ? '' : 's' }} before continuing.\n </div>\n</div>\n" }]
|
|
144
336
|
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { mode: [{
|
|
145
337
|
type: Input
|
|
146
338
|
}], profileName: [{
|
|
147
339
|
type: Input
|
|
148
|
-
}],
|
|
340
|
+
}], searchFn: [{
|
|
341
|
+
type: Input
|
|
342
|
+
}], pageSize: [{
|
|
149
343
|
type: Input
|
|
150
344
|
}], assignedEnvironments: [{
|
|
151
345
|
type: Input
|
|
152
346
|
}], defaultSourceEnvId: [{
|
|
153
347
|
type: Input
|
|
348
|
+
}], listScrollRef: [{
|
|
349
|
+
type: ViewChild,
|
|
350
|
+
args: ['listScroll', { static: false }]
|
|
154
351
|
}] } });
|
|
155
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assign-environments-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/assign-environments-dialog/assign-environments-dialog.component.ts","../../../../../src/lib/assign-environments-dialog/assign-environments-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\n\nimport {\n  DynamicSelectFieldConfig,\n  SelectOption,\n} from '../dynamic-select/dynamic-select-field.component';\nimport {\n  AssignEnvironmentOption,\n  AssignEnvironmentsDialogValue,\n  AssignEnvironmentsMode,\n} from './assign-environments-dialog.models';\n\n@Component({\n  selector: 'cqa-assign-environments-dialog',\n  templateUrl: './assign-environments-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class AssignEnvironmentsDialogComponent implements OnInit {\n  @Input() mode: AssignEnvironmentsMode = 'assign';\n  @Input() profileName: string = '';\n  @Input() environments: AssignEnvironmentOption[] = [];\n  @Input() assignedEnvironments: AssignEnvironmentOption[] = [];\n  @Input() defaultSourceEnvId?: number;\n\n  public readonly selected = new Set<number>();\n\n  public readonly sourceForm = new FormGroup({\n    sourceEnvId: new FormControl(null),\n  });\n  public sourceConfig: DynamicSelectFieldConfig = this.buildSourceConfig([]);\n\n  public sourceError: string | null = null;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    // Source picker appears in both modes when there are already-attached envs\n    // to copy from. In clone mode it's required; in assign mode it's optional\n    // (falling back to the base profile rows when no source is picked).\n    if (this.showSourcePicker) {\n      const initial = this.defaultSourceEnvId != null\n        ? this.defaultSourceEnvId\n        : (this.mode === 'clone' && this.assignedEnvironments && this.assignedEnvironments.length\n          ? this.assignedEnvironments[0].id\n          : null);\n      this.sourceForm.get('sourceEnvId')!.setValue(initial);\n      this.sourceConfig = this.buildSourceConfig(this.assignedEnvironments ?? []);\n    }\n  }\n\n  public get showSourcePicker(): boolean {\n    return (this.assignedEnvironments && this.assignedEnvironments.length > 0);\n  }\n\n  public get sourceLabel(): string {\n    return this.mode === 'clone' ? 'Copy from which environment' : 'Copy rows from (optional)';\n  }\n\n  public get sourceHelper(): string {\n    if (this.sourceError) { return this.sourceError; }\n    return this.mode === 'clone'\n      ? \"The selected environment's rows will be copied into each target environment below.\"\n      : 'Pick an environment to copy its rows into the new ones. Leave blank to seed with the base profile rows.';\n  }\n\n  public get title(): string {\n    return this.mode === 'clone' ? 'Clone to other environments' : 'Assign to environments';\n  }\n\n  public get subtitle(): string {\n    const name = this.profileName ? `\"${this.profileName}\"` : 'this profile';\n    return this.mode === 'clone'\n      ? `Duplicate ${name} into other environments. Column structure is shared; row values are independent per environment.`\n      : `Make ${name} available in the selected environments. Columns are shared — each environment gets an independent row set.`;\n  }\n\n  public get primaryButtonLabel(): string {\n    const count = this.selected.size;\n    const verb = this.mode === 'clone' ? 'Clone to' : 'Assign to';\n    return `${verb} ${count} environment${count === 1 ? '' : 's'}`;\n  }\n\n  public get primaryDisabled(): boolean {\n    if (this.selected.size === 0) { return true; }\n    if (this.mode === 'clone' && this.sourceForm.get('sourceEnvId')!.value == null) { return true; }\n    return false;\n  }\n\n  public get selectableEnvironments(): AssignEnvironmentOption[] {\n    return this.environments ?? [];\n  }\n\n  public get hasAnySelectable(): boolean {\n    return (this.environments ?? []).some(e => !e.alreadyAssigned);\n  }\n\n  public get sourceValue(): number | null {\n    return this.sourceForm.get('sourceEnvId')!.value ?? null;\n  }\n\n  public isSelected(id: number): boolean {\n    return this.selected.has(id);\n  }\n\n  public toggle(env: AssignEnvironmentOption): void {\n    if (env.alreadyAssigned) { return; }\n    if (this.selected.has(env.id)) {\n      this.selected.delete(env.id);\n    } else {\n      this.selected.add(env.id);\n    }\n    this.cdr.markForCheck();\n  }\n\n  public helperFor(env: AssignEnvironmentOption): string {\n    if (env.alreadyAssigned) { return 'Already assigned to this environment'; }\n    return env.description || '';\n  }\n\n  public getValue(): AssignEnvironmentsDialogValue | null {\n    if (this.selected.size === 0) { return null; }\n    const src = this.sourceValue;\n    if (this.mode === 'clone') {\n      if (src == null) {\n        this.sourceError = 'Pick the environment whose rows should be copied.';\n        this.cdr.markForCheck();\n        return null;\n      }\n      this.sourceError = null;\n      return { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) };\n    }\n    // Assign mode: source is optional — include it when the user picked one.\n    this.sourceError = null;\n    return src != null\n      ? { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) }\n      : { selectedIds: Array.from(this.selected) };\n  }\n\n  private buildSourceConfig(envs: AssignEnvironmentOption[]): DynamicSelectFieldConfig {\n    const options: SelectOption[] = (envs || []).map(e => ({\n      id: e.id,\n      value: e.id,\n      name: e.name,\n      label: e.name,\n      statusColor: e.color,\n    }));\n    return {\n      key: 'sourceEnvId',\n      label: '',\n      placeholder: 'Select source environment…',\n      multiple: false,\n      searchable: true,\n      options,\n    };\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n  <!-- Source env picker: required in clone mode, optional in assign mode -->\n  <div *ngIf=\"showSourcePicker\" class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">{{ sourceLabel }}</label>\n    <cqa-dynamic-select\n      [form]=\"sourceForm\"\n      [config]=\"sourceConfig\">\n    </cqa-dynamic-select>\n    <span [class.cqa-text-red-600]=\"sourceError\" [class.cqa-text-gray-500]=\"!sourceError\" class=\"cqa-text-xs\">\n      {{ sourceHelper }}\n    </span>\n  </div>\n\n  <!-- Selectable envs -->\n  <div\n    *ngIf=\"hasAnySelectable || (environments?.length || 0) > 0\"\n    class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[340px] cqa-overflow-y-auto\">\n    <label\n      *ngFor=\"let env of selectableEnvironments\"\n      class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3 cqa-py-2.5 cqa-border cqa-rounded-[10px] cqa-bg-white cqa-transition-colors\"\n      [class.cqa-border-gray-200]=\"!isSelected(env.id) && !env.alreadyAssigned\"\n      [class.hover:cqa-border-indigo-200]=\"!env.alreadyAssigned && !isSelected(env.id)\"\n      [class.cqa-border-indigo-400]=\"isSelected(env.id)\"\n      [class.cqa-bg-indigo-50]=\"isSelected(env.id)\"\n      [class.cqa-opacity-50]=\"env.alreadyAssigned\"\n      [class.cqa-cursor-pointer]=\"!env.alreadyAssigned\"\n      [class.cqa-cursor-not-allowed]=\"env.alreadyAssigned\"\n      (click)=\"toggle(env)\">\n      <input\n        type=\"checkbox\"\n        class=\"cqa-w-4 cqa-h-4 cqa-cursor-pointer\"\n        [checked]=\"isSelected(env.id)\"\n        [disabled]=\"!!env.alreadyAssigned\"\n        (click)=\"$event.stopPropagation()\"\n        (change)=\"toggle(env)\" />\n\n      <span\n        class=\"cqa-inline-block cqa-w-2 cqa-h-2 cqa-rounded-full cqa-flex-none\"\n        [style.background]=\"env.color || '#3F43EE'\"></span>\n\n      <div class=\"cqa-flex cqa-flex-col cqa-min-w-0 cqa-flex-1\">\n        <div class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">{{ env.name }}</div>\n        <div *ngIf=\"helperFor(env)\" class=\"cqa-text-xs cqa-text-gray-500 cqa-leading-[1.4]\">{{ helperFor(env) }}</div>\n      </div>\n    </label>\n  </div>\n\n  <div\n    *ngIf=\"(environments?.length || 0) === 0\"\n    class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-sm cqa-text-gray-500\">\n    All environments are already assigned to this profile.\n  </div>\n</div>\n"]}
|
|
352
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assign-environments-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/assign-environments-dialog/assign-environments-dialog.component.ts","../../../../../src/lib/assign-environments-dialog/assign-environments-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  OnDestroy,\n  OnInit,\n  ViewChild,\n} from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { Observable, Subject, of } from 'rxjs';\nimport {\n  catchError,\n  debounceTime,\n  distinctUntilChanged,\n  switchMap,\n  takeUntil,\n} from 'rxjs/operators';\n\nimport {\n  DynamicSelectFieldConfig,\n  SelectOption,\n} from '../dynamic-select/dynamic-select-field.component';\nimport { SegmentOption } from '../segment-control/segment-control.component';\nimport {\n  AssignEnvSearchFn,\n  AssignEnvSearchPage,\n  AssignEnvironmentOption,\n  AssignEnvironmentsDialogValue,\n  AssignEnvironmentsMode,\n} from './assign-environments-dialog.models';\n\ntype ViewMode = 'all' | 'selected';\n\n@Component({\n  selector: 'cqa-assign-environments-dialog',\n  templateUrl: './assign-environments-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class AssignEnvironmentsDialogComponent implements OnInit, OnDestroy {\n  @Input() mode: AssignEnvironmentsMode = 'assign';\n  @Input() profileName: string = '';\n\n  /** Paginated search callback. Fires `('', 0, pageSize)` on open and on every\n   *  scroll-near-bottom (next page) or distinct search term (page reset). */\n  @Input() searchFn?: AssignEnvSearchFn;\n  @Input() pageSize = 50;\n\n  /** Source-env candidates the per-row \"Copy from which environment\" picker\n   *  populates from. Synchronous list — typically bounded by the profile's\n   *  current attachments (small, often < 10). */\n  @Input() assignedEnvironments: AssignEnvironmentOption[] = [];\n\n  /** Pre-select this source for newly-toggled targets. */\n  @Input() defaultSourceEnvId?: number;\n\n  public readonly selected = new Set<number>();\n  public selectedSnapshot: AssignEnvironmentOption[] = [];\n\n  public searchTerm = '';\n  public isInitialLoading = false;\n  public isLoadingMore = false;\n  public currentResults: AssignEnvironmentOption[] = [];\n  public totalElements = 0;\n\n  public viewMode: ViewMode = 'all';\n  /** Stable segment array — recomputed only when selection size flips zero ↔ non-zero. */\n  public viewSegments: SegmentOption[] = this.computeViewSegments();\n\n  /** One FormGroup, controls keyed `tgt_${targetEnvId}`. */\n  public readonly sourceForm = new FormGroup({});\n  public readonly sourceConfigById = new Map<number, DynamicSelectFieldConfig>();\n\n  public readonly skeletonRows = [0, 1, 2, 3, 4];\n\n  @ViewChild('listScroll', { static: false }) private listScrollRef?: ElementRef<HTMLElement>;\n\n  private pageIndex = 0;\n  private readonly searchInput$ = new Subject<string>();\n  private readonly destroy$ = new Subject<void>();\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    // Search-term pipeline: every distinct, debounced term resets the page and\n    // replaces the visible list.\n    this.searchInput$\n      .pipe(\n        debounceTime(300),\n        distinctUntilChanged(),\n        switchMap(term => this.fetchPage(term, 0)),\n        takeUntil(this.destroy$),\n      )\n      .subscribe(({ term, page, payload }) => {\n        this.applyPage(term, page, /* append */ false, payload);\n      });\n\n    // Initial page-0 fetch on dialog open.\n    this.runFetch('', 0, /* append */ false);\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  // -- Public template state -----------------------------------------------\n\n  public get title(): string {\n    return this.mode === 'clone' ? 'Clone to other environments' : 'Assign to environments';\n  }\n\n  public get subtitle(): string {\n    const name = this.profileName ? `\"${this.profileName}\"` : 'this profile';\n    return this.mode === 'clone'\n      ? `Duplicate ${name} into other environments. Column structure is shared; row values are independent per environment.`\n      : `Make ${name} available in the selected environments. Columns are shared — each environment gets an independent row set.`;\n  }\n\n  public get showPerRowSource(): boolean {\n    return (this.assignedEnvironments?.length ?? 0) > 0;\n  }\n\n  public get isEmpty(): boolean {\n    return !this.currentResults || this.currentResults.length === 0;\n  }\n\n  public get hasMore(): boolean {\n    return this.currentResults.length < this.totalElements;\n  }\n\n  public get displayedRows(): AssignEnvironmentOption[] {\n    return this.viewMode === 'selected' ? this.selectedSnapshot : this.currentResults;\n  }\n\n  public get unresolvedSourcesCount(): number {\n    if (!this.showPerRowSource) { return 0; }\n    let n = 0;\n    for (const env of this.selectedSnapshot) {\n      if (env.alreadyAssigned) { continue; }\n      if (!this.isSourceResolved(env.id)) { n++; }\n    }\n    return n;\n  }\n\n  public get primaryButtonLabel(): string {\n    const count = this.selected.size;\n    const verb = this.mode === 'clone' ? 'Clone to' : 'Assign to';\n    return `${verb} ${count} environment${count === 1 ? '' : 's'}`;\n  }\n\n  public get primaryDisabled(): boolean {\n    if (this.selected.size === 0) { return true; }\n    if (this.unresolvedSourcesCount > 0) { return true; }\n    return false;\n  }\n\n  public isSelected(id: number): boolean {\n    return this.selected.has(id);\n  }\n\n  public isSourceResolved(targetEnvId: number): boolean {\n    if (!this.showPerRowSource) { return true; }\n    const ctrl = this.sourceForm.get(this.controlKey(targetEnvId));\n    return ctrl?.value != null;\n  }\n\n  public getSourceConfig(targetEnvId: number): DynamicSelectFieldConfig | undefined {\n    return this.sourceConfigById.get(targetEnvId);\n  }\n\n  public pickedSourceName(targetEnvId: number): string {\n    const value = this.sourceForm.get(this.controlKey(targetEnvId))?.value;\n    if (value == null) { return ''; }\n    const env = (this.assignedEnvironments || []).find(e => Number(e.id) === Number(value));\n    return env ? env.name : '';\n  }\n\n  public helperFor(env: AssignEnvironmentOption): string {\n    if (env.alreadyAssigned) { return 'Already assigned to this environment'; }\n    return env.description || '';\n  }\n\n  public trackById(_index: number, env: AssignEnvironmentOption): number {\n    return env.id;\n  }\n\n  // -- Public template events ---------------------------------------------\n\n  public onSearchChange(term: string): void {\n    this.searchTerm = term ?? '';\n    this.isInitialLoading = true;\n    this.currentResults = [];\n    this.cdr.markForCheck();\n    this.searchInput$.next(this.searchTerm);\n  }\n\n  public onViewModeChange(value: string): void {\n    this.viewMode = value === 'selected' ? 'selected' : 'all';\n    this.cdr.markForCheck();\n  }\n\n  public toggle(env: AssignEnvironmentOption): void {\n    if (env?.alreadyAssigned) { return; }\n    const id = Number(env.id);\n    const wasZero = this.selected.size === 0;\n    if (this.selected.has(id)) {\n      this.selected.delete(id);\n      this.selectedSnapshot = this.selectedSnapshot.filter(e => Number(e.id) !== id);\n      this.removeSourceState(id);\n      // If the Selected view just emptied, fall back to All so the user\n      // doesn't sit on a blank screen.\n      if (this.viewMode === 'selected' && this.selectedSnapshot.length === 0) {\n        this.viewMode = 'all';\n      }\n    } else {\n      this.selected.add(id);\n      this.selectedSnapshot = [...this.selectedSnapshot, env];\n      this.ensureSourceState(id);\n    }\n    const isZero = this.selected.size === 0;\n    if (wasZero !== isZero) {\n      this.viewSegments = this.computeViewSegments();\n    } else {\n      // Keep the count up to date even when zero ↔ non-zero didn't flip.\n      this.viewSegments = this.computeViewSegments();\n    }\n    this.cdr.markForCheck();\n  }\n\n  public onListScroll(): void {\n    const el = this.listScrollRef?.nativeElement;\n    if (!el) { return; }\n    if (this.viewMode === 'selected') { return; }\n    if (this.isInitialLoading || this.isLoadingMore) { return; }\n    if (!this.hasMore) { return; }\n    const distanceFromBottom = el.scrollHeight - (el.scrollTop + el.clientHeight);\n    if (distanceFromBottom < 80) {\n      this.runFetch(this.searchTerm, this.pageIndex + 1, /* append */ true);\n    }\n  }\n\n  public getValue(): AssignEnvironmentsDialogValue | null {\n    if (this.selected.size === 0) { return null; }\n    if (this.unresolvedSourcesCount > 0) { return null; }\n\n    if (!this.showPerRowSource) {\n      return { selectedIds: Array.from(this.selected), sources: [] };\n    }\n\n    const sources = Array.from(this.selected).map(id => {\n      const value = this.sourceForm.get(this.controlKey(id))?.value;\n      return { targetEnvId: id, sourceEnvId: Number(value) };\n    });\n    return { selectedIds: Array.from(this.selected), sources };\n  }\n\n  // -- Internals -----------------------------------------------------------\n\n  private controlKey(targetEnvId: number): string {\n    return `tgt_${targetEnvId}`;\n  }\n\n  private computeViewSegments(): SegmentOption[] {\n    const count = this.selected.size;\n    return [\n      { label: 'All', value: 'all' },\n      count > 0\n        ? { label: 'Selected', value: 'selected', count }\n        : { label: 'Selected', value: 'selected' },\n    ];\n  }\n\n  private ensureSourceState(targetEnvId: number): void {\n    if (!this.showPerRowSource) { return; }\n    if (!this.sourceForm.contains(this.controlKey(targetEnvId))) {\n      this.sourceForm.addControl(this.controlKey(targetEnvId), new FormControl(null));\n    }\n    if (!this.sourceConfigById.has(targetEnvId)) {\n      this.sourceConfigById.set(targetEnvId, this.buildSourceConfig(targetEnvId));\n    }\n    // Pre-select the default source if the caller suggested one (and it's a\n    // valid candidate). Saves the user a click when there's an obvious choice.\n    const ctrl = this.sourceForm.get(this.controlKey(targetEnvId));\n    if (ctrl && ctrl.value == null) {\n      const defaultId = this.resolveDefaultSourceId(targetEnvId);\n      if (defaultId != null) {\n        ctrl.setValue(defaultId);\n      }\n    }\n  }\n\n  private removeSourceState(targetEnvId: number): void {\n    const key = this.controlKey(targetEnvId);\n    if (this.sourceForm.contains(key)) { this.sourceForm.removeControl(key); }\n    this.sourceConfigById.delete(targetEnvId);\n  }\n\n  private buildSourceConfig(targetEnvId: number): DynamicSelectFieldConfig {\n    // Filter the target out of its own source list — copying an env into\n    // itself is non-sensical.\n    const candidates = (this.assignedEnvironments || []).filter(e => Number(e.id) !== Number(targetEnvId));\n    const options: SelectOption[] = candidates.map(e => ({\n      id: e.id,\n      value: e.id,\n      name: e.name,\n      label: e.name,\n      statusColor: e.color,\n    }));\n    return {\n      key: this.controlKey(targetEnvId),\n      label: '',\n      placeholder: 'Select source environment…',\n      multiple: false,\n      searchable: options.length > 6,\n      options,\n    };\n  }\n\n  private resolveDefaultSourceId(targetEnvId: number): number | null {\n    const candidates = (this.assignedEnvironments || []).filter(e => Number(e.id) !== Number(targetEnvId));\n    if (candidates.length === 0) { return null; }\n    if (this.defaultSourceEnvId != null) {\n      const match = candidates.find(e => Number(e.id) === Number(this.defaultSourceEnvId));\n      if (match) { return Number(match.id); }\n    }\n    if (this.mode === 'clone') {\n      // Clone implies a source was already in scope — pick the first attached env.\n      return Number(candidates[0].id);\n    }\n    return null;\n  }\n\n  // -- Fetch pipeline ------------------------------------------------------\n\n  private runFetch(term: string, page: number, append: boolean): void {\n    if (append) {\n      this.isLoadingMore = true;\n    } else {\n      this.isInitialLoading = true;\n    }\n    this.cdr.markForCheck();\n\n    this.fetchPage(term, page)\n      .pipe(takeUntil(this.destroy$))\n      .subscribe(result => {\n        this.applyPage(result.term, result.page, append, result.payload);\n      });\n  }\n\n  private fetchPage(term: string, page: number): Observable<{\n    term: string;\n    page: number;\n    payload: AssignEnvSearchPage;\n  }> {\n    if (!this.searchFn) {\n      return of({ term, page, payload: { items: [], total: 0 } });\n    }\n    return this.searchFn(term, page, this.pageSize).pipe(\n      catchError(() => of<AssignEnvSearchPage>({ items: [], total: 0 })),\n      switchMap(payload => of({ term, page, payload })),\n    );\n  }\n\n  private applyPage(\n    term: string,\n    page: number,\n    append: boolean,\n    payload?: AssignEnvSearchPage,\n  ): void {\n    // Drop responses whose term no longer matches the active search.\n    if (term !== this.searchTerm) { return; }\n\n    const items = payload?.items ?? [];\n    const total = payload?.total ?? items.length;\n\n    if (append) {\n      this.currentResults = [...this.currentResults, ...items];\n    } else {\n      this.currentResults = [...items];\n    }\n    this.totalElements = total;\n    this.pageIndex = page;\n    this.isInitialLoading = false;\n    this.isLoadingMore = false;\n    this.cdr.markForCheck();\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-w-full\">\n\n  <!-- All / Selected (N) toggle. Visible only when source picking is enabled\n       (otherwise this is a single-purpose pick list and a toggle is noise). -->\n  <div *ngIf=\"showPerRowSource\" class=\"cqa-flex cqa-items-center\">\n    <cqa-segment-control\n      [segments]=\"viewSegments\"\n      [value]=\"viewMode\"\n      (valueChange)=\"onViewModeChange($event)\">\n    </cqa-segment-control>\n  </div>\n\n  <div *ngIf=\"viewMode === 'all'\">\n    <cqa-search-bar\n      placeholder=\"Search environments...\"\n      [value]=\"searchTerm\"\n      [showClear]=\"true\"\n      [fullWidth]=\"true\"\n      (valueChange)=\"onSearchChange($event)\"\n      (search)=\"onSearchChange($event)\"\n      (cleared)=\"onSearchChange('')\">\n    </cqa-search-bar>\n  </div>\n\n  <!-- Loading skeleton — only relevant in the searchable \"All\" view. -->\n  <div *ngIf=\"viewMode === 'all' && isInitialLoading\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n    <div *ngFor=\"let _ of skeletonRows\"\n         class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3.5 cqa-py-3 cqa-rounded-[10px]\"\n         [style.border]=\"'1px solid #E5E7EB'\"\n         [style.background]=\"'#FFFFFF'\">\n      <div class=\"cqa-aed-shimmer cqa-rounded-[4px]\" [style.width.px]=\"16\" [style.height.px]=\"16\"></div>\n      <div class=\"cqa-aed-shimmer cqa-rounded-full\" [style.width.px]=\"10\" [style.height.px]=\"10\"></div>\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-flex-1\">\n        <div class=\"cqa-aed-shimmer cqa-rounded-[3px]\" style=\"height: 10px; width: 160px;\"></div>\n        <div class=\"cqa-aed-shimmer cqa-rounded-[3px]\" style=\"height: 8px; width: 220px;\"></div>\n      </div>\n    </div>\n  </div>\n\n  <div\n    *ngIf=\"viewMode === 'all' && !isInitialLoading && isEmpty\"\n    class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-[13px]\"\n    [style.color]=\"'#64748B'\">\n    <ng-container *ngIf=\"searchTerm; else noEnvsTpl\">\n      No environments match \"{{ searchTerm }}\".\n    </ng-container>\n    <ng-template #noEnvsTpl>\n      No environments available to assign.\n    </ng-template>\n  </div>\n\n  <div\n    *ngIf=\"viewMode === 'selected' && selectedSnapshot.length === 0\"\n    class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-[13px]\"\n    [style.color]=\"'#64748B'\">\n    Nothing selected yet — switch back to \"All\" to pick environments.\n  </div>\n\n  <div\n    *ngIf=\"(viewMode === 'all' && !isInitialLoading && !isEmpty) || (viewMode === 'selected' && selectedSnapshot.length > 0)\"\n    #listScroll\n    class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-overflow-y-auto cqa-overflow-x-hidden\"\n    style=\"max-height: 420px;\"\n    (scroll)=\"onListScroll()\">\n\n    <div\n      *ngFor=\"let env of displayedRows; trackBy: trackById\"\n      class=\"cqa-flex cqa-flex-col cqa-rounded-[10px] cqa-transition-colors cqa-min-w-0\"\n      [style.border]=\"'1px solid ' + (isSelected(env.id) ? '#8A8CF4' : '#E5E7EB')\"\n      [style.background]=\"isSelected(env.id) ? '#EEF0FF' : '#FFFFFF'\"\n      [style.opacity]=\"env.alreadyAssigned ? 0.5 : 1\">\n\n      <!-- Click-target region: checkbox + name/desc ONLY. The source picker\n           below is a SIBLING div (not nested in this region) so clicks on\n           the dropdown never trigger the toggle handler. NB: don't wrap the\n           whole row in a <label> — the label's native click → checkbox\n           association fires even when bubbling is stopped. -->\n      <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3.5 cqa-py-3\"\n           [style.cursor]=\"env.alreadyAssigned ? 'not-allowed' : 'pointer'\"\n           (click)=\"toggle(env)\">\n        <input\n          type=\"checkbox\"\n          class=\"cqa-cursor-pointer cqa-flex-none\"\n          [style.width.px]=\"16\"\n          [style.height.px]=\"16\"\n          [checked]=\"isSelected(env.id)\"\n          [disabled]=\"!!env.alreadyAssigned\"\n          (click)=\"$event.stopPropagation(); toggle(env)\"\n          [attr.aria-label]=\"env.alreadyAssigned ? env.name + ' (already assigned)' : 'Select ' + env.name\" />\n\n        <span\n          class=\"cqa-inline-block cqa-rounded-full cqa-flex-none\"\n          [style.width.px]=\"10\"\n          [style.height.px]=\"10\"\n          [style.background]=\"env.color || '#3F43EE'\"></span>\n\n        <div class=\"cqa-flex cqa-flex-col cqa-min-w-0 cqa-flex-1\">\n          <div class=\"cqa-text-[13px] cqa-font-medium cqa-truncate\" [style.color]=\"'#0F172A'\">{{ env.name }}</div>\n          <div *ngIf=\"helperFor(env)\"\n               class=\"cqa-text-[11px] cqa-leading-[1.4]\"\n               style=\"color: #64748B; display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical; overflow: hidden; word-break: break-word; overflow-wrap: anywhere;\"\n               [innerHTML]=\"helperFor(env)\"></div>\n        </div>\n      </div>\n\n      <!-- Inline \"Copy from which environment\" picker — sits inside the card,\n           below the head. Sibling of the head (not nested) so interacting with\n           the dropdown never reaches the toggle handler. -->\n      <div\n        *ngIf=\"showPerRowSource && isSelected(env.id) && !env.alreadyAssigned\"\n        class=\"cqa-aed-source cqa-flex cqa-flex-col cqa-gap-1.5 cqa-px-3.5 cqa-pb-3 cqa-pt-2.5\"\n        style=\"border-top: 1px dashed #C7D2FE;\"\n        (click)=\"$event.stopPropagation()\">\n        <div class=\"cqa-text-[12px] cqa-font-semibold\" [style.color]=\"'#4338CA'\">Copy from which environment</div>\n\n        <div *ngIf=\"getSourceConfig(env.id) as cfg\">\n          <cqa-dynamic-select\n            [form]=\"sourceForm\"\n            [config]=\"cfg\">\n          </cqa-dynamic-select>\n        </div>\n\n        <div\n          *ngIf=\"pickedSourceName(env.id) as pickedName; else pickPromptTpl\"\n          class=\"cqa-flex cqa-items-start cqa-gap-1.5 cqa-text-[11px] cqa-leading-[1.4]\"\n          [style.color]=\"'#15803D'\">\n          <mat-icon class=\"cqa-flex-none\" style=\"font-size: 14px; width: 14px; height: 14px; color: #16A34A;\">check_circle</mat-icon>\n          <span>\n            Rows will be copied from <strong>{{ pickedName }}</strong> into <strong>{{ env.name }}</strong>.\n          </span>\n        </div>\n        <ng-template #pickPromptTpl>\n          <div class=\"cqa-text-[11px] cqa-leading-[1.4]\" [style.color]=\"'#64748B'\">\n            Pick where this environment's rows should be copied from.\n          </div>\n        </ng-template>\n      </div>\n    </div>\n\n    <!-- Loading-more indicator at list bottom while paginating (All view only). -->\n    <div *ngIf=\"viewMode === 'all' && isLoadingMore\"\n         class=\"cqa-flex cqa-items-center cqa-justify-center cqa-gap-2 cqa-py-3 cqa-text-[12px]\"\n         [style.color]=\"'#64748B'\"\n         aria-live=\"polite\">\n      <span class=\"cqa-aed-spinner\" aria-hidden=\"true\"></span>\n      Loading more environments…\n    </div>\n\n    <!-- End-of-list marker when the user has scrolled through all results. -->\n    <div\n      *ngIf=\"viewMode === 'all' && !isLoadingMore && !hasMore && currentResults.length > 0\"\n      class=\"cqa-text-center cqa-py-2 cqa-text-[11px]\"\n      [style.color]=\"'#94A3B8'\">\n      End of list — {{ currentResults.length }} of {{ totalElements }} shown.\n    </div>\n  </div>\n\n  <!-- Inline note: tells the user why the primary button stays disabled. -->\n  <div\n    *ngIf=\"showPerRowSource && selected.size > 0 && unresolvedSourcesCount > 0\"\n    class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-px-3 cqa-py-2 cqa-rounded-[8px] cqa-text-[12px]\"\n    style=\"background: #FEF3C7; color: #92400E; border: 1px solid #FDE68A;\">\n    Pick a source for {{ unresolvedSourcesCount }} environment{{ unresolvedSourcesCount === 1 ? '' : 's' }} before continuing.\n  </div>\n</div>\n"]}
|