@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.
@@ -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 "../dynamic-select/dynamic-select-field.component";
5
- import * as i2 from "@angular/common";
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.environments = [];
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.sourceForm = new FormGroup({
15
- sourceEnvId: new FormControl(null),
16
- });
17
- this.sourceConfig = this.buildSourceConfig([]);
18
- this.sourceError = null;
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
- // Source picker appears in both modes when there are already-attached envs
22
- // to copy from. In clone mode it's required; in assign mode it's optional
23
- // (falling back to the base profile rows when no source is picked).
24
- if (this.showSourcePicker) {
25
- const initial = this.defaultSourceEnvId != null
26
- ? this.defaultSourceEnvId
27
- : (this.mode === 'clone' && this.assignedEnvironments && this.assignedEnvironments.length
28
- ? this.assignedEnvironments[0].id
29
- : null);
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
- get sourceHelper() {
41
- if (this.sourceError) {
42
- return this.sourceError;
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.mode === 'clone' && this.sourceForm.get('sourceEnvId').value == null) {
100
+ if (this.unresolvedSourcesCount > 0) {
67
101
  return true;
68
102
  }
69
103
  return false;
70
104
  }
71
- get selectableEnvironments() {
72
- return this.environments ?? [];
105
+ isSelected(id) {
106
+ return this.selected.has(id);
73
107
  }
74
- get hasAnySelectable() {
75
- return (this.environments ?? []).some(e => !e.alreadyAssigned);
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
- get sourceValue() {
78
- return this.sourceForm.get('sourceEnvId').value ?? null;
115
+ getSourceConfig(targetEnvId) {
116
+ return this.sourceConfigById.get(targetEnvId);
79
117
  }
80
- isSelected(id) {
81
- return this.selected.has(id);
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
- toggle(env) {
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
- if (this.selected.has(env.id)) {
88
- this.selected.delete(env.id);
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
- this.selected.add(env.id);
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
- helperFor(env) {
96
- if (env.alreadyAssigned) {
97
- return 'Already assigned to this environment';
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
- const src = this.sourceValue;
106
- if (this.mode === 'clone') {
107
- if (src == null) {
108
- this.sourceError = 'Pick the environment whose rows should be copied.';
109
- this.cdr.markForCheck();
110
- return null;
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
- buildSourceConfig(envs) {
122
- const options = (envs || []).map(e => ({
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: 'sourceEnvId',
265
+ key: this.controlKey(targetEnvId),
131
266
  label: '',
132
267
  placeholder: 'Select source environment…',
133
268
  multiple: false,
134
- searchable: true,
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", environments: "environments", assignedEnvironments: "assignedEnvironments", defaultSourceEnvId: "defaultSourceEnvId" }, 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 <!-- 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", components: [{ type: i1.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
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-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" }]
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
- }], environments: [{
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"]}