ngx-histaff-alpha 2.3.8 → 2.4.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.
@@ -1,4 +1,4 @@
1
- import { Component, Input, ViewChild, isDevMode, forwardRef } from '@angular/core';
1
+ import { Component, Input, ViewChild, isDevMode } from '@angular/core';
2
2
  import { CoreFormControlBaseComponent } from '../../core-form-control-base/core-form-control-base.component';
3
3
  import { FormsModule, NG_VALUE_ACCESSOR, NgControl } from '@angular/forms';
4
4
  import { alertOptions, noneAutoClosedAlertOptions } from '../../../constants/alertOptions';
@@ -16,13 +16,14 @@ import * as i1 from "../../alert/alert.service";
16
16
  import * as i2 from "../../../services/multi-language.service";
17
17
  import * as i3 from "../../../services/app-config.service";
18
18
  import * as i4 from "../../../services/uploaded-file.service";
19
- import * as i5 from "@angular/common";
20
- import * as i6 from "@angular/forms";
19
+ import * as i5 from "../../services/dom.service";
20
+ import * as i6 from "@angular/common";
21
+ import * as i7 from "@angular/forms";
21
22
  export class CoreAttachmentComponent extends CoreFormControlBaseComponent {
22
23
  writeValue(obj) {
23
- this.valueToShow = obj?.serverFileName;
24
+ this.valueToShow = obj?.serverFileName || '';
24
25
  }
25
- constructor(injector, alertService, mls, appConfigService, uploadedFileService, renderer) {
26
+ constructor(injector, alertService, mls, appConfigService, uploadedFileService, renderer, domService) {
26
27
  super();
27
28
  this.injector = injector;
28
29
  this.alertService = alertService;
@@ -30,8 +31,10 @@ export class CoreAttachmentComponent extends CoreFormControlBaseComponent {
30
31
  this.appConfigService = appConfigService;
31
32
  this.uploadedFileService = uploadedFileService;
32
33
  this.renderer = renderer;
34
+ this.domService = domService;
33
35
  this.dropdownHeight = 0;
34
36
  this.dropdownPadding = 0;
37
+ this.dropdownZIndex = 1;
35
38
  }
36
39
  ngOnChanges(changes) {
37
40
  if (changes['valueToShow']) {
@@ -40,7 +43,7 @@ export class CoreAttachmentComponent extends CoreFormControlBaseComponent {
40
43
  }
41
44
  }
42
45
  ngOnInit() {
43
- setTimeout(() => this.ngControl = this.injector.get(NgControl), 200);
46
+ this.ngControl = this.injector.get(NgControl);
44
47
  if (!!!this.assignTo && !!!this.readonly && isDevMode()) {
45
48
  this.alertService.error(`
46
49
  CoreAttachmentComponent required input 'assignTo'
@@ -55,6 +58,7 @@ export class CoreAttachmentComponent extends CoreFormControlBaseComponent {
55
58
  this.dropdownPadding = 0;
56
59
  }
57
60
  });
61
+ this.dropdownZIndex = this.domService.getMaxZIndex() + 1;
58
62
  });
59
63
  }
60
64
  onClickUpload() {
@@ -137,14 +141,28 @@ export class CoreAttachmentComponent extends CoreFormControlBaseComponent {
137
141
  }
138
142
  }
139
143
  }
140
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: CoreAttachmentComponent, deps: [{ token: i0.Injector }, { token: i1.AlertService }, { token: i2.MultiLanguageService }, { token: i3.AppConfigService }, { token: i4.UploadedFileService }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); }
144
+ clear() {
145
+ this.value = null;
146
+ this.valueToShow = '';
147
+ this.onChange(null);
148
+ const sibling = this.ngControl.control?.parent?.get(this.assignTo);
149
+ if (!sibling) {
150
+ if (isDevMode()) {
151
+ this.alertService.error(`Chưa khai báo field '${this.assignTo}'`, noneAutoClosedAlertOptions);
152
+ }
153
+ }
154
+ else {
155
+ sibling.patchValue(null);
156
+ }
157
+ }
158
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: CoreAttachmentComponent, deps: [{ token: i0.Injector }, { token: i1.AlertService }, { token: i2.MultiLanguageService }, { token: i3.AppConfigService }, { token: i4.UploadedFileService }, { token: i0.Renderer2 }, { token: i5.DomService }], target: i0.ɵɵFactoryTarget.Component }); }
141
159
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.3", type: CoreAttachmentComponent, isStandalone: true, selector: "core-attachment", inputs: { assignTo: "assignTo", readonly: "readonly", valueToShow: "valueToShow" }, providers: [
142
160
  {
143
161
  provide: NG_VALUE_ACCESSOR,
144
162
  multi: true,
145
- useExisting: forwardRef(() => CoreAttachmentComponent),
163
+ useExisting: CoreAttachmentComponent,
146
164
  }
147
- ], viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true }, { propertyName: "rawInput", first: true, predicate: ["rawInput"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div #container class=\"core-attachment-container\">\r\n <input id=\"rawInput\" #rawInput type=\"file\" (change)=\"handleInputChange($event)\">\r\n <div class=\"form-control\" type=\"text\" [(ngModel)]=\"valueToShow\" readonly [class.form-control-disabled]=\"disabled\">\r\n <div class=\"text-to-show\" [appTooltip]=\"valueToShow | uploadedFilenameCutoff\">{{ valueToShow |\r\n uploadedFilenameCutoff}}</div>\r\n </div>\r\n <div class=\"upload-icon-wrapper\" (click)=\"onClickUpload()\" [class.opacity0]=\"readonly\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-upload\"></i>\r\n </div>\r\n </div>\r\n @if (!!valueToShow && !!!touched) {\r\n <div class=\"download-icon-wrapper\" (click)=\"onClickDownload()\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-download\"></i>\r\n </div>\r\n </div>\r\n }\r\n\r\n <div class=\"dropdown\" [ngStyle]=\"{ height: dropdownHeight + 'px', padding: dropdownPadding + 'px' }\">\r\n @if (previewable) {\r\n <img [src]=\"valueToShow | mapAttachmentToServer\" imageErrorResolver />\r\n }\r\n </div>\r\n\r\n</div>", styles: [".core-attachment-container{--height: 35px;display:block;position:relative;background-color:#fff;width:100%}.core-attachment-container .dropdown{box-sizing:border-box;display:block;position:absolute;width:100%;overflow:hidden;background-color:#fff;font-size:15px;box-shadow:0 1rem 3rem #0000002e;transition:height .25s linear;z-index:calc(var(--max-z-index) + 1)}.core-attachment-container .dropdown img{transform:none;max-width:100%;max-height:170px}.core-attachment-container .dropdown img:hover{transform:none}.core-attachment-container #rawInput{display:none}.core-attachment-container .text-to-show{width:calc(100% - 60px);height:calc(var(--height) - 12px);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;transform:translate(30px)}.core-attachment-container>.upload-icon-wrapper{right:3px;top:3px}.core-attachment-container>.download-icon-wrapper{left:3px;top:3px}.core-attachment-container>.upload-icon-wrapper,.core-attachment-container>.download-icon-wrapper{display:block;position:absolute;background-color:#e9e9e9;width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);border-radius:.275rem;cursor:pointer}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper,.core-attachment-container>.download-icon-wrapper>.flex-wrapper{width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);display:flex;align-items:center;justify-content:center}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper>i,.core-attachment-container>.download-icon-wrapper>.flex-wrapper>i{color:#848484;font-size:20px}.core-attachment-container .source-container{display:block;background-color:var(--color-bg-main);padding:var(--size-layout-block-cell-spacing);max-width:1200px;max-height:800px;overflow:hidden;border-radius:10px}.core-attachment-container .source-container i{font-size:20px;cursor:pointer;float:right}.core-attachment-container .source-container.d-none{display:none}.core-attachment-container .source-container>caption{display:block;width:100%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i6.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: TooltipDirective, selector: "[appTooltip]", inputs: ["appTooltip", "showAnyway", "position"] }, { kind: "pipe", type: UploadedFilenameCutoffPipe, name: "uploadedFilenameCutoff" }, { kind: "pipe", type: MapAttachmentToServerPipe, name: "mapAttachmentToServer" }] }); }
165
+ ], viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true }, { propertyName: "rawInput", first: true, predicate: ["rawInput"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div #container class=\"core-attachment-container\">\r\n <input id=\"rawInput\" #rawInput type=\"file\" (change)=\"handleInputChange($event)\">\r\n <div class=\"form-control\" type=\"text\" ngDefaultControl [(ngModel)]=\"valueToShow\" readonly\r\n [class.form-control-disabled]=\"disabled\">\r\n <div class=\"text-to-show\" [appTooltip]=\"valueToShow | uploadedFilenameCutoff\" [ngStyle]=\"{\r\n transform: (!!valueToShow && !!!touched) ? 'translate(30px)' : 'translate(0px)'\r\n }\">\r\n {{ valueToShow | uploadedFilenameCutoff }}\r\n </div>\r\n @if (!!valueToShow) {\r\n <i class=\"feather-x\" (click)=\"clear()\"></i>\r\n }\r\n </div>\r\n <div class=\"upload-icon-wrapper\" (click)=\"onClickUpload()\" [class.opacity0]=\"readonly\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-upload\"></i>\r\n </div>\r\n </div>\r\n @if (!!valueToShow && !!!touched) {\r\n <div class=\"download-icon-wrapper\" (click)=\"onClickDownload()\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-download\"></i>\r\n </div>\r\n </div>\r\n }\r\n\r\n <div class=\"dropdown\" [ngStyle]=\"{ height: dropdownHeight + 'px', padding: dropdownPadding + 'px', zIndex: dropdownZIndex }\">\r\n @if (previewable) {\r\n <img [src]=\"valueToShow | mapAttachmentToServer\" imageErrorResolver />\r\n }\r\n </div>\r\n\r\n</div>", styles: [".core-attachment-container{--height: 35px;display:block;position:relative;background-color:#fff;width:100%}.core-attachment-container .dropdown{box-sizing:border-box;display:block;position:absolute;width:100%;overflow:hidden;background-color:#fff;font-size:15px;box-shadow:0 1rem 3rem #0000002e;transition:height .25s linear;z-index:calc(var(--max-z-index) + 1)}.core-attachment-container .dropdown img{transform:none;max-width:100%;max-height:170px}.core-attachment-container .dropdown img:hover{transform:none}.core-attachment-container #rawInput{display:none}.core-attachment-container .text-to-show{width:calc(100% - 70px);height:calc(var(--height) - 12px);overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.core-attachment-container i.feather-x{display:block;position:absolute;right:35px;top:5px;z-index:1;font-size:24px;color:#d3d3d3;cursor:pointer}.core-attachment-container>.upload-icon-wrapper{right:3px;top:3px}.core-attachment-container>.download-icon-wrapper{left:3px;top:3px}.core-attachment-container>.upload-icon-wrapper,.core-attachment-container>.download-icon-wrapper{display:block;position:absolute;background-color:#e9e9e9;width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);border-radius:.275rem;cursor:pointer}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper,.core-attachment-container>.download-icon-wrapper>.flex-wrapper{width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);display:flex;align-items:center;justify-content:center}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper>i,.core-attachment-container>.download-icon-wrapper>.flex-wrapper>i{color:#848484;font-size:20px}.core-attachment-container .source-container{display:block;background-color:var(--color-bg-main);padding:var(--size-layout-block-cell-spacing);max-width:1200px;max-height:800px;overflow:hidden;border-radius:10px}.core-attachment-container .source-container i{font-size:20px;cursor:pointer;float:right}.core-attachment-container .source-container.d-none{display:none}.core-attachment-container .source-container>caption{display:block;width:100%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i6.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i7.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: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: TooltipDirective, selector: "[appTooltip]", inputs: ["appTooltip", "showAnyway", "position"] }, { kind: "pipe", type: UploadedFilenameCutoffPipe, name: "uploadedFilenameCutoff" }, { kind: "pipe", type: MapAttachmentToServerPipe, name: "mapAttachmentToServer" }] }); }
148
166
  }
149
167
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: CoreAttachmentComponent, decorators: [{
150
168
  type: Component,
@@ -159,10 +177,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.3", ngImpor
159
177
  {
160
178
  provide: NG_VALUE_ACCESSOR,
161
179
  multi: true,
162
- useExisting: forwardRef(() => CoreAttachmentComponent),
180
+ useExisting: CoreAttachmentComponent,
163
181
  }
164
- ], template: "<div #container class=\"core-attachment-container\">\r\n <input id=\"rawInput\" #rawInput type=\"file\" (change)=\"handleInputChange($event)\">\r\n <div class=\"form-control\" type=\"text\" [(ngModel)]=\"valueToShow\" readonly [class.form-control-disabled]=\"disabled\">\r\n <div class=\"text-to-show\" [appTooltip]=\"valueToShow | uploadedFilenameCutoff\">{{ valueToShow |\r\n uploadedFilenameCutoff}}</div>\r\n </div>\r\n <div class=\"upload-icon-wrapper\" (click)=\"onClickUpload()\" [class.opacity0]=\"readonly\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-upload\"></i>\r\n </div>\r\n </div>\r\n @if (!!valueToShow && !!!touched) {\r\n <div class=\"download-icon-wrapper\" (click)=\"onClickDownload()\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-download\"></i>\r\n </div>\r\n </div>\r\n }\r\n\r\n <div class=\"dropdown\" [ngStyle]=\"{ height: dropdownHeight + 'px', padding: dropdownPadding + 'px' }\">\r\n @if (previewable) {\r\n <img [src]=\"valueToShow | mapAttachmentToServer\" imageErrorResolver />\r\n }\r\n </div>\r\n\r\n</div>", styles: [".core-attachment-container{--height: 35px;display:block;position:relative;background-color:#fff;width:100%}.core-attachment-container .dropdown{box-sizing:border-box;display:block;position:absolute;width:100%;overflow:hidden;background-color:#fff;font-size:15px;box-shadow:0 1rem 3rem #0000002e;transition:height .25s linear;z-index:calc(var(--max-z-index) + 1)}.core-attachment-container .dropdown img{transform:none;max-width:100%;max-height:170px}.core-attachment-container .dropdown img:hover{transform:none}.core-attachment-container #rawInput{display:none}.core-attachment-container .text-to-show{width:calc(100% - 60px);height:calc(var(--height) - 12px);overflow:hidden;white-space:nowrap;text-overflow:ellipsis;transform:translate(30px)}.core-attachment-container>.upload-icon-wrapper{right:3px;top:3px}.core-attachment-container>.download-icon-wrapper{left:3px;top:3px}.core-attachment-container>.upload-icon-wrapper,.core-attachment-container>.download-icon-wrapper{display:block;position:absolute;background-color:#e9e9e9;width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);border-radius:.275rem;cursor:pointer}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper,.core-attachment-container>.download-icon-wrapper>.flex-wrapper{width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);display:flex;align-items:center;justify-content:center}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper>i,.core-attachment-container>.download-icon-wrapper>.flex-wrapper>i{color:#848484;font-size:20px}.core-attachment-container .source-container{display:block;background-color:var(--color-bg-main);padding:var(--size-layout-block-cell-spacing);max-width:1200px;max-height:800px;overflow:hidden;border-radius:10px}.core-attachment-container .source-container i{font-size:20px;cursor:pointer;float:right}.core-attachment-container .source-container.d-none{display:none}.core-attachment-container .source-container>caption{display:block;width:100%}\n"] }]
165
- }], ctorParameters: () => [{ type: i0.Injector }, { type: i1.AlertService }, { type: i2.MultiLanguageService }, { type: i3.AppConfigService }, { type: i4.UploadedFileService }, { type: i0.Renderer2 }], propDecorators: { assignTo: [{
182
+ ], template: "<div #container class=\"core-attachment-container\">\r\n <input id=\"rawInput\" #rawInput type=\"file\" (change)=\"handleInputChange($event)\">\r\n <div class=\"form-control\" type=\"text\" ngDefaultControl [(ngModel)]=\"valueToShow\" readonly\r\n [class.form-control-disabled]=\"disabled\">\r\n <div class=\"text-to-show\" [appTooltip]=\"valueToShow | uploadedFilenameCutoff\" [ngStyle]=\"{\r\n transform: (!!valueToShow && !!!touched) ? 'translate(30px)' : 'translate(0px)'\r\n }\">\r\n {{ valueToShow | uploadedFilenameCutoff }}\r\n </div>\r\n @if (!!valueToShow) {\r\n <i class=\"feather-x\" (click)=\"clear()\"></i>\r\n }\r\n </div>\r\n <div class=\"upload-icon-wrapper\" (click)=\"onClickUpload()\" [class.opacity0]=\"readonly\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-upload\"></i>\r\n </div>\r\n </div>\r\n @if (!!valueToShow && !!!touched) {\r\n <div class=\"download-icon-wrapper\" (click)=\"onClickDownload()\">\r\n <div class=\"flex-wrapper\">\r\n <i class=\"feather-download\"></i>\r\n </div>\r\n </div>\r\n }\r\n\r\n <div class=\"dropdown\" [ngStyle]=\"{ height: dropdownHeight + 'px', padding: dropdownPadding + 'px', zIndex: dropdownZIndex }\">\r\n @if (previewable) {\r\n <img [src]=\"valueToShow | mapAttachmentToServer\" imageErrorResolver />\r\n }\r\n </div>\r\n\r\n</div>", styles: [".core-attachment-container{--height: 35px;display:block;position:relative;background-color:#fff;width:100%}.core-attachment-container .dropdown{box-sizing:border-box;display:block;position:absolute;width:100%;overflow:hidden;background-color:#fff;font-size:15px;box-shadow:0 1rem 3rem #0000002e;transition:height .25s linear;z-index:calc(var(--max-z-index) + 1)}.core-attachment-container .dropdown img{transform:none;max-width:100%;max-height:170px}.core-attachment-container .dropdown img:hover{transform:none}.core-attachment-container #rawInput{display:none}.core-attachment-container .text-to-show{width:calc(100% - 70px);height:calc(var(--height) - 12px);overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.core-attachment-container i.feather-x{display:block;position:absolute;right:35px;top:5px;z-index:1;font-size:24px;color:#d3d3d3;cursor:pointer}.core-attachment-container>.upload-icon-wrapper{right:3px;top:3px}.core-attachment-container>.download-icon-wrapper{left:3px;top:3px}.core-attachment-container>.upload-icon-wrapper,.core-attachment-container>.download-icon-wrapper{display:block;position:absolute;background-color:#e9e9e9;width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);border-radius:.275rem;cursor:pointer}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper,.core-attachment-container>.download-icon-wrapper>.flex-wrapper{width:calc(var(--height) - 6px);height:calc(var(--height) - 6px);display:flex;align-items:center;justify-content:center}.core-attachment-container>.upload-icon-wrapper>.flex-wrapper>i,.core-attachment-container>.download-icon-wrapper>.flex-wrapper>i{color:#848484;font-size:20px}.core-attachment-container .source-container{display:block;background-color:var(--color-bg-main);padding:var(--size-layout-block-cell-spacing);max-width:1200px;max-height:800px;overflow:hidden;border-radius:10px}.core-attachment-container .source-container i{font-size:20px;cursor:pointer;float:right}.core-attachment-container .source-container.d-none{display:none}.core-attachment-container .source-container>caption{display:block;width:100%}\n"] }]
183
+ }], ctorParameters: () => [{ type: i0.Injector }, { type: i1.AlertService }, { type: i2.MultiLanguageService }, { type: i3.AppConfigService }, { type: i4.UploadedFileService }, { type: i0.Renderer2 }, { type: i5.DomService }], propDecorators: { assignTo: [{
166
184
  type: Input
167
185
  }], readonly: [{
168
186
  type: Input
@@ -175,4 +193,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.3", ngImpor
175
193
  type: ViewChild,
176
194
  args: ['rawInput']
177
195
  }] } });
178
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"core-attachment.component.js","sourceRoot":"","sources":["../../../../../../../../projects/ngx-histaff-alpha/src/lib/app/libraries/core-attachment/core-attachment/core-attachment.component.ts","../../../../../../../../projects/ngx-histaff-alpha/src/lib/app/libraries/core-attachment/core-attachment/core-attachment.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, Renderer2, ElementRef, Injector, Input, OnChanges, OnInit, SimpleChanges, ViewChild, isDevMode, forwardRef } from '@angular/core';\r\nimport { CoreFormControlBaseComponent } from '../../core-form-control-base/core-form-control-base.component';\r\nimport { FormsModule, NG_VALUE_ACCESSOR, NgControl } from '@angular/forms';\r\nimport { AlertService } from '../../alert/alert.service';\r\nimport { MultiLanguageService } from '../../../services/multi-language.service';\r\nimport { AppConfigService } from '../../../services/app-config.service';\r\nimport { alertOptions, noneAutoClosedAlertOptions } from '../../../constants/alertOptions';\r\nimport { attachmentOptions } from '../../../constants/attachmentOptions';\r\nimport { EnumTranslateKey } from \"alpha-global-constants\";\r\nimport { blob_to_base64_script } from '../../../../assets/js/blob2base64_wk';\r\nimport { TooltipDirective } from '../../tooltip/tooltip.directive';\r\nimport { CommonModule } from '@angular/common';\r\n\r\nimport { coreFileUtils } from '../../../../assets/js/coreFileUtils';\r\n\r\nimport { UploadedFilenameCutoffPipe } from '../../../app-pipes/uploaded-filename-cutoff.pipe';\r\nimport { UploadedFileService } from '../../../services/uploaded-file.service';\r\nimport { MapAttachmentToServerPipe } from '../../../app-pipes/map-attachment-to-server.pipe';\r\nimport { ImageErrorResolverDirective } from '../../../directives/image-error-resolver.directive';\r\n\r\nexport interface ICoreAttachment {\r\n  assignTo: string;\r\n  serverFileName?: string;\r\n  clientFileName: string;\r\n  clientFileType: string;\r\n  clientFileData: string;\r\n}\r\n\r\n@Component({\r\n  selector: 'core-attachment',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    FormsModule,\r\n    TooltipDirective,\r\n    UploadedFilenameCutoffPipe,\r\n    MapAttachmentToServerPipe,\r\n    ImageErrorResolverDirective\r\n  ],\r\n  templateUrl: './core-attachment.component.html',\r\n  styleUrls: ['./core-attachment.component.scss'],\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      multi: true,\r\n      useExisting: forwardRef(() => CoreAttachmentComponent),\r\n    }\r\n  ]\r\n})\r\nexport class CoreAttachmentComponent extends CoreFormControlBaseComponent implements OnChanges, OnInit, AfterViewInit {\r\n  @Input() assignTo!: string;\r\n  @Input() override readonly!: boolean;\r\n  @Input() valueToShow!: string;\r\n\r\n  @ViewChild('container') container!: ElementRef;\r\n\r\n  previewable!: boolean;\r\n\r\n  override value!: ICoreAttachment | null;\r\n  override onChange!: (_: ICoreAttachment | null) => void;\r\n\r\n  override writeValue(obj: ICoreAttachment | null): void {\r\n    this.valueToShow = obj?.serverFileName!\r\n  }\r\n\r\n  ngControl!: NgControl;\r\n\r\n  worker!: Worker;\r\n\r\n  @ViewChild('rawInput') rawInput!: ElementRef;\r\n\r\n  listenerFn!: () => void;\r\n\r\n  dropdownHeight: number = 0;\r\n  dropdownPadding: number = 0;\r\n\r\n  constructor(\r\n    private injector: Injector,\r\n    private alertService: AlertService,\r\n    private mls: MultiLanguageService,\r\n    private appConfigService: AppConfigService,\r\n    private uploadedFileService: UploadedFileService,\r\n    private renderer: Renderer2,\r\n  ) {\r\n    super()\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['valueToShow']) {\r\n      this.valueToShow = changes['valueToShow']?.currentValue;\r\n      this.previewable = this.uploadedFileService.canPreviewAsImage(this.valueToShow);\r\n    }\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    setTimeout(() => this.ngControl = this.injector.get(NgControl), 200);\r\n\r\n    if (!!!this.assignTo && !!!this.readonly && isDevMode()) {\r\n      this.alertService.error(`\r\n      CoreAttachmentComponent required input 'assignTo'\r\n      `, noneAutoClosedAlertOptions)\r\n    }\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    setTimeout(() => {\r\n      this.listenerFn = this.renderer.listen('window', 'click', (e: Event) => {\r\n        if (this.container && !!!this.container.nativeElement.contains(e.target)) {\r\n          this.dropdownHeight = 0;\r\n          this.dropdownPadding = 0;\r\n        }\r\n      });\r\n    })\r\n  }\r\n\r\n  onClickUpload(): void {\r\n    if (this.disabled || this.readonly) return;\r\n    this.dropdownHeight = 0;\r\n    this.dropdownPadding = 0;\r\n    this.rawInput.nativeElement.dispatchEvent(new MouseEvent('click'));\r\n  }\r\n\r\n  onClickDownload(): void {\r\n\r\n    if (this.uploadedFileService.canPreviewAsImage(this.valueToShow)) {\r\n\r\n      this.dropdownHeight = this.dropdownHeight === 0 ? 200 : 0;\r\n      this.dropdownPadding = this.dropdownPadding === 0 ? 15 : 0;\r\n\r\n    } else {\r\n      var uri = this.appConfigService.BASE_URL + '/' + this.appConfigService.STATIC_FOLDER + '/attachments/' + this.valueToShow\r\n      fetch(uri)\r\n        .then(response => response.blob())\r\n        .then(blob => {\r\n          const link = document.createElement(\"a\");\r\n          link.href = URL.createObjectURL(blob);\r\n          link.download = this.valueToShow.substring(52);\r\n          link.click();\r\n        })\r\n        .catch(console.error);\r\n    }\r\n\r\n  }\r\n\r\n  isPreviewable(): boolean {\r\n    return this.previewable\r\n  }\r\n\r\n  handleInputChange(e: any) {\r\n\r\n    if (!!e.target.files.length) {\r\n      const file = e.target.files[0];\r\n\r\n      if (file.size > attachmentOptions.maxFileSize) {\r\n        this.alertService.error(`${this.mls.trans(EnumTranslateKey.UI_CORE_ATTACHMENT_MAXIMUM_ATTACHMENT_FILE_SIZE_ERROR)} (${attachmentOptions.maxFileSize} bytes)`, alertOptions);\r\n        return;\r\n      }\r\n\r\n      console.log(\"file\", file)\r\n      const clientFileName = file.name\r\n      const clientFileType = file.type\r\n      let clientFileData = \"\"\r\n\r\n      const blob = new Blob([file]);\r\n      if (typeof Worker !== 'undefined') {\r\n        this.worker = new Worker(blob_to_base64_script)\r\n        this.worker.addEventListener(\"message\", e => {\r\n\r\n          console.log(\"worker message\", e)\r\n          clientFileData = e.data\r\n          const index = (clientFileData as string).indexOf('base64,');\r\n          if (index >= 0) {\r\n            clientFileData = (clientFileData as string).substring(index + 7);\r\n          }\r\n          this.onChange({\r\n            assignTo: this.assignTo,\r\n            serverFileName: this.value?.serverFileName,\r\n            clientFileName,\r\n            clientFileType,\r\n            clientFileData,\r\n          });\r\n          this.valueToShow = clientFileName;\r\n          this.markAsTouched();\r\n        })\r\n        this.worker.postMessage(blob);\r\n      } else {\r\n        coreFileUtils.prototype.blobToBase64(blob).then((clientFileData: any) => {\r\n          const index = (clientFileData as string).indexOf('base64,');\r\n          if (index >= 0) {\r\n            clientFileData = (clientFileData as string).substring(index + 7);\r\n          }\r\n          this.onChange({\r\n            assignTo: this.assignTo,\r\n            serverFileName: this.value?.serverFileName,\r\n            clientFileName,\r\n            clientFileType,\r\n            clientFileData,\r\n          });\r\n          this.valueToShow = clientFileName;\r\n          this.markAsTouched();\r\n        });\r\n      }\r\n    }\r\n  }\r\n}\r\n\r\n\r\n","<div #container class=\"core-attachment-container\">\r\n    <input id=\"rawInput\" #rawInput type=\"file\" (change)=\"handleInputChange($event)\">\r\n    <div class=\"form-control\" type=\"text\" [(ngModel)]=\"valueToShow\" readonly [class.form-control-disabled]=\"disabled\">\r\n        <div class=\"text-to-show\" [appTooltip]=\"valueToShow | uploadedFilenameCutoff\">{{ valueToShow |\r\n            uploadedFilenameCutoff}}</div>\r\n    </div>\r\n    <div class=\"upload-icon-wrapper\" (click)=\"onClickUpload()\" [class.opacity0]=\"readonly\">\r\n        <div class=\"flex-wrapper\">\r\n            <i class=\"feather-upload\"></i>\r\n        </div>\r\n    </div>\r\n    @if (!!valueToShow && !!!touched) {\r\n    <div class=\"download-icon-wrapper\" (click)=\"onClickDownload()\">\r\n        <div class=\"flex-wrapper\">\r\n            <i class=\"feather-download\"></i>\r\n        </div>\r\n    </div>\r\n    }\r\n\r\n    <div class=\"dropdown\" [ngStyle]=\"{ height: dropdownHeight + 'px', padding: dropdownPadding + 'px' }\">\r\n        @if (previewable) {\r\n        <img [src]=\"valueToShow | mapAttachmentToServer\" imageErrorResolver />\r\n        }\r\n    </div>\r\n\r\n</div>"]}
196
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"core-attachment.component.js","sourceRoot":"","sources":["../../../../../../../../projects/ngx-histaff-alpha/src/lib/app/libraries/core-attachment/core-attachment/core-attachment.component.ts","../../../../../../../../projects/ngx-histaff-alpha/src/lib/app/libraries/core-attachment/core-attachment/core-attachment.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, Renderer2, ElementRef, Injector, Input, OnChanges, OnInit, SimpleChanges, ViewChild, isDevMode } from '@angular/core';\r\nimport { CoreFormControlBaseComponent } from '../../core-form-control-base/core-form-control-base.component';\r\nimport { FormsModule, NG_VALUE_ACCESSOR, NgControl, ReactiveFormsModule } from '@angular/forms';\r\nimport { AlertService } from '../../alert/alert.service';\r\nimport { MultiLanguageService } from '../../../services/multi-language.service';\r\nimport { AppConfigService } from '../../../services/app-config.service';\r\nimport { alertOptions, noneAutoClosedAlertOptions } from '../../../constants/alertOptions';\r\nimport { attachmentOptions } from '../../../constants/attachmentOptions';\r\nimport { EnumTranslateKey } from \"alpha-global-constants\";\r\nimport { blob_to_base64_script } from '../../../../assets/js/blob2base64_wk';\r\nimport { TooltipDirective } from '../../tooltip/tooltip.directive';\r\nimport { CommonModule } from '@angular/common';\r\n\r\nimport { coreFileUtils } from '../../../../assets/js/coreFileUtils';\r\n\r\nimport { UploadedFilenameCutoffPipe } from '../../../app-pipes/uploaded-filename-cutoff.pipe';\r\nimport { UploadedFileService } from '../../../services/uploaded-file.service';\r\nimport { MapAttachmentToServerPipe } from '../../../app-pipes/map-attachment-to-server.pipe';\r\nimport { ImageErrorResolverDirective } from '../../../directives/image-error-resolver.directive';\r\nimport { DomService } from '../../services/dom.service';\r\n\r\nexport interface ICoreAttachment {\r\n  assignTo: string;\r\n  serverFileName?: string;\r\n  clientFileName: string;\r\n  clientFileType: string;\r\n  clientFileData: string;\r\n}\r\n\r\n@Component({\r\n  selector: 'core-attachment',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    FormsModule,\r\n    TooltipDirective,\r\n    UploadedFilenameCutoffPipe,\r\n    MapAttachmentToServerPipe,\r\n    ImageErrorResolverDirective\r\n  ],\r\n  templateUrl: './core-attachment.component.html',\r\n  styleUrls: ['./core-attachment.component.scss'],\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      multi: true,\r\n      useExisting: CoreAttachmentComponent,\r\n    }\r\n  ]\r\n})\r\nexport class CoreAttachmentComponent extends CoreFormControlBaseComponent implements OnChanges, OnInit, AfterViewInit {\r\n  ngControl!: NgControl;\r\n\r\n  @Input() assignTo!: string;\r\n  @Input() override readonly!: boolean;\r\n  @Input() valueToShow!: string;\r\n\r\n  @ViewChild('container') container!: ElementRef;\r\n\r\n  previewable!: boolean;\r\n\r\n  override value!: ICoreAttachment | null;\r\n  override onChange!: (_: ICoreAttachment | null) => void;\r\n\r\n  override writeValue(obj: ICoreAttachment | null): void {\r\n    this.valueToShow = obj?.serverFileName || ''\r\n  }\r\n\r\n  worker!: Worker;\r\n\r\n  @ViewChild('rawInput') rawInput!: ElementRef;\r\n\r\n  listenerFn!: () => void;\r\n\r\n  dropdownHeight: number = 0;\r\n  dropdownPadding: number = 0;\r\n  dropdownZIndex: number = 1;\r\n\r\n  constructor(\r\n    private injector: Injector,\r\n    private alertService: AlertService,\r\n    private mls: MultiLanguageService,\r\n    private appConfigService: AppConfigService,\r\n    private uploadedFileService: UploadedFileService,\r\n    private renderer: Renderer2,\r\n    private domService: DomService,\r\n  ) {\r\n    super()\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['valueToShow']) {\r\n      this.valueToShow = changes['valueToShow']?.currentValue;\r\n      this.previewable = this.uploadedFileService.canPreviewAsImage(this.valueToShow);\r\n    }\r\n  }\r\n\r\n  ngOnInit(): void {\r\n\r\n    this.ngControl = this.injector.get(NgControl);\r\n\r\n    if (!!!this.assignTo && !!!this.readonly && isDevMode()) {\r\n      this.alertService.error(`\r\n      CoreAttachmentComponent required input 'assignTo'\r\n      `, noneAutoClosedAlertOptions)\r\n    }\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    setTimeout(() => {\r\n      this.listenerFn = this.renderer.listen('window', 'click', (e: Event) => {\r\n        if (this.container && !!!this.container.nativeElement.contains(e.target)) {\r\n          this.dropdownHeight = 0;\r\n          this.dropdownPadding = 0;\r\n        }\r\n      });\r\n      this.dropdownZIndex = this.domService.getMaxZIndex() + 1;\r\n    })\r\n  }\r\n\r\n  onClickUpload(): void {\r\n    if (this.disabled || this.readonly) return;\r\n    this.dropdownHeight = 0;\r\n    this.dropdownPadding = 0;\r\n    this.rawInput.nativeElement.dispatchEvent(new MouseEvent('click'));\r\n  }\r\n\r\n  onClickDownload(): void {\r\n\r\n    if (this.uploadedFileService.canPreviewAsImage(this.valueToShow)) {\r\n\r\n      this.dropdownHeight = this.dropdownHeight === 0 ? 200 : 0;\r\n      this.dropdownPadding = this.dropdownPadding === 0 ? 15 : 0;\r\n\r\n    } else {\r\n      var uri = this.appConfigService.BASE_URL + '/' + this.appConfigService.STATIC_FOLDER + '/attachments/' + this.valueToShow\r\n      fetch(uri)\r\n        .then(response => response.blob())\r\n        .then(blob => {\r\n          const link = document.createElement(\"a\");\r\n          link.href = URL.createObjectURL(blob);\r\n          link.download = this.valueToShow.substring(52);\r\n          link.click();\r\n        })\r\n        .catch(console.error);\r\n    }\r\n\r\n  }\r\n\r\n  isPreviewable(): boolean {\r\n    return this.previewable\r\n  }\r\n\r\n  handleInputChange(e: any) {\r\n\r\n    if (!!e.target.files.length) {\r\n      const file = e.target.files[0];\r\n\r\n      if (file.size > attachmentOptions.maxFileSize) {\r\n        this.alertService.error(`${this.mls.trans(EnumTranslateKey.UI_CORE_ATTACHMENT_MAXIMUM_ATTACHMENT_FILE_SIZE_ERROR)} (${attachmentOptions.maxFileSize} bytes)`, alertOptions);\r\n        return;\r\n      }\r\n\r\n      console.log(\"file\", file)\r\n      const clientFileName = file.name\r\n      const clientFileType = file.type\r\n      let clientFileData = \"\"\r\n\r\n      const blob = new Blob([file]);\r\n      if (typeof Worker !== 'undefined') {\r\n        this.worker = new Worker(blob_to_base64_script)\r\n        this.worker.addEventListener(\"message\", e => {\r\n\r\n          console.log(\"worker message\", e)\r\n          clientFileData = e.data\r\n          const index = (clientFileData as string).indexOf('base64,');\r\n          if (index >= 0) {\r\n            clientFileData = (clientFileData as string).substring(index + 7);\r\n          }\r\n          this.onChange({\r\n            assignTo: this.assignTo,\r\n            serverFileName: this.value?.serverFileName,\r\n            clientFileName,\r\n            clientFileType,\r\n            clientFileData,\r\n          });\r\n          this.valueToShow = clientFileName;\r\n          this.markAsTouched();\r\n        })\r\n        this.worker.postMessage(blob);\r\n      } else {\r\n        coreFileUtils.prototype.blobToBase64(blob).then((clientFileData: any) => {\r\n          const index = (clientFileData as string).indexOf('base64,');\r\n          if (index >= 0) {\r\n            clientFileData = (clientFileData as string).substring(index + 7);\r\n          }\r\n          this.onChange({\r\n            assignTo: this.assignTo,\r\n            serverFileName: this.value?.serverFileName,\r\n            clientFileName,\r\n            clientFileType,\r\n            clientFileData,\r\n          });\r\n          this.valueToShow = clientFileName;\r\n          this.markAsTouched();\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  clear():void {\r\n    this.value = null;\r\n    this.valueToShow = '';\r\n    this.onChange(null);\r\n    const sibling = this.ngControl.control?.parent?.get(this.assignTo);\r\n    if (!sibling) {\r\n      if (isDevMode()) {\r\n        this.alertService.error(`Chưa khai báo field '${this.assignTo}'`, noneAutoClosedAlertOptions);\r\n      }\r\n    } else {\r\n      sibling.patchValue(null);\r\n    }\r\n  }\r\n\r\n}\r\n\r\n\r\n","<div #container class=\"core-attachment-container\">\r\n    <input id=\"rawInput\" #rawInput type=\"file\" (change)=\"handleInputChange($event)\">\r\n    <div class=\"form-control\" type=\"text\" ngDefaultControl [(ngModel)]=\"valueToShow\" readonly\r\n        [class.form-control-disabled]=\"disabled\">\r\n        <div class=\"text-to-show\" [appTooltip]=\"valueToShow | uploadedFilenameCutoff\" [ngStyle]=\"{\r\n                transform: (!!valueToShow && !!!touched) ? 'translate(30px)' : 'translate(0px)'\r\n            }\">\r\n            {{ valueToShow | uploadedFilenameCutoff }}\r\n        </div>\r\n        @if (!!valueToShow) {\r\n            <i class=\"feather-x\" (click)=\"clear()\"></i>\r\n        }\r\n    </div>\r\n    <div class=\"upload-icon-wrapper\" (click)=\"onClickUpload()\" [class.opacity0]=\"readonly\">\r\n        <div class=\"flex-wrapper\">\r\n            <i class=\"feather-upload\"></i>\r\n        </div>\r\n    </div>\r\n    @if (!!valueToShow && !!!touched) {\r\n    <div class=\"download-icon-wrapper\" (click)=\"onClickDownload()\">\r\n        <div class=\"flex-wrapper\">\r\n            <i class=\"feather-download\"></i>\r\n        </div>\r\n    </div>\r\n    }\r\n\r\n    <div class=\"dropdown\" [ngStyle]=\"{ height: dropdownHeight + 'px', padding: dropdownPadding + 'px', zIndex: dropdownZIndex }\">\r\n        @if (previewable) {\r\n        <img [src]=\"valueToShow | mapAttachmentToServer\" imageErrorResolver />\r\n        }\r\n    </div>\r\n\r\n</div>"]}