@cqa-lib/cqa-ui 1.1.529 → 1.1.530

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,146 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { FormControl, FormGroup } from '@angular/forms';
3
+ import { Subject } from 'rxjs';
4
+ import { takeUntil } from 'rxjs/operators';
5
+ import { ALL_FILTER_VALUE, } from './audit-log-drawer.models';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/material/icon";
8
+ import * as i2 from "../dynamic-select/dynamic-select-field.component";
9
+ import * as i3 from "../full-table-loader/full-table-loader.component";
10
+ import * as i4 from "../empty-state/empty-state.component";
11
+ import * as i5 from "./audit-log-entry-card.component";
12
+ import * as i6 from "@angular/common";
13
+ export class AuditLogDrawerComponent {
14
+ constructor() {
15
+ this.entries = [];
16
+ this.entityTypeOptions = [];
17
+ this.envOptions = [];
18
+ this.userOptions = [];
19
+ this.defaultRange = '7d';
20
+ this.isLoading = false;
21
+ this.filterChange = new EventEmitter();
22
+ this.close = new EventEmitter();
23
+ this.ALL = ALL_FILTER_VALUE;
24
+ this.filtersForm = new FormGroup({
25
+ entityType: new FormControl(ALL_FILTER_VALUE),
26
+ env: new FormControl(ALL_FILTER_VALUE),
27
+ user: new FormControl(ALL_FILTER_VALUE),
28
+ range: new FormControl('7d'),
29
+ });
30
+ this.trackById = (_, e) => e.id;
31
+ this.destroy$ = new Subject();
32
+ this.rangeMs = {
33
+ '24h': 24 * 60 * 60 * 1000,
34
+ '7d': 7 * 24 * 60 * 60 * 1000,
35
+ '30d': 30 * 24 * 60 * 60 * 1000,
36
+ 'all': Number.POSITIVE_INFINITY,
37
+ };
38
+ }
39
+ ngOnInit() {
40
+ this.filtersForm.patchValue({ range: this.defaultRange });
41
+ this.entityConfig = this.buildSelectConfig('entityType', 'All entity types', this.entityTypeOptions);
42
+ this.envConfig = this.buildSelectConfig('env', 'All environments', this.envOptions);
43
+ this.userConfig = this.buildSelectConfig('user', 'All users', this.userOptions);
44
+ this.rangeConfig = this.buildSelectConfig('range', 'Last 7 days', [
45
+ { value: '24h', label: 'Last 24h' },
46
+ { value: '7d', label: 'Last 7 days' },
47
+ { value: '30d', label: 'Last 30 days' },
48
+ { value: 'all', label: 'All time' },
49
+ ], false);
50
+ this.filtersForm.valueChanges
51
+ .pipe(takeUntil(this.destroy$))
52
+ .subscribe(() => this.filterChange.emit(this.currentFilterState()));
53
+ }
54
+ ngOnDestroy() {
55
+ this.destroy$.next();
56
+ this.destroy$.complete();
57
+ }
58
+ get filteredEntries() {
59
+ const f = this.currentFilterState();
60
+ const cutoff = this.rangeMs[f.range];
61
+ const now = Date.now();
62
+ return (this.entries || []).filter(e => {
63
+ if (f.entityType !== ALL_FILTER_VALUE && e.entityType !== f.entityType) {
64
+ return false;
65
+ }
66
+ if (f.env !== ALL_FILTER_VALUE && !this.entryMatchesEnv(e, f.env)) {
67
+ return false;
68
+ }
69
+ if (f.user !== ALL_FILTER_VALUE && e.user !== f.user) {
70
+ return false;
71
+ }
72
+ if (cutoff !== Number.POSITIVE_INFINITY) {
73
+ const ts = this.parseTs(e.ts);
74
+ if (ts != null && now - ts > cutoff) {
75
+ return false;
76
+ }
77
+ }
78
+ return true;
79
+ });
80
+ }
81
+ onClose() {
82
+ this.close.emit();
83
+ }
84
+ entryMatchesEnv(entry, env) {
85
+ return (entry.entityName || '').toLowerCase().includes(env.toLowerCase());
86
+ }
87
+ parseTs(ts) {
88
+ if (!ts) {
89
+ return null;
90
+ }
91
+ const t = Date.parse(ts);
92
+ return Number.isNaN(t) ? null : t;
93
+ }
94
+ currentFilterState() {
95
+ const v = this.filtersForm.value;
96
+ return {
97
+ entityType: v.entityType ?? ALL_FILTER_VALUE,
98
+ env: v.env ?? ALL_FILTER_VALUE,
99
+ user: v.user ?? ALL_FILTER_VALUE,
100
+ range: v.range ?? '7d',
101
+ };
102
+ }
103
+ buildSelectConfig(key, placeholder, items, includeAll = true) {
104
+ const options = [];
105
+ if (includeAll) {
106
+ options.push({ id: ALL_FILTER_VALUE, value: ALL_FILTER_VALUE, name: placeholder, label: placeholder });
107
+ }
108
+ for (const it of items) {
109
+ options.push({ id: it.value, value: it.value, name: it.label, label: it.label });
110
+ }
111
+ return {
112
+ key,
113
+ label: '',
114
+ placeholder,
115
+ multiple: false,
116
+ searchable: false,
117
+ options,
118
+ };
119
+ }
120
+ }
121
+ AuditLogDrawerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AuditLogDrawerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
122
+ AuditLogDrawerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AuditLogDrawerComponent, selector: "cqa-audit-log-drawer", inputs: { entries: "entries", entityTypeOptions: "entityTypeOptions", envOptions: "envOptions", userOptions: "userOptions", defaultRange: "defaultRange", isLoading: "isLoading" }, outputs: { filterChange: "filterChange", close: "close" }, host: { styleAttribute: "display:flex;flex-direction:column;height:100%;width:560px;max-width:100vw;background:#fff;box-shadow:-8px 0 32px rgba(15,23,42,.12);", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<header class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-3 cqa-px-[22px] cqa-py-[18px] cqa-border-b cqa-border-[#E5E7EB] cqa-flex-shrink-0 cqa-font-inter\">\n <div class=\"cqa-min-w-0\">\n <h3 class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-m-0 cqa-text-[15px] cqa-font-semibold cqa-text-[#0F172A]\">\n <mat-icon class=\"cqa-text-primary\" style=\"font-size:18px;width:18px;height:18px;line-height:18px;color:#3F43EE;\">history</mat-icon>\n Audit log\n </h3>\n <p class=\"cqa-text-xs cqa-text-[#64748B] cqa-mt-0.5 cqa-mb-0\">\n All changes across environments, profiles, and globals. Read-only, append-only.\n </p>\n </div>\n <button type=\"button\" class=\"cqa-audit-log-close-btn\" (click)=\"onClose()\" aria-label=\"Close audit log\">\n <mat-icon>close</mat-icon>\n </button>\n</header>\n\n<section class=\"cqa-grid cqa-gap-2 cqa-px-[22px] cqa-py-3 cqa-border-b cqa-border-[#E5E7EB] cqa-flex-shrink-0\"\n style=\"grid-template-columns: repeat(4, minmax(0, 1fr)); background:#FAFBFC;\">\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"entityConfig\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"envConfig\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"userConfig\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"rangeConfig\"></cqa-dynamic-select>\n</section>\n\n<div class=\"cqa-flex-1 cqa-overflow-y-auto\">\n <cqa-full-table-loader *ngIf=\"isLoading\"></cqa-full-table-loader>\n <ng-container *ngIf=\"!isLoading\">\n <cqa-empty-state\n *ngIf=\"filteredEntries.length === 0; else listTpl\"\n title=\"No entries match your filters\"\n description=\"Adjust the filters above to widen the search.\">\n </cqa-empty-state>\n <ng-template #listTpl>\n <cqa-audit-log-entry-card\n *ngFor=\"let e of filteredEntries; trackBy: trackById\"\n [entry]=\"e\">\n </cqa-audit-log-entry-card>\n </ng-template>\n </ng-container>\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i3.FullTableLoaderComponent, selector: "cqa-full-table-loader", inputs: ["label"] }, { type: i4.EmptyStateComponent, selector: "cqa-empty-state", inputs: ["preset", "imageUrl", "title", "description", "actions"], outputs: ["actionClick"] }, { type: i5.AuditLogEntryCardComponent, selector: "cqa-audit-log-entry-card", inputs: ["entry"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
123
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AuditLogDrawerComponent, decorators: [{
124
+ type: Component,
125
+ args: [{ selector: 'cqa-audit-log-drawer', host: {
126
+ class: 'cqa-ui-root',
127
+ style: 'display:flex;flex-direction:column;height:100%;width:560px;max-width:100vw;background:#fff;box-shadow:-8px 0 32px rgba(15,23,42,.12);',
128
+ }, template: "<header class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-3 cqa-px-[22px] cqa-py-[18px] cqa-border-b cqa-border-[#E5E7EB] cqa-flex-shrink-0 cqa-font-inter\">\n <div class=\"cqa-min-w-0\">\n <h3 class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-m-0 cqa-text-[15px] cqa-font-semibold cqa-text-[#0F172A]\">\n <mat-icon class=\"cqa-text-primary\" style=\"font-size:18px;width:18px;height:18px;line-height:18px;color:#3F43EE;\">history</mat-icon>\n Audit log\n </h3>\n <p class=\"cqa-text-xs cqa-text-[#64748B] cqa-mt-0.5 cqa-mb-0\">\n All changes across environments, profiles, and globals. Read-only, append-only.\n </p>\n </div>\n <button type=\"button\" class=\"cqa-audit-log-close-btn\" (click)=\"onClose()\" aria-label=\"Close audit log\">\n <mat-icon>close</mat-icon>\n </button>\n</header>\n\n<section class=\"cqa-grid cqa-gap-2 cqa-px-[22px] cqa-py-3 cqa-border-b cqa-border-[#E5E7EB] cqa-flex-shrink-0\"\n style=\"grid-template-columns: repeat(4, minmax(0, 1fr)); background:#FAFBFC;\">\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"entityConfig\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"envConfig\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"userConfig\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"rangeConfig\"></cqa-dynamic-select>\n</section>\n\n<div class=\"cqa-flex-1 cqa-overflow-y-auto\">\n <cqa-full-table-loader *ngIf=\"isLoading\"></cqa-full-table-loader>\n <ng-container *ngIf=\"!isLoading\">\n <cqa-empty-state\n *ngIf=\"filteredEntries.length === 0; else listTpl\"\n title=\"No entries match your filters\"\n description=\"Adjust the filters above to widen the search.\">\n </cqa-empty-state>\n <ng-template #listTpl>\n <cqa-audit-log-entry-card\n *ngFor=\"let e of filteredEntries; trackBy: trackById\"\n [entry]=\"e\">\n </cqa-audit-log-entry-card>\n </ng-template>\n </ng-container>\n</div>\n" }]
129
+ }], propDecorators: { entries: [{
130
+ type: Input
131
+ }], entityTypeOptions: [{
132
+ type: Input
133
+ }], envOptions: [{
134
+ type: Input
135
+ }], userOptions: [{
136
+ type: Input
137
+ }], defaultRange: [{
138
+ type: Input
139
+ }], isLoading: [{
140
+ type: Input
141
+ }], filterChange: [{
142
+ type: Output
143
+ }], close: [{
144
+ type: Output
145
+ }] } });
146
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit-log-drawer.component.js","sourceRoot":"","sources":["../../../../../src/lib/audit-log-drawer/audit-log-drawer.component.ts","../../../../../src/lib/audit-log-drawer/audit-log-drawer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport { DynamicSelectFieldConfig, SelectOption } from '../dynamic-select/dynamic-select-field.component';\nimport {\n  ALL_FILTER_VALUE,\n  AuditDateRange,\n  AuditEntry,\n  AuditFilterOption,\n  AuditLogFilterState,\n} from './audit-log-drawer.models';\n\n@Component({\n  selector: 'cqa-audit-log-drawer',\n  templateUrl: './audit-log-drawer.component.html',\n  host: {\n    class: 'cqa-ui-root',\n    style: 'display:flex;flex-direction:column;height:100%;width:560px;max-width:100vw;background:#fff;box-shadow:-8px 0 32px rgba(15,23,42,.12);',\n  },\n})\nexport class AuditLogDrawerComponent implements OnInit, OnDestroy {\n  @Input() entries: AuditEntry[] = [];\n  @Input() entityTypeOptions: AuditFilterOption[] = [];\n  @Input() envOptions: AuditFilterOption[] = [];\n  @Input() userOptions: AuditFilterOption[] = [];\n  @Input() defaultRange: AuditDateRange = '7d';\n  @Input() isLoading = false;\n\n  @Output() filterChange = new EventEmitter<AuditLogFilterState>();\n  @Output() close = new EventEmitter<void>();\n\n  public readonly ALL = ALL_FILTER_VALUE;\n\n  public readonly filtersForm = new FormGroup({\n    entityType: new FormControl(ALL_FILTER_VALUE),\n    env:        new FormControl(ALL_FILTER_VALUE),\n    user:       new FormControl(ALL_FILTER_VALUE),\n    range:      new FormControl('7d' as AuditDateRange),\n  });\n\n  public entityConfig!: DynamicSelectFieldConfig;\n  public envConfig!: DynamicSelectFieldConfig;\n  public userConfig!: DynamicSelectFieldConfig;\n  public rangeConfig!: DynamicSelectFieldConfig;\n\n  public trackById = (_: number, e: AuditEntry): string => e.id;\n\n  private readonly destroy$ = new Subject<void>();\n  private readonly rangeMs: Record<AuditDateRange, number> = {\n    '24h': 24 * 60 * 60 * 1000,\n    '7d':  7  * 24 * 60 * 60 * 1000,\n    '30d': 30 * 24 * 60 * 60 * 1000,\n    'all': Number.POSITIVE_INFINITY,\n  };\n\n  ngOnInit(): void {\n    this.filtersForm.patchValue({ range: this.defaultRange });\n\n    this.entityConfig = this.buildSelectConfig('entityType', 'All entity types', this.entityTypeOptions);\n    this.envConfig    = this.buildSelectConfig('env',        'All environments', this.envOptions);\n    this.userConfig   = this.buildSelectConfig('user',       'All users',        this.userOptions);\n    this.rangeConfig  = this.buildSelectConfig('range', 'Last 7 days', [\n      { value: '24h', label: 'Last 24h' },\n      { value: '7d',  label: 'Last 7 days' },\n      { value: '30d', label: 'Last 30 days' },\n      { value: 'all', label: 'All time' },\n    ], false);\n\n    this.filtersForm.valueChanges\n      .pipe(takeUntil(this.destroy$))\n      .subscribe(() => this.filterChange.emit(this.currentFilterState()));\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  public get filteredEntries(): AuditEntry[] {\n    const f = this.currentFilterState();\n    const cutoff = this.rangeMs[f.range];\n    const now = Date.now();\n    return (this.entries || []).filter(e => {\n      if (f.entityType !== ALL_FILTER_VALUE && e.entityType !== f.entityType) { return false; }\n      if (f.env !== ALL_FILTER_VALUE && !this.entryMatchesEnv(e, f.env)) { return false; }\n      if (f.user !== ALL_FILTER_VALUE && e.user !== f.user) { return false; }\n      if (cutoff !== Number.POSITIVE_INFINITY) {\n        const ts = this.parseTs(e.ts);\n        if (ts != null && now - ts > cutoff) { return false; }\n      }\n      return true;\n    });\n  }\n\n  public onClose(): void {\n    this.close.emit();\n  }\n\n  private entryMatchesEnv(entry: AuditEntry, env: string): boolean {\n    return (entry.entityName || '').toLowerCase().includes(env.toLowerCase());\n  }\n\n  private parseTs(ts: string): number | null {\n    if (!ts) { return null; }\n    const t = Date.parse(ts);\n    return Number.isNaN(t) ? null : t;\n  }\n\n  private currentFilterState(): AuditLogFilterState {\n    const v = this.filtersForm.value;\n    return {\n      entityType: v.entityType ?? ALL_FILTER_VALUE,\n      env:        v.env        ?? ALL_FILTER_VALUE,\n      user:       v.user       ?? ALL_FILTER_VALUE,\n      range:      (v.range as AuditDateRange) ?? '7d',\n    };\n  }\n\n  private buildSelectConfig(\n    key: string,\n    placeholder: string,\n    items: AuditFilterOption[],\n    includeAll: boolean = true,\n  ): DynamicSelectFieldConfig {\n    const options: SelectOption[] = [];\n    if (includeAll) {\n      options.push({ id: ALL_FILTER_VALUE, value: ALL_FILTER_VALUE, name: placeholder, label: placeholder });\n    }\n    for (const it of items) {\n      options.push({ id: it.value, value: it.value, name: it.label, label: it.label });\n    }\n    return {\n      key,\n      label: '',\n      placeholder,\n      multiple: false,\n      searchable: false,\n      options,\n    };\n  }\n}\n","<header class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-3 cqa-px-[22px] cqa-py-[18px] cqa-border-b cqa-border-[#E5E7EB] cqa-flex-shrink-0 cqa-font-inter\">\n  <div class=\"cqa-min-w-0\">\n    <h3 class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-m-0 cqa-text-[15px] cqa-font-semibold cqa-text-[#0F172A]\">\n      <mat-icon class=\"cqa-text-primary\" style=\"font-size:18px;width:18px;height:18px;line-height:18px;color:#3F43EE;\">history</mat-icon>\n      Audit log\n    </h3>\n    <p class=\"cqa-text-xs cqa-text-[#64748B] cqa-mt-0.5 cqa-mb-0\">\n      All changes across environments, profiles, and globals. Read-only, append-only.\n    </p>\n  </div>\n  <button type=\"button\" class=\"cqa-audit-log-close-btn\" (click)=\"onClose()\" aria-label=\"Close audit log\">\n    <mat-icon>close</mat-icon>\n  </button>\n</header>\n\n<section class=\"cqa-grid cqa-gap-2 cqa-px-[22px] cqa-py-3 cqa-border-b cqa-border-[#E5E7EB] cqa-flex-shrink-0\"\n         style=\"grid-template-columns: repeat(4, minmax(0, 1fr)); background:#FAFBFC;\">\n  <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"entityConfig\"></cqa-dynamic-select>\n  <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"envConfig\"></cqa-dynamic-select>\n  <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"userConfig\"></cqa-dynamic-select>\n  <cqa-dynamic-select [form]=\"filtersForm\" [config]=\"rangeConfig\"></cqa-dynamic-select>\n</section>\n\n<div class=\"cqa-flex-1 cqa-overflow-y-auto\">\n  <cqa-full-table-loader *ngIf=\"isLoading\"></cqa-full-table-loader>\n  <ng-container *ngIf=\"!isLoading\">\n    <cqa-empty-state\n      *ngIf=\"filteredEntries.length === 0; else listTpl\"\n      title=\"No entries match your filters\"\n      description=\"Adjust the filters above to widen the search.\">\n    </cqa-empty-state>\n    <ng-template #listTpl>\n      <cqa-audit-log-entry-card\n        *ngFor=\"let e of filteredEntries; trackBy: trackById\"\n        [entry]=\"e\">\n      </cqa-audit-log-entry-card>\n    </ng-template>\n  </ng-container>\n</div>\n"]}
@@ -0,0 +1,2 @@
1
+ export const ALL_FILTER_VALUE = '__all__';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXVkaXQtbG9nLWRyYXdlci5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2F1ZGl0LWxvZy1kcmF3ZXIvYXVkaXQtbG9nLWRyYXdlci5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNkNBLE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHLFNBQVMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE9ic2VydmFibGUsIFN1YmplY3QgfSBmcm9tICdyeGpzJztcblxuZXhwb3J0IHR5cGUgQXVkaXRFbnRpdHlUeXBlID0gJ0Vudmlyb25tZW50IFZhcmlhYmxlJyB8ICdUZXN0IERhdGEgUHJvZmlsZScgfCAnR2xvYmFsIERhdGEnO1xuZXhwb3J0IHR5cGUgQXVkaXREYXRlUmFuZ2UgPSAnMjRoJyB8ICc3ZCcgfCAnMzBkJyB8ICdhbGwnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEF1ZGl0RW50cnkge1xuICBpZDogc3RyaW5nO1xuICB0czogc3RyaW5nO1xuICB1c2VyOiBzdHJpbmc7XG4gIGVudGl0eVR5cGU6IEF1ZGl0RW50aXR5VHlwZTtcbiAgZW50aXR5TmFtZTogc3RyaW5nO1xuICBjaGFuZ2U6IHN0cmluZztcbiAgYmVmb3JlPzogc3RyaW5nIHwgbnVsbDtcbiAgYWZ0ZXI/OiBzdHJpbmcgfCBudWxsO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEF1ZGl0RmlsdGVyT3B0aW9uIHtcbiAgdmFsdWU6IHN0cmluZztcbiAgbGFiZWw6IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBdWRpdExvZ0RyYXdlcklucHV0cyB7XG4gIGVudHJpZXM6IEF1ZGl0RW50cnlbXTtcbiAgZW50aXR5VHlwZU9wdGlvbnM/OiBBdWRpdEZpbHRlck9wdGlvbltdO1xuICBlbnZPcHRpb25zPzogQXVkaXRGaWx0ZXJPcHRpb25bXTtcbiAgdXNlck9wdGlvbnM/OiBBdWRpdEZpbHRlck9wdGlvbltdO1xuICBkZWZhdWx0UmFuZ2U/OiBBdWRpdERhdGVSYW5nZTtcbiAgaXNMb2FkaW5nPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBdWRpdExvZ0ZpbHRlclN0YXRlIHtcbiAgZW50aXR5VHlwZTogc3RyaW5nO1xuICBlbnY6IHN0cmluZztcbiAgdXNlcjogc3RyaW5nO1xuICByYW5nZTogQXVkaXREYXRlUmFuZ2U7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXVkaXRMb2dEcmF3ZXJSZWYge1xuICBmaWx0ZXJDaGFuZ2UkOiBPYnNlcnZhYmxlPEF1ZGl0TG9nRmlsdGVyU3RhdGU+O1xuICBhZnRlckNsb3NlZCQ6IE9ic2VydmFibGU8dm9pZD47XG4gIGNsb3NlKCk6IHZvaWQ7XG4gIHVwZGF0ZUVudHJpZXMoZW50cmllczogQXVkaXRFbnRyeVtdKTogdm9pZDtcbiAgdXBkYXRlTG9hZGluZyhpc0xvYWRpbmc6IGJvb2xlYW4pOiB2b2lkO1xufVxuXG5leHBvcnQgY29uc3QgQUxMX0ZJTFRFUl9WQUxVRSA9ICdfX2FsbF9fJztcblxuZXhwb3J0IGludGVyZmFjZSBBdWRpdExvZ0RyYXdlckludGVybmFsUmVmIGV4dGVuZHMgQXVkaXRMb2dEcmF3ZXJSZWYge1xuICByZWFkb25seSBmaWx0ZXJDaGFuZ2U6IFN1YmplY3Q8QXVkaXRMb2dGaWx0ZXJTdGF0ZT47XG4gIHJlYWRvbmx5IGFmdGVyQ2xvc2VkOiBTdWJqZWN0PHZvaWQ+O1xufVxuIl19
@@ -0,0 +1,84 @@
1
+ import { Injectable } from '@angular/core';
2
+ import { OverlayConfig } from '@angular/cdk/overlay';
3
+ import { ComponentPortal } from '@angular/cdk/portal';
4
+ import { Subject } from 'rxjs';
5
+ import { filter } from 'rxjs/operators';
6
+ import { AuditLogDrawerComponent } from './audit-log-drawer.component';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/cdk/overlay";
9
+ export class AuditLogDrawerService {
10
+ constructor(overlay, injector) {
11
+ this.overlay = overlay;
12
+ this.injector = injector;
13
+ this.currentOverlayRef = null;
14
+ }
15
+ open(inputs) {
16
+ if (this.currentOverlayRef) {
17
+ this.currentOverlayRef.dispose();
18
+ this.currentOverlayRef = null;
19
+ }
20
+ const positionStrategy = this.overlay.position()
21
+ .global()
22
+ .right('0')
23
+ .top('0')
24
+ .bottom('0');
25
+ const overlayRef = this.overlay.create(new OverlayConfig({
26
+ hasBackdrop: true,
27
+ backdropClass: ['cqa-audit-log-drawer-backdrop'],
28
+ scrollStrategy: this.overlay.scrollStrategies.block(),
29
+ positionStrategy,
30
+ panelClass: ['cqa-audit-log-drawer-panel', 'cqa-ui-root'],
31
+ maxWidth: '100vw',
32
+ height: '100%',
33
+ }));
34
+ const portal = new ComponentPortal(AuditLogDrawerComponent, null, this.injector);
35
+ const componentRef = overlayRef.attach(portal);
36
+ const instance = componentRef.instance;
37
+ instance.entries = inputs.entries ?? [];
38
+ instance.entityTypeOptions = inputs.entityTypeOptions ?? [];
39
+ instance.envOptions = inputs.envOptions ?? [];
40
+ instance.userOptions = inputs.userOptions ?? [];
41
+ instance.defaultRange = inputs.defaultRange ?? '7d';
42
+ instance.isLoading = !!inputs.isLoading;
43
+ componentRef.changeDetectorRef.markForCheck();
44
+ const filterChange = new Subject();
45
+ const afterClosed = new Subject();
46
+ instance.filterChange.subscribe((s) => filterChange.next(s));
47
+ const closeAll = () => {
48
+ if (!this.currentOverlayRef) {
49
+ return;
50
+ }
51
+ overlayRef.dispose();
52
+ this.currentOverlayRef = null;
53
+ afterClosed.next();
54
+ afterClosed.complete();
55
+ filterChange.complete();
56
+ };
57
+ instance.close.subscribe(() => closeAll());
58
+ overlayRef.backdropClick().subscribe(() => closeAll());
59
+ overlayRef.keydownEvents()
60
+ .pipe(filter((e) => e.key === 'Escape' || e.key === 'Esc'))
61
+ .subscribe(() => closeAll());
62
+ this.currentOverlayRef = overlayRef;
63
+ return {
64
+ filterChange$: filterChange.asObservable(),
65
+ afterClosed$: afterClosed.asObservable(),
66
+ close: () => closeAll(),
67
+ updateEntries: (entries) => {
68
+ instance.entries = entries ?? [];
69
+ componentRef.changeDetectorRef.markForCheck();
70
+ },
71
+ updateLoading: (isLoading) => {
72
+ instance.isLoading = !!isLoading;
73
+ componentRef.changeDetectorRef.markForCheck();
74
+ },
75
+ };
76
+ }
77
+ }
78
+ AuditLogDrawerService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AuditLogDrawerService, deps: [{ token: i1.Overlay }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Injectable });
79
+ AuditLogDrawerService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AuditLogDrawerService, providedIn: 'root' });
80
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AuditLogDrawerService, decorators: [{
81
+ type: Injectable,
82
+ args: [{ providedIn: 'root' }]
83
+ }], ctorParameters: function () { return [{ type: i1.Overlay }, { type: i0.Injector }]; } });
84
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit-log-drawer.service.js","sourceRoot":"","sources":["../../../../../src/lib/audit-log-drawer/audit-log-drawer.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Injector } from '@angular/core';\nimport { Overlay, OverlayConfig, OverlayRef } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { Subject } from 'rxjs';\nimport { filter } from 'rxjs/operators';\n\nimport { AuditLogDrawerComponent } from './audit-log-drawer.component';\nimport {\n  AuditLogDrawerInputs,\n  AuditLogDrawerRef,\n  AuditLogFilterState,\n} from './audit-log-drawer.models';\n\n@Injectable({ providedIn: 'root' })\nexport class AuditLogDrawerService {\n  private currentOverlayRef: OverlayRef | null = null;\n\n  constructor(private readonly overlay: Overlay, private readonly injector: Injector) {}\n\n  public open(inputs: AuditLogDrawerInputs): AuditLogDrawerRef {\n    if (this.currentOverlayRef) {\n      this.currentOverlayRef.dispose();\n      this.currentOverlayRef = null;\n    }\n\n    const positionStrategy = this.overlay.position()\n      .global()\n      .right('0')\n      .top('0')\n      .bottom('0');\n\n    const overlayRef = this.overlay.create(new OverlayConfig({\n      hasBackdrop: true,\n      backdropClass: ['cqa-audit-log-drawer-backdrop'],\n      scrollStrategy: this.overlay.scrollStrategies.block(),\n      positionStrategy,\n      panelClass: ['cqa-audit-log-drawer-panel', 'cqa-ui-root'],\n      maxWidth: '100vw',\n      height: '100%',\n    }));\n\n    const portal = new ComponentPortal(AuditLogDrawerComponent, null, this.injector);\n    const componentRef = overlayRef.attach(portal);\n    const instance = componentRef.instance;\n\n    instance.entries = inputs.entries ?? [];\n    instance.entityTypeOptions = inputs.entityTypeOptions ?? [];\n    instance.envOptions = inputs.envOptions ?? [];\n    instance.userOptions = inputs.userOptions ?? [];\n    instance.defaultRange = inputs.defaultRange ?? '7d';\n    instance.isLoading = !!inputs.isLoading;\n    componentRef.changeDetectorRef.markForCheck();\n\n    const filterChange = new Subject<AuditLogFilterState>();\n    const afterClosed = new Subject<void>();\n\n    instance.filterChange.subscribe((s: AuditLogFilterState) => filterChange.next(s));\n\n    const closeAll = () => {\n      if (!this.currentOverlayRef) { return; }\n      overlayRef.dispose();\n      this.currentOverlayRef = null;\n      afterClosed.next();\n      afterClosed.complete();\n      filterChange.complete();\n    };\n\n    instance.close.subscribe(() => closeAll());\n    overlayRef.backdropClick().subscribe(() => closeAll());\n    overlayRef.keydownEvents()\n      .pipe(filter((e: KeyboardEvent) => e.key === 'Escape' || e.key === 'Esc'))\n      .subscribe(() => closeAll());\n\n    this.currentOverlayRef = overlayRef;\n\n    return {\n      filterChange$: filterChange.asObservable(),\n      afterClosed$: afterClosed.asObservable(),\n      close: () => closeAll(),\n      updateEntries: (entries) => {\n        instance.entries = entries ?? [];\n        componentRef.changeDetectorRef.markForCheck();\n      },\n      updateLoading: (isLoading) => {\n        instance.isLoading = !!isLoading;\n        componentRef.changeDetectorRef.markForCheck();\n      },\n    };\n  }\n}\n"]}
@@ -0,0 +1,30 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/common";
4
+ export class AuditLogEntryCardComponent {
5
+ get userShortName() {
6
+ const u = this.entry?.user || '';
7
+ const idx = u.indexOf('@');
8
+ return idx > 0 ? u.substring(0, idx) : u;
9
+ }
10
+ get etypeClass() {
11
+ switch (this.entry?.entityType) {
12
+ case 'Environment Variable': return 'cqa-audit-log-etype-ev';
13
+ case 'Test Data Profile': return 'cqa-audit-log-etype-tdp';
14
+ case 'Global Data': return 'cqa-audit-log-etype-gd';
15
+ default: return '';
16
+ }
17
+ }
18
+ get hasDiff() {
19
+ return this.entry?.before != null || this.entry?.after != null;
20
+ }
21
+ }
22
+ AuditLogEntryCardComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AuditLogEntryCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
23
+ AuditLogEntryCardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AuditLogEntryCardComponent, selector: "cqa-audit-log-entry-card", inputs: { entry: "entry" }, host: { classAttribute: "cqa-ui-root cqa-audit-log-row" }, ngImport: i0, template: "<div class=\"cqa-audit-log-meta cqa-font-inter\">\n <div class=\"cqa-audit-log-ts\">{{ entry?.ts }}</div>\n <div class=\"cqa-audit-log-user\">{{ userShortName }}</div>\n</div>\n<div class=\"cqa-min-w-0\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-flex-wrap cqa-mb-1\">\n <span class=\"cqa-audit-log-etype\" [ngClass]=\"etypeClass\">{{ entry?.entityType }}</span>\n <span class=\"cqa-text-[13px] cqa-font-medium cqa-text-[#0F172A]\">{{ entry?.entityName }}</span>\n </div>\n <div class=\"cqa-text-xs cqa-text-[#64748B]\">{{ entry?.change }}</div>\n <div *ngIf=\"hasDiff\" class=\"cqa-audit-log-diff\">\n <span class=\"cqa-audit-log-before\">{{ entry?.before || '\u2014' }}</span>\n <span class=\"cqa-audit-log-arrow\">\u2192</span>\n <span class=\"cqa-audit-log-after\">{{ entry?.after || '\u2014' }}</span>\n </div>\n</div>\n", directives: [{ type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
24
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AuditLogEntryCardComponent, decorators: [{
25
+ type: Component,
26
+ args: [{ selector: 'cqa-audit-log-entry-card', host: { class: 'cqa-ui-root cqa-audit-log-row' }, template: "<div class=\"cqa-audit-log-meta cqa-font-inter\">\n <div class=\"cqa-audit-log-ts\">{{ entry?.ts }}</div>\n <div class=\"cqa-audit-log-user\">{{ userShortName }}</div>\n</div>\n<div class=\"cqa-min-w-0\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-flex-wrap cqa-mb-1\">\n <span class=\"cqa-audit-log-etype\" [ngClass]=\"etypeClass\">{{ entry?.entityType }}</span>\n <span class=\"cqa-text-[13px] cqa-font-medium cqa-text-[#0F172A]\">{{ entry?.entityName }}</span>\n </div>\n <div class=\"cqa-text-xs cqa-text-[#64748B]\">{{ entry?.change }}</div>\n <div *ngIf=\"hasDiff\" class=\"cqa-audit-log-diff\">\n <span class=\"cqa-audit-log-before\">{{ entry?.before || '\u2014' }}</span>\n <span class=\"cqa-audit-log-arrow\">\u2192</span>\n <span class=\"cqa-audit-log-after\">{{ entry?.after || '\u2014' }}</span>\n </div>\n</div>\n" }]
27
+ }], propDecorators: { entry: [{
28
+ type: Input
29
+ }] } });
30
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXVkaXQtbG9nLWVudHJ5LWNhcmQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9hdWRpdC1sb2ctZHJhd2VyL2F1ZGl0LWxvZy1lbnRyeS1jYXJkLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvYXVkaXQtbG9nLWRyYXdlci9hdWRpdC1sb2ctZW50cnktY2FyZC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBUWpELE1BQU0sT0FBTywwQkFBMEI7SUFHckMsSUFBVyxhQUFhO1FBQ3RCLE1BQU0sQ0FBQyxHQUFHLElBQUksQ0FBQyxLQUFLLEVBQUUsSUFBSSxJQUFJLEVBQUUsQ0FBQztRQUNqQyxNQUFNLEdBQUcsR0FBRyxDQUFDLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQzNCLE9BQU8sR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUMzQyxDQUFDO0lBRUQsSUFBVyxVQUFVO1FBQ25CLFFBQVEsSUFBSSxDQUFDLEtBQUssRUFBRSxVQUFVLEVBQUU7WUFDOUIsS0FBSyxzQkFBc0IsQ0FBQyxDQUFDLE9BQU8sd0JBQXdCLENBQUM7WUFDN0QsS0FBSyxtQkFBbUIsQ0FBQyxDQUFJLE9BQU8seUJBQXlCLENBQUM7WUFDOUQsS0FBSyxhQUFhLENBQUMsQ0FBVSxPQUFPLHdCQUF3QixDQUFDO1lBQzdELE9BQU8sQ0FBQyxDQUFxQixPQUFPLEVBQUUsQ0FBQztTQUN4QztJQUNILENBQUM7SUFFRCxJQUFXLE9BQU87UUFDaEIsT0FBTyxJQUFJLENBQUMsS0FBSyxFQUFFLE1BQU0sSUFBSSxJQUFJLElBQUksSUFBSSxDQUFDLEtBQUssRUFBRSxLQUFLLElBQUksSUFBSSxDQUFDO0lBQ2pFLENBQUM7O3VIQXBCVSwwQkFBMEI7MkdBQTFCLDBCQUEwQix1SkNSdkMsODFCQWdCQTsyRkRSYSwwQkFBMEI7a0JBTHRDLFNBQVM7K0JBQ0UsMEJBQTBCLFFBRTlCLEVBQUUsS0FBSyxFQUFFLCtCQUErQixFQUFFOzhCQUd2QyxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBBdWRpdEVudHJ5IH0gZnJvbSAnLi9hdWRpdC1sb2ctZHJhd2VyLm1vZGVscyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NxYS1hdWRpdC1sb2ctZW50cnktY2FyZCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9hdWRpdC1sb2ctZW50cnktY2FyZC5jb21wb25lbnQuaHRtbCcsXG4gIGhvc3Q6IHsgY2xhc3M6ICdjcWEtdWktcm9vdCBjcWEtYXVkaXQtbG9nLXJvdycgfSxcbn0pXG5leHBvcnQgY2xhc3MgQXVkaXRMb2dFbnRyeUNhcmRDb21wb25lbnQge1xuICBASW5wdXQoKSBlbnRyeSE6IEF1ZGl0RW50cnk7XG5cbiAgcHVibGljIGdldCB1c2VyU2hvcnROYW1lKCk6IHN0cmluZyB7XG4gICAgY29uc3QgdSA9IHRoaXMuZW50cnk/LnVzZXIgfHwgJyc7XG4gICAgY29uc3QgaWR4ID0gdS5pbmRleE9mKCdAJyk7XG4gICAgcmV0dXJuIGlkeCA+IDAgPyB1LnN1YnN0cmluZygwLCBpZHgpIDogdTtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgZXR5cGVDbGFzcygpOiBzdHJpbmcge1xuICAgIHN3aXRjaCAodGhpcy5lbnRyeT8uZW50aXR5VHlwZSkge1xuICAgICAgY2FzZSAnRW52aXJvbm1lbnQgVmFyaWFibGUnOiByZXR1cm4gJ2NxYS1hdWRpdC1sb2ctZXR5cGUtZXYnO1xuICAgICAgY2FzZSAnVGVzdCBEYXRhIFByb2ZpbGUnOiAgICByZXR1cm4gJ2NxYS1hdWRpdC1sb2ctZXR5cGUtdGRwJztcbiAgICAgIGNhc2UgJ0dsb2JhbCBEYXRhJzogICAgICAgICAgcmV0dXJuICdjcWEtYXVkaXQtbG9nLWV0eXBlLWdkJztcbiAgICAgIGRlZmF1bHQ6ICAgICAgICAgICAgICAgICAgICAgcmV0dXJuICcnO1xuICAgIH1cbiAgfVxuXG4gIHB1YmxpYyBnZXQgaGFzRGlmZigpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5lbnRyeT8uYmVmb3JlICE9IG51bGwgfHwgdGhpcy5lbnRyeT8uYWZ0ZXIgIT0gbnVsbDtcbiAgfVxufVxuIiwiPGRpdiBjbGFzcz1cImNxYS1hdWRpdC1sb2ctbWV0YSBjcWEtZm9udC1pbnRlclwiPlxuICA8ZGl2IGNsYXNzPVwiY3FhLWF1ZGl0LWxvZy10c1wiPnt7IGVudHJ5Py50cyB9fTwvZGl2PlxuICA8ZGl2IGNsYXNzPVwiY3FhLWF1ZGl0LWxvZy11c2VyXCI+e3sgdXNlclNob3J0TmFtZSB9fTwvZGl2PlxuPC9kaXY+XG48ZGl2IGNsYXNzPVwiY3FhLW1pbi13LTBcIj5cbiAgPGRpdiBjbGFzcz1cImNxYS1mbGV4IGNxYS1pdGVtcy1jZW50ZXIgY3FhLWdhcC0yIGNxYS1mbGV4LXdyYXAgY3FhLW1iLTFcIj5cbiAgICA8c3BhbiBjbGFzcz1cImNxYS1hdWRpdC1sb2ctZXR5cGVcIiBbbmdDbGFzc109XCJldHlwZUNsYXNzXCI+e3sgZW50cnk/LmVudGl0eVR5cGUgfX08L3NwYW4+XG4gICAgPHNwYW4gY2xhc3M9XCJjcWEtdGV4dC1bMTNweF0gY3FhLWZvbnQtbWVkaXVtIGNxYS10ZXh0LVsjMEYxNzJBXVwiPnt7IGVudHJ5Py5lbnRpdHlOYW1lIH19PC9zcGFuPlxuICA8L2Rpdj5cbiAgPGRpdiBjbGFzcz1cImNxYS10ZXh0LXhzIGNxYS10ZXh0LVsjNjQ3NDhCXVwiPnt7IGVudHJ5Py5jaGFuZ2UgfX08L2Rpdj5cbiAgPGRpdiAqbmdJZj1cImhhc0RpZmZcIiBjbGFzcz1cImNxYS1hdWRpdC1sb2ctZGlmZlwiPlxuICAgIDxzcGFuIGNsYXNzPVwiY3FhLWF1ZGl0LWxvZy1iZWZvcmVcIj57eyBlbnRyeT8uYmVmb3JlIHx8ICfigJQnIH19PC9zcGFuPlxuICAgIDxzcGFuIGNsYXNzPVwiY3FhLWF1ZGl0LWxvZy1hcnJvd1wiPuKGkjwvc3Bhbj5cbiAgICA8c3BhbiBjbGFzcz1cImNxYS1hdWRpdC1sb2ctYWZ0ZXJcIj57eyBlbnRyeT8uYWZ0ZXIgfHwgJ+KAlCcgfX08L3NwYW4+XG4gIDwvZGl2PlxuPC9kaXY+XG4iXX0=
@@ -0,0 +1,133 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/material/icon";
4
+ import * as i2 from "../custom-input/custom-input.component";
5
+ import * as i3 from "../button/button.component";
6
+ import * as i4 from "ngx-drag-drop";
7
+ import * as i5 from "@angular/common";
8
+ export class ManageColumnsDialogComponent {
9
+ constructor() {
10
+ this.columns = [];
11
+ this.lockedColumns = [];
12
+ this.items = [];
13
+ this.newColName = '';
14
+ this.columnsError = null;
15
+ this.trackByUid = (_, item) => item.uid;
16
+ this.nextUid = 1;
17
+ }
18
+ ngOnInit() {
19
+ const initial = Array.isArray(this.columns) ? this.columns : [];
20
+ this.items = initial.map(name => ({
21
+ uid: this.nextUid++,
22
+ originalName: name,
23
+ name,
24
+ }));
25
+ }
26
+ isLocked(item) {
27
+ if (!this.lockedColumns || this.lockedColumns.length === 0) {
28
+ return false;
29
+ }
30
+ if (item.originalName != null && this.lockedColumns.indexOf(item.originalName) !== -1) {
31
+ return true;
32
+ }
33
+ return this.lockedColumns.indexOf(item.name) !== -1;
34
+ }
35
+ onDndDrop(event) {
36
+ const dragged = event.data;
37
+ if (!dragged || event.index == null) {
38
+ return;
39
+ }
40
+ const from = this.items.findIndex(it => it.uid === dragged.uid);
41
+ if (from < 0) {
42
+ return;
43
+ }
44
+ const next = this.items.slice();
45
+ next.splice(from, 1);
46
+ const to = event.index > from ? event.index - 1 : event.index;
47
+ next.splice(to, 0, dragged);
48
+ this.items = next;
49
+ }
50
+ onMove(index, dir) {
51
+ const target = index + dir;
52
+ if (target < 0 || target >= this.items.length) {
53
+ return;
54
+ }
55
+ const next = this.items.slice();
56
+ [next[index], next[target]] = [next[target], next[index]];
57
+ this.items = next;
58
+ }
59
+ onRename(index, value) {
60
+ if (!this.items[index]) {
61
+ return;
62
+ }
63
+ this.items = this.items.map((it, i) => i === index ? { ...it, name: value } : it);
64
+ this.columnsError = null;
65
+ }
66
+ onRemove(index) {
67
+ const item = this.items[index];
68
+ if (!item || this.isLocked(item)) {
69
+ return;
70
+ }
71
+ this.items = this.items.filter((_, i) => i !== index);
72
+ this.columnsError = null;
73
+ }
74
+ onNewColNameChange(value) {
75
+ this.newColName = value;
76
+ }
77
+ onAdd() {
78
+ const trimmed = (this.newColName || '').trim();
79
+ if (!trimmed || this.itemsHasName(trimmed)) {
80
+ return;
81
+ }
82
+ this.items = [...this.items, {
83
+ uid: this.nextUid++,
84
+ originalName: null,
85
+ name: trimmed,
86
+ }];
87
+ this.newColName = '';
88
+ this.columnsError = null;
89
+ }
90
+ get canAdd() {
91
+ const trimmed = (this.newColName || '').trim();
92
+ if (!trimmed) {
93
+ return false;
94
+ }
95
+ return !this.itemsHasName(trimmed);
96
+ }
97
+ getValue() {
98
+ const trimmed = this.items.map(it => ({
99
+ originalName: it.originalName,
100
+ name: (it.name || '').trim(),
101
+ }));
102
+ if (trimmed.length === 0) {
103
+ this.columnsError = 'At least one column is required.';
104
+ return null;
105
+ }
106
+ if (trimmed.some(c => c.name.length === 0)) {
107
+ this.columnsError = 'Column names cannot be empty.';
108
+ return null;
109
+ }
110
+ const lowered = trimmed.map(c => c.name.toLowerCase());
111
+ if (new Set(lowered).size !== trimmed.length) {
112
+ this.columnsError = 'Column names must be unique.';
113
+ return null;
114
+ }
115
+ this.columnsError = null;
116
+ return { columns: trimmed };
117
+ }
118
+ itemsHasName(candidate) {
119
+ const lowered = candidate.toLowerCase();
120
+ return this.items.some(it => (it.name || '').trim().toLowerCase() === lowered);
121
+ }
122
+ }
123
+ ManageColumnsDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ManageColumnsDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
124
+ ManageColumnsDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: ManageColumnsDialogComponent, selector: "cqa-manage-columns-dialog", inputs: { columns: "columns", lockedColumns: "lockedColumns" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-w-full\">\n\n <div class=\"manage-cols-drop-list cqa-flex cqa-flex-col cqa-gap-2\"\n [dndDropzone]=\"['col']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"manage-cols-placeholder\">Drop here</div>\n <div *ngFor=\"let item of items; let i = index; trackBy: trackByUid\"\n class=\"manage-cols-row cqa-flex cqa-items-center cqa-gap-2 cqa-px-2 cqa-py-1.5 cqa-rounded-md cqa-border cqa-border-gray-200 cqa-bg-gray-50\">\n <span class=\"manage-cols-handle cqa-flex cqa-items-center cqa-justify-center cqa-text-gray-400\"\n [dndDraggable]=\"item\"\n [dndDisableIf]=\"isLocked(item)\"\n dndType=\"col\"\n dndEffectAllowed=\"move\">\n <mat-icon style=\"font-size:18px;width:18px;height:18px;line-height:18px;\">drag_indicator</mat-icon>\n </span>\n <cqa-custom-input\n class=\"cqa-flex-1\"\n [value]=\"item.name\"\n [fullWidth]=\"true\"\n [disabled]=\"isLocked(item)\"\n inputInlineStyle=\"font-family: 'Inter', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onRename(i, $event)\">\n </cqa-custom-input>\n <button type=\"button\" class=\"manage-cols-icon-btn\"\n title=\"Move up\"\n [disabled]=\"i === 0\"\n (click)=\"onMove(i, -1)\">\n <mat-icon>arrow_upward</mat-icon>\n </button>\n <button type=\"button\" class=\"manage-cols-icon-btn\"\n title=\"Move down\"\n [disabled]=\"i === items.length - 1\"\n (click)=\"onMove(i, 1)\">\n <mat-icon>arrow_downward</mat-icon>\n </button>\n <button type=\"button\" class=\"manage-cols-icon-btn manage-cols-icon-btn-danger\"\n title=\"Delete column\"\n [disabled]=\"isLocked(item)\"\n (click)=\"onRemove(i)\">\n <mat-icon>delete</mat-icon>\n </button>\n </div>\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-pt-3 cqa-mt-1 cqa-border-t cqa-border-dashed cqa-border-gray-200\">\n <cqa-custom-input\n class=\"cqa-flex-1\"\n [value]=\"newColName\"\n [fullWidth]=\"true\"\n placeholder=\"New column name (e.g. currency)\"\n inputInlineStyle=\"font-family: 'Inter', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onNewColNameChange($event)\"\n (enterPressed)=\"onAdd()\">\n </cqa-custom-input>\n <cqa-button\n variant=\"outlined\"\n btnSize=\"sm\"\n icon=\"add\"\n text=\"Add column\"\n [disabled]=\"!canAdd\"\n (clicked)=\"onAdd()\">\n </cqa-button>\n </div>\n\n <span *ngIf=\"columnsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ columnsError }}</span>\n\n <div class=\"cqa-px-3 cqa-py-2 cqa-rounded-md cqa-bg-primary-surface cqa-border cqa-border-success-100 cqa-text-xs cqa-text-gray-700\">\n <strong class=\"cqa-text-gray-900\">Heads up:</strong>\n Deleting a column removes its data from every environment. Renaming keeps the data in place.\n </div>\n\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "loading", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i4.DndDropzoneDirective, selector: "[dndDropzone]", inputs: ["dndDropzone", "dndEffectAllowed", "dndAllowExternal", "dndHorizontal", "dndDragoverClass", "dndDropzoneDisabledClass", "dndDisableIf", "dndDisableDropIf"], outputs: ["dndDragover", "dndDrop"] }, { type: i4.DndPlaceholderRefDirective, selector: "[dndPlaceholderRef]" }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.DndDraggableDirective, selector: "[dndDraggable]", inputs: ["dndDraggable", "dndEffectAllowed", "dndType", "dndDraggingClass", "dndDraggingSourceClass", "dndDraggableDisabledClass", "dndDragImageOffsetFunction", "dndDisableIf", "dndDisableDragIf"], outputs: ["dndStart", "dndDrag", "dndEnd", "dndMoved", "dndCopied", "dndLinked", "dndCanceled"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
125
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ManageColumnsDialogComponent, decorators: [{
126
+ type: Component,
127
+ args: [{ selector: 'cqa-manage-columns-dialog', 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 <div class=\"manage-cols-drop-list cqa-flex cqa-flex-col cqa-gap-2\"\n [dndDropzone]=\"['col']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"manage-cols-placeholder\">Drop here</div>\n <div *ngFor=\"let item of items; let i = index; trackBy: trackByUid\"\n class=\"manage-cols-row cqa-flex cqa-items-center cqa-gap-2 cqa-px-2 cqa-py-1.5 cqa-rounded-md cqa-border cqa-border-gray-200 cqa-bg-gray-50\">\n <span class=\"manage-cols-handle cqa-flex cqa-items-center cqa-justify-center cqa-text-gray-400\"\n [dndDraggable]=\"item\"\n [dndDisableIf]=\"isLocked(item)\"\n dndType=\"col\"\n dndEffectAllowed=\"move\">\n <mat-icon style=\"font-size:18px;width:18px;height:18px;line-height:18px;\">drag_indicator</mat-icon>\n </span>\n <cqa-custom-input\n class=\"cqa-flex-1\"\n [value]=\"item.name\"\n [fullWidth]=\"true\"\n [disabled]=\"isLocked(item)\"\n inputInlineStyle=\"font-family: 'Inter', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onRename(i, $event)\">\n </cqa-custom-input>\n <button type=\"button\" class=\"manage-cols-icon-btn\"\n title=\"Move up\"\n [disabled]=\"i === 0\"\n (click)=\"onMove(i, -1)\">\n <mat-icon>arrow_upward</mat-icon>\n </button>\n <button type=\"button\" class=\"manage-cols-icon-btn\"\n title=\"Move down\"\n [disabled]=\"i === items.length - 1\"\n (click)=\"onMove(i, 1)\">\n <mat-icon>arrow_downward</mat-icon>\n </button>\n <button type=\"button\" class=\"manage-cols-icon-btn manage-cols-icon-btn-danger\"\n title=\"Delete column\"\n [disabled]=\"isLocked(item)\"\n (click)=\"onRemove(i)\">\n <mat-icon>delete</mat-icon>\n </button>\n </div>\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-pt-3 cqa-mt-1 cqa-border-t cqa-border-dashed cqa-border-gray-200\">\n <cqa-custom-input\n class=\"cqa-flex-1\"\n [value]=\"newColName\"\n [fullWidth]=\"true\"\n placeholder=\"New column name (e.g. currency)\"\n inputInlineStyle=\"font-family: 'Inter', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onNewColNameChange($event)\"\n (enterPressed)=\"onAdd()\">\n </cqa-custom-input>\n <cqa-button\n variant=\"outlined\"\n btnSize=\"sm\"\n icon=\"add\"\n text=\"Add column\"\n [disabled]=\"!canAdd\"\n (clicked)=\"onAdd()\">\n </cqa-button>\n </div>\n\n <span *ngIf=\"columnsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ columnsError }}</span>\n\n <div class=\"cqa-px-3 cqa-py-2 cqa-rounded-md cqa-bg-primary-surface cqa-border cqa-border-success-100 cqa-text-xs cqa-text-gray-700\">\n <strong class=\"cqa-text-gray-900\">Heads up:</strong>\n Deleting a column removes its data from every environment. Renaming keeps the data in place.\n </div>\n\n</div>\n" }]
128
+ }], propDecorators: { columns: [{
129
+ type: Input
130
+ }], lockedColumns: [{
131
+ type: Input
132
+ }] } });
133
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"manage-columns-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/manage-columns-dialog/manage-columns-dialog.component.ts","../../../../../src/lib/manage-columns-dialog/manage-columns-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\nimport { DndDropEvent } from 'ngx-drag-drop';\n\nimport {\n  ManageColumnsDialogColumn,\n  ManageColumnsDialogValue,\n} from './manage-columns-dialog.models';\n\ninterface ManageColumnsItem {\n  uid: number;\n  originalName: string | null;\n  name: string;\n}\n\n@Component({\n  selector: 'cqa-manage-columns-dialog',\n  templateUrl: './manage-columns-dialog.component.html',\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class ManageColumnsDialogComponent implements OnInit {\n  @Input() columns: string[] = [];\n  @Input() lockedColumns: string[] = [];\n\n  public items: ManageColumnsItem[] = [];\n  public newColName = '';\n  public columnsError: string | null = null;\n\n  public trackByUid = (_: number, item: ManageColumnsItem): number => item.uid;\n\n  private nextUid = 1;\n\n  ngOnInit(): void {\n    const initial = Array.isArray(this.columns) ? this.columns : [];\n    this.items = initial.map(name => ({\n      uid: this.nextUid++,\n      originalName: name,\n      name,\n    }));\n  }\n\n  public isLocked(item: ManageColumnsItem): boolean {\n    if (!this.lockedColumns || this.lockedColumns.length === 0) { return false; }\n    if (item.originalName != null && this.lockedColumns.indexOf(item.originalName) !== -1) {\n      return true;\n    }\n    return this.lockedColumns.indexOf(item.name) !== -1;\n  }\n\n  public onDndDrop(event: DndDropEvent): void {\n    const dragged = event.data as ManageColumnsItem | null;\n    if (!dragged || event.index == null) { return; }\n    const from = this.items.findIndex(it => it.uid === dragged.uid);\n    if (from < 0) { return; }\n    const next = this.items.slice();\n    next.splice(from, 1);\n    const to = event.index > from ? event.index - 1 : event.index;\n    next.splice(to, 0, dragged);\n    this.items = next;\n  }\n\n  public onMove(index: number, dir: -1 | 1): void {\n    const target = index + dir;\n    if (target < 0 || target >= this.items.length) { return; }\n    const next = this.items.slice();\n    [next[index], next[target]] = [next[target], next[index]];\n    this.items = next;\n  }\n\n  public onRename(index: number, value: string): void {\n    if (!this.items[index]) { return; }\n    this.items = this.items.map((it, i) => i === index ? { ...it, name: value } : it);\n    this.columnsError = null;\n  }\n\n  public onRemove(index: number): void {\n    const item = this.items[index];\n    if (!item || this.isLocked(item)) { return; }\n    this.items = this.items.filter((_, i) => i !== index);\n    this.columnsError = null;\n  }\n\n  public onNewColNameChange(value: string): void {\n    this.newColName = value;\n  }\n\n  public onAdd(): void {\n    const trimmed = (this.newColName || '').trim();\n    if (!trimmed || this.itemsHasName(trimmed)) { return; }\n    this.items = [...this.items, {\n      uid: this.nextUid++,\n      originalName: null,\n      name: trimmed,\n    }];\n    this.newColName = '';\n    this.columnsError = null;\n  }\n\n  public get canAdd(): boolean {\n    const trimmed = (this.newColName || '').trim();\n    if (!trimmed) { return false; }\n    return !this.itemsHasName(trimmed);\n  }\n\n  public getValue(): ManageColumnsDialogValue | null {\n    const trimmed: ManageColumnsDialogColumn[] = this.items.map(it => ({\n      originalName: it.originalName,\n      name: (it.name || '').trim(),\n    }));\n\n    if (trimmed.length === 0) {\n      this.columnsError = 'At least one column is required.';\n      return null;\n    }\n    if (trimmed.some(c => c.name.length === 0)) {\n      this.columnsError = 'Column names cannot be empty.';\n      return null;\n    }\n    const lowered = trimmed.map(c => c.name.toLowerCase());\n    if (new Set(lowered).size !== trimmed.length) {\n      this.columnsError = 'Column names must be unique.';\n      return null;\n    }\n\n    this.columnsError = null;\n    return { columns: trimmed };\n  }\n\n  private itemsHasName(candidate: string): boolean {\n    const lowered = candidate.toLowerCase();\n    return this.items.some(it => (it.name || '').trim().toLowerCase() === lowered);\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-w-full\">\n\n  <div class=\"manage-cols-drop-list cqa-flex cqa-flex-col cqa-gap-2\"\n       [dndDropzone]=\"['col']\"\n       dndEffectAllowed=\"move\"\n       dndDragoverClass=\"dndDragover\"\n       (dndDrop)=\"onDndDrop($event)\">\n    <div dndPlaceholderRef class=\"manage-cols-placeholder\">Drop here</div>\n    <div *ngFor=\"let item of items; let i = index; trackBy: trackByUid\"\n         class=\"manage-cols-row cqa-flex cqa-items-center cqa-gap-2 cqa-px-2 cqa-py-1.5 cqa-rounded-md cqa-border cqa-border-gray-200 cqa-bg-gray-50\">\n      <span class=\"manage-cols-handle cqa-flex cqa-items-center cqa-justify-center cqa-text-gray-400\"\n            [dndDraggable]=\"item\"\n            [dndDisableIf]=\"isLocked(item)\"\n            dndType=\"col\"\n            dndEffectAllowed=\"move\">\n        <mat-icon style=\"font-size:18px;width:18px;height:18px;line-height:18px;\">drag_indicator</mat-icon>\n      </span>\n      <cqa-custom-input\n        class=\"cqa-flex-1\"\n        [value]=\"item.name\"\n        [fullWidth]=\"true\"\n        [disabled]=\"isLocked(item)\"\n        inputInlineStyle=\"font-family: 'Inter', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n        (valueChange)=\"onRename(i, $event)\">\n      </cqa-custom-input>\n      <button type=\"button\" class=\"manage-cols-icon-btn\"\n              title=\"Move up\"\n              [disabled]=\"i === 0\"\n              (click)=\"onMove(i, -1)\">\n        <mat-icon>arrow_upward</mat-icon>\n      </button>\n      <button type=\"button\" class=\"manage-cols-icon-btn\"\n              title=\"Move down\"\n              [disabled]=\"i === items.length - 1\"\n              (click)=\"onMove(i, 1)\">\n        <mat-icon>arrow_downward</mat-icon>\n      </button>\n      <button type=\"button\" class=\"manage-cols-icon-btn manage-cols-icon-btn-danger\"\n              title=\"Delete column\"\n              [disabled]=\"isLocked(item)\"\n              (click)=\"onRemove(i)\">\n        <mat-icon>delete</mat-icon>\n      </button>\n    </div>\n  </div>\n\n  <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-pt-3 cqa-mt-1 cqa-border-t cqa-border-dashed cqa-border-gray-200\">\n    <cqa-custom-input\n      class=\"cqa-flex-1\"\n      [value]=\"newColName\"\n      [fullWidth]=\"true\"\n      placeholder=\"New column name (e.g. currency)\"\n      inputInlineStyle=\"font-family: 'Inter', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n      (valueChange)=\"onNewColNameChange($event)\"\n      (enterPressed)=\"onAdd()\">\n    </cqa-custom-input>\n    <cqa-button\n      variant=\"outlined\"\n      btnSize=\"sm\"\n      icon=\"add\"\n      text=\"Add column\"\n      [disabled]=\"!canAdd\"\n      (clicked)=\"onAdd()\">\n    </cqa-button>\n  </div>\n\n  <span *ngIf=\"columnsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ columnsError }}</span>\n\n  <div class=\"cqa-px-3 cqa-py-2 cqa-rounded-md cqa-bg-primary-surface cqa-border cqa-border-success-100 cqa-text-xs cqa-text-gray-700\">\n    <strong class=\"cqa-text-gray-900\">Heads up:</strong>\n    Deleting a column removes its data from every environment. Renaming keeps the data in place.\n  </div>\n\n</div>\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFuYWdlLWNvbHVtbnMtZGlhbG9nLm1vZGVscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvbWFuYWdlLWNvbHVtbnMtZGlhbG9nL21hbmFnZS1jb2x1bW5zLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgTWFuYWdlQ29sdW1uc0RpYWxvZ0NvbHVtbiB7XG4gIG9yaWdpbmFsTmFtZTogc3RyaW5nIHwgbnVsbDtcbiAgbmFtZTogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1hbmFnZUNvbHVtbnNEaWFsb2dWYWx1ZSB7XG4gIGNvbHVtbnM6IE1hbmFnZUNvbHVtbnNEaWFsb2dDb2x1bW5bXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNYW5hZ2VDb2x1bW5zRGlhbG9nSW5wdXRzIHtcbiAgY29sdW1ucz86IHN0cmluZ1tdO1xuICBsb2NrZWRDb2x1bW5zPzogc3RyaW5nW107XG59XG4iXX0=