@dso-design-system/ui 0.0.1 → 0.1.0

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.
Files changed (65) hide show
  1. package/esm2022/lib/alert/alert.component.mjs +54 -0
  2. package/esm2022/lib/badge/badge.component.mjs +22 -0
  3. package/esm2022/lib/breadcrumb/breadcrumb.component.mjs +29 -0
  4. package/esm2022/lib/checkbox/checkbox.component.mjs +82 -0
  5. package/esm2022/lib/datepicker/datepicker.component.mjs +161 -0
  6. package/esm2022/lib/dialog/dialog.component.mjs +25 -0
  7. package/esm2022/lib/directives/truncate.directive.mjs +71 -0
  8. package/esm2022/lib/dropdown-list/dropdown-list.component.mjs +89 -0
  9. package/esm2022/lib/file-upload-items/file-upload-items.component.mjs +65 -0
  10. package/esm2022/lib/file-upload-multiple/file-upload-multiple.component.mjs +232 -0
  11. package/esm2022/lib/file-upload-multiple/upload-item.model.mjs +2 -0
  12. package/esm2022/lib/file-upload-multiple/upload-simulator.service.mjs +76 -0
  13. package/esm2022/lib/file-upload-single/file-upload-single.component.mjs +100 -0
  14. package/esm2022/lib/input-text/input-text.component.mjs +93 -0
  15. package/esm2022/lib/pagination/pagination.component.mjs +115 -0
  16. package/esm2022/lib/progress-bar/progress-bar.component.mjs +25 -0
  17. package/esm2022/lib/radio/radio.component.mjs +41 -0
  18. package/esm2022/lib/select-dropdown/select-dropdown.component.mjs +228 -0
  19. package/esm2022/lib/service/toast.service.mjs +20 -0
  20. package/esm2022/lib/side-navigation-bar/side-navigation-bar.component.mjs +113 -0
  21. package/esm2022/lib/spinner/spinner.component.mjs +60 -0
  22. package/esm2022/lib/table/table.component.mjs +136 -0
  23. package/esm2022/lib/tabs/tab.component.mjs +20 -0
  24. package/esm2022/lib/tabs/tabs.component.mjs +40 -0
  25. package/esm2022/lib/tag/tag.component.mjs +27 -0
  26. package/esm2022/lib/text-area/text-area.component.mjs +74 -0
  27. package/esm2022/lib/toast/toast.component.mjs +36 -0
  28. package/esm2022/lib/tooltip/tooltip.component.mjs +38 -0
  29. package/esm2022/lib/tooltip/tooltip.directive.mjs +105 -0
  30. package/esm2022/lib/top-navigation-bar/top-navigation-bar.component.mjs +24 -0
  31. package/esm2022/public-api.mjs +27 -2
  32. package/fesm2022/dso-design-system-ui.mjs +2056 -3
  33. package/fesm2022/dso-design-system-ui.mjs.map +1 -1
  34. package/lib/alert/alert.component.d.ts +20 -0
  35. package/lib/badge/badge.component.d.ts +8 -0
  36. package/lib/breadcrumb/breadcrumb.component.d.ts +15 -0
  37. package/lib/checkbox/checkbox.component.d.ts +42 -0
  38. package/lib/datepicker/datepicker.component.d.ts +48 -0
  39. package/lib/dialog/dialog.component.d.ts +10 -0
  40. package/lib/directives/truncate.directive.d.ts +23 -0
  41. package/lib/dropdown-list/dropdown-list.component.d.ts +33 -0
  42. package/lib/file-upload-items/file-upload-items.component.d.ts +27 -0
  43. package/lib/file-upload-multiple/file-upload-multiple.component.d.ts +44 -0
  44. package/lib/file-upload-multiple/upload-item.model.d.ts +7 -0
  45. package/lib/file-upload-multiple/upload-simulator.service.d.ts +34 -0
  46. package/lib/file-upload-single/file-upload-single.component.d.ts +28 -0
  47. package/lib/input-text/input-text.component.d.ts +24 -0
  48. package/lib/pagination/pagination.component.d.ts +31 -0
  49. package/lib/progress-bar/progress-bar.component.d.ts +11 -0
  50. package/lib/radio/radio.component.d.ts +14 -0
  51. package/lib/select-dropdown/select-dropdown.component.d.ts +78 -0
  52. package/lib/service/toast.service.d.ts +16 -0
  53. package/lib/side-navigation-bar/side-navigation-bar.component.d.ts +74 -0
  54. package/lib/spinner/spinner.component.d.ts +23 -0
  55. package/lib/table/table.component.d.ts +43 -0
  56. package/lib/tabs/tab.component.d.ts +9 -0
  57. package/lib/tabs/tabs.component.d.ts +15 -0
  58. package/lib/tag/tag.component.d.ts +10 -0
  59. package/lib/text-area/text-area.component.d.ts +21 -0
  60. package/lib/toast/toast.component.d.ts +13 -0
  61. package/lib/tooltip/tooltip.component.d.ts +15 -0
  62. package/lib/tooltip/tooltip.directive.d.ts +19 -0
  63. package/lib/top-navigation-bar/top-navigation-bar.component.d.ts +16 -0
  64. package/package.json +13 -7
  65. package/public-api.d.ts +25 -0
@@ -0,0 +1,89 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, Input, Output, EventEmitter, HostListener, ViewChild } from '@angular/core';
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "@angular/common";
5
+ export class DropdownListComponent {
6
+ // -------------------------------
7
+ // ELEMENT REFERENCE
8
+ // -------------------------------
9
+ /** Reference to the root <div> to detect clicks inside/outside dropdown */
10
+ root;
11
+ // -------------------------------
12
+ // INPUT PROPERTIES
13
+ // -------------------------------
14
+ /** Options to display in the dropdown. Each option has a label and a value */
15
+ options = [];
16
+ /** Placeholder text when no option is selected */
17
+ placeholder = 'Select';
18
+ /** Currently selected value */
19
+ value = null;
20
+ /** Disable dropdown interaction */
21
+ disabled = false;
22
+ // -------------------------------
23
+ // OUTPUT EVENTS
24
+ // -------------------------------
25
+ /** Emits the selected value when user clicks an option */
26
+ selectionChange = new EventEmitter();
27
+ // -------------------------------
28
+ // INTERNAL STATE
29
+ // -------------------------------
30
+ /** Tracks whether the dropdown list is open or closed */
31
+ isOpen = false;
32
+ // -------------------------------
33
+ // METHODS
34
+ // -------------------------------
35
+ /** Toggle dropdown open/close when trigger is clicked */
36
+ toggle() {
37
+ if (!this.disabled) {
38
+ this.isOpen = !this.isOpen;
39
+ }
40
+ }
41
+ /** Handle option selection */
42
+ select(option) {
43
+ this.value = option.value; // Update internal state
44
+ this.selectionChange.emit(this.value); // Notify parent component
45
+ this.isOpen = false; // Close dropdown
46
+ }
47
+ // -------------------------------
48
+ // HANDLE CLICKS OUTSIDE DROPDOWN
49
+ // -------------------------------
50
+ handleOutsideClick(event) {
51
+ if (!this.isOpen)
52
+ return;
53
+ const target = event.target;
54
+ if (!this.root.nativeElement.contains(target)) {
55
+ this.isOpen = false;
56
+ }
57
+ }
58
+ // -------------------------------
59
+ // HELPER GETTER
60
+ // -------------------------------
61
+ /** Returns label of selected value or placeholder if none selected */
62
+ get selectedLabel() {
63
+ const found = this.options.find(o => o.value === this.value);
64
+ return found ? found.label : this.placeholder;
65
+ }
66
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DropdownListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
67
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: DropdownListComponent, isStandalone: true, selector: "dso-dropdown-list", inputs: { options: "options", placeholder: "placeholder", value: "value", disabled: "disabled" }, outputs: { selectionChange: "selectionChange" }, host: { listeners: { "document:click": "handleOutsideClick($event)" } }, viewQueries: [{ propertyName: "root", first: true, predicate: ["root"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"dropdown\" #root [class.disabled]=\"disabled\">\r\n <!--\r\n #root: Linked to ViewChild in TS for detecting outside clicks\r\n [class.disabled]: Adds CSS class when dropdown is disabled\r\n -->\r\n\r\n <!-- Trigger area: clicking toggles dropdown open/close -->\r\n <div class=\"dropdown-trigger\" (click)=\"toggle()\">\r\n <!-- Display selected label or placeholder -->\r\n <span>{{ selectedLabel }}</span>\r\n <!-- Arrow icon that rotates when dropdown is open -->\r\n <span class=\"icon\" [class.open]=\"isOpen\">\u25BE</span>\r\n </div>\r\n\r\n <!-- Dropdown menu: visible only when isOpen is true -->\r\n <ul class=\"dropdown-menu\" *ngIf=\"isOpen\">\r\n <!-- Render each option -->\r\n <li \r\n *ngFor=\"let option of options\"\r\n (click)=\"select(option)\"\r\n [class.selected]=\"option.value === value\"> <!-- Highlight selected option -->\r\n {{ option.label }}\r\n </li>\r\n </ul>\r\n</div>\r\n", styles: [".dropdown{position:relative;display:inline-flex;font-family:inherit}.dropdown.disabled{opacity:.5;pointer-events:none}.dropdown-trigger{border:1px solid #ccc;padding:8px 12px;background:#fff;cursor:pointer;display:inline-flex;gap:8px;align-items:center;border-radius:4px}.dropdown-trigger:hover{border-color:#888}.dropdown-menu{position:absolute;top:calc(100% + 4px);left:0;right:0;background:#fff;border:1px solid #ccc;border-radius:4px;list-style:none;padding:4px 0;margin:0;max-height:200px;overflow-y:auto;z-index:100}.dropdown-menu li{padding:8px 12px;cursor:pointer}.dropdown-menu li:hover{background:#f2f2f2}.dropdown-menu li.selected{background:#e6e6e6;font-weight:500}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
68
+ }
69
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DropdownListComponent, decorators: [{
70
+ type: Component,
71
+ args: [{ selector: 'dso-dropdown-list', standalone: true, imports: [CommonModule], template: "<div class=\"dropdown\" #root [class.disabled]=\"disabled\">\r\n <!--\r\n #root: Linked to ViewChild in TS for detecting outside clicks\r\n [class.disabled]: Adds CSS class when dropdown is disabled\r\n -->\r\n\r\n <!-- Trigger area: clicking toggles dropdown open/close -->\r\n <div class=\"dropdown-trigger\" (click)=\"toggle()\">\r\n <!-- Display selected label or placeholder -->\r\n <span>{{ selectedLabel }}</span>\r\n <!-- Arrow icon that rotates when dropdown is open -->\r\n <span class=\"icon\" [class.open]=\"isOpen\">\u25BE</span>\r\n </div>\r\n\r\n <!-- Dropdown menu: visible only when isOpen is true -->\r\n <ul class=\"dropdown-menu\" *ngIf=\"isOpen\">\r\n <!-- Render each option -->\r\n <li \r\n *ngFor=\"let option of options\"\r\n (click)=\"select(option)\"\r\n [class.selected]=\"option.value === value\"> <!-- Highlight selected option -->\r\n {{ option.label }}\r\n </li>\r\n </ul>\r\n</div>\r\n", styles: [".dropdown{position:relative;display:inline-flex;font-family:inherit}.dropdown.disabled{opacity:.5;pointer-events:none}.dropdown-trigger{border:1px solid #ccc;padding:8px 12px;background:#fff;cursor:pointer;display:inline-flex;gap:8px;align-items:center;border-radius:4px}.dropdown-trigger:hover{border-color:#888}.dropdown-menu{position:absolute;top:calc(100% + 4px);left:0;right:0;background:#fff;border:1px solid #ccc;border-radius:4px;list-style:none;padding:4px 0;margin:0;max-height:200px;overflow-y:auto;z-index:100}.dropdown-menu li{padding:8px 12px;cursor:pointer}.dropdown-menu li:hover{background:#f2f2f2}.dropdown-menu li.selected{background:#e6e6e6;font-weight:500}\n"] }]
72
+ }], propDecorators: { root: [{
73
+ type: ViewChild,
74
+ args: ['root', { static: true }]
75
+ }], options: [{
76
+ type: Input
77
+ }], placeholder: [{
78
+ type: Input
79
+ }], value: [{
80
+ type: Input
81
+ }], disabled: [{
82
+ type: Input
83
+ }], selectionChange: [{
84
+ type: Output
85
+ }], handleOutsideClick: [{
86
+ type: HostListener,
87
+ args: ['document:click', ['$event']]
88
+ }] } });
89
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-list.component.js","sourceRoot":"","sources":["../../../../../projects/ui/src/lib/dropdown-list/dropdown-list.component.ts","../../../../../projects/ui/src/lib/dropdown-list/dropdown-list.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,YAAY,EAAE,YAAY,EAAc,SAAS,EAAE,MAAM,eAAe,CAAC;;;AAS5G,MAAM,OAAO,qBAAqB;IAEhC,kCAAkC;IAClC,oBAAoB;IACpB,kCAAkC;IAClC,2EAA2E;IACtC,IAAI,CAAc;IAEvD,kCAAkC;IAClC,mBAAmB;IACnB,kCAAkC;IAClC,8EAA8E;IACrE,OAAO,GAAoC,EAAE,CAAC;IAEvD,kDAAkD;IACzC,WAAW,GAAG,QAAQ,CAAC;IAEhC,+BAA+B;IACtB,KAAK,GAAQ,IAAI,CAAC;IAE3B,mCAAmC;IAC1B,QAAQ,GAAG,KAAK,CAAC;IAE1B,kCAAkC;IAClC,gBAAgB;IAChB,kCAAkC;IAClC,0DAA0D;IAChD,eAAe,GAAG,IAAI,YAAY,EAAO,CAAC;IAEpD,kCAAkC;IAClC,iBAAiB;IACjB,kCAAkC;IAClC,yDAAyD;IACzD,MAAM,GAAG,KAAK,CAAC;IAEf,kCAAkC;IAClC,UAAU;IACV,kCAAkC;IAElC,yDAAyD;IACzD,MAAM;QACJ,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACnB,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,8BAA8B;IAC9B,MAAM,CAAC,MAAqC;QAC1C,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAe,wBAAwB;QACjE,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAG,0BAA0B;QACnE,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC,CAAqB,iBAAiB;IAC5D,CAAC;IAED,kCAAkC;IAClC,iCAAiC;IACjC,kCAAkC;IAElC,kBAAkB,CAAC,KAAY;QAC7B,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,OAAO;QAEzB,MAAM,MAAM,GAAG,KAAK,CAAC,MAAqB,CAAC;QAC3C,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;YAC9C,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QACtB,CAAC;IACH,CAAC;IAED,kCAAkC;IAClC,gBAAgB;IAChB,kCAAkC;IAClC,sEAAsE;IACtE,IAAI,aAAa;QACf,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7D,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC;IAChD,CAAC;wGAzEU,qBAAqB;4FAArB,qBAAqB,qZCVlC,+8BAyBA,guBDjBY,YAAY;;4FAEX,qBAAqB;kBAPjC,SAAS;+BACE,mBAAmB,cAGjB,IAAI,WACP,CAAC,YAAY,CAAC;8BAQc,IAAI;sBAAxC,SAAS;uBAAC,MAAM,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBAM1B,OAAO;sBAAf,KAAK;gBAGG,WAAW;sBAAnB,KAAK;gBAGG,KAAK;sBAAb,KAAK;gBAGG,QAAQ;sBAAhB,KAAK;gBAMI,eAAe;sBAAxB,MAAM;gBA8BP,kBAAkB;sBADjB,YAAY;uBAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, Input, Output, EventEmitter, HostListener, ElementRef, ViewChild } from '@angular/core';\r\n\r\n@Component({\r\n  selector: 'dso-dropdown-list',\r\n  templateUrl: './dropdown-list.component.html',\r\n  styleUrls: ['./dropdown-list.component.scss'],\r\n  standalone: true,\r\n  imports: [CommonModule]\r\n})\r\nexport class DropdownListComponent {\r\n\r\n  // -------------------------------\r\n  // ELEMENT REFERENCE\r\n  // -------------------------------\r\n  /** Reference to the root <div> to detect clicks inside/outside dropdown */\r\n  @ViewChild('root', { static: true }) root!: ElementRef;\r\n\r\n  // -------------------------------\r\n  // INPUT PROPERTIES\r\n  // -------------------------------\r\n  /** Options to display in the dropdown. Each option has a label and a value */\r\n  @Input() options: { label: string; value: any }[] = [];\r\n\r\n  /** Placeholder text when no option is selected */\r\n  @Input() placeholder = 'Select';\r\n\r\n  /** Currently selected value */\r\n  @Input() value: any = null;\r\n\r\n  /** Disable dropdown interaction */\r\n  @Input() disabled = false;\r\n\r\n  // -------------------------------\r\n  // OUTPUT EVENTS\r\n  // -------------------------------\r\n  /** Emits the selected value when user clicks an option */\r\n  @Output() selectionChange = new EventEmitter<any>();\r\n\r\n  // -------------------------------\r\n  // INTERNAL STATE\r\n  // -------------------------------\r\n  /** Tracks whether the dropdown list is open or closed */\r\n  isOpen = false;\r\n\r\n  // -------------------------------\r\n  // METHODS\r\n  // -------------------------------\r\n\r\n  /** Toggle dropdown open/close when trigger is clicked */\r\n  toggle(): void {\r\n    if (!this.disabled) {\r\n      this.isOpen = !this.isOpen;\r\n    }\r\n  }\r\n\r\n  /** Handle option selection */\r\n  select(option: { label: string; value: any }): void {\r\n    this.value = option.value;               // Update internal state\r\n    this.selectionChange.emit(this.value);   // Notify parent component\r\n    this.isOpen = false;                     // Close dropdown\r\n  }\r\n\r\n  // -------------------------------\r\n  // HANDLE CLICKS OUTSIDE DROPDOWN\r\n  // -------------------------------\r\n  @HostListener('document:click', ['$event'])\r\n  handleOutsideClick(event: Event): void {\r\n    if (!this.isOpen) return;\r\n\r\n    const target = event.target as HTMLElement;\r\n    if (!this.root.nativeElement.contains(target)) {\r\n      this.isOpen = false;\r\n    }\r\n  }\r\n\r\n  // -------------------------------\r\n  // HELPER GETTER\r\n  // -------------------------------\r\n  /** Returns label of selected value or placeholder if none selected */\r\n  get selectedLabel(): string {\r\n    const found = this.options.find(o => o.value === this.value);\r\n    return found ? found.label : this.placeholder;\r\n  }\r\n}\r\n","<div class=\"dropdown\" #root [class.disabled]=\"disabled\">\r\n  <!--\r\n    #root: Linked to ViewChild in TS for detecting outside clicks\r\n    [class.disabled]: Adds CSS class when dropdown is disabled\r\n  -->\r\n\r\n  <!-- Trigger area: clicking toggles dropdown open/close -->\r\n  <div class=\"dropdown-trigger\" (click)=\"toggle()\">\r\n    <!-- Display selected label or placeholder -->\r\n    <span>{{ selectedLabel }}</span>\r\n    <!-- Arrow icon that rotates when dropdown is open -->\r\n    <span class=\"icon\" [class.open]=\"isOpen\">▾</span>\r\n  </div>\r\n\r\n  <!-- Dropdown menu: visible only when isOpen is true -->\r\n  <ul class=\"dropdown-menu\" *ngIf=\"isOpen\">\r\n    <!-- Render each option -->\r\n    <li \r\n      *ngFor=\"let option of options\"\r\n      (click)=\"select(option)\"\r\n      [class.selected]=\"option.value === value\"> <!-- Highlight selected option -->\r\n      {{ option.label }}\r\n    </li>\r\n  </ul>\r\n</div>\r\n"]}
@@ -0,0 +1,65 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ButtonComponent } from '../button/button.component';
4
+ import { IconComponent } from '../icon/icon.component';
5
+ import { ProgressBarComponent } from '../progress-bar/progress-bar.component';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/common";
8
+ export class FileUploadItemComponent {
9
+ /** The file upload item to display */
10
+ item;
11
+ /** Emitted when the user retries a failed upload */
12
+ retry = new EventEmitter();
13
+ /** Emitted when the user removes the file from the list */
14
+ remove = new EventEmitter();
15
+ /** Emitted when the user cancels an ongoing upload */
16
+ cancel = new EventEmitter();
17
+ /**
18
+ * Format the file size into a human-readable string
19
+ * @param size - size in bytes
20
+ * @returns formatted string like '12.3 KB', '1.5 MB'
21
+ */
22
+ formatSize(size) {
23
+ if (size < 1024)
24
+ return size + ' B';
25
+ if (size < 1024 * 1024)
26
+ return (size / 1024).toFixed(1) + ' KB';
27
+ return (size / (1024 * 1024)).toFixed(1) + ' MB';
28
+ }
29
+ /**
30
+ * Get an icon name for the file based on its type or extension
31
+ * @param item - the UploadItem
32
+ * @returns icon name string
33
+ */
34
+ getFileIcon(item) {
35
+ const file = item.file;
36
+ const type = file.type.toLowerCase();
37
+ const name = file.name.toLowerCase();
38
+ // Image MIME types
39
+ if (type.startsWith('image/')) {
40
+ return 'icon-gallery';
41
+ }
42
+ // Image file extensions fallback
43
+ const imageExtensions = ['.png', '.jpg', '.jpeg', '.gif', '.webp', '.bmp', '.svg'];
44
+ if (imageExtensions.some(ext => name.endsWith(ext))) {
45
+ return 'icon-gallery';
46
+ }
47
+ // Default icon for non-image files
48
+ return 'icon-van';
49
+ }
50
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FileUploadItemComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
51
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: FileUploadItemComponent, isStandalone: true, selector: "dso-file-item", inputs: { item: "item" }, outputs: { retry: "retry", remove: "remove", cancel: "cancel" }, ngImport: i0, template: "<div class=\"file-row\">\r\n\r\n <!-- LEFT SECTION: Thumbnail + filename -->\r\n <div class=\"thumbnail-filename-wrapper\">\r\n <div class=\"file-thumb\">\r\n <dso-icon \r\n [iconName]=\"getFileIcon(item)\"\r\n size=\"small\"\r\n class=\"thumb-icon\">\r\n </dso-icon>\r\n </div>\r\n\r\n <div class=\"file-info\">\r\n <div class=\"file-name\">{{ item.file.name }}</div>\r\n <div class=\"file-size\">{{ formatSize(item.file.size) }}</div>\r\n </div>\r\n </div>\r\n\r\n\r\n <!-- MIDDLE SECTION: Progress or status -->\r\n <div class=\"filecontent-loader-wrapper\">\r\n \r\n <!-- Progress bar -->\r\n <div class=\"file-progress\" *ngIf=\"!item.hideProgressBar\">\r\n <dso-progress-bar\r\n [value]=\"\r\n item.status === 'uploading' ? item.progress :\r\n item.status === 'completed' ? 100 :\r\n null\r\n \"\r\n [color]=\"item.status === 'completed' ? 'success' : 'primary'\"\r\n ></dso-progress-bar>\r\n </div>\r\n\r\n <!-- Status text -->\r\n <div class=\"file-status\" *ngIf=\"item.hideProgressBar\">\r\n\r\n <!-- Success -->\r\n <div *ngIf=\"item.status === 'completed'\" class=\"status success\">\r\n <dso-icon iconName=\"icon-van\" size=\"small\"></dso-icon>\r\n <span>Uploaded</span>\r\n </div>\r\n\r\n <!-- Failed -->\r\n <div *ngIf=\"item.status === 'failed'\" class=\"status error\">\r\n <dso-icon iconName=\"icon-center-align\" size=\"small\"></dso-icon>\r\n <span>Upload failed</span>\r\n </div>\r\n\r\n </div>\r\n\r\n <!-- RIGHT SECTION: Action buttons -->\r\n <div class=\"file-actions\">\r\n\r\n <!-- Uploading -->\r\n <ng-container *ngIf=\"item.status === 'uploading'\">\r\n <dso-button\r\n btnLabel=\"Cancel\"\r\n btnType=\"text\"\r\n (onClick)=\"cancel.emit()\">\r\n </dso-button>\r\n </ng-container>\r\n\r\n <!-- Failed -->\r\n <ng-container *ngIf=\"item.status === 'failed'\">\r\n <dso-button\r\n btnLabel=\"Retry\"\r\n btnType=\"text\"\r\n (onClick)=\"retry.emit()\">\r\n </dso-button>\r\n\r\n <dso-button\r\n btnLabel=\"Remove\"\r\n btnType=\"text\"\r\n (onClick)=\"remove.emit()\">\r\n </dso-button>\r\n </ng-container>\r\n\r\n <!-- Completed -->\r\n <ng-container *ngIf=\"item.status === 'completed'\">\r\n <dso-button\r\n btnLabel=\"Remove\"\r\n btnType=\"text\"\r\n (onClick)=\"remove.emit()\">\r\n </dso-button>\r\n </ng-container>\r\n\r\n </div>\r\n\r\n </div>\r\n\r\n</div>\r\n", styles: [".file-row{display:flex;justify-content:space-between;align-items:center;padding:.75rem 1rem;background:#fafafa;border-radius:4px}.file-info{display:flex;flex-direction:column}.file-info .file-name{font-weight:300}.file-info .file-size{font-size:.85rem;color:#666}.file-progress{padding-right:1rem;flex:1 0 0;display:flex;justify-content:center;align-items:center;min-width:260px;max-width:320px}.file-actions{display:flex;width:160px;gap:.5rem;justify-content:flex-end}.file-status{display:flex;align-items:center;font-size:14px;flex:1 0 0;min-width:260px;max-width:320px}.file-status .status{display:flex;align-items:center;gap:6px;font-weight:500}.file-status .status.success{color:#16a34a}.file-status .status.error{color:#dc2626}.file-status .status .status-icon{width:16px;height:16px}.file-thumb{width:40px;height:40px;flex-shrink:0;border-radius:6px;overflow:hidden;background:#f3f4f6;display:flex;align-items:center;justify-content:center}.file-thumb img{width:100%;height:100%;object-fit:cover}.thumbnail-filename-wrapper{display:flex;gap:24px}.filecontent-loader-wrapper{display:flex;gap:20px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.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: IconComponent, selector: "dso-icon", inputs: ["iconName", "size"] }, { kind: "component", type: ProgressBarComponent, selector: "dso-progress-bar", inputs: ["value", "color", "animated"] }] });
52
+ }
53
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FileUploadItemComponent, decorators: [{
54
+ type: Component,
55
+ args: [{ selector: 'dso-file-item', standalone: true, imports: [CommonModule, ButtonComponent, IconComponent, ProgressBarComponent], template: "<div class=\"file-row\">\r\n\r\n <!-- LEFT SECTION: Thumbnail + filename -->\r\n <div class=\"thumbnail-filename-wrapper\">\r\n <div class=\"file-thumb\">\r\n <dso-icon \r\n [iconName]=\"getFileIcon(item)\"\r\n size=\"small\"\r\n class=\"thumb-icon\">\r\n </dso-icon>\r\n </div>\r\n\r\n <div class=\"file-info\">\r\n <div class=\"file-name\">{{ item.file.name }}</div>\r\n <div class=\"file-size\">{{ formatSize(item.file.size) }}</div>\r\n </div>\r\n </div>\r\n\r\n\r\n <!-- MIDDLE SECTION: Progress or status -->\r\n <div class=\"filecontent-loader-wrapper\">\r\n \r\n <!-- Progress bar -->\r\n <div class=\"file-progress\" *ngIf=\"!item.hideProgressBar\">\r\n <dso-progress-bar\r\n [value]=\"\r\n item.status === 'uploading' ? item.progress :\r\n item.status === 'completed' ? 100 :\r\n null\r\n \"\r\n [color]=\"item.status === 'completed' ? 'success' : 'primary'\"\r\n ></dso-progress-bar>\r\n </div>\r\n\r\n <!-- Status text -->\r\n <div class=\"file-status\" *ngIf=\"item.hideProgressBar\">\r\n\r\n <!-- Success -->\r\n <div *ngIf=\"item.status === 'completed'\" class=\"status success\">\r\n <dso-icon iconName=\"icon-van\" size=\"small\"></dso-icon>\r\n <span>Uploaded</span>\r\n </div>\r\n\r\n <!-- Failed -->\r\n <div *ngIf=\"item.status === 'failed'\" class=\"status error\">\r\n <dso-icon iconName=\"icon-center-align\" size=\"small\"></dso-icon>\r\n <span>Upload failed</span>\r\n </div>\r\n\r\n </div>\r\n\r\n <!-- RIGHT SECTION: Action buttons -->\r\n <div class=\"file-actions\">\r\n\r\n <!-- Uploading -->\r\n <ng-container *ngIf=\"item.status === 'uploading'\">\r\n <dso-button\r\n btnLabel=\"Cancel\"\r\n btnType=\"text\"\r\n (onClick)=\"cancel.emit()\">\r\n </dso-button>\r\n </ng-container>\r\n\r\n <!-- Failed -->\r\n <ng-container *ngIf=\"item.status === 'failed'\">\r\n <dso-button\r\n btnLabel=\"Retry\"\r\n btnType=\"text\"\r\n (onClick)=\"retry.emit()\">\r\n </dso-button>\r\n\r\n <dso-button\r\n btnLabel=\"Remove\"\r\n btnType=\"text\"\r\n (onClick)=\"remove.emit()\">\r\n </dso-button>\r\n </ng-container>\r\n\r\n <!-- Completed -->\r\n <ng-container *ngIf=\"item.status === 'completed'\">\r\n <dso-button\r\n btnLabel=\"Remove\"\r\n btnType=\"text\"\r\n (onClick)=\"remove.emit()\">\r\n </dso-button>\r\n </ng-container>\r\n\r\n </div>\r\n\r\n </div>\r\n\r\n</div>\r\n", styles: [".file-row{display:flex;justify-content:space-between;align-items:center;padding:.75rem 1rem;background:#fafafa;border-radius:4px}.file-info{display:flex;flex-direction:column}.file-info .file-name{font-weight:300}.file-info .file-size{font-size:.85rem;color:#666}.file-progress{padding-right:1rem;flex:1 0 0;display:flex;justify-content:center;align-items:center;min-width:260px;max-width:320px}.file-actions{display:flex;width:160px;gap:.5rem;justify-content:flex-end}.file-status{display:flex;align-items:center;font-size:14px;flex:1 0 0;min-width:260px;max-width:320px}.file-status .status{display:flex;align-items:center;gap:6px;font-weight:500}.file-status .status.success{color:#16a34a}.file-status .status.error{color:#dc2626}.file-status .status .status-icon{width:16px;height:16px}.file-thumb{width:40px;height:40px;flex-shrink:0;border-radius:6px;overflow:hidden;background:#f3f4f6;display:flex;align-items:center;justify-content:center}.file-thumb img{width:100%;height:100%;object-fit:cover}.thumbnail-filename-wrapper{display:flex;gap:24px}.filecontent-loader-wrapper{display:flex;gap:20px}\n"] }]
56
+ }], propDecorators: { item: [{
57
+ type: Input
58
+ }], retry: [{
59
+ type: Output
60
+ }], remove: [{
61
+ type: Output
62
+ }], cancel: [{
63
+ type: Output
64
+ }] } });
65
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload-items.component.js","sourceRoot":"","sources":["../../../../../projects/ui/src/lib/file-upload-items/file-upload-items.component.ts","../../../../../projects/ui/src/lib/file-upload-items/file-upload-items.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACvE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAE/C,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC7D,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,oBAAoB,EAAE,MAAM,wCAAwC,CAAC;;;AAS9E,MAAM,OAAO,uBAAuB;IAElC,sCAAsC;IAC7B,IAAI,CAAc;IAE3B,oDAAoD;IAC1C,KAAK,GAAG,IAAI,YAAY,EAAQ,CAAC;IAE3C,2DAA2D;IACjD,MAAM,GAAG,IAAI,YAAY,EAAQ,CAAC;IAE5C,sDAAsD;IAC5C,MAAM,GAAG,IAAI,YAAY,EAAQ,CAAC;IAE5C;;;;OAIG;IACH,UAAU,CAAC,IAAY;QACrB,IAAI,IAAI,GAAG,IAAI;YAAE,OAAO,IAAI,GAAG,IAAI,CAAC;QACpC,IAAI,IAAI,GAAG,IAAI,GAAG,IAAI;YAAE,OAAO,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;QAChE,OAAO,CAAC,IAAI,GAAG,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;IACnD,CAAC;IAED;;;;OAIG;IACH,WAAW,CAAC,IAAgB;QAC1B,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QACvB,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;QACrC,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;QAErC,mBAAmB;QACnB,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC9B,OAAO,cAAc,CAAC;QACxB,CAAC;QAED,iCAAiC;QACjC,MAAM,eAAe,GAAG,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC;QACnF,IAAI,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC;YACpD,OAAO,cAAc,CAAC;QACxB,CAAC;QAED,mCAAmC;QACnC,OAAO,UAAU,CAAC;IACpB,CAAC;wGAhDU,uBAAuB;4FAAvB,uBAAuB,oKCdpC,4lFA4FA,0oCDlFY,YAAY,mIAAE,eAAe,kKAAE,aAAa,mFAAE,oBAAoB;;4FAIjE,uBAAuB;kBAPnC,SAAS;+BACE,eAAe,cACb,IAAI,WACP,CAAC,YAAY,EAAE,eAAe,EAAE,aAAa,EAAE,oBAAoB,CAAC;8BAOpE,IAAI;sBAAZ,KAAK;gBAGI,KAAK;sBAAd,MAAM;gBAGG,MAAM;sBAAf,MAAM;gBAGG,MAAM;sBAAf,MAAM","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { UploadItem } from '../file-upload-multiple/upload-item.model';\r\nimport { ButtonComponent } from '../button/button.component';\r\nimport { IconComponent } from '../icon/icon.component';\r\nimport { ProgressBarComponent } from '../progress-bar/progress-bar.component';\r\n\r\n@Component({\r\n  selector: 'dso-file-item',\r\n  standalone: true,\r\n  imports: [CommonModule, ButtonComponent, IconComponent, ProgressBarComponent],\r\n  templateUrl: './file-upload-items.component.html',\r\n  styleUrls: ['./file-upload-items.component.scss']\r\n})\r\nexport class FileUploadItemComponent {\r\n\r\n  /** The file upload item to display */\r\n  @Input() item!: UploadItem;\r\n\r\n  /** Emitted when the user retries a failed upload */\r\n  @Output() retry = new EventEmitter<void>();\r\n  \r\n  /** Emitted when the user removes the file from the list */\r\n  @Output() remove = new EventEmitter<void>();\r\n  \r\n  /** Emitted when the user cancels an ongoing upload */\r\n  @Output() cancel = new EventEmitter<void>();\r\n\r\n  /**\r\n   * Format the file size into a human-readable string\r\n   * @param size - size in bytes\r\n   * @returns formatted string like '12.3 KB', '1.5 MB'\r\n   */\r\n  formatSize(size: number): string {\r\n    if (size < 1024) return size + ' B';\r\n    if (size < 1024 * 1024) return (size / 1024).toFixed(1) + ' KB';\r\n    return (size / (1024 * 1024)).toFixed(1) + ' MB';\r\n  }\r\n\r\n  /**\r\n   * Get an icon name for the file based on its type or extension\r\n   * @param item - the UploadItem\r\n   * @returns icon name string\r\n   */\r\n  getFileIcon(item: UploadItem): string {\r\n    const file = item.file;\r\n    const type = file.type.toLowerCase();\r\n    const name = file.name.toLowerCase();\r\n\r\n    // Image MIME types\r\n    if (type.startsWith('image/')) {\r\n      return 'icon-gallery';\r\n    }\r\n\r\n    // Image file extensions fallback\r\n    const imageExtensions = ['.png', '.jpg', '.jpeg', '.gif', '.webp', '.bmp', '.svg'];\r\n    if (imageExtensions.some(ext => name.endsWith(ext))) {\r\n      return 'icon-gallery';\r\n    }\r\n\r\n    // Default icon for non-image files\r\n    return 'icon-van';\r\n  }\r\n}\r\n","<div class=\"file-row\">\r\n\r\n  <!-- LEFT SECTION: Thumbnail + filename -->\r\n  <div class=\"thumbnail-filename-wrapper\">\r\n    <div class=\"file-thumb\">\r\n      <dso-icon \r\n        [iconName]=\"getFileIcon(item)\"\r\n        size=\"small\"\r\n        class=\"thumb-icon\">\r\n      </dso-icon>\r\n    </div>\r\n\r\n    <div class=\"file-info\">\r\n      <div class=\"file-name\">{{ item.file.name }}</div>\r\n      <div class=\"file-size\">{{ formatSize(item.file.size) }}</div>\r\n    </div>\r\n  </div>\r\n\r\n\r\n  <!-- MIDDLE SECTION: Progress or status -->\r\n  <div class=\"filecontent-loader-wrapper\">\r\n    \r\n    <!-- Progress bar -->\r\n    <div class=\"file-progress\" *ngIf=\"!item.hideProgressBar\">\r\n      <dso-progress-bar\r\n        [value]=\"\r\n          item.status === 'uploading' ? item.progress :\r\n          item.status === 'completed' ? 100 :\r\n          null\r\n        \"\r\n        [color]=\"item.status === 'completed' ? 'success' : 'primary'\"\r\n      ></dso-progress-bar>\r\n    </div>\r\n\r\n    <!-- Status text -->\r\n    <div class=\"file-status\" *ngIf=\"item.hideProgressBar\">\r\n\r\n      <!-- Success -->\r\n      <div *ngIf=\"item.status === 'completed'\" class=\"status success\">\r\n        <dso-icon iconName=\"icon-van\" size=\"small\"></dso-icon>\r\n        <span>Uploaded</span>\r\n      </div>\r\n\r\n      <!-- Failed -->\r\n      <div *ngIf=\"item.status === 'failed'\" class=\"status error\">\r\n        <dso-icon iconName=\"icon-center-align\" size=\"small\"></dso-icon>\r\n        <span>Upload failed</span>\r\n      </div>\r\n\r\n    </div>\r\n\r\n  <!-- RIGHT SECTION: Action buttons -->\r\n  <div class=\"file-actions\">\r\n\r\n    <!-- Uploading -->\r\n    <ng-container *ngIf=\"item.status === 'uploading'\">\r\n      <dso-button\r\n        btnLabel=\"Cancel\"\r\n        btnType=\"text\"\r\n        (onClick)=\"cancel.emit()\">\r\n      </dso-button>\r\n    </ng-container>\r\n\r\n    <!-- Failed -->\r\n    <ng-container *ngIf=\"item.status === 'failed'\">\r\n      <dso-button\r\n        btnLabel=\"Retry\"\r\n        btnType=\"text\"\r\n        (onClick)=\"retry.emit()\">\r\n      </dso-button>\r\n\r\n      <dso-button\r\n        btnLabel=\"Remove\"\r\n        btnType=\"text\"\r\n        (onClick)=\"remove.emit()\">\r\n      </dso-button>\r\n    </ng-container>\r\n\r\n    <!-- Completed -->\r\n    <ng-container *ngIf=\"item.status === 'completed'\">\r\n      <dso-button\r\n        btnLabel=\"Remove\"\r\n        btnType=\"text\"\r\n        (onClick)=\"remove.emit()\">\r\n      </dso-button>\r\n    </ng-container>\r\n\r\n  </div>\r\n\r\n  </div>\r\n\r\n</div>\r\n"]}
@@ -0,0 +1,232 @@
1
+ // file-upload-multi.component.ts
2
+ import { Component } from '@angular/core';
3
+ import { CommonModule } from '@angular/common';
4
+ import { ButtonComponent } from '../button/button.component';
5
+ import { IconComponent } from '../icon/icon.component';
6
+ import { FileUploadItemComponent } from '../file-upload-items/file-upload-items.component';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "./upload-simulator.service";
9
+ import * as i2 from "@angular/common";
10
+ /**
11
+ * FileUploadMultiComponent
12
+ * -------------------------
13
+ * This component handles:
14
+ * - Drag & drop file uploads
15
+ * - Selecting multiple files through <input type="file">
16
+ * - Upload progress per file
17
+ * - Cancelling an upload
18
+ * - Retrying failed uploads
19
+ * - Removing items from the queue
20
+ *
21
+ * UploadSimulatorService is used to mimic real upload behavior.
22
+ * Later, you can replace it with a real HTTP upload service.
23
+ */
24
+ export class FileUploadMultiComponent {
25
+ sim;
26
+ // ngOnInit() {
27
+ // // Fake completed file
28
+ // this.uploadQueue.push({
29
+ // file: new File([""], "example-image.jpg", { type: "image/jpeg" }),
30
+ // progress: 100,
31
+ // status: "completed",
32
+ // hideProgressBar: true
33
+ // });
34
+ // // Fake failed file
35
+ // this.uploadQueue.push({
36
+ // file: new File([""], "broken-file.pdf", { type: "application/pdf" }),
37
+ // progress: null,
38
+ // status: "failed",
39
+ // hideProgressBar: true
40
+ // });
41
+ // // Fake uploading file
42
+ // this.uploadQueue.push({
43
+ // file: new File([""], "uploading.png", { type: "image/png" }),
44
+ // progress: 40,
45
+ // status: "uploading",
46
+ // hideProgressBar: false
47
+ // });
48
+ // }
49
+ /** All files being managed (uploaded, uploading, failed, completed). */
50
+ uploadQueue = [];
51
+ /** True while user drags a file over the drop zone. */
52
+ isDragOver = false;
53
+ constructor(sim) {
54
+ this.sim = sim;
55
+ // console.log('%c[COMPONENT] FileUploadMultiComponent initialized', 'color: purple;');
56
+ }
57
+ /* ===========================================================
58
+ * DRAG & DROP EVENTS
59
+ * =========================================================== */
60
+ onDragOver(event) {
61
+ event.preventDefault();
62
+ this.isDragOver = true;
63
+ // console.log('%c[DRAG] Dragging over drop zone...', 'color: #00aaff;');
64
+ }
65
+ onDragLeave(event) {
66
+ this.isDragOver = false;
67
+ // console.log('%c[DRAG] Drag left drop zone', 'color: #888;');
68
+ }
69
+ onDrop(event) {
70
+ event.preventDefault();
71
+ this.isDragOver = false;
72
+ // console.log('%c[DRAG] Files dropped', 'color: #00cc66;');
73
+ if (event.dataTransfer?.files) {
74
+ // console.log('[DRAG] Dropped files:', event.dataTransfer.files);
75
+ this.handleFiles(event.dataTransfer.files);
76
+ }
77
+ }
78
+ /* ===========================================================
79
+ * FILE INPUT SELECT
80
+ * =========================================================== */
81
+ onFileSelected(event) {
82
+ const input = event.target;
83
+ if (input.files) {
84
+ // console.log('%c[INPUT] Files selected:', 'color: #0099ff;', input.files);
85
+ this.handleFiles(input.files);
86
+ // Clears <input> so user can re-select same file
87
+ input.value = '';
88
+ }
89
+ }
90
+ /* ===========================================================
91
+ * PROCESSING NEWLY ADDED FILES
92
+ * =========================================================== */
93
+ handleFiles(fileList) {
94
+ // console.log('%c[FILES] Processing new files...', 'color: #ffaa00;');
95
+ Array.from(fileList).forEach(file => {
96
+ // console.log(`%c[FILES] Added: ${file.name} (${file.size} bytes)`, 'color: #ffaa00;');
97
+ const item = {
98
+ file,
99
+ progress: 0,
100
+ status: 'queued'
101
+ };
102
+ this.uploadQueue.push(item);
103
+ this.startUpload(item);
104
+ });
105
+ }
106
+ /* ===========================================================
107
+ * UPLOAD LOGIC
108
+ * =========================================================== */
109
+ startUpload(item) {
110
+ // console.log(`%c[UPLOAD] Starting upload for ${item.file.name}`, 'color: #00cc99;');
111
+ item.status = 'uploading';
112
+ item.progress = 0;
113
+ this.sim.simulateUpload(item,
114
+ // ---- onProgress callback ----
115
+ (progress) => {
116
+ item.progress = progress;
117
+ // console.log(`%c[UPLOAD] ${item.file.name}: ${progress.toFixed(1)}%`, 'color: #0099ff;');
118
+ },
119
+ // ---- onComplete callback ----
120
+ () => {
121
+ item.status = 'completed';
122
+ item.progress = 100;
123
+ // Hide progress bar after a very small delay (100–300ms)
124
+ setTimeout(() => item.hideProgressBar = true, 300);
125
+ // console.log(`%c[UPLOAD] COMPLETED: ${item.file.name}`, 'color: green; font-weight: bold;');
126
+ },
127
+ // ---- onError callback ----
128
+ () => {
129
+ item.status = 'failed';
130
+ item.progress = null;
131
+ // Hide progress bar so status text can show
132
+ setTimeout(() => item.hideProgressBar = true, 300);
133
+ // console.log(`%c[UPLOAD] FAILED: ${item.file.name}`, 'color: red; font-weight: bold;');
134
+ });
135
+ }
136
+ onDropZoneClick(fileInput) {
137
+ // Prevent click if user is dragging files over the drop-zone
138
+ if (this.isDragOver)
139
+ return;
140
+ console.log('%c[CLICK] Drop zone clicked → opening file dialog', 'color: #0077ff;');
141
+ fileInput.click();
142
+ }
143
+ onBrowseButtonClick(event, fileInput) {
144
+ event.stopPropagation(); // stops click from bubbling to drop-zone
145
+ // console.log('%c[BUTTON] Browse clicked → opening file dialog', 'color: #8844ff;');
146
+ fileInput.click();
147
+ }
148
+ /* ===========================================================
149
+ * BUTTON ACTIONS (CANCEL / RETRY / REMOVE)
150
+ * =========================================================== */
151
+ cancelUpload(item) {
152
+ // console.log(`%c[CANCEL] Cancelling ${item.file.name}`, 'color: orange; font-weight: bold;');
153
+ // item.status = 'cancelled';
154
+ // item.progress = null;
155
+ this.removeItem(item);
156
+ }
157
+ retryUpload(item) {
158
+ // console.log('[RETRY] Restarting upload for', item.file.name);
159
+ // Reset state
160
+ item.status = 'queued';
161
+ item.progress = 0;
162
+ item.hideProgressBar = false; // <-- IMPORTANT
163
+ // Start upload again
164
+ this.startUpload(item);
165
+ }
166
+ removeItem(item) {
167
+ // console.log(`%c[REMOVE] Removing ${item.file.name}`, 'color: gray;');
168
+ this.uploadQueue = this.uploadQueue.filter(i => i !== item);
169
+ }
170
+ /* ===========================================================
171
+ * UTILITY FUNCTIONS
172
+ * =========================================================== */
173
+ /**
174
+ * Converts bytes → human readable sizes.
175
+ */
176
+ formatSize(size) {
177
+ if (size < 1024)
178
+ return size + ' B';
179
+ if (size < 1024 * 1024)
180
+ return (size / 1024).toFixed(1) + ' KB';
181
+ return (size / (1024 * 1024)).toFixed(1) + ' MB';
182
+ }
183
+ // getThumbnail(item: UploadItem): string | null {
184
+ // const file = item.file;
185
+ // // If actual image → return preview URL
186
+ // if (file.type.startsWith('image/')) {
187
+ // return URL.createObjectURL(file);
188
+ // }
189
+ // // Otherwise return an icon path depending on type
190
+ // if (file.type === 'application/pdf') return 'assets/icons/pdf.svg';
191
+ // if (file.type.includes('word')) return 'assets/icons/doc.svg';
192
+ // if (file.type.includes('spreadsheet')) return 'assets/icons/xlsx.svg';
193
+ // // default icon
194
+ // return 'assets/icons/file.svg';
195
+ // }
196
+ getThumbnail(item) {
197
+ const file = item.file;
198
+ // Image preview for image file types
199
+ if (file.type.startsWith('image/')) {
200
+ return URL.createObjectURL(file);
201
+ }
202
+ // Not an image → no thumbnail
203
+ return null;
204
+ }
205
+ getFileIcon(item) {
206
+ const file = item.file;
207
+ const type = file.type.toLowerCase();
208
+ const name = file.name.toLowerCase();
209
+ console.log('%c[DEBUG] File name:', 'color: blue;', file.name);
210
+ console.log('%c[DEBUG] File type:', 'color: green;', file.type);
211
+ // Check MIME type first
212
+ if (type.startsWith('image/')) {
213
+ console.log('%c[DEBUG] Detected as image by MIME type', 'color: purple;');
214
+ return 'icon-gallery';
215
+ }
216
+ // Fallback: check extension for image types including SVG
217
+ const imageExtensions = ['.png', '.jpg', '.jpeg', '.gif', '.webp', '.bmp', '.svg'];
218
+ if (imageExtensions.some(ext => name.endsWith(ext))) {
219
+ console.log('%c[DEBUG] Detected as image by extension', 'color: orange;');
220
+ return 'icon-gallery';
221
+ }
222
+ console.log('%c[DEBUG] Defaulting to document icon', 'color: red;');
223
+ return 'icon-van';
224
+ }
225
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FileUploadMultiComponent, deps: [{ token: i1.UploadSimulatorService }], target: i0.ɵɵFactoryTarget.Component });
226
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: FileUploadMultiComponent, isStandalone: true, selector: "dso-file-upload-multi", ngImport: i0, template: "<div class=\"upload-container\">\r\n\r\n <!-- DRAG & DROP AREA -->\r\n <div\r\n class=\"drop-zone\"\r\n [class.drag-over]=\"isDragOver\"\r\n (dragover)=\"onDragOver($event)\"\r\n (dragleave)=\"onDragLeave($event)\"\r\n (drop)=\"onDrop($event)\"\r\n (click)=\"onDropZoneClick(fileInput)\"\r\n >\r\n\r\n <dso-button \r\n style=\"padding: 0; \"\r\n btnLabel=\"Select files\"\r\n btnType=\"link\"\r\n (onClick)=\"onBrowseButtonClick($event, fileInput)\">\r\n </dso-button>\r\n or drag & drop files \r\n <input \r\n type=\"file\" \r\n multiple \r\n hidden \r\n #fileInput \r\n (change)=\"onFileSelected($event)\"\r\n />\r\n </div>\r\n\r\n <!-- FILE LIST -->\r\n <div class=\"file-list\" *ngIf=\"uploadQueue.length > 0\">\r\n\r\n <dso-file-item\r\n *ngFor=\"let item of uploadQueue\"\r\n [item]=\"item\"\r\n (retry)=\"retryUpload(item)\"\r\n (remove)=\"removeItem(item)\"\r\n (cancel)=\"cancelUpload(item)\"\r\n ></dso-file-item>\r\n\r\n\r\n </div>\r\n</div>\r\n", styles: [".upload-container{display:flex;flex-direction:column;gap:8px}.drop-zone{border:2px dashed #ccc;padding:2rem;text-align:center;border-radius:6px;transition:border-color .2s,background-color .2s;cursor:pointer}.drop-zone.drag-over{border-color:#007bff;background-color:#f0f7ff}.file-list{display:flex;flex-direction:column;gap:.75rem;max-height:360px;overflow-y:auto;padding-right:4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { 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"] }] });
227
+ }
228
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FileUploadMultiComponent, decorators: [{
229
+ type: Component,
230
+ args: [{ selector: 'dso-file-upload-multi', standalone: true, imports: [CommonModule, ButtonComponent, IconComponent, FileUploadItemComponent], template: "<div class=\"upload-container\">\r\n\r\n <!-- DRAG & DROP AREA -->\r\n <div\r\n class=\"drop-zone\"\r\n [class.drag-over]=\"isDragOver\"\r\n (dragover)=\"onDragOver($event)\"\r\n (dragleave)=\"onDragLeave($event)\"\r\n (drop)=\"onDrop($event)\"\r\n (click)=\"onDropZoneClick(fileInput)\"\r\n >\r\n\r\n <dso-button \r\n style=\"padding: 0; \"\r\n btnLabel=\"Select files\"\r\n btnType=\"link\"\r\n (onClick)=\"onBrowseButtonClick($event, fileInput)\">\r\n </dso-button>\r\n or drag & drop files \r\n <input \r\n type=\"file\" \r\n multiple \r\n hidden \r\n #fileInput \r\n (change)=\"onFileSelected($event)\"\r\n />\r\n </div>\r\n\r\n <!-- FILE LIST -->\r\n <div class=\"file-list\" *ngIf=\"uploadQueue.length > 0\">\r\n\r\n <dso-file-item\r\n *ngFor=\"let item of uploadQueue\"\r\n [item]=\"item\"\r\n (retry)=\"retryUpload(item)\"\r\n (remove)=\"removeItem(item)\"\r\n (cancel)=\"cancelUpload(item)\"\r\n ></dso-file-item>\r\n\r\n\r\n </div>\r\n</div>\r\n", styles: [".upload-container{display:flex;flex-direction:column;gap:8px}.drop-zone{border:2px dashed #ccc;padding:2rem;text-align:center;border-radius:6px;transition:border-color .2s,background-color .2s;cursor:pointer}.drop-zone.drag-over{border-color:#007bff;background-color:#f0f7ff}.file-list{display:flex;flex-direction:column;gap:.75rem;max-height:360px;overflow-y:auto;padding-right:4px}\n"] }]
231
+ }], ctorParameters: () => [{ type: i1.UploadSimulatorService }] });
232
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload-multiple.component.js","sourceRoot":"","sources":["../../../../../projects/ui/src/lib/file-upload-multiple/file-upload-multiple.component.ts","../../../../../projects/ui/src/lib/file-upload-multiple/file-upload-multiple.component.html"],"names":[],"mappings":"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","sourcesContent":["// file-upload-multi.component.ts\r\nimport { Component } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { UploadItem } from './upload-item.model';\r\nimport { UploadSimulatorService } from './upload-simulator.service';\r\nimport { ButtonComponent } from '../button/button.component';\r\nimport { IconComponent } from '../icon/icon.component';\r\nimport { FileUploadItemComponent } from '../file-upload-items/file-upload-items.component';\r\n\r\n/**\r\n * FileUploadMultiComponent\r\n * -------------------------\r\n * This component handles:\r\n * - Drag & drop file uploads\r\n * - Selecting multiple files through <input type=\"file\">\r\n * - Upload progress per file\r\n * - Cancelling an upload\r\n * - Retrying failed uploads\r\n * - Removing items from the queue\r\n *\r\n * UploadSimulatorService is used to mimic real upload behavior.\r\n * Later, you can replace it with a real HTTP upload service.\r\n */\r\n@Component({\r\n  selector: 'dso-file-upload-multi',\r\n  standalone: true,\r\n  imports: [CommonModule, ButtonComponent, IconComponent, FileUploadItemComponent],\r\n  templateUrl: './file-upload-multiple.component.html',\r\n  styleUrls: ['./file-upload-multiple.component.scss']\r\n})\r\nexport class FileUploadMultiComponent {\r\n\r\n  //   ngOnInit() {\r\n  //   // Fake completed file\r\n  //   this.uploadQueue.push({\r\n  //     file: new File([\"\"], \"example-image.jpg\", { type: \"image/jpeg\" }),\r\n  //     progress: 100,\r\n  //     status: \"completed\",\r\n  //     hideProgressBar: true\r\n  //   });\r\n\r\n  //   // Fake failed file\r\n  //   this.uploadQueue.push({\r\n  //     file: new File([\"\"], \"broken-file.pdf\", { type: \"application/pdf\" }),\r\n  //     progress: null,\r\n  //     status: \"failed\",\r\n  //     hideProgressBar: true\r\n  //   });\r\n\r\n  //   // Fake uploading file\r\n  //   this.uploadQueue.push({\r\n  //     file: new File([\"\"], \"uploading.png\", { type: \"image/png\" }),\r\n  //     progress: 40,\r\n  //     status: \"uploading\",\r\n  //     hideProgressBar: false\r\n  //   });\r\n  // }\r\n\r\n\r\n  /** All files being managed (uploaded, uploading, failed, completed). */\r\n  uploadQueue: UploadItem[] = [];\r\n\r\n  /** True while user drags a file over the drop zone. */\r\n  isDragOver = false;\r\n\r\n  constructor(private sim: UploadSimulatorService) {\r\n    // console.log('%c[COMPONENT] FileUploadMultiComponent initialized', 'color: purple;');\r\n  }\r\n\r\n  /* ===========================================================\r\n   *                     DRAG & DROP EVENTS\r\n   * =========================================================== */\r\n\r\n  onDragOver(event: DragEvent) {\r\n    event.preventDefault();\r\n    this.isDragOver = true;\r\n    // console.log('%c[DRAG] Dragging over drop zone...', 'color: #00aaff;');\r\n  }\r\n\r\n  onDragLeave(event: DragEvent) {\r\n    this.isDragOver = false;\r\n    // console.log('%c[DRAG] Drag left drop zone', 'color: #888;');\r\n  }\r\n\r\n  onDrop(event: DragEvent) {\r\n    event.preventDefault();\r\n    this.isDragOver = false;\r\n\r\n    // console.log('%c[DRAG] Files dropped', 'color: #00cc66;');\r\n\r\n    if (event.dataTransfer?.files) {\r\n      // console.log('[DRAG] Dropped files:', event.dataTransfer.files);\r\n      this.handleFiles(event.dataTransfer.files);\r\n    }\r\n  }\r\n\r\n  /* ===========================================================\r\n   *                     FILE INPUT SELECT\r\n   * =========================================================== */\r\n\r\n  onFileSelected(event: Event) {\r\n    const input = event.target as HTMLInputElement;\r\n\r\n    if (input.files) {\r\n      // console.log('%c[INPUT] Files selected:', 'color: #0099ff;', input.files);\r\n      this.handleFiles(input.files);\r\n\r\n      // Clears <input> so user can re-select same file\r\n      input.value = '';\r\n    }\r\n  }\r\n\r\n  /* ===========================================================\r\n   *                PROCESSING NEWLY ADDED FILES\r\n   * =========================================================== */\r\n\r\n  handleFiles(fileList: FileList) {\r\n    // console.log('%c[FILES] Processing new files...', 'color: #ffaa00;');\r\n\r\n    Array.from(fileList).forEach(file => {\r\n      // console.log(`%c[FILES] Added: ${file.name} (${file.size} bytes)`, 'color: #ffaa00;');\r\n\r\n      const item: UploadItem = {\r\n        file,\r\n        progress: 0,\r\n        status: 'queued'\r\n      };\r\n\r\n      this.uploadQueue.push(item);\r\n\r\n      this.startUpload(item);\r\n    });\r\n  }\r\n\r\n  /* ===========================================================\r\n   *                        UPLOAD LOGIC\r\n   * =========================================================== */\r\n\r\n  startUpload(item: UploadItem) {\r\n    // console.log(`%c[UPLOAD] Starting upload for ${item.file.name}`, 'color: #00cc99;');\r\n\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        // console.log(`%c[UPLOAD] ${item.file.name}: ${progress.toFixed(1)}%`, 'color: #0099ff;');\r\n      },\r\n\r\n      // ---- onComplete callback ----\r\n      () => {\r\n        item.status = 'completed';\r\n        item.progress = 100;\r\n        \r\n        // Hide progress bar after a very small delay (100–300ms)\r\n        setTimeout(() => item.hideProgressBar = true, 300);\r\n        // console.log(`%c[UPLOAD] COMPLETED: ${item.file.name}`, 'color: green; font-weight: bold;');\r\n      },\r\n\r\n      // ---- onError callback ----\r\n      () => {\r\n        item.status = 'failed';\r\n        item.progress = null;\r\n\r\n        // Hide progress bar so status text can show\r\n        setTimeout(() => item.hideProgressBar = true, 300);\r\n        // console.log(`%c[UPLOAD] FAILED: ${item.file.name}`, 'color: red; font-weight: bold;');\r\n      }\r\n    );\r\n  }\r\n\r\n  onDropZoneClick(fileInput: HTMLInputElement) {\r\n  // Prevent click if user is dragging files over the drop-zone\r\n  if (this.isDragOver) return;\r\n\r\n  console.log('%c[CLICK] Drop zone clicked → opening file dialog', 'color: #0077ff;');\r\n  fileInput.click();\r\n}\r\n\r\nonBrowseButtonClick(event: Event, fileInput: HTMLInputElement) {\r\n  event.stopPropagation(); // stops click from bubbling to drop-zone\r\n  // console.log('%c[BUTTON] Browse clicked → opening file dialog', 'color: #8844ff;');\r\n  fileInput.click();\r\n}\r\n\r\n  /* ===========================================================\r\n   *                BUTTON ACTIONS (CANCEL / RETRY / REMOVE)\r\n   * =========================================================== */\r\n\r\n  cancelUpload(item: UploadItem) {\r\n    // console.log(`%c[CANCEL] Cancelling ${item.file.name}`, 'color: orange; font-weight: bold;');\r\n    // item.status = 'cancelled';\r\n    // item.progress = null;\r\n    this.removeItem(item);\r\n\r\n  }\r\n\r\n  retryUpload(item: UploadItem) {\r\n    // console.log('[RETRY] Restarting upload for', item.file.name);\r\n\r\n    // Reset state\r\n    item.status = 'queued';\r\n    item.progress = 0;\r\n    item.hideProgressBar = false; // <-- IMPORTANT\r\n\r\n    // Start upload again\r\n    this.startUpload(item);\r\n  }\r\n\r\n  removeItem(item: UploadItem) {\r\n    // console.log(`%c[REMOVE] Removing ${item.file.name}`, 'color: gray;');\r\n    this.uploadQueue = this.uploadQueue.filter(i => i !== item);\r\n  }\r\n\r\n  /* ===========================================================\r\n   *                     UTILITY FUNCTIONS\r\n   * =========================================================== */\r\n\r\n  /**\r\n   * Converts bytes → human readable sizes.\r\n   */\r\n  formatSize(size: number): string {\r\n    if (size < 1024) return size + ' B';\r\n    if (size < 1024 * 1024) return (size / 1024).toFixed(1) + ' KB';\r\n    return (size / (1024 * 1024)).toFixed(1) + ' MB';\r\n  }\r\n\r\n  // getThumbnail(item: UploadItem): string | null {\r\n  //   const file = item.file;\r\n\r\n  //   // If actual image → return preview URL\r\n  //   if (file.type.startsWith('image/')) {\r\n  //     return URL.createObjectURL(file);\r\n  //   }\r\n\r\n  //   // Otherwise return an icon path depending on type\r\n  //   if (file.type === 'application/pdf') return 'assets/icons/pdf.svg';\r\n  //   if (file.type.includes('word')) return 'assets/icons/doc.svg';\r\n  //   if (file.type.includes('spreadsheet')) return 'assets/icons/xlsx.svg';\r\n\r\n  //   // default icon\r\n  //   return 'assets/icons/file.svg';\r\n  // }\r\n\r\n  getThumbnail(item: UploadItem): string | null {\r\n  const file = item.file;\r\n\r\n  // Image preview for image file types\r\n  if (file.type.startsWith('image/')) {\r\n    return URL.createObjectURL(file);\r\n  }\r\n\r\n  // Not an image → no thumbnail\r\n  return null;\r\n}\r\n\r\n  getFileIcon(item: UploadItem): string {\r\n    const file = item.file;\r\n    const type = file.type.toLowerCase();\r\n    const name = file.name.toLowerCase();\r\n\r\n    console.log('%c[DEBUG] File name:', 'color: blue;', file.name);\r\n    console.log('%c[DEBUG] File type:', 'color: green;', file.type);\r\n\r\n    // Check MIME type first\r\n    if (type.startsWith('image/')) {\r\n      console.log('%c[DEBUG] Detected as image by MIME type', 'color: purple;');\r\n      return 'icon-gallery';\r\n    }\r\n\r\n    // Fallback: check extension for image types including SVG\r\n    const imageExtensions = ['.png', '.jpg', '.jpeg', '.gif', '.webp', '.bmp', '.svg'];\r\n    if (imageExtensions.some(ext => name.endsWith(ext))) {\r\n      console.log('%c[DEBUG] Detected as image by extension', 'color: orange;');\r\n      return 'icon-gallery';\r\n    }\r\n\r\n    console.log('%c[DEBUG] Defaulting to document icon', 'color: red;');\r\n    return 'icon-van';\r\n  }\r\n}\r\n","<div class=\"upload-container\">\r\n\r\n  <!-- DRAG & DROP AREA -->\r\n  <div\r\n    class=\"drop-zone\"\r\n    [class.drag-over]=\"isDragOver\"\r\n    (dragover)=\"onDragOver($event)\"\r\n    (dragleave)=\"onDragLeave($event)\"\r\n    (drop)=\"onDrop($event)\"\r\n    (click)=\"onDropZoneClick(fileInput)\"\r\n  >\r\n\r\n    <dso-button \r\n    style=\"padding: 0; \"\r\n    btnLabel=\"Select files\"\r\n    btnType=\"link\"\r\n    (onClick)=\"onBrowseButtonClick($event, fileInput)\">\r\n    </dso-button>\r\n    or drag & drop files \r\n    <input \r\n      type=\"file\" \r\n      multiple \r\n      hidden \r\n      #fileInput \r\n      (change)=\"onFileSelected($event)\"\r\n    />\r\n  </div>\r\n\r\n  <!-- FILE LIST -->\r\n  <div class=\"file-list\" *ngIf=\"uploadQueue.length > 0\">\r\n\r\n    <dso-file-item\r\n    *ngFor=\"let item of uploadQueue\"\r\n    [item]=\"item\"\r\n    (retry)=\"retryUpload(item)\"\r\n    (remove)=\"removeItem(item)\"\r\n    (cancel)=\"cancelUpload(item)\"\r\n    ></dso-file-item>\r\n\r\n\r\n  </div>\r\n</div>\r\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBsb2FkLWl0ZW0ubW9kZWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy91aS9zcmMvbGliL2ZpbGUtdXBsb2FkLW11bHRpcGxlL3VwbG9hZC1pdGVtLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIFVwbG9hZEl0ZW0ge1xyXG4gIGZpbGU6IEZpbGU7XHJcbiAgcHJvZ3Jlc3M6IG51bWJlciB8IG51bGw7ICAvLyBudWxsID0gcXVldWVkIC8gbm90IHN0YXJ0ZWQgeWV0XHJcbiAgc3RhdHVzOiAncXVldWVkJyB8ICd1cGxvYWRpbmcnIHwgJ2NvbXBsZXRlZCcgfCAnZmFpbGVkJyB8ICdjYW5jZWxsZWQnO1xyXG4gIGNvbnRyb2xsZXI/OiBBYm9ydENvbnRyb2xsZXI7IC8vIHVzZWQgbGF0ZXIgZm9yIHJlYWwgdXBsb2Fkc1xyXG4gIGhpZGVQcm9ncmVzc0Jhcj86IGJvb2xlYW47XHJcbn1cclxuIl19
@@ -0,0 +1,76 @@
1
+ // upload-simulator.service.ts
2
+ import { Injectable } from '@angular/core';
3
+ import * as i0 from "@angular/core";
4
+ /**
5
+ * UploadSimulatorService
6
+ * -----------------------
7
+ * This service simulates an upload process.
8
+ * It is ONLY for demo and UI development. Later, it can
9
+ * be replaced with a real HTTP upload service without
10
+ * changing the UI components.
11
+ *
12
+ * How it works:
13
+ * - Every 300ms, a timer increases the upload progress.
14
+ * - Progress increases by a random amount (to look realistic).
15
+ * - When it reaches 100%, the upload is marked complete.
16
+ * - There is a 10% chance the upload "fails" to simulate errors.
17
+ * - If an upload is cancelled, the interval stops immediately.
18
+ *
19
+ * This pattern mimics:
20
+ * - progress events (like HttpClient upload events)
21
+ * - cancellation (AbortController / unsubscribe)
22
+ */
23
+ export class UploadSimulatorService {
24
+ /**
25
+ * Simulates uploading a single file.
26
+ *
27
+ * @param item The UploadItem being uploaded
28
+ * @param onProgress Callback fired whenever progress changes
29
+ * @param onComplete Callback fired when upload reaches 100%
30
+ * @param onError Callback fired when upload "fails"
31
+ */
32
+ simulateUpload(item, onProgress, onComplete, onError) {
33
+ console.log('%c[SIMULATOR] Starting upload:', 'color: green;', item.file.name);
34
+ let progress = 0;
35
+ // This interval simulates a real upload stream
36
+ const interval = setInterval(() => {
37
+ // If cancelled externally, we stop updating
38
+ if (item.status === 'cancelled') {
39
+ console.log('%c[SIMULATOR] Upload CANCELLED:', 'color: orange;', item.file.name);
40
+ clearInterval(interval);
41
+ return;
42
+ }
43
+ // 🔼 Simulate uploading by adding random progress increments
44
+ const increment = Math.random() * 15 + 5;
45
+ progress += increment;
46
+ console.log(`%c[SIMULATOR] ${item.file.name} +${increment.toFixed(1)}% → ${progress.toFixed(1)}%`, 'color: #1e90ff;');
47
+ // 🟦 If progress reaches 100%, finalize upload
48
+ if (progress >= 100) {
49
+ progress = 100;
50
+ onProgress(progress);
51
+ clearInterval(interval);
52
+ console.log('%c[SIMULATOR] Upload reached 100% for ' + item.file.name, 'color: green; font-weight: bold;');
53
+ // 🔥 50% chance to simulate failure
54
+ const randomFail = Math.random() < 0.5;
55
+ if (randomFail) {
56
+ console.log('%c[SIMULATOR] Simulated FAILURE for ' + item.file.name, 'color: red; font-weight: bold;');
57
+ onError();
58
+ }
59
+ else {
60
+ console.log('%c[SIMULATOR] Upload SUCCESS for ' + item.file.name, 'color: limegreen; font-weight: bold;');
61
+ onComplete();
62
+ }
63
+ return;
64
+ }
65
+ // Emit progress update
66
+ onProgress(progress);
67
+ }, 300); // runs every 300ms like a real upload tick
68
+ }
69
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UploadSimulatorService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
70
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UploadSimulatorService, providedIn: 'root' });
71
+ }
72
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UploadSimulatorService, decorators: [{
73
+ type: Injectable,
74
+ args: [{ providedIn: 'root' }]
75
+ }] });
76
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBsb2FkLXNpbXVsYXRvci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdWkvc3JjL2xpYi9maWxlLXVwbG9hZC1tdWx0aXBsZS91cGxvYWQtc2ltdWxhdG9yLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsOEJBQThCO0FBQzlCLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRzNDOzs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FrQkc7QUFFSCxNQUFNLE9BQU8sc0JBQXNCO0lBRWpDOzs7Ozs7O09BT0c7SUFDSCxjQUFjLENBQ1osSUFBZ0IsRUFDaEIsVUFBbUMsRUFDbkMsVUFBc0IsRUFDdEIsT0FBbUI7UUFFbkIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQ0FBZ0MsRUFBRSxlQUFlLEVBQUUsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUUvRSxJQUFJLFFBQVEsR0FBRyxDQUFDLENBQUM7UUFFakIsK0NBQStDO1FBQy9DLE1BQU0sUUFBUSxHQUFHLFdBQVcsQ0FBQyxHQUFHLEVBQUU7WUFFaEMsNENBQTRDO1lBQzVDLElBQUksSUFBSSxDQUFDLE1BQU0sS0FBSyxXQUFXLEVBQUUsQ0FBQztnQkFDaEMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxpQ0FBaUMsRUFBRSxnQkFBZ0IsRUFBRSxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO2dCQUNqRixhQUFhLENBQUMsUUFBUSxDQUFDLENBQUM7Z0JBQ3hCLE9BQU87WUFDVCxDQUFDO1lBRUQsNkRBQTZEO1lBQzdELE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDO1lBQ3pDLFFBQVEsSUFBSSxTQUFTLENBQUM7WUFFdEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxpQkFBaUIsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEtBQUssU0FBUyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsT0FBTyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxHQUFHLEVBQy9GLGlCQUFpQixDQUFDLENBQUM7WUFFckIsK0NBQStDO1lBQy9DLElBQUksUUFBUSxJQUFJLEdBQUcsRUFBRSxDQUFDO2dCQUNwQixRQUFRLEdBQUcsR0FBRyxDQUFDO2dCQUNmLFVBQVUsQ0FBQyxRQUFRLENBQUMsQ0FBQztnQkFDckIsYUFBYSxDQUFDLFFBQVEsQ0FBQyxDQUFDO2dCQUV4QixPQUFPLENBQUMsR0FBRyxDQUFDLHdDQUF3QyxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUNuRSxrQ0FBa0MsQ0FBQyxDQUFDO2dCQUV0QyxvQ0FBb0M7Z0JBQ3BDLE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxHQUFHLENBQUM7Z0JBRXZDLElBQUksVUFBVSxFQUFFLENBQUM7b0JBQ2YsT0FBTyxDQUFDLEdBQUcsQ0FBQyxzQ0FBc0MsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksRUFDakUsZ0NBQWdDLENBQUMsQ0FBQztvQkFDcEMsT0FBTyxFQUFFLENBQUM7Z0JBQ1osQ0FBQztxQkFBTSxDQUFDO29CQUNOLE9BQU8sQ0FBQyxHQUFHLENBQUMsbUNBQW1DLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQzlELHNDQUFzQyxDQUFDLENBQUM7b0JBQzFDLFVBQVUsRUFBRSxDQUFDO2dCQUNmLENBQUM7Z0JBRUQsT0FBTztZQUNULENBQUM7WUFFRCx1QkFBdUI7WUFDdkIsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBRXZCLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLDJDQUEyQztJQUN0RCxDQUFDO3dHQWxFVSxzQkFBc0I7NEdBQXRCLHNCQUFzQixjQURULE1BQU07OzRGQUNuQixzQkFBc0I7a0JBRGxDLFVBQVU7bUJBQUMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiLy8gdXBsb2FkLXNpbXVsYXRvci5zZXJ2aWNlLnRzXHJcbmltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgVXBsb2FkSXRlbSB9IGZyb20gJy4vdXBsb2FkLWl0ZW0ubW9kZWwnO1xyXG5cclxuLyoqXHJcbiAqIFVwbG9hZFNpbXVsYXRvclNlcnZpY2VcclxuICogLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cclxuICogVGhpcyBzZXJ2aWNlIHNpbXVsYXRlcyBhbiB1cGxvYWQgcHJvY2Vzcy5cclxuICogSXQgaXMgT05MWSBmb3IgZGVtbyBhbmQgVUkgZGV2ZWxvcG1lbnQuIExhdGVyLCBpdCBjYW5cclxuICogYmUgcmVwbGFjZWQgd2l0aCBhIHJlYWwgSFRUUCB1cGxvYWQgc2VydmljZSB3aXRob3V0XHJcbiAqIGNoYW5naW5nIHRoZSBVSSBjb21wb25lbnRzLlxyXG4gKlxyXG4gKiBIb3cgaXQgd29ya3M6XHJcbiAqIC0gRXZlcnkgMzAwbXMsIGEgdGltZXIgaW5jcmVhc2VzIHRoZSB1cGxvYWQgcHJvZ3Jlc3MuXHJcbiAqIC0gUHJvZ3Jlc3MgaW5jcmVhc2VzIGJ5IGEgcmFuZG9tIGFtb3VudCAodG8gbG9vayByZWFsaXN0aWMpLlxyXG4gKiAtIFdoZW4gaXQgcmVhY2hlcyAxMDAlLCB0aGUgdXBsb2FkIGlzIG1hcmtlZCBjb21wbGV0ZS5cclxuICogLSBUaGVyZSBpcyBhIDEwJSBjaGFuY2UgdGhlIHVwbG9hZCBcImZhaWxzXCIgdG8gc2ltdWxhdGUgZXJyb3JzLlxyXG4gKiAtIElmIGFuIHVwbG9hZCBpcyBjYW5jZWxsZWQsIHRoZSBpbnRlcnZhbCBzdG9wcyBpbW1lZGlhdGVseS5cclxuICpcclxuICogVGhpcyBwYXR0ZXJuIG1pbWljczpcclxuICogLSBwcm9ncmVzcyBldmVudHMgKGxpa2UgSHR0cENsaWVudCB1cGxvYWQgZXZlbnRzKVxyXG4gKiAtIGNhbmNlbGxhdGlvbiAoQWJvcnRDb250cm9sbGVyIC8gdW5zdWJzY3JpYmUpXHJcbiAqL1xyXG5ASW5qZWN0YWJsZSh7IHByb3ZpZGVkSW46ICdyb290JyB9KVxyXG5leHBvcnQgY2xhc3MgVXBsb2FkU2ltdWxhdG9yU2VydmljZSB7XHJcblxyXG4gIC8qKlxyXG4gICAqIFNpbXVsYXRlcyB1cGxvYWRpbmcgYSBzaW5nbGUgZmlsZS5cclxuICAgKlxyXG4gICAqIEBwYXJhbSBpdGVtICAgICAgICBUaGUgVXBsb2FkSXRlbSBiZWluZyB1cGxvYWRlZFxyXG4gICAqIEBwYXJhbSBvblByb2dyZXNzICBDYWxsYmFjayBmaXJlZCB3aGVuZXZlciBwcm9ncmVzcyBjaGFuZ2VzXHJcbiAgICogQHBhcmFtIG9uQ29tcGxldGUgIENhbGxiYWNrIGZpcmVkIHdoZW4gdXBsb2FkIHJlYWNoZXMgMTAwJVxyXG4gICAqIEBwYXJhbSBvbkVycm9yICAgICBDYWxsYmFjayBmaXJlZCB3aGVuIHVwbG9hZCBcImZhaWxzXCJcclxuICAgKi9cclxuICBzaW11bGF0ZVVwbG9hZChcclxuICAgIGl0ZW06IFVwbG9hZEl0ZW0sXHJcbiAgICBvblByb2dyZXNzOiAodmFsdWU6IG51bWJlcikgPT4gdm9pZCxcclxuICAgIG9uQ29tcGxldGU6ICgpID0+IHZvaWQsXHJcbiAgICBvbkVycm9yOiAoKSA9PiB2b2lkXHJcbiAgKSB7XHJcbiAgICBjb25zb2xlLmxvZygnJWNbU0lNVUxBVE9SXSBTdGFydGluZyB1cGxvYWQ6JywgJ2NvbG9yOiBncmVlbjsnLCBpdGVtLmZpbGUubmFtZSk7XHJcblxyXG4gICAgbGV0IHByb2dyZXNzID0gMDtcclxuXHJcbiAgICAvLyBUaGlzIGludGVydmFsIHNpbXVsYXRlcyBhIHJlYWwgdXBsb2FkIHN0cmVhbVxyXG4gICAgY29uc3QgaW50ZXJ2YWwgPSBzZXRJbnRlcnZhbCgoKSA9PiB7XHJcblxyXG4gICAgICAvLyBJZiBjYW5jZWxsZWQgZXh0ZXJuYWxseSwgd2Ugc3RvcCB1cGRhdGluZ1xyXG4gICAgICBpZiAoaXRlbS5zdGF0dXMgPT09ICdjYW5jZWxsZWQnKSB7XHJcbiAgICAgICAgY29uc29sZS5sb2coJyVjW1NJTVVMQVRPUl0gVXBsb2FkIENBTkNFTExFRDonLCAnY29sb3I6IG9yYW5nZTsnLCBpdGVtLmZpbGUubmFtZSk7XHJcbiAgICAgICAgY2xlYXJJbnRlcnZhbChpbnRlcnZhbCk7XHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgICB9XHJcblxyXG4gICAgICAvLyDwn5S8IFNpbXVsYXRlIHVwbG9hZGluZyBieSBhZGRpbmcgcmFuZG9tIHByb2dyZXNzIGluY3JlbWVudHNcclxuICAgICAgY29uc3QgaW5jcmVtZW50ID0gTWF0aC5yYW5kb20oKSAqIDE1ICsgNTtcclxuICAgICAgcHJvZ3Jlc3MgKz0gaW5jcmVtZW50O1xyXG5cclxuICAgICAgY29uc29sZS5sb2coYCVjW1NJTVVMQVRPUl0gJHtpdGVtLmZpbGUubmFtZX0gKyR7aW5jcmVtZW50LnRvRml4ZWQoMSl9JSDihpIgJHtwcm9ncmVzcy50b0ZpeGVkKDEpfSVgLFxyXG4gICAgICAgICdjb2xvcjogIzFlOTBmZjsnKTtcclxuXHJcbiAgICAgIC8vIPCfn6YgSWYgcHJvZ3Jlc3MgcmVhY2hlcyAxMDAlLCBmaW5hbGl6ZSB1cGxvYWRcclxuICAgICAgaWYgKHByb2dyZXNzID49IDEwMCkge1xyXG4gICAgICAgIHByb2dyZXNzID0gMTAwO1xyXG4gICAgICAgIG9uUHJvZ3Jlc3MocHJvZ3Jlc3MpO1xyXG4gICAgICAgIGNsZWFySW50ZXJ2YWwoaW50ZXJ2YWwpO1xyXG5cclxuICAgICAgICBjb25zb2xlLmxvZygnJWNbU0lNVUxBVE9SXSBVcGxvYWQgcmVhY2hlZCAxMDAlIGZvciAnICsgaXRlbS5maWxlLm5hbWUsXHJcbiAgICAgICAgICAnY29sb3I6IGdyZWVuOyBmb250LXdlaWdodDogYm9sZDsnKTtcclxuXHJcbiAgICAgICAgLy8g8J+UpSA1MCUgY2hhbmNlIHRvIHNpbXVsYXRlIGZhaWx1cmVcclxuICAgICAgICBjb25zdCByYW5kb21GYWlsID0gTWF0aC5yYW5kb20oKSA8IDAuNTtcclxuXHJcbiAgICAgICAgaWYgKHJhbmRvbUZhaWwpIHtcclxuICAgICAgICAgIGNvbnNvbGUubG9nKCclY1tTSU1VTEFUT1JdIFNpbXVsYXRlZCBGQUlMVVJFIGZvciAnICsgaXRlbS5maWxlLm5hbWUsXHJcbiAgICAgICAgICAgICdjb2xvcjogcmVkOyBmb250LXdlaWdodDogYm9sZDsnKTtcclxuICAgICAgICAgIG9uRXJyb3IoKTtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgY29uc29sZS5sb2coJyVjW1NJTVVMQVRPUl0gVXBsb2FkIFNVQ0NFU1MgZm9yICcgKyBpdGVtLmZpbGUubmFtZSxcclxuICAgICAgICAgICAgJ2NvbG9yOiBsaW1lZ3JlZW47IGZvbnQtd2VpZ2h0OiBib2xkOycpO1xyXG4gICAgICAgICAgb25Db21wbGV0ZSgpO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgcmV0dXJuO1xyXG4gICAgICB9XHJcblxyXG4gICAgICAvLyBFbWl0IHByb2dyZXNzIHVwZGF0ZVxyXG4gICAgICBvblByb2dyZXNzKHByb2dyZXNzKTtcclxuXHJcbiAgICB9LCAzMDApOyAvLyBydW5zIGV2ZXJ5IDMwMG1zIGxpa2UgYSByZWFsIHVwbG9hZCB0aWNrXHJcbiAgfVxyXG59XHJcbiJdfQ==