@dso-design-system/ui 0.0.2 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/alert/alert.component.mjs +54 -0
- package/esm2022/lib/badge/badge.component.mjs +22 -0
- package/esm2022/lib/badge/badge.directive.mjs +68 -0
- package/esm2022/lib/breadcrumb/breadcrumb.component.mjs +29 -0
- package/esm2022/lib/checkbox/checkbox.component.mjs +82 -0
- package/esm2022/lib/datepicker/datepicker.component.mjs +161 -0
- package/esm2022/lib/dialog/dialog.component.mjs +25 -0
- package/esm2022/lib/dropdown-list/dropdown-list.component.mjs +89 -0
- package/esm2022/lib/file-upload-items/file-upload-items.component.mjs +65 -0
- package/esm2022/lib/file-upload-multiple/file-upload-multiple.component.mjs +232 -0
- package/esm2022/lib/file-upload-multiple/upload-item.model.mjs +2 -0
- package/esm2022/lib/file-upload-multiple/upload-simulator.service.mjs +76 -0
- package/esm2022/lib/file-upload-single/file-upload-single.component.mjs +100 -0
- package/esm2022/lib/input-text/input-text.component.mjs +93 -0
- package/esm2022/lib/pagination/pagination.component.mjs +115 -0
- package/esm2022/lib/progress-bar/progress-bar.component.mjs +25 -0
- package/esm2022/lib/radio/radio.component.mjs +41 -0
- package/esm2022/lib/select-dropdown/select-dropdown.component.mjs +228 -0
- package/esm2022/lib/service/toast.service.mjs +20 -0
- package/esm2022/lib/side-navigation-bar/side-navigation-bar.component.mjs +113 -0
- package/esm2022/lib/spinner/spinner.component.mjs +60 -0
- package/esm2022/lib/table/table.component.mjs +136 -0
- package/esm2022/lib/tabs/tab.component.mjs +20 -0
- package/esm2022/lib/tabs/tabs.component.mjs +40 -0
- package/esm2022/lib/tag/tag.component.mjs +27 -0
- package/esm2022/lib/text-area/text-area.component.mjs +74 -0
- package/esm2022/lib/toast/toast.component.mjs +36 -0
- package/esm2022/lib/tooltip/tooltip.component.mjs +38 -0
- package/esm2022/lib/tooltip/tooltip.directive.mjs +105 -0
- package/esm2022/lib/top-navigation-bar/top-navigation-bar.component.mjs +24 -0
- package/esm2022/public-api.mjs +29 -2
- package/fesm2022/dso-design-system-ui.mjs +2053 -3
- package/fesm2022/dso-design-system-ui.mjs.map +1 -1
- package/lib/alert/alert.component.d.ts +20 -0
- package/lib/badge/badge.component.d.ts +8 -0
- package/lib/badge/badge.directive.d.ts +15 -0
- package/lib/breadcrumb/breadcrumb.component.d.ts +15 -0
- package/lib/checkbox/checkbox.component.d.ts +42 -0
- package/lib/datepicker/datepicker.component.d.ts +48 -0
- package/lib/dialog/dialog.component.d.ts +10 -0
- package/lib/dropdown-list/dropdown-list.component.d.ts +33 -0
- package/lib/file-upload-items/file-upload-items.component.d.ts +27 -0
- package/lib/file-upload-multiple/file-upload-multiple.component.d.ts +44 -0
- package/lib/file-upload-multiple/upload-item.model.d.ts +7 -0
- package/lib/file-upload-multiple/upload-simulator.service.d.ts +34 -0
- package/lib/file-upload-single/file-upload-single.component.d.ts +28 -0
- package/lib/input-text/input-text.component.d.ts +24 -0
- package/lib/pagination/pagination.component.d.ts +31 -0
- package/lib/progress-bar/progress-bar.component.d.ts +11 -0
- package/lib/radio/radio.component.d.ts +14 -0
- package/lib/select-dropdown/select-dropdown.component.d.ts +78 -0
- package/lib/service/toast.service.d.ts +16 -0
- package/lib/side-navigation-bar/side-navigation-bar.component.d.ts +74 -0
- package/lib/spinner/spinner.component.d.ts +23 -0
- package/lib/table/table.component.d.ts +43 -0
- package/lib/tabs/tab.component.d.ts +9 -0
- package/lib/tabs/tabs.component.d.ts +15 -0
- package/lib/tag/tag.component.d.ts +10 -0
- package/lib/text-area/text-area.component.d.ts +21 -0
- package/lib/toast/toast.component.d.ts +13 -0
- package/lib/tooltip/tooltip.component.d.ts +15 -0
- package/lib/tooltip/tooltip.directive.d.ts +19 -0
- package/lib/top-navigation-bar/top-navigation-bar.component.d.ts +16 -0
- package/package.json +1 -1
- package/public-api.d.ts +27 -0
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { ButtonComponent } from '../button/button.component';
|
|
4
|
+
import { FileUploadItemComponent } from '../file-upload-items/file-upload-items.component';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "../file-upload-multiple/upload-simulator.service";
|
|
7
|
+
import * as i2 from "@angular/common";
|
|
8
|
+
export class FileUploadSingleComponent {
|
|
9
|
+
sim;
|
|
10
|
+
/** Button label text */
|
|
11
|
+
btnLabel = 'Choose File';
|
|
12
|
+
/** Allowed file types, e.g. ".png,.jpg,.pdf" */
|
|
13
|
+
accept = '';
|
|
14
|
+
/** Emit the selected file to parent component */
|
|
15
|
+
fileSelected = new EventEmitter();
|
|
16
|
+
/** The currently uploaded file shown in the UI */
|
|
17
|
+
item = null;
|
|
18
|
+
constructor(sim) {
|
|
19
|
+
this.sim = sim;
|
|
20
|
+
}
|
|
21
|
+
/** Trigger the hidden file input click */
|
|
22
|
+
onButtonClick(input) {
|
|
23
|
+
input.click();
|
|
24
|
+
}
|
|
25
|
+
/** Handle file selection from input */
|
|
26
|
+
onFileChange(event) {
|
|
27
|
+
const input = event.target;
|
|
28
|
+
if (!input.files || input.files.length === 0)
|
|
29
|
+
return;
|
|
30
|
+
const file = input.files[0];
|
|
31
|
+
// Wrap file in UploadItem structure (same as multi-upload)
|
|
32
|
+
this.item = {
|
|
33
|
+
file,
|
|
34
|
+
progress: 0,
|
|
35
|
+
status: 'queued',
|
|
36
|
+
hideProgressBar: false
|
|
37
|
+
};
|
|
38
|
+
// Emit file to parent
|
|
39
|
+
this.fileSelected.emit(file);
|
|
40
|
+
// Start simulated upload
|
|
41
|
+
this.startUpload();
|
|
42
|
+
// Reset input so user can pick same file again
|
|
43
|
+
input.value = '';
|
|
44
|
+
}
|
|
45
|
+
/** Start upload simulation */
|
|
46
|
+
startUpload() {
|
|
47
|
+
if (!this.item)
|
|
48
|
+
return;
|
|
49
|
+
const item = this.item;
|
|
50
|
+
item.status = 'uploading';
|
|
51
|
+
item.progress = 0;
|
|
52
|
+
this.sim.simulateUpload(item,
|
|
53
|
+
// onProgress callback
|
|
54
|
+
(progress) => {
|
|
55
|
+
item.progress = progress;
|
|
56
|
+
},
|
|
57
|
+
// onComplete callback
|
|
58
|
+
() => {
|
|
59
|
+
item.status = 'completed';
|
|
60
|
+
item.progress = 100;
|
|
61
|
+
setTimeout(() => item.hideProgressBar = true, 300);
|
|
62
|
+
},
|
|
63
|
+
// onError callback
|
|
64
|
+
() => {
|
|
65
|
+
item.status = 'failed';
|
|
66
|
+
item.progress = null;
|
|
67
|
+
setTimeout(() => item.hideProgressBar = true, 300);
|
|
68
|
+
});
|
|
69
|
+
}
|
|
70
|
+
/** Retry failed upload */
|
|
71
|
+
retry() {
|
|
72
|
+
if (!this.item)
|
|
73
|
+
return;
|
|
74
|
+
this.item.status = 'queued';
|
|
75
|
+
this.item.progress = 0;
|
|
76
|
+
this.item.hideProgressBar = false;
|
|
77
|
+
this.startUpload();
|
|
78
|
+
}
|
|
79
|
+
/** Cancel or remove the file from UI */
|
|
80
|
+
clearFile() {
|
|
81
|
+
this.item = null;
|
|
82
|
+
}
|
|
83
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FileUploadSingleComponent, deps: [{ token: i1.UploadSimulatorService }], target: i0.ɵɵFactoryTarget.Component });
|
|
84
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: FileUploadSingleComponent, isStandalone: true, selector: "dso-file-upload-single", inputs: { btnLabel: "btnLabel", accept: "accept" }, outputs: { fileSelected: "fileSelected" }, ngImport: i0, template: "<div class=\"file-upload-single\">\r\n\r\n <!-- Hidden file input -->\r\n <input\r\n type=\"file\"\r\n #fileInput\r\n [accept]=\"accept\"\r\n (change)=\"onFileChange($event)\"\r\n hidden\r\n />\r\n\r\n <!-- Main button -->\r\n <dso-button\r\n [btnLabel]=\"btnLabel\"\r\n btnType=\"filled\"\r\n (onClick)=\"onButtonClick(fileInput)\"\r\n ></dso-button>\r\n\r\n <!-- Render file info using shared reusable component -->\r\n <dso-file-item\r\n *ngIf=\"item\"\r\n [item]=\"item\"\r\n (cancel)=\"clearFile()\"\r\n (remove)=\"clearFile()\"\r\n (retry)=\"retry()\"\r\n ></dso-file-item>\r\n\r\n</div>\r\n", styles: [".file-upload-single{display:flex;flex-direction:column;gap:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ButtonComponent, selector: "dso-button", inputs: ["btnLabel", "btnType", "btnSize", "btnIconName", "isDisabled", "isActive"], outputs: ["onClick"] }, { kind: "component", type: FileUploadItemComponent, selector: "dso-file-item", inputs: ["item"], outputs: ["retry", "remove", "cancel"] }] });
|
|
85
|
+
}
|
|
86
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FileUploadSingleComponent, decorators: [{
|
|
87
|
+
type: Component,
|
|
88
|
+
args: [{ selector: 'dso-file-upload-single', standalone: true, imports: [
|
|
89
|
+
CommonModule,
|
|
90
|
+
ButtonComponent,
|
|
91
|
+
FileUploadItemComponent
|
|
92
|
+
], template: "<div class=\"file-upload-single\">\r\n\r\n <!-- Hidden file input -->\r\n <input\r\n type=\"file\"\r\n #fileInput\r\n [accept]=\"accept\"\r\n (change)=\"onFileChange($event)\"\r\n hidden\r\n />\r\n\r\n <!-- Main button -->\r\n <dso-button\r\n [btnLabel]=\"btnLabel\"\r\n btnType=\"filled\"\r\n (onClick)=\"onButtonClick(fileInput)\"\r\n ></dso-button>\r\n\r\n <!-- Render file info using shared reusable component -->\r\n <dso-file-item\r\n *ngIf=\"item\"\r\n [item]=\"item\"\r\n (cancel)=\"clearFile()\"\r\n (remove)=\"clearFile()\"\r\n (retry)=\"retry()\"\r\n ></dso-file-item>\r\n\r\n</div>\r\n", styles: [".file-upload-single{display:flex;flex-direction:column;gap:8px}\n"] }]
|
|
93
|
+
}], ctorParameters: () => [{ type: i1.UploadSimulatorService }], propDecorators: { btnLabel: [{
|
|
94
|
+
type: Input
|
|
95
|
+
}], accept: [{
|
|
96
|
+
type: Input
|
|
97
|
+
}], fileSelected: [{
|
|
98
|
+
type: Output
|
|
99
|
+
}] } });
|
|
100
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload-single.component.js","sourceRoot":"","sources":["../../../../../projects/ui/src/lib/file-upload-single/file-upload-single.component.ts","../../../../../projects/ui/src/lib/file-upload-single/file-upload-single.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { ButtonComponent } from '../button/button.component';\r\nimport { UploadItem } from '../file-upload-multiple/upload-item.model';\r\nimport { UploadSimulatorService } from '../file-upload-multiple/upload-simulator.service';\r\nimport { FileUploadItemComponent } from '../file-upload-items/file-upload-items.component';\r\n\r\n@Component({\r\n  selector: 'dso-file-upload-single',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    ButtonComponent,\r\n    FileUploadItemComponent\r\n  ],\r\n  templateUrl: './file-upload-single.component.html',\r\n  styleUrls: ['./file-upload-single.component.scss']\r\n})\r\nexport class FileUploadSingleComponent {\r\n\r\n  /** Button label text */\r\n  @Input() btnLabel: string = 'Choose File';\r\n  \r\n  /** Allowed file types, e.g. \".png,.jpg,.pdf\" */\r\n  @Input() accept: string = '';\r\n  \r\n  /** Emit the selected file to parent component */\r\n  @Output() fileSelected = new EventEmitter<File>();\r\n\r\n  /** The currently uploaded file shown in the UI */\r\n  item: UploadItem | null = null;\r\n\r\n  constructor(private sim: UploadSimulatorService) {}\r\n\r\n  /** Trigger the hidden file input click */\r\n  onButtonClick(input: HTMLInputElement) {\r\n    input.click();\r\n  }\r\n\r\n  /** Handle file selection from input */\r\n  onFileChange(event: Event) {\r\n    const input = event.target as HTMLInputElement;\r\n    if (!input.files || input.files.length === 0) return;\r\n\r\n    const file = input.files[0];\r\n\r\n    // Wrap file in UploadItem structure (same as multi-upload)\r\n    this.item = {\r\n      file,\r\n      progress: 0,\r\n      status: 'queued',\r\n      hideProgressBar: false\r\n    };\r\n\r\n    // Emit file to parent\r\n    this.fileSelected.emit(file);\r\n\r\n    // Start simulated upload\r\n    this.startUpload();\r\n\r\n    // Reset input so user can pick same file again\r\n    input.value = '';\r\n  }\r\n\r\n  /** Start upload simulation */\r\n  startUpload() {\r\n    if (!this.item) return;\r\n\r\n    const item = this.item;\r\n    item.status = 'uploading';\r\n    item.progress = 0;\r\n\r\n    this.sim.simulateUpload(\r\n      item,\r\n\r\n      // onProgress callback\r\n      (progress) => {\r\n        item.progress = progress;\r\n      },\r\n\r\n      // onComplete callback\r\n      () => {\r\n        item.status = 'completed';\r\n        item.progress = 100;\r\n        setTimeout(() => item.hideProgressBar = true, 300);\r\n      },\r\n\r\n      // onError callback\r\n      () => {\r\n        item.status = 'failed';\r\n        item.progress = null;\r\n        setTimeout(() => item.hideProgressBar = true, 300);\r\n      }\r\n    );\r\n  }\r\n\r\n  /** Retry failed upload */\r\n  retry() {\r\n    if (!this.item) return;\r\n\r\n    this.item.status = 'queued';\r\n    this.item.progress = 0;\r\n    this.item.hideProgressBar = false;\r\n    this.startUpload();\r\n  }\r\n\r\n  /** Cancel or remove the file from UI */\r\n  clearFile() {\r\n    this.item = null;\r\n  }\r\n}\r\n","<div class=\"file-upload-single\">\r\n\r\n  <!-- Hidden file input -->\r\n  <input\r\n    type=\"file\"\r\n    #fileInput\r\n    [accept]=\"accept\"\r\n    (change)=\"onFileChange($event)\"\r\n    hidden\r\n  />\r\n\r\n  <!-- Main button -->\r\n  <dso-button\r\n    [btnLabel]=\"btnLabel\"\r\n    btnType=\"filled\"\r\n    (onClick)=\"onButtonClick(fileInput)\"\r\n  ></dso-button>\r\n\r\n  <!-- Render file info using shared reusable component -->\r\n  <dso-file-item\r\n    *ngIf=\"item\"\r\n    [item]=\"item\"\r\n    (cancel)=\"clearFile()\"\r\n    (remove)=\"clearFile()\"\r\n    (retry)=\"retry()\"\r\n  ></dso-file-item>\r\n\r\n</div>\r\n"]}
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/common";
|
|
6
|
+
import * as i2 from "@angular/forms";
|
|
7
|
+
export class TextInputComponent {
|
|
8
|
+
// -------------------------------
|
|
9
|
+
// INPUT PROPERTIES
|
|
10
|
+
// -------------------------------
|
|
11
|
+
inputLabel = ''; // Optional label displayed above input
|
|
12
|
+
value = ''; // Two-way bound input value
|
|
13
|
+
placeholder = 'Input value'; // Placeholder text
|
|
14
|
+
type = ''; // Input type (text, email, etc.)
|
|
15
|
+
isDisabled = false; // Disable the input
|
|
16
|
+
helperText = ''; // Optional helper text below input
|
|
17
|
+
errorText = ''; // Optional error message
|
|
18
|
+
enableValidation = true; // Flag to enable/disable validation
|
|
19
|
+
maxLength = 80; // Maximum characters allowed
|
|
20
|
+
// -------------------------------
|
|
21
|
+
// OUTPUT EVENTS
|
|
22
|
+
// -------------------------------
|
|
23
|
+
valueChange = new EventEmitter(); // Emit input changes to parent
|
|
24
|
+
// -------------------------------
|
|
25
|
+
// INTERNAL STATE
|
|
26
|
+
// -------------------------------
|
|
27
|
+
isTouched = false; // Tracks if user has interacted with input
|
|
28
|
+
isInvalid = false; // True if validation fails
|
|
29
|
+
charCount = 0; // Tracks current character count
|
|
30
|
+
isExceeded = false; // True if charCount exceeds maxLength
|
|
31
|
+
// -------------------------------
|
|
32
|
+
// EVENT HANDLERS
|
|
33
|
+
// -------------------------------
|
|
34
|
+
onInputChange(event) {
|
|
35
|
+
this.value = event.target.value; // Update value
|
|
36
|
+
const inputValue = event.target.value;
|
|
37
|
+
const charCount = inputValue.length;
|
|
38
|
+
// Track if max length exceeded
|
|
39
|
+
this.isExceeded = charCount > this.maxLength;
|
|
40
|
+
this.isTouched = true; // Mark input as touched
|
|
41
|
+
this.valueChange.emit(this.value); // Notify parent
|
|
42
|
+
this.checkValidity(); // Validate input
|
|
43
|
+
}
|
|
44
|
+
onBlur() {
|
|
45
|
+
this.isTouched = true; // Input lost focus → mark as touched
|
|
46
|
+
this.checkValidity(); // Validate when blurred
|
|
47
|
+
}
|
|
48
|
+
// -------------------------------
|
|
49
|
+
// HELPER METHODS
|
|
50
|
+
// -------------------------------
|
|
51
|
+
getCharCountStyle() {
|
|
52
|
+
const charCount = this.value.length;
|
|
53
|
+
if (this.isExceeded) {
|
|
54
|
+
return 'char-count-error'; // Apply error style if limit exceeded
|
|
55
|
+
}
|
|
56
|
+
// Optional: style differently if more than 50% of maxLength used
|
|
57
|
+
if (charCount > this.maxLength / 2) {
|
|
58
|
+
return '';
|
|
59
|
+
}
|
|
60
|
+
return ''; // Default style
|
|
61
|
+
}
|
|
62
|
+
checkValidity() {
|
|
63
|
+
// If touched and value is empty, mark as invalid
|
|
64
|
+
this.isInvalid = this.isTouched && !this.value.trim();
|
|
65
|
+
}
|
|
66
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TextInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
67
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: TextInputComponent, isStandalone: true, selector: "dso-input-text", inputs: { inputLabel: "inputLabel", value: "value", placeholder: "placeholder", type: "type", isDisabled: "isDisabled", helperText: "helperText", errorText: "errorText", enableValidation: "enableValidation", maxLength: "maxLength" }, outputs: { valueChange: "valueChange" }, ngImport: i0, template: "<div class=\"input-container\">\r\n <!-- Optional label -->\r\n <div class=\"text-wrapper\">\r\n <!-- Optional label -->\r\n <label *ngIf=\"inputLabel\">\r\n {{ inputLabel }}\r\n </label> \r\n <!-- Character count inside the container -->\r\n <div class=\"char-count\" [ngClass]=\"getCharCountStyle()\" *ngIf=\"maxLength\">\r\n <span>\r\n {{ value.length }} \r\n </span>\r\n / {{ maxLength }}\r\n </div>\r\n </div> \r\n <input\r\n [type]=\"type\" \r\n [placeholder]=\"placeholder\"\r\n [(ngModel)]=\"value\"\r\n (input)=\"onInputChange($event)\" \r\n (blur)=\"onBlur()\"\r\n [disabled]=\"isDisabled\"\r\n [class.disabled]=\"isDisabled\"\r\n [class.invalid]=\"(isInvalid && isTouched && enableValidation) || isExceeded\" \r\n />\r\n\r\n <!-- Helper Text \r\n This condition checks if the helper text should be displayed:\r\n - If the character limit is **not exceeded** and validation is **not enabled**, show the helper text.\r\n - If validation **is enabled**, show the helper text only if the input is **not invalid** and `helperText` is provided.\r\n - If the character limit is exceeded, the helper text will be replaced with an error message. -->\r\n <div *ngIf=\"!isExceeded && !enableValidation || ((!isInvalid && helperText) && !isExceeded)\" class=\"helper-text\">\r\n {{ helperText }}\r\n </div>\r\n\r\n <!-- Specified Error Text -->\r\n <div *ngIf=\"isInvalid && isTouched && enableValidation\" class=\"error-message\">\r\n {{ errorText }}\r\n </div>\r\n\r\n <!-- Error message if character limit exceeded -->\r\n <div *ngIf=\"isExceeded\" class=\"error-message\">\r\n Character limit exceeded.\r\n </div>\r\n</div>\r\n\r\n", styles: [":host{display:inline-block}.input-container{width:500px;display:flex;flex-direction:column;gap:8px}.text-wrapper{display:flex;justify-content:space-between}input{width:100%;height:48px;padding:12px 8px;font-size:16px;border:1px solid #ccc;border-radius:4px}input:hover{border:1px solid #071d35}input:focus{border-color:#007bff;outline:none}input.invalid{border-color:red}.error-message{color:red;font-size:16px}input.disabled{cursor:not-allowed;background-color:#f0f0f0;border:1px solid #ddd;color:#000}.helper-text{font-size:16px;color:#6c757d}.error-message{font-size:16px;color:red}.char-count{font-size:12px;color:#666;align-self:flex-end}.char-count span,.char-count-error{font-size:12px;color:#666}.char-count-error span{font-size:12px;color:red}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] });
|
|
68
|
+
}
|
|
69
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TextInputComponent, decorators: [{
|
|
70
|
+
type: Component,
|
|
71
|
+
args: [{ selector: 'dso-input-text', standalone: true, imports: [CommonModule, FormsModule], template: "<div class=\"input-container\">\r\n <!-- Optional label -->\r\n <div class=\"text-wrapper\">\r\n <!-- Optional label -->\r\n <label *ngIf=\"inputLabel\">\r\n {{ inputLabel }}\r\n </label> \r\n <!-- Character count inside the container -->\r\n <div class=\"char-count\" [ngClass]=\"getCharCountStyle()\" *ngIf=\"maxLength\">\r\n <span>\r\n {{ value.length }} \r\n </span>\r\n / {{ maxLength }}\r\n </div>\r\n </div> \r\n <input\r\n [type]=\"type\" \r\n [placeholder]=\"placeholder\"\r\n [(ngModel)]=\"value\"\r\n (input)=\"onInputChange($event)\" \r\n (blur)=\"onBlur()\"\r\n [disabled]=\"isDisabled\"\r\n [class.disabled]=\"isDisabled\"\r\n [class.invalid]=\"(isInvalid && isTouched && enableValidation) || isExceeded\" \r\n />\r\n\r\n <!-- Helper Text \r\n This condition checks if the helper text should be displayed:\r\n - If the character limit is **not exceeded** and validation is **not enabled**, show the helper text.\r\n - If validation **is enabled**, show the helper text only if the input is **not invalid** and `helperText` is provided.\r\n - If the character limit is exceeded, the helper text will be replaced with an error message. -->\r\n <div *ngIf=\"!isExceeded && !enableValidation || ((!isInvalid && helperText) && !isExceeded)\" class=\"helper-text\">\r\n {{ helperText }}\r\n </div>\r\n\r\n <!-- Specified Error Text -->\r\n <div *ngIf=\"isInvalid && isTouched && enableValidation\" class=\"error-message\">\r\n {{ errorText }}\r\n </div>\r\n\r\n <!-- Error message if character limit exceeded -->\r\n <div *ngIf=\"isExceeded\" class=\"error-message\">\r\n Character limit exceeded.\r\n </div>\r\n</div>\r\n\r\n", styles: [":host{display:inline-block}.input-container{width:500px;display:flex;flex-direction:column;gap:8px}.text-wrapper{display:flex;justify-content:space-between}input{width:100%;height:48px;padding:12px 8px;font-size:16px;border:1px solid #ccc;border-radius:4px}input:hover{border:1px solid #071d35}input:focus{border-color:#007bff;outline:none}input.invalid{border-color:red}.error-message{color:red;font-size:16px}input.disabled{cursor:not-allowed;background-color:#f0f0f0;border:1px solid #ddd;color:#000}.helper-text{font-size:16px;color:#6c757d}.error-message{font-size:16px;color:red}.char-count{font-size:12px;color:#666;align-self:flex-end}.char-count span,.char-count-error{font-size:12px;color:#666}.char-count-error span{font-size:12px;color:red}\n"] }]
|
|
72
|
+
}], propDecorators: { inputLabel: [{
|
|
73
|
+
type: Input
|
|
74
|
+
}], value: [{
|
|
75
|
+
type: Input
|
|
76
|
+
}], placeholder: [{
|
|
77
|
+
type: Input
|
|
78
|
+
}], type: [{
|
|
79
|
+
type: Input
|
|
80
|
+
}], isDisabled: [{
|
|
81
|
+
type: Input
|
|
82
|
+
}], helperText: [{
|
|
83
|
+
type: Input
|
|
84
|
+
}], errorText: [{
|
|
85
|
+
type: Input
|
|
86
|
+
}], enableValidation: [{
|
|
87
|
+
type: Input
|
|
88
|
+
}], maxLength: [{
|
|
89
|
+
type: Input
|
|
90
|
+
}], valueChange: [{
|
|
91
|
+
type: Output
|
|
92
|
+
}] } });
|
|
93
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-text.component.js","sourceRoot":"","sources":["../../../../../projects/ui/src/lib/input-text/input-text.component.ts","../../../../../projects/ui/src/lib/input-text/input-text.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, Input, Output, EventEmitter } from '@angular/core';\r\nimport { FormsModule } from '@angular/forms';\r\n\r\n@Component({\r\n  selector: 'dso-input-text',\r\n  standalone: true,\r\n  templateUrl: './input-text.component.html',\r\n  styleUrls: ['./input-text.component.scss'],\r\n  imports: [ CommonModule, FormsModule ]\r\n})\r\nexport class TextInputComponent {\r\n  // -------------------------------\r\n  // INPUT PROPERTIES\r\n  // -------------------------------\r\n  @Input() inputLabel: string = ''; // Optional label displayed above input\r\n  @Input() value: string = '';      // Two-way bound input value\r\n  @Input() placeholder: string = 'Input value'; // Placeholder text\r\n  @Input() type: string = '';       // Input type (text, email, etc.)\r\n  @Input() isDisabled: boolean = false; // Disable the input\r\n  @Input() helperText: string = '';     // Optional helper text below input\r\n  @Input() errorText: string = '';      // Optional error message\r\n  @Input() enableValidation: boolean = true; // Flag to enable/disable validation\r\n  @Input() maxLength: number = 80;      // Maximum characters allowed\r\n\r\n  // -------------------------------\r\n  // OUTPUT EVENTS\r\n  // -------------------------------\r\n  @Output() valueChange = new EventEmitter<string>(); // Emit input changes to parent\r\n\r\n  // -------------------------------\r\n  // INTERNAL STATE\r\n  // -------------------------------\r\n  isTouched: boolean = false;  // Tracks if user has interacted with input\r\n  isInvalid: boolean = false;  // True if validation fails\r\n  charCount: number = 0;       // Tracks current character count\r\n  isExceeded: boolean = false; // True if charCount exceeds maxLength\r\n\r\n  // -------------------------------\r\n  // EVENT HANDLERS\r\n  // -------------------------------\r\n  onInputChange(event: any): void {\r\n    this.value = event.target.value; // Update value\r\n    const inputValue = (event.target as HTMLTextAreaElement).value;\r\n    const charCount = inputValue.length;\r\n\r\n    // Track if max length exceeded\r\n    this.isExceeded = charCount > this.maxLength;\r\n\r\n    this.isTouched = true;            // Mark input as touched\r\n    this.valueChange.emit(this.value); // Notify parent\r\n    this.checkValidity();             // Validate input\r\n  }\r\n\r\n  onBlur(): void {\r\n    this.isTouched = true;  // Input lost focus → mark as touched\r\n    this.checkValidity();   // Validate when blurred\r\n  }\r\n\r\n  // -------------------------------\r\n  // HELPER METHODS\r\n  // -------------------------------\r\n  getCharCountStyle() {\r\n    const charCount = this.value.length;\r\n\r\n    if (this.isExceeded) {\r\n      return 'char-count-error'; // Apply error style if limit exceeded\r\n    }\r\n\r\n    // Optional: style differently if more than 50% of maxLength used\r\n    if (charCount > this.maxLength / 2) {\r\n      return '';\r\n    }\r\n\r\n    return ''; // Default style\r\n  }\r\n\r\n  checkValidity(): void {\r\n    // If touched and value is empty, mark as invalid\r\n    this.isInvalid = this.isTouched && !this.value.trim();\r\n  }\r\n}\r\n","<div class=\"input-container\">\r\n  <!-- Optional label -->\r\n    <div class=\"text-wrapper\">\r\n        <!-- Optional label -->\r\n        <label *ngIf=\"inputLabel\">\r\n            {{ inputLabel }}\r\n        </label>  \r\n        <!-- Character count inside the container -->\r\n        <div class=\"char-count\" [ngClass]=\"getCharCountStyle()\" *ngIf=\"maxLength\">\r\n            <span>\r\n                {{ value.length }}  \r\n            </span>\r\n            / {{ maxLength }}\r\n        </div>\r\n    </div> \r\n  <input\r\n    [type]=\"type\" \r\n    [placeholder]=\"placeholder\"\r\n    [(ngModel)]=\"value\"\r\n    (input)=\"onInputChange($event)\" \r\n    (blur)=\"onBlur()\"\r\n    [disabled]=\"isDisabled\"\r\n    [class.disabled]=\"isDisabled\"\r\n    [class.invalid]=\"(isInvalid && isTouched && enableValidation) || isExceeded\" \r\n    />\r\n\r\n    <!-- Helper Text \r\n        This condition checks if the helper text should be displayed:\r\n          - If the character limit is **not exceeded** and validation is **not enabled**, show the helper text.\r\n          - If validation **is enabled**, show the helper text only if the input is **not invalid** and `helperText` is provided.\r\n          - If the character limit is exceeded, the helper text will be replaced with an error message. -->\r\n    <div *ngIf=\"!isExceeded && !enableValidation || ((!isInvalid && helperText) && !isExceeded)\" class=\"helper-text\">\r\n      {{ helperText }}\r\n    </div>\r\n\r\n    <!-- Specified Error Text -->\r\n    <div *ngIf=\"isInvalid && isTouched && enableValidation\" class=\"error-message\">\r\n      {{ errorText }}\r\n    </div>\r\n\r\n    <!-- Error message if character limit exceeded -->\r\n    <div *ngIf=\"isExceeded\" class=\"error-message\">\r\n        Character limit exceeded.\r\n    </div>\r\n</div>\r\n\r\n"]}
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import { ButtonComponent } from '../button/button.component';
|
|
5
|
+
import { DropdownListComponent } from '../dropdown-list/dropdown-list.component';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/common";
|
|
8
|
+
import * as i2 from "@angular/forms";
|
|
9
|
+
export class PaginationComponent {
|
|
10
|
+
/* =======================
|
|
11
|
+
Inputs (controlled state)
|
|
12
|
+
======================= */
|
|
13
|
+
page = 1;
|
|
14
|
+
pageSize = 10;
|
|
15
|
+
totalItems = 0;
|
|
16
|
+
pageSizes = [10, 20, 50, 100];
|
|
17
|
+
/* =======================
|
|
18
|
+
Outputs (user intent)
|
|
19
|
+
======================= */
|
|
20
|
+
pageChange = new EventEmitter();
|
|
21
|
+
pageSizeChange = new EventEmitter();
|
|
22
|
+
/* =======================
|
|
23
|
+
Internal UI state
|
|
24
|
+
======================= */
|
|
25
|
+
pageNumbers = [];
|
|
26
|
+
jumpPageNumber = null;
|
|
27
|
+
pageSizeOptions = [];
|
|
28
|
+
/* =======================
|
|
29
|
+
Lifecycle
|
|
30
|
+
======================= */
|
|
31
|
+
ngOnChanges() {
|
|
32
|
+
this.buildPageSizeOptions();
|
|
33
|
+
this.generatePageNumbers();
|
|
34
|
+
}
|
|
35
|
+
/* =======================
|
|
36
|
+
Derived values
|
|
37
|
+
======================= */
|
|
38
|
+
get totalPages() {
|
|
39
|
+
return Math.max(Math.ceil(this.totalItems / this.pageSize), 1);
|
|
40
|
+
}
|
|
41
|
+
get startItem() {
|
|
42
|
+
return this.totalItems === 0 ? 0 : (this.page - 1) * this.pageSize + 1;
|
|
43
|
+
}
|
|
44
|
+
get endItem() {
|
|
45
|
+
return Math.min(this.page * this.pageSize, this.totalItems);
|
|
46
|
+
}
|
|
47
|
+
/* =======================
|
|
48
|
+
UI helpers
|
|
49
|
+
======================= */
|
|
50
|
+
buildPageSizeOptions() {
|
|
51
|
+
this.pageSizeOptions = this.pageSizes.map(size => ({
|
|
52
|
+
label: `${size} per page`,
|
|
53
|
+
value: size
|
|
54
|
+
}));
|
|
55
|
+
}
|
|
56
|
+
generatePageNumbers() {
|
|
57
|
+
const total = this.totalPages;
|
|
58
|
+
let start = Math.max(this.page - 2, 1);
|
|
59
|
+
let end = Math.min(start + 4, total);
|
|
60
|
+
start = Math.max(end - 4, 1);
|
|
61
|
+
this.pageNumbers = [];
|
|
62
|
+
for (let i = start; i <= end; i++) {
|
|
63
|
+
this.pageNumbers.push(i);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
/* =======================
|
|
67
|
+
Navigation (emit only)
|
|
68
|
+
======================= */
|
|
69
|
+
goToPage(page) {
|
|
70
|
+
if (page >= 1 && page <= this.totalPages && page !== this.page) {
|
|
71
|
+
this.pageChange.emit(page);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
goFirst() {
|
|
75
|
+
this.goToPage(1);
|
|
76
|
+
}
|
|
77
|
+
goPrevious() {
|
|
78
|
+
this.goToPage(this.page - 1);
|
|
79
|
+
}
|
|
80
|
+
goNext() {
|
|
81
|
+
this.goToPage(this.page + 1);
|
|
82
|
+
}
|
|
83
|
+
goLast() {
|
|
84
|
+
this.goToPage(this.totalPages);
|
|
85
|
+
}
|
|
86
|
+
onPageSizeChange(newSize) {
|
|
87
|
+
this.pageSizeChange.emit(newSize);
|
|
88
|
+
}
|
|
89
|
+
jumpToPage() {
|
|
90
|
+
if (!this.jumpPageNumber)
|
|
91
|
+
return;
|
|
92
|
+
const page = Math.max(1, Math.min(this.jumpPageNumber, this.totalPages));
|
|
93
|
+
this.pageChange.emit(page);
|
|
94
|
+
this.jumpPageNumber = null;
|
|
95
|
+
}
|
|
96
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PaginationComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
97
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: PaginationComponent, isStandalone: true, selector: "dso-pagination", inputs: { page: "page", pageSize: "pageSize", totalItems: "totalItems", pageSizes: "pageSizes" }, outputs: { pageChange: "pageChange", pageSizeChange: "pageSizeChange" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"pagination-container\">\r\n\r\n <!-- First / Previous -->\r\n <dso-button\r\n btnLabel=\"\u00AB\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === 1\"\r\n (onClick)=\"goFirst()\">\r\n </dso-button>\r\n\r\n <dso-button\r\n btnLabel=\"\u2039\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === 1\"\r\n (onClick)=\"goPrevious()\">\r\n </dso-button>\r\n\r\n <!-- Page numbers -->\r\n <div class=\"page-buttons\">\r\n <ng-container *ngFor=\"let p of pageNumbers\">\r\n <dso-button\r\n [btnLabel]=\"p.toString()\"\r\n btnType=\"ghost\"\r\n btnSize=\"mdBtn\"\r\n [isDisabled]=\"page === p\"\r\n [isActive]=\"page === p\"\r\n (onClick)=\"goToPage(p)\">\r\n </dso-button>\r\n </ng-container>\r\n </div>\r\n\r\n <!-- Next / Last -->\r\n <dso-button\r\n btnLabel=\"\u203A\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === totalPages\"\r\n (onClick)=\"goNext()\">\r\n </dso-button>\r\n\r\n <dso-button\r\n btnLabel=\"\u00BB\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === totalPages\"\r\n (onClick)=\"goLast()\">\r\n </dso-button>\r\n\r\n <!-- Page size -->\r\n <dso-dropdown-list\r\n [options]=\"pageSizeOptions\"\r\n [value]=\"pageSize\"\r\n (selectionChange)=\"onPageSizeChange($event)\">\r\n </dso-dropdown-list>\r\n\r\n <!-- Range info -->\r\n <span class=\"range-info\">\r\n Showing {{ startItem }}\u2013{{ endItem }} of {{ totalItems }} results\r\n </span>\r\n\r\n <!-- Jump to page -->\r\n <div class=\"jump-to-page\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n [(ngModel)]=\"jumpPageNumber\"\r\n (keydown.enter)=\"jumpToPage()\" />\r\n\r\n <dso-button\r\n btnLabel=\"Go\"\r\n btnType=\"ghost\"\r\n btnSize=\"mdBtn\"\r\n [isDisabled]=\"!jumpPageNumber\"\r\n (onClick)=\"jumpToPage()\">\r\n </dso-button>\r\n </div>\r\n\r\n</div>\r\n", styles: [".pagination-container{display:flex;align-items:center;gap:8px;font-size:14px}.range-info{margin-right:12px}button{padding:6px 10px;border-radius:4px;border:1px solid #ccc;background:#f0f0f0;cursor:pointer}button:disabled{opacity:.4;cursor:not-allowed}button.page-number.active{background-color:#007bff;color:#fff;font-weight:700}.page-buttons{display:flex;gap:4px}.page-buttons dso-button{flex:0 0 44px;text-align:center}.jump-to-page{display:flex;align-items:center;gap:4px}.jump-to-page input{width:48px;padding:6px 8px;border-radius:4px;border:1px solid #ccc;text-align:center}input::-webkit-outer-spin-button,input::-webkit-inner-spin-button{-webkit-appearance:none;margin:0}\n"], dependencies: [{ kind: "component", type: ButtonComponent, selector: "dso-button", inputs: ["btnLabel", "btnType", "btnSize", "btnIconName", "isDisabled", "isActive"], outputs: ["onClick"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: DropdownListComponent, selector: "dso-dropdown-list", inputs: ["options", "placeholder", "value", "disabled"], outputs: ["selectionChange"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] });
|
|
98
|
+
}
|
|
99
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PaginationComponent, decorators: [{
|
|
100
|
+
type: Component,
|
|
101
|
+
args: [{ selector: 'dso-pagination', standalone: true, imports: [ButtonComponent, CommonModule, DropdownListComponent, FormsModule], template: "<div class=\"pagination-container\">\r\n\r\n <!-- First / Previous -->\r\n <dso-button\r\n btnLabel=\"\u00AB\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === 1\"\r\n (onClick)=\"goFirst()\">\r\n </dso-button>\r\n\r\n <dso-button\r\n btnLabel=\"\u2039\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === 1\"\r\n (onClick)=\"goPrevious()\">\r\n </dso-button>\r\n\r\n <!-- Page numbers -->\r\n <div class=\"page-buttons\">\r\n <ng-container *ngFor=\"let p of pageNumbers\">\r\n <dso-button\r\n [btnLabel]=\"p.toString()\"\r\n btnType=\"ghost\"\r\n btnSize=\"mdBtn\"\r\n [isDisabled]=\"page === p\"\r\n [isActive]=\"page === p\"\r\n (onClick)=\"goToPage(p)\">\r\n </dso-button>\r\n </ng-container>\r\n </div>\r\n\r\n <!-- Next / Last -->\r\n <dso-button\r\n btnLabel=\"\u203A\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === totalPages\"\r\n (onClick)=\"goNext()\">\r\n </dso-button>\r\n\r\n <dso-button\r\n btnLabel=\"\u00BB\"\r\n btnType=\"outlined\"\r\n [isDisabled]=\"page === totalPages\"\r\n (onClick)=\"goLast()\">\r\n </dso-button>\r\n\r\n <!-- Page size -->\r\n <dso-dropdown-list\r\n [options]=\"pageSizeOptions\"\r\n [value]=\"pageSize\"\r\n (selectionChange)=\"onPageSizeChange($event)\">\r\n </dso-dropdown-list>\r\n\r\n <!-- Range info -->\r\n <span class=\"range-info\">\r\n Showing {{ startItem }}\u2013{{ endItem }} of {{ totalItems }} results\r\n </span>\r\n\r\n <!-- Jump to page -->\r\n <div class=\"jump-to-page\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n [(ngModel)]=\"jumpPageNumber\"\r\n (keydown.enter)=\"jumpToPage()\" />\r\n\r\n <dso-button\r\n btnLabel=\"Go\"\r\n btnType=\"ghost\"\r\n btnSize=\"mdBtn\"\r\n [isDisabled]=\"!jumpPageNumber\"\r\n (onClick)=\"jumpToPage()\">\r\n </dso-button>\r\n </div>\r\n\r\n</div>\r\n", styles: [".pagination-container{display:flex;align-items:center;gap:8px;font-size:14px}.range-info{margin-right:12px}button{padding:6px 10px;border-radius:4px;border:1px solid #ccc;background:#f0f0f0;cursor:pointer}button:disabled{opacity:.4;cursor:not-allowed}button.page-number.active{background-color:#007bff;color:#fff;font-weight:700}.page-buttons{display:flex;gap:4px}.page-buttons dso-button{flex:0 0 44px;text-align:center}.jump-to-page{display:flex;align-items:center;gap:4px}.jump-to-page input{width:48px;padding:6px 8px;border-radius:4px;border:1px solid #ccc;text-align:center}input::-webkit-outer-spin-button,input::-webkit-inner-spin-button{-webkit-appearance:none;margin:0}\n"] }]
|
|
102
|
+
}], propDecorators: { page: [{
|
|
103
|
+
type: Input
|
|
104
|
+
}], pageSize: [{
|
|
105
|
+
type: Input
|
|
106
|
+
}], totalItems: [{
|
|
107
|
+
type: Input
|
|
108
|
+
}], pageSizes: [{
|
|
109
|
+
type: Input
|
|
110
|
+
}], pageChange: [{
|
|
111
|
+
type: Output
|
|
112
|
+
}], pageSizeChange: [{
|
|
113
|
+
type: Output
|
|
114
|
+
}] } });
|
|
115
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.component.js","sourceRoot":"","sources":["../../../../../projects/ui/src/lib/pagination/pagination.component.ts","../../../../../projects/ui/src/lib/pagination/pagination.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnChanges } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { ButtonComponent } from '../button/button.component';\r\nimport { DropdownListComponent } from '../dropdown-list/dropdown-list.component';\r\n\r\n@Component({\r\n  selector: 'dso-pagination',\r\n  standalone: true,\r\n  imports: [ButtonComponent, CommonModule, DropdownListComponent, FormsModule],\r\n  templateUrl: './pagination.component.html',\r\n  styleUrls: ['./pagination.component.scss']\r\n})\r\nexport class PaginationComponent implements OnChanges {\r\n\r\n  /* =======================\r\n     Inputs (controlled state)\r\n     ======================= */\r\n\r\n  @Input() page = 1;\r\n  @Input() pageSize = 10;\r\n  @Input() totalItems = 0;\r\n  @Input() pageSizes: number[] = [10, 20, 50, 100];\r\n\r\n  /* =======================\r\n     Outputs (user intent)\r\n     ======================= */\r\n\r\n  @Output() pageChange = new EventEmitter<number>();\r\n  @Output() pageSizeChange = new EventEmitter<number>();\r\n\r\n  /* =======================\r\n     Internal UI state\r\n     ======================= */\r\n\r\n  pageNumbers: number[] = [];\r\n  jumpPageNumber: number | null = null;\r\n  pageSizeOptions: { label: string; value: number }[] = [];\r\n\r\n  /* =======================\r\n     Lifecycle\r\n     ======================= */\r\n\r\n  ngOnChanges(): void {\r\n    this.buildPageSizeOptions();\r\n    this.generatePageNumbers();\r\n  }\r\n\r\n  /* =======================\r\n     Derived values\r\n     ======================= */\r\n\r\n  get totalPages(): number {\r\n    return Math.max(Math.ceil(this.totalItems / this.pageSize), 1);\r\n  }\r\n\r\n  get startItem(): number {\r\n    return this.totalItems === 0 ? 0 : (this.page - 1) * this.pageSize + 1;\r\n  }\r\n\r\n  get endItem(): number {\r\n    return Math.min(this.page * this.pageSize, this.totalItems);\r\n  }\r\n\r\n  /* =======================\r\n     UI helpers\r\n     ======================= */\r\n\r\n  private buildPageSizeOptions(): void {\r\n    this.pageSizeOptions = this.pageSizes.map(size => ({\r\n      label: `${size} per page`,\r\n      value: size\r\n    }));\r\n  }\r\n\r\n  private generatePageNumbers(): void {\r\n    const total = this.totalPages;\r\n    let start = Math.max(this.page - 2, 1);\r\n    let end = Math.min(start + 4, total);\r\n\r\n    start = Math.max(end - 4, 1);\r\n\r\n    this.pageNumbers = [];\r\n    for (let i = start; i <= end; i++) {\r\n      this.pageNumbers.push(i);\r\n    }\r\n  }\r\n\r\n  /* =======================\r\n     Navigation (emit only)\r\n     ======================= */\r\n\r\n  goToPage(page: number): void {\r\n    if (page >= 1 && page <= this.totalPages && page !== this.page) {\r\n      this.pageChange.emit(page);\r\n    }\r\n  }\r\n\r\n  goFirst(): void {\r\n    this.goToPage(1);\r\n  }\r\n\r\n  goPrevious(): void {\r\n    this.goToPage(this.page - 1);\r\n  }\r\n\r\n  goNext(): void {\r\n    this.goToPage(this.page + 1);\r\n  }\r\n\r\n  goLast(): void {\r\n    this.goToPage(this.totalPages);\r\n  }\r\n\r\n  onPageSizeChange(newSize: number): void {\r\n    this.pageSizeChange.emit(newSize);\r\n  }\r\n\r\n  jumpToPage(): void {\r\n    if (!this.jumpPageNumber) return;\r\n\r\n    const page = Math.max(1, Math.min(this.jumpPageNumber, this.totalPages));\r\n    this.pageChange.emit(page);\r\n    this.jumpPageNumber = null;\r\n  }\r\n}\r\n","<div class=\"pagination-container\">\r\n\r\n  <!-- First / Previous -->\r\n  <dso-button\r\n    btnLabel=\"«\"\r\n    btnType=\"outlined\"\r\n    [isDisabled]=\"page === 1\"\r\n    (onClick)=\"goFirst()\">\r\n  </dso-button>\r\n\r\n  <dso-button\r\n    btnLabel=\"‹\"\r\n    btnType=\"outlined\"\r\n    [isDisabled]=\"page === 1\"\r\n    (onClick)=\"goPrevious()\">\r\n  </dso-button>\r\n\r\n  <!-- Page numbers -->\r\n  <div class=\"page-buttons\">\r\n    <ng-container *ngFor=\"let p of pageNumbers\">\r\n      <dso-button\r\n        [btnLabel]=\"p.toString()\"\r\n        btnType=\"ghost\"\r\n        btnSize=\"mdBtn\"\r\n        [isDisabled]=\"page === p\"\r\n        [isActive]=\"page === p\"\r\n        (onClick)=\"goToPage(p)\">\r\n      </dso-button>\r\n    </ng-container>\r\n  </div>\r\n\r\n  <!-- Next / Last -->\r\n  <dso-button\r\n    btnLabel=\"›\"\r\n    btnType=\"outlined\"\r\n    [isDisabled]=\"page === totalPages\"\r\n    (onClick)=\"goNext()\">\r\n  </dso-button>\r\n\r\n  <dso-button\r\n    btnLabel=\"»\"\r\n    btnType=\"outlined\"\r\n    [isDisabled]=\"page === totalPages\"\r\n    (onClick)=\"goLast()\">\r\n  </dso-button>\r\n\r\n  <!-- Page size -->\r\n  <dso-dropdown-list\r\n    [options]=\"pageSizeOptions\"\r\n    [value]=\"pageSize\"\r\n    (selectionChange)=\"onPageSizeChange($event)\">\r\n  </dso-dropdown-list>\r\n\r\n  <!-- Range info -->\r\n  <span class=\"range-info\">\r\n    Showing {{ startItem }}–{{ endItem }} of {{ totalItems }} results\r\n  </span>\r\n\r\n  <!-- Jump to page -->\r\n  <div class=\"jump-to-page\">\r\n    <input\r\n      type=\"number\"\r\n      min=\"1\"\r\n      [(ngModel)]=\"jumpPageNumber\"\r\n      (keydown.enter)=\"jumpToPage()\" />\r\n\r\n    <dso-button\r\n      btnLabel=\"Go\"\r\n      btnType=\"ghost\"\r\n      btnSize=\"mdBtn\"\r\n      [isDisabled]=\"!jumpPageNumber\"\r\n      (onClick)=\"jumpToPage()\">\r\n    </dso-button>\r\n  </div>\r\n\r\n</div>\r\n"]}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, Input } from '@angular/core';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
export class ProgressBarComponent {
|
|
6
|
+
/** Progress value (0–100). If null, shows indeterminate animation. */
|
|
7
|
+
value = null;
|
|
8
|
+
/** Color variants */
|
|
9
|
+
color = 'primary';
|
|
10
|
+
/** Optional animation toggle */
|
|
11
|
+
animated = true;
|
|
12
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ProgressBarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
13
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: ProgressBarComponent, isStandalone: true, selector: "dso-progress-bar", inputs: { value: "value", color: "color", animated: "animated" }, ngImport: i0, template: "<div class=\"progress-bar-container\">\r\n <div\r\n class=\"progress-bar-fill\"\r\n [ngClass]=\"[\r\n color,\r\n animated ? 'animated' : '',\r\n value === null ? 'indeterminate' : ''\r\n ]\"\r\n [style.width.%]=\"value !== null ? value : 50\"\r\n ></div>\r\n</div>\r\n", styles: ["@charset \"UTF-8\";.progress-bar-container{width:100%;height:8px;background-color:#e9ecef;border-radius:4px;overflow:hidden;position:relative;min-width:240px}.progress-bar-fill{position:absolute;height:100%;border-radius:4px;transition:width .4s ease}.progress-bar-fill.primary{background-color:#007bff}.progress-bar-fill.success{background-color:#28a745}.progress-bar-fill.warning{background-color:#ffc107}.progress-bar-fill.danger{background-color:#dc3545}.progress-bar-fill.animated{transition:width .4s ease-in-out}.progress-bar-fill.indeterminate{width:30%;animation:scrollLoop 1.5s linear infinite}@keyframes scrollLoop{0%{transform:translate(-100%)}to{transform:translate(400%)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] });
|
|
14
|
+
}
|
|
15
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ProgressBarComponent, decorators: [{
|
|
16
|
+
type: Component,
|
|
17
|
+
args: [{ selector: 'dso-progress-bar', standalone: true, imports: [CommonModule], template: "<div class=\"progress-bar-container\">\r\n <div\r\n class=\"progress-bar-fill\"\r\n [ngClass]=\"[\r\n color,\r\n animated ? 'animated' : '',\r\n value === null ? 'indeterminate' : ''\r\n ]\"\r\n [style.width.%]=\"value !== null ? value : 50\"\r\n ></div>\r\n</div>\r\n", styles: ["@charset \"UTF-8\";.progress-bar-container{width:100%;height:8px;background-color:#e9ecef;border-radius:4px;overflow:hidden;position:relative;min-width:240px}.progress-bar-fill{position:absolute;height:100%;border-radius:4px;transition:width .4s ease}.progress-bar-fill.primary{background-color:#007bff}.progress-bar-fill.success{background-color:#28a745}.progress-bar-fill.warning{background-color:#ffc107}.progress-bar-fill.danger{background-color:#dc3545}.progress-bar-fill.animated{transition:width .4s ease-in-out}.progress-bar-fill.indeterminate{width:30%;animation:scrollLoop 1.5s linear infinite}@keyframes scrollLoop{0%{transform:translate(-100%)}to{transform:translate(400%)}}\n"] }]
|
|
18
|
+
}], propDecorators: { value: [{
|
|
19
|
+
type: Input
|
|
20
|
+
}], color: [{
|
|
21
|
+
type: Input
|
|
22
|
+
}], animated: [{
|
|
23
|
+
type: Input
|
|
24
|
+
}] } });
|
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvZ3Jlc3MtYmFyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3VpL3NyYy9saWIvcHJvZ3Jlc3MtYmFyL3Byb2dyZXNzLWJhci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy91aS9zcmMvbGliL3Byb2dyZXNzLWJhci9wcm9ncmVzcy1iYXIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFTakQsTUFBTSxPQUFPLG9CQUFvQjtJQUMvQixzRUFBc0U7SUFDN0QsS0FBSyxHQUFrQixJQUFJLENBQUM7SUFFckMscUJBQXFCO0lBQ1osS0FBSyxHQUFpRCxTQUFTLENBQUM7SUFFekUsZ0NBQWdDO0lBQ3ZCLFFBQVEsR0FBWSxJQUFJLENBQUM7d0dBUnZCLG9CQUFvQjs0RkFBcEIsb0JBQW9CLDhJQ1ZqQywyU0FXQSx5dUJESFksWUFBWTs7NEZBRVgsb0JBQW9CO2tCQVBoQyxTQUFTOytCQUNFLGtCQUFrQixjQUdoQixJQUFJLFdBQ1AsQ0FBQyxZQUFZLENBQUM7OEJBSWQsS0FBSztzQkFBYixLQUFLO2dCQUdHLEtBQUs7c0JBQWIsS0FBSztnQkFHRyxRQUFRO3NCQUFoQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICdkc28tcHJvZ3Jlc3MtYmFyJyxcclxuICB0ZW1wbGF0ZVVybDogJy4vcHJvZ3Jlc3MtYmFyLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybHM6IFsnLi9wcm9ncmVzcy1iYXIuY29tcG9uZW50LnNjc3MnXSxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBQcm9ncmVzc0JhckNvbXBvbmVudCB7XHJcbiAgLyoqIFByb2dyZXNzIHZhbHVlICgw4oCTMTAwKS4gSWYgbnVsbCwgc2hvd3MgaW5kZXRlcm1pbmF0ZSBhbmltYXRpb24uICovXHJcbiAgQElucHV0KCkgdmFsdWU6IG51bWJlciB8IG51bGwgPSBudWxsO1xyXG5cclxuICAvKiogQ29sb3IgdmFyaWFudHMgKi9cclxuICBASW5wdXQoKSBjb2xvcjogJ3ByaW1hcnknIHwgJ3N1Y2Nlc3MnIHwgJ3dhcm5pbmcnIHwgJ2RhbmdlcicgPSAncHJpbWFyeSc7XHJcblxyXG4gIC8qKiBPcHRpb25hbCBhbmltYXRpb24gdG9nZ2xlICovXHJcbiAgQElucHV0KCkgYW5pbWF0ZWQ6IGJvb2xlYW4gPSB0cnVlO1xyXG59IiwiPGRpdiBjbGFzcz1cInByb2dyZXNzLWJhci1jb250YWluZXJcIj5cclxuICA8ZGl2XHJcbiAgICBjbGFzcz1cInByb2dyZXNzLWJhci1maWxsXCJcclxuICAgIFtuZ0NsYXNzXT1cIltcclxuICAgICAgY29sb3IsXHJcbiAgICAgIGFuaW1hdGVkID8gJ2FuaW1hdGVkJyA6ICcnLFxyXG4gICAgICB2YWx1ZSA9PT0gbnVsbCA/ICdpbmRldGVybWluYXRlJyA6ICcnXHJcbiAgICBdXCJcclxuICAgIFtzdHlsZS53aWR0aC4lXT1cInZhbHVlICE9PSBudWxsID8gdmFsdWUgOiA1MFwiXHJcbiAgPjwvZGl2PlxyXG48L2Rpdj5cclxuIl19
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/common";
|
|
6
|
+
export class RadioComponent {
|
|
7
|
+
label = 'Radio label';
|
|
8
|
+
value;
|
|
9
|
+
name; // for grouping radios
|
|
10
|
+
disabled = false;
|
|
11
|
+
error = false;
|
|
12
|
+
isChecked = false;
|
|
13
|
+
change = new EventEmitter();
|
|
14
|
+
// Method to handle change in radio state
|
|
15
|
+
toggleRadio(event) {
|
|
16
|
+
// console.log(this.value);
|
|
17
|
+
event.stopPropagation();
|
|
18
|
+
this.change.emit(this.value); // Emit the updated value to the parent
|
|
19
|
+
}
|
|
20
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RadioComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
21
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: RadioComponent, isStandalone: true, selector: "dso-radio", inputs: { label: "label", value: "value", name: "name", disabled: "disabled", error: "error", isChecked: "isChecked" }, outputs: { change: "change" }, ngImport: i0, template: "<label class=\"radio-wrapper\" \r\n[class.disabled]=\"disabled\" \r\n[class.error]=\"error\">\r\n <input\r\n class=\"radio-input\"\r\n type=\"radio\"\r\n [attr.name]=\"name\"\r\n [value]=\"value\"\r\n [checked]=\"isChecked\"\r\n [disabled]=\"disabled\"\r\n (change)=\"toggleRadio($event)\"\r\n />\r\n <span class=\"custom-radio\"></span>\r\n <span *ngIf=\"label\" class=\"radio-label\">{{ label }}</span>\r\n</label>\r\n", styles: [".radio-wrapper{display:flex;align-items:center;gap:8px;cursor:pointer}.radio-wrapper.disabled{cursor:not-allowed}.radio-wrapper.disabled .custom-radio{background-color:#f0f0f0;border-color:#d0d0d0}.radio-wrapper.disabled .radio-input:checked~.custom-radio{border:1px solid rgba(0,0,255,.5)}.radio-wrapper.disabled .custom-radio:after{opacity:.5}.radio-input{position:absolute;width:1px;height:1px;margin:-1px;border:0;padding:0;clip:rect(0 0 0 0);clip-path:inset(100%);overflow:hidden;white-space:nowrap}.custom-radio{display:flex;align-items:center;justify-content:center;width:24px;height:24px;border:1px solid #666666;border-radius:50%}.radio-input:checked~.custom-radio{border:1px solid blue}input[type=radio]:checked+.custom-radio:after{display:block;content:\"\";width:100%;height:100%;background:#00f;border:5px solid blue;background:transparent;box-sizing:border-box;border-radius:50%}.radio-label{font-size:14px}.error .custom-radio{border-color:red}.radio-wrapper:not(.disabled):hover .custom-radio{background-color:#eef5f9;border-color:#0000ffc0;cursor:pointer}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
22
|
+
}
|
|
23
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RadioComponent, decorators: [{
|
|
24
|
+
type: Component,
|
|
25
|
+
args: [{ selector: 'dso-radio', standalone: true, imports: [FormsModule, CommonModule], template: "<label class=\"radio-wrapper\" \r\n[class.disabled]=\"disabled\" \r\n[class.error]=\"error\">\r\n <input\r\n class=\"radio-input\"\r\n type=\"radio\"\r\n [attr.name]=\"name\"\r\n [value]=\"value\"\r\n [checked]=\"isChecked\"\r\n [disabled]=\"disabled\"\r\n (change)=\"toggleRadio($event)\"\r\n />\r\n <span class=\"custom-radio\"></span>\r\n <span *ngIf=\"label\" class=\"radio-label\">{{ label }}</span>\r\n</label>\r\n", styles: [".radio-wrapper{display:flex;align-items:center;gap:8px;cursor:pointer}.radio-wrapper.disabled{cursor:not-allowed}.radio-wrapper.disabled .custom-radio{background-color:#f0f0f0;border-color:#d0d0d0}.radio-wrapper.disabled .radio-input:checked~.custom-radio{border:1px solid rgba(0,0,255,.5)}.radio-wrapper.disabled .custom-radio:after{opacity:.5}.radio-input{position:absolute;width:1px;height:1px;margin:-1px;border:0;padding:0;clip:rect(0 0 0 0);clip-path:inset(100%);overflow:hidden;white-space:nowrap}.custom-radio{display:flex;align-items:center;justify-content:center;width:24px;height:24px;border:1px solid #666666;border-radius:50%}.radio-input:checked~.custom-radio{border:1px solid blue}input[type=radio]:checked+.custom-radio:after{display:block;content:\"\";width:100%;height:100%;background:#00f;border:5px solid blue;background:transparent;box-sizing:border-box;border-radius:50%}.radio-label{font-size:14px}.error .custom-radio{border-color:red}.radio-wrapper:not(.disabled):hover .custom-radio{background-color:#eef5f9;border-color:#0000ffc0;cursor:pointer}\n"] }]
|
|
26
|
+
}], propDecorators: { label: [{
|
|
27
|
+
type: Input
|
|
28
|
+
}], value: [{
|
|
29
|
+
type: Input
|
|
30
|
+
}], name: [{
|
|
31
|
+
type: Input
|
|
32
|
+
}], disabled: [{
|
|
33
|
+
type: Input
|
|
34
|
+
}], error: [{
|
|
35
|
+
type: Input
|
|
36
|
+
}], isChecked: [{
|
|
37
|
+
type: Input
|
|
38
|
+
}], change: [{
|
|
39
|
+
type: Output
|
|
40
|
+
}] } });
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFkaW8uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdWkvc3JjL2xpYi9yYWRpby9yYWRpby5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy91aS9zcmMvbGliL3JhZGlvL3JhZGlvLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsWUFBWSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3ZFLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7O0FBUzdDLE1BQU0sT0FBTyxjQUFjO0lBQ2hCLEtBQUssR0FBWSxhQUFhLENBQUM7SUFDL0IsS0FBSyxDQUFVO0lBQ2YsSUFBSSxDQUFVLENBQUMsc0JBQXNCO0lBQ3JDLFFBQVEsR0FBRyxLQUFLLENBQUM7SUFDakIsS0FBSyxHQUFHLEtBQUssQ0FBQztJQUNkLFNBQVMsR0FBWSxLQUFLLENBQUM7SUFFMUIsTUFBTSxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7SUFHNUMseUNBQXlDO0lBQzNDLFdBQVcsQ0FBQyxLQUFZO1FBQ3BCLDJCQUEyQjtRQUMzQixLQUFLLENBQUMsZUFBZSxFQUFFLENBQUM7UUFDeEIsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsdUNBQXVDO0lBQ3pFLENBQUM7d0dBaEJVLGNBQWM7NEZBQWQsY0FBYyw0TkNYM0IsK2JBZUEsMm1DRE5hLFdBQVcsOEJBQUUsWUFBWTs7NEZBRXpCLGNBQWM7a0JBUDFCLFNBQVM7K0JBQ0UsV0FBVyxjQUNULElBQUksV0FHUCxDQUFFLFdBQVcsRUFBRSxZQUFZLENBQUU7OEJBRzdCLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxLQUFLO3NCQUFiLEtBQUs7Z0JBQ0csSUFBSTtzQkFBWixLQUFLO2dCQUNHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBQ0csS0FBSztzQkFBYixLQUFLO2dCQUNHLFNBQVM7c0JBQWpCLEtBQUs7Z0JBRUksTUFBTTtzQkFBZixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgT3V0cHV0LCBFdmVudEVtaXR0ZXIgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ2Rzby1yYWRpbycsXHJcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcclxuICB0ZW1wbGF0ZVVybDogJy4vcmFkaW8uY29tcG9uZW50Lmh0bWwnLFxyXG4gIHN0eWxlVXJsczogWycuL3JhZGlvLmNvbXBvbmVudC5zY3NzJ10sXHJcbiAgaW1wb3J0czogWyBGb3Jtc01vZHVsZSwgQ29tbW9uTW9kdWxlIF1cclxufSlcclxuZXhwb3J0IGNsYXNzIFJhZGlvQ29tcG9uZW50IHtcclxuICBASW5wdXQoKSBsYWJlbD86IHN0cmluZyA9ICdSYWRpbyBsYWJlbCc7XHJcbiAgQElucHV0KCkgdmFsdWUhOiBzdHJpbmc7XHJcbiAgQElucHV0KCkgbmFtZSE6IHN0cmluZzsgLy8gZm9yIGdyb3VwaW5nIHJhZGlvc1xyXG4gIEBJbnB1dCgpIGRpc2FibGVkID0gZmFsc2U7XHJcbiAgQElucHV0KCkgZXJyb3IgPSBmYWxzZTtcclxuICBASW5wdXQoKSBpc0NoZWNrZWQ6IGJvb2xlYW4gPSBmYWxzZTtcclxuXHJcbiAgQE91dHB1dCgpIGNoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8c3RyaW5nPigpO1xyXG5cclxuICBcclxuICAgIC8vIE1ldGhvZCB0byBoYW5kbGUgY2hhbmdlIGluIHJhZGlvIHN0YXRlXHJcbiAgdG9nZ2xlUmFkaW8oZXZlbnQ6IEV2ZW50KSB7XHJcbiAgICAgIC8vIGNvbnNvbGUubG9nKHRoaXMudmFsdWUpO1xyXG4gICAgICBldmVudC5zdG9wUHJvcGFnYXRpb24oKTsgXHJcbiAgICAgIHRoaXMuY2hhbmdlLmVtaXQodGhpcy52YWx1ZSk7IC8vIEVtaXQgdGhlIHVwZGF0ZWQgdmFsdWUgdG8gdGhlIHBhcmVudFxyXG4gIH1cclxuXHJcbn1cclxuIiwiPGxhYmVsIGNsYXNzPVwicmFkaW8td3JhcHBlclwiIFxyXG5bY2xhc3MuZGlzYWJsZWRdPVwiZGlzYWJsZWRcIiBcclxuW2NsYXNzLmVycm9yXT1cImVycm9yXCI+XHJcbiAgPGlucHV0XHJcbiAgICBjbGFzcz1cInJhZGlvLWlucHV0XCJcclxuICAgIHR5cGU9XCJyYWRpb1wiXHJcbiAgICBbYXR0ci5uYW1lXT1cIm5hbWVcIlxyXG4gICAgW3ZhbHVlXT1cInZhbHVlXCJcclxuICAgIFtjaGVja2VkXT1cImlzQ2hlY2tlZFwiXHJcbiAgICBbZGlzYWJsZWRdPVwiZGlzYWJsZWRcIlxyXG4gICAgKGNoYW5nZSk9XCJ0b2dnbGVSYWRpbygkZXZlbnQpXCJcclxuICAvPlxyXG4gIDxzcGFuIGNsYXNzPVwiY3VzdG9tLXJhZGlvXCI+PC9zcGFuPlxyXG4gIDxzcGFuICpuZ0lmPVwibGFiZWxcIiBjbGFzcz1cInJhZGlvLWxhYmVsXCI+e3sgbGFiZWwgfX08L3NwYW4+XHJcbjwvbGFiZWw+XHJcbiJdfQ==
|