@fovestta2/web-angular 1.0.23 → 1.0.25
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/fv-document-field/fv-document-field.component.mjs +38 -6
- package/esm2022/lib/fv-file-selector/fv-file-selector.component.mjs +34 -4
- package/esm2022/lib/fv-image-selector/fv-image-selector.component.mjs +22 -4
- package/esm2022/lib/fv-name-code/fv-name-code.component.mjs +7 -2
- package/fesm2022/fovestta2-web-angular.mjs +97 -12
- package/fesm2022/fovestta2-web-angular.mjs.map +1 -1
- package/lib/fv-document-field/fv-document-field.component.d.ts +2 -0
- package/lib/fv-file-selector/fv-file-selector.component.d.ts +1 -0
- package/lib/fv-image-selector/fv-image-selector.component.d.ts +1 -0
- package/package.json +1 -1
|
@@ -93,13 +93,18 @@ export class FvNameCodeComponent {
|
|
|
93
93
|
this.highlightedIndex = -1;
|
|
94
94
|
}
|
|
95
95
|
writeValue(value) {
|
|
96
|
+
if (value === this.value) {
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
96
99
|
this.value = value;
|
|
97
100
|
const selectedOption = this.options.find(o => o.value === value);
|
|
98
101
|
if (selectedOption) {
|
|
99
102
|
this.searchControl.setValue(`${selectedOption.code} - ${selectedOption.name}`, { emitEvent: false });
|
|
100
103
|
}
|
|
101
104
|
else {
|
|
102
|
-
|
|
105
|
+
if (value === null || value === undefined || value === '') {
|
|
106
|
+
this.searchControl.setValue('', { emitEvent: false });
|
|
107
|
+
}
|
|
103
108
|
}
|
|
104
109
|
}
|
|
105
110
|
registerOnChange(fn) {
|
|
@@ -318,4 +323,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
318
323
|
type: HostListener,
|
|
319
324
|
args: ['keydown', ['$event']]
|
|
320
325
|
}] } });
|
|
321
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fv-name-code.component.js","sourceRoot":"","sources":["../../../../../projects/fv-controls/src/lib/fv-name-code/fv-name-code.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, forwardRef, HostListener, ElementRef, Output, EventEmitter, OnChanges, SimpleChanges } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, ReactiveFormsModule, FormControl } from '@angular/forms';\r\nimport { ValidationSchema } from '@fovestta2/validation-engine';\r\n\r\nexport interface SearchSelectOption {\r\n  code: string;\r\n  name: string;\r\n  value: any;\r\n}\r\n\r\n@Component({\r\n  selector: 'fv-name-code',\r\n  standalone: true,\r\n  imports: [CommonModule, ReactiveFormsModule],\r\n  template: `\r\n    <div class=\"fv-name-code-container\">\r\n      <label *ngIf=\"label\" class=\"fv-label\">\r\n        {{ label }}\r\n        <span *ngIf=\"isRequired()\" class=\"required-asterisk\">*</span>\r\n      </label>\r\n\r\n      <div class=\"search-box-wrapper\">\r\n        <input\r\n          type=\"text\"\r\n          class=\"fv-input\"\r\n          [placeholder]=\"(searchControl.value || isOpen) ? '' : (placeholder || 'Search by Code or Name')\"\r\n          [formControl]=\"searchControl\"\r\n          (focus)=\"openDropdown()\"\r\n          (blur)=\"onBlur()\"\r\n          [class.error]=\"isInvalid()\"\r\n        />\r\n      </div>\r\n\r\n      <div class=\"dropdown-list\" *ngIf=\"isOpen && !disabled\">\r\n        <div \r\n          class=\"dropdown-item\" \r\n          *ngFor=\"let option of filteredOptions; let i = index\"\r\n          [class.highlighted]=\"i === highlightedIndex\"\r\n          (click)=\"selectOption(option)\"\r\n        >\r\n          <span class=\"code\">{{ option.code }}</span>\r\n          <span class=\"name\">{{ option.name }}</span>\r\n        </div>\r\n        <div class=\"no-results\" *ngIf=\"filteredOptions.length === 0\">\r\n          No results found\r\n        </div>\r\n      </div>\r\n\r\n      <div *ngIf=\"isInvalid()\" class=\"error-message\">\r\n        ⚠ {{ getErrorMessage() }}\r\n      </div>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    @import url('https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;500;600;700&display=swap');\r\n\r\n    * { font-family: 'Poppins', sans-serif; }\r\n\r\n    .fv-name-code-container {\r\n      position: relative;\r\n      width: 100%;\r\n      margin-bottom: 0;\r\n    }\r\n\r\n    .fv-label {\r\n      display: block;\r\n      font-size: 14px;\r\n      font-weight: 600;\r\n      color: #151D48;\r\n      margin-bottom: 6px;\r\n    }\r\n\r\n    .required-asterisk {\r\n      color: #e74c3c;\r\n      margin-left: 2px;\r\n    }\r\n\r\n    .search-box-wrapper {\r\n      position: relative;\r\n      display: flex;\r\n      align-items: center;\r\n    }\r\n\r\n    .fv-input {\r\n      width: 100%;\r\n      padding: 5px 10px;\r\n      height: 34px;\r\n      border: 1px solid #8CBBA8;\r\n      border-radius: 8px;\r\n      font-size: 14px;\r\n      color: #1f2b41;\r\n      outline: none;\r\n      transition: all 0.2s;\r\n      background-color: #fff;\r\n      box-sizing: border-box;\r\n      font-family: 'Poppins', sans-serif;\r\n      min-width: 0;\r\n    }\r\n\r\n    .fv-input:focus {\r\n      border-color: #3498db;\r\n      box-shadow: 0 0 0 3px rgba(52, 152, 219, 0.1);\r\n    }\r\n\r\n    .fv-input.error {\r\n      border-color: #e74c3c;\r\n    }\r\n\r\n    .fv-input:disabled {\r\n      background-color: #f5f5f5;\r\n      cursor: not-allowed;\r\n    }\r\n\r\n    .dropdown-list {\r\n      position: absolute;\r\n      top: 100%;\r\n      left: 0;\r\n      min-width: 100%;\r\n      width: max-content;\r\n      max-width: 300px;\r\n      background: white;\r\n      border: 1px solid #ddd;\r\n      border-radius: 8px;\r\n      margin-top: 4px;\r\n      max-height: 200px;\r\n      overflow-y: auto;\r\n      overflow-x: hidden;\r\n      z-index: 1000;\r\n      box-shadow: 0 4px 6px rgba(0,0,0,0.1);\r\n    }\r\n\r\n    .dropdown-item {\r\n      padding: 10px 12px;\r\n      display: flex;\r\n      justify-content: space-between;\r\n      align-items: center;\r\n      cursor: pointer;\r\n      border-bottom: 1px solid #f0f0f0;\r\n      gap: 12px;\r\n    }\r\n\r\n    .dropdown-item:last-child {\r\n      border-bottom: none;\r\n    }\r\n\r\n    .dropdown-item:hover, .dropdown-item.highlighted {\r\n      background-color: #f0f7ff;\r\n    }\r\n\r\n    .code {\r\n      color: #0056b3;\r\n      font-weight: 600;\r\n      font-size: 14px;\r\n      white-space: nowrap;\r\n    }\r\n\r\n    .name {\r\n      color: #333;\r\n      font-size: 14px;\r\n      white-space: nowrap;\r\n      overflow: hidden;\r\n      text-overflow: ellipsis;\r\n      text-align: right;\r\n    }\r\n\r\n    .no-results {\r\n      padding: 10px;\r\n      color: #999;\r\n      text-align: center;\r\n      font-size: 13px;\r\n    }\r\n\r\n    .error-message {\r\n      color: #e74c3c;\r\n      font-size: 12px;\r\n      margin-top: 4px;\r\n      display: flex;\r\n      align-items: center;\r\n      gap: 4px;\r\n    }\r\n  `],\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => FvNameCodeComponent),\r\n      multi: true\r\n    }\r\n  ]\r\n})\r\nexport class FvNameCodeComponent implements ControlValueAccessor, OnChanges {\r\n  @Input() label: string = '';\r\n  @Input() placeholder: string = '';\r\n  @Input() options: SearchSelectOption[] = [];\r\n  @Input() schema?: ValidationSchema;\r\n  @Input() control?: FormControl;\r\n  @Input() disabled: boolean = false;\r\n\r\n  @Output() valueChange = new EventEmitter<any>();\r\n\r\n  searchControl = new FormControl('');\r\n  filteredOptions: SearchSelectOption[] = [];\r\n  isOpen: boolean = false;\r\n  highlightedIndex: number = -1;\r\n\r\n  private onChange: (value: any) => void = () => { };\r\n  private onTouched: () => void = () => { };\r\n\r\n  value: any = null;\r\n\r\n  ngOnChanges(changes: SimpleChanges) {\r\n    if (changes['options']) {\r\n      this.filteredOptions = this.options;\r\n      if (this.value !== null && this.value !== undefined) {\r\n        this.writeValue(this.value);\r\n      }\r\n    }\r\n  }\r\n\r\n  ngOnInit() {\r\n    this.filteredOptions = this.options;\r\n\r\n    this.searchControl.valueChanges.subscribe(term => {\r\n      this.filterOptions(term || '');\r\n    });\r\n\r\n    if (this.control) {\r\n      if (this.control.value !== null && this.control.value !== undefined) {\r\n        this.writeValue(this.control.value);\r\n      }\r\n      this.control.valueChanges.subscribe(val => {\r\n        this.writeValue(val);\r\n      });\r\n      if (this.control.disabled !== this.disabled) {\r\n        this.setDisabledState(this.control.disabled);\r\n      }\r\n    }\r\n  }\r\n\r\n  filterOptions(term: string) {\r\n    if (!term) {\r\n      this.filteredOptions = this.options;\r\n      return;\r\n    }\r\n    const lowerTerm = term.toLowerCase();\r\n    this.filteredOptions = this.options.filter(opt =>\r\n      opt.code.toLowerCase().includes(lowerTerm) ||\r\n      opt.name.toLowerCase().includes(lowerTerm)\r\n    );\r\n  }\r\n\r\n  openDropdown() {\r\n    if (!this.disabled) {\r\n      this.isOpen = true;\r\n      this.filterOptions(this.searchControl.value || '');\r\n      this.highlightedIndex = this.filteredOptions.length === 1 ? 0 : -1;\r\n    }\r\n  }\r\n\r\n  onBlur() {\r\n    this.onTouched();\r\n  }\r\n\r\n  closeDropdown() {\r\n    this.isOpen = false;\r\n    this.highlightedIndex = -1;\r\n    const selectedOption = this.options.find(o => o.value === this.value);\r\n    if (selectedOption) {\r\n      this.searchControl.setValue(`${selectedOption.code} - ${selectedOption.name}`, { emitEvent: false });\r\n    } else {\r\n      if (!this.value) {\r\n        this.searchControl.setValue('', { emitEvent: false });\r\n      }\r\n    }\r\n    this.onTouched();\r\n  }\r\n\r\n  selectOption(option: SearchSelectOption) {\r\n    if (!option) return;\r\n    this.value = option.value;\r\n    this.onChange(this.value);\r\n\r\n    if (this.control) {\r\n      this.control.setValue(this.value);\r\n      this.control.markAsDirty();\r\n      this.control.markAsTouched();\r\n    }\r\n\r\n    this.searchControl.setValue(`${option.code} - ${option.name}`, { emitEvent: false });\r\n    this.valueChange.emit(this.value);\r\n    this.isOpen = false;\r\n    this.highlightedIndex = -1;\r\n  }\r\n\r\n  writeValue(value: any): void {\r\n    this.value = value;\r\n    const selectedOption = this.options.find(o => o.value === value);\r\n    if (selectedOption) {\r\n      this.searchControl.setValue(`${selectedOption.code} - ${selectedOption.name}`, { emitEvent: false });\r\n    } else {\r\n      this.searchControl.setValue('', { emitEvent: false });\r\n    }\r\n  }\r\n\r\n  registerOnChange(fn: any): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: any): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.disabled = isDisabled;\r\n    if (isDisabled) {\r\n      this.searchControl.disable({ emitEvent: false });\r\n    } else {\r\n      this.searchControl.enable({ emitEvent: false });\r\n    }\r\n  }\r\n\r\n  isInvalid(): boolean {\r\n    if (this.control) {\r\n      return this.control.invalid && (this.control.dirty || this.control.touched);\r\n    }\r\n    return false;\r\n  }\r\n\r\n  isRequired(): boolean {\r\n    return this.schema?.rules?.some(r => r.name === 'required') || false;\r\n  }\r\n\r\n  getErrorMessage(): string {\r\n    if (!this.control || !this.control.errors || !this.schema) return '';\r\n\r\n    for (const errorKey of this.schema.errorPriority) {\r\n      if (this.control.hasError(errorKey)) {\r\n        const rule = this.schema.rules.find(r => r.name === (errorKey === 'required' ? 'required' : errorKey));\r\n        if (rule) return this.getReadableError(rule.errorKey || '');\r\n      }\r\n    }\r\n    if (this.control.hasError('required')) return 'This field is required';\r\n\r\n    return 'Invalid value';\r\n  }\r\n\r\n  getReadableError(key: string): string {\r\n    const map: { [key: string]: string } = {\r\n      'ERR_REQUIRED': 'This field is required',\r\n      'ERR_MIN_LENGTH': 'Value is too short',\r\n    };\r\n    return map[key] || 'Invalid value';\r\n  }\r\n\r\n  @HostListener('document:click', ['$event'])\r\n  onClickOutside(event: MouseEvent) {\r\n    const clickedInside = (event.target as HTMLElement).closest('.fv-name-code-container');\r\n    if (!clickedInside) {\r\n      this.closeDropdown();\r\n    }\r\n  }\r\n\r\n  @HostListener('keydown', ['$event'])\r\n  onKeyDown(event: KeyboardEvent) {\r\n    if (!this.isOpen && (event.key === 'ArrowDown' || event.key === 'ArrowUp' || event.key === 'Enter')) {\r\n      this.openDropdown();\r\n      event.preventDefault();\r\n      return;\r\n    }\r\n\r\n    if (!this.isOpen) return;\r\n\r\n    switch (event.key) {\r\n      case 'ArrowDown':\r\n        event.preventDefault();\r\n        this.highlightedIndex = (this.highlightedIndex + 1) % this.filteredOptions.length;\r\n        this.scrollToHighlighted();\r\n        break;\r\n      case 'ArrowUp':\r\n        event.preventDefault();\r\n        this.highlightedIndex = (this.highlightedIndex - 1 + this.filteredOptions.length) % this.filteredOptions.length;\r\n        this.scrollToHighlighted();\r\n        break;\r\n      case 'Enter':\r\n        event.preventDefault();\r\n        if (this.filteredOptions.length === 1) {\r\n          this.selectOption(this.filteredOptions[0]);\r\n        } else if (this.highlightedIndex >= 0 && this.highlightedIndex < this.filteredOptions.length) {\r\n          this.selectOption(this.filteredOptions[this.highlightedIndex]);\r\n        }\r\n        break;\r\n      case 'Escape':\r\n        this.closeDropdown();\r\n        break;\r\n      case 'Tab':\r\n        this.closeDropdown();\r\n        break;\r\n    }\r\n  }\r\n\r\n  private scrollToHighlighted() {\r\n    const list = document.querySelector('.dropdown-list');\r\n    const item = document.querySelectorAll('.dropdown-item')[this.highlightedIndex];\r\n    if (list && item) {\r\n      const listRect = list.getBoundingClientRect();\r\n      const itemRect = item.getBoundingClientRect();\r\n\r\n      if (itemRect.bottom > listRect.bottom) {\r\n        list.scrollTop += (itemRect.bottom - listRect.bottom);\r\n      } else if (itemRect.top < listRect.top) {\r\n        list.scrollTop -= (listRect.top - itemRect.top);\r\n      }\r\n    }\r\n  }\r\n}\r\n"]}
|
|
326
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fv-name-code.component.js","sourceRoot":"","sources":["../../../../../projects/fv-controls/src/lib/fv-name-code/fv-name-code.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, forwardRef, HostListener, ElementRef, Output, EventEmitter, OnChanges, SimpleChanges } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, ReactiveFormsModule, FormControl } from '@angular/forms';\r\nimport { ValidationSchema } from '@fovestta2/validation-engine';\r\n\r\nexport interface SearchSelectOption {\r\n  code: string;\r\n  name: string;\r\n  value: any;\r\n}\r\n\r\n@Component({\r\n  selector: 'fv-name-code',\r\n  standalone: true,\r\n  imports: [CommonModule, ReactiveFormsModule],\r\n  template: `\r\n    <div class=\"fv-name-code-container\">\r\n      <label *ngIf=\"label\" class=\"fv-label\">\r\n        {{ label }}\r\n        <span *ngIf=\"isRequired()\" class=\"required-asterisk\">*</span>\r\n      </label>\r\n\r\n      <div class=\"search-box-wrapper\">\r\n        <input\r\n          type=\"text\"\r\n          class=\"fv-input\"\r\n          [placeholder]=\"(searchControl.value || isOpen) ? '' : (placeholder || 'Search by Code or Name')\"\r\n          [formControl]=\"searchControl\"\r\n          (focus)=\"openDropdown()\"\r\n          (blur)=\"onBlur()\"\r\n          [class.error]=\"isInvalid()\"\r\n        />\r\n      </div>\r\n\r\n      <div class=\"dropdown-list\" *ngIf=\"isOpen && !disabled\">\r\n        <div \r\n          class=\"dropdown-item\" \r\n          *ngFor=\"let option of filteredOptions; let i = index\"\r\n          [class.highlighted]=\"i === highlightedIndex\"\r\n          (click)=\"selectOption(option)\"\r\n        >\r\n          <span class=\"code\">{{ option.code }}</span>\r\n          <span class=\"name\">{{ option.name }}</span>\r\n        </div>\r\n        <div class=\"no-results\" *ngIf=\"filteredOptions.length === 0\">\r\n          No results found\r\n        </div>\r\n      </div>\r\n\r\n      <div *ngIf=\"isInvalid()\" class=\"error-message\">\r\n        ⚠ {{ getErrorMessage() }}\r\n      </div>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    @import url('https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;500;600;700&display=swap');\r\n\r\n    * { font-family: 'Poppins', sans-serif; }\r\n\r\n    .fv-name-code-container {\r\n      position: relative;\r\n      width: 100%;\r\n      margin-bottom: 0;\r\n    }\r\n\r\n    .fv-label {\r\n      display: block;\r\n      font-size: 14px;\r\n      font-weight: 600;\r\n      color: #151D48;\r\n      margin-bottom: 6px;\r\n    }\r\n\r\n    .required-asterisk {\r\n      color: #e74c3c;\r\n      margin-left: 2px;\r\n    }\r\n\r\n    .search-box-wrapper {\r\n      position: relative;\r\n      display: flex;\r\n      align-items: center;\r\n    }\r\n\r\n    .fv-input {\r\n      width: 100%;\r\n      padding: 5px 10px;\r\n      height: 34px;\r\n      border: 1px solid #8CBBA8;\r\n      border-radius: 8px;\r\n      font-size: 14px;\r\n      color: #1f2b41;\r\n      outline: none;\r\n      transition: all 0.2s;\r\n      background-color: #fff;\r\n      box-sizing: border-box;\r\n      font-family: 'Poppins', sans-serif;\r\n      min-width: 0;\r\n    }\r\n\r\n    .fv-input:focus {\r\n      border-color: #3498db;\r\n      box-shadow: 0 0 0 3px rgba(52, 152, 219, 0.1);\r\n    }\r\n\r\n    .fv-input.error {\r\n      border-color: #e74c3c;\r\n    }\r\n\r\n    .fv-input:disabled {\r\n      background-color: #f5f5f5;\r\n      cursor: not-allowed;\r\n    }\r\n\r\n    .dropdown-list {\r\n      position: absolute;\r\n      top: 100%;\r\n      left: 0;\r\n      min-width: 100%;\r\n      width: max-content;\r\n      max-width: 300px;\r\n      background: white;\r\n      border: 1px solid #ddd;\r\n      border-radius: 8px;\r\n      margin-top: 4px;\r\n      max-height: 200px;\r\n      overflow-y: auto;\r\n      overflow-x: hidden;\r\n      z-index: 1000;\r\n      box-shadow: 0 4px 6px rgba(0,0,0,0.1);\r\n    }\r\n\r\n    .dropdown-item {\r\n      padding: 10px 12px;\r\n      display: flex;\r\n      justify-content: space-between;\r\n      align-items: center;\r\n      cursor: pointer;\r\n      border-bottom: 1px solid #f0f0f0;\r\n      gap: 12px;\r\n    }\r\n\r\n    .dropdown-item:last-child {\r\n      border-bottom: none;\r\n    }\r\n\r\n    .dropdown-item:hover, .dropdown-item.highlighted {\r\n      background-color: #f0f7ff;\r\n    }\r\n\r\n    .code {\r\n      color: #0056b3;\r\n      font-weight: 600;\r\n      font-size: 14px;\r\n      white-space: nowrap;\r\n    }\r\n\r\n    .name {\r\n      color: #333;\r\n      font-size: 14px;\r\n      white-space: nowrap;\r\n      overflow: hidden;\r\n      text-overflow: ellipsis;\r\n      text-align: right;\r\n    }\r\n\r\n    .no-results {\r\n      padding: 10px;\r\n      color: #999;\r\n      text-align: center;\r\n      font-size: 13px;\r\n    }\r\n\r\n    .error-message {\r\n      color: #e74c3c;\r\n      font-size: 12px;\r\n      margin-top: 4px;\r\n      display: flex;\r\n      align-items: center;\r\n      gap: 4px;\r\n    }\r\n  `],\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => FvNameCodeComponent),\r\n      multi: true\r\n    }\r\n  ]\r\n})\r\nexport class FvNameCodeComponent implements ControlValueAccessor, OnChanges {\r\n  @Input() label: string = '';\r\n  @Input() placeholder: string = '';\r\n  @Input() options: SearchSelectOption[] = [];\r\n  @Input() schema?: ValidationSchema;\r\n  @Input() control?: FormControl;\r\n  @Input() disabled: boolean = false;\r\n\r\n  @Output() valueChange = new EventEmitter<any>();\r\n\r\n  searchControl = new FormControl('');\r\n  filteredOptions: SearchSelectOption[] = [];\r\n  isOpen: boolean = false;\r\n  highlightedIndex: number = -1;\r\n\r\n  private onChange: (value: any) => void = () => { };\r\n  private onTouched: () => void = () => { };\r\n\r\n  value: any = null;\r\n\r\n  ngOnChanges(changes: SimpleChanges) {\r\n    if (changes['options']) {\r\n      this.filteredOptions = this.options;\r\n      if (this.value !== null && this.value !== undefined) {\r\n        this.writeValue(this.value);\r\n      }\r\n    }\r\n  }\r\n\r\n  ngOnInit() {\r\n    this.filteredOptions = this.options;\r\n\r\n    this.searchControl.valueChanges.subscribe(term => {\r\n      this.filterOptions(term || '');\r\n    });\r\n\r\n    if (this.control) {\r\n      if (this.control.value !== null && this.control.value !== undefined) {\r\n        this.writeValue(this.control.value);\r\n      }\r\n      this.control.valueChanges.subscribe(val => {\r\n        this.writeValue(val);\r\n      });\r\n      if (this.control.disabled !== this.disabled) {\r\n        this.setDisabledState(this.control.disabled);\r\n      }\r\n    }\r\n  }\r\n\r\n  filterOptions(term: string) {\r\n    if (!term) {\r\n      this.filteredOptions = this.options;\r\n      return;\r\n    }\r\n    const lowerTerm = term.toLowerCase();\r\n    this.filteredOptions = this.options.filter(opt =>\r\n      opt.code.toLowerCase().includes(lowerTerm) ||\r\n      opt.name.toLowerCase().includes(lowerTerm)\r\n    );\r\n  }\r\n\r\n  openDropdown() {\r\n    if (!this.disabled) {\r\n      this.isOpen = true;\r\n      this.filterOptions(this.searchControl.value || '');\r\n      this.highlightedIndex = this.filteredOptions.length === 1 ? 0 : -1;\r\n    }\r\n  }\r\n\r\n  onBlur() {\r\n    this.onTouched();\r\n  }\r\n\r\n  closeDropdown() {\r\n    this.isOpen = false;\r\n    this.highlightedIndex = -1;\r\n    const selectedOption = this.options.find(o => o.value === this.value);\r\n    if (selectedOption) {\r\n      this.searchControl.setValue(`${selectedOption.code} - ${selectedOption.name}`, { emitEvent: false });\r\n    } else {\r\n      if (!this.value) {\r\n        this.searchControl.setValue('', { emitEvent: false });\r\n      }\r\n    }\r\n    this.onTouched();\r\n  }\r\n\r\n  selectOption(option: SearchSelectOption) {\r\n    if (!option) return;\r\n    this.value = option.value;\r\n    this.onChange(this.value);\r\n\r\n    if (this.control) {\r\n      this.control.setValue(this.value);\r\n      this.control.markAsDirty();\r\n      this.control.markAsTouched();\r\n    }\r\n\r\n    this.searchControl.setValue(`${option.code} - ${option.name}`, { emitEvent: false });\r\n    this.valueChange.emit(this.value);\r\n    this.isOpen = false;\r\n    this.highlightedIndex = -1;\r\n  }\r\n\r\n  writeValue(value: any): void {\r\n    if (value === this.value) {\r\n      return;\r\n    }\r\n    this.value = value;\r\n    const selectedOption = this.options.find(o => o.value === value);\r\n    if (selectedOption) {\r\n      this.searchControl.setValue(`${selectedOption.code} - ${selectedOption.name}`, { emitEvent: false });\r\n    } else {\r\n      if (value === null || value === undefined || value === '') {\r\n        this.searchControl.setValue('', { emitEvent: false });\r\n      }\r\n    }\r\n  }\r\n\r\n  registerOnChange(fn: any): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: any): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.disabled = isDisabled;\r\n    if (isDisabled) {\r\n      this.searchControl.disable({ emitEvent: false });\r\n    } else {\r\n      this.searchControl.enable({ emitEvent: false });\r\n    }\r\n  }\r\n\r\n  isInvalid(): boolean {\r\n    if (this.control) {\r\n      return this.control.invalid && (this.control.dirty || this.control.touched);\r\n    }\r\n    return false;\r\n  }\r\n\r\n  isRequired(): boolean {\r\n    return this.schema?.rules?.some(r => r.name === 'required') || false;\r\n  }\r\n\r\n  getErrorMessage(): string {\r\n    if (!this.control || !this.control.errors || !this.schema) return '';\r\n\r\n    for (const errorKey of this.schema.errorPriority) {\r\n      if (this.control.hasError(errorKey)) {\r\n        const rule = this.schema.rules.find(r => r.name === (errorKey === 'required' ? 'required' : errorKey));\r\n        if (rule) return this.getReadableError(rule.errorKey || '');\r\n      }\r\n    }\r\n    if (this.control.hasError('required')) return 'This field is required';\r\n\r\n    return 'Invalid value';\r\n  }\r\n\r\n  getReadableError(key: string): string {\r\n    const map: { [key: string]: string } = {\r\n      'ERR_REQUIRED': 'This field is required',\r\n      'ERR_MIN_LENGTH': 'Value is too short',\r\n    };\r\n    return map[key] || 'Invalid value';\r\n  }\r\n\r\n  @HostListener('document:click', ['$event'])\r\n  onClickOutside(event: MouseEvent) {\r\n    const clickedInside = (event.target as HTMLElement).closest('.fv-name-code-container');\r\n    if (!clickedInside) {\r\n      this.closeDropdown();\r\n    }\r\n  }\r\n\r\n  @HostListener('keydown', ['$event'])\r\n  onKeyDown(event: KeyboardEvent) {\r\n    if (!this.isOpen && (event.key === 'ArrowDown' || event.key === 'ArrowUp' || event.key === 'Enter')) {\r\n      this.openDropdown();\r\n      event.preventDefault();\r\n      return;\r\n    }\r\n\r\n    if (!this.isOpen) return;\r\n\r\n    switch (event.key) {\r\n      case 'ArrowDown':\r\n        event.preventDefault();\r\n        this.highlightedIndex = (this.highlightedIndex + 1) % this.filteredOptions.length;\r\n        this.scrollToHighlighted();\r\n        break;\r\n      case 'ArrowUp':\r\n        event.preventDefault();\r\n        this.highlightedIndex = (this.highlightedIndex - 1 + this.filteredOptions.length) % this.filteredOptions.length;\r\n        this.scrollToHighlighted();\r\n        break;\r\n      case 'Enter':\r\n        event.preventDefault();\r\n        if (this.filteredOptions.length === 1) {\r\n          this.selectOption(this.filteredOptions[0]);\r\n        } else if (this.highlightedIndex >= 0 && this.highlightedIndex < this.filteredOptions.length) {\r\n          this.selectOption(this.filteredOptions[this.highlightedIndex]);\r\n        }\r\n        break;\r\n      case 'Escape':\r\n        this.closeDropdown();\r\n        break;\r\n      case 'Tab':\r\n        this.closeDropdown();\r\n        break;\r\n    }\r\n  }\r\n\r\n  private scrollToHighlighted() {\r\n    const list = document.querySelector('.dropdown-list');\r\n    const item = document.querySelectorAll('.dropdown-item')[this.highlightedIndex];\r\n    if (list && item) {\r\n      const listRect = list.getBoundingClientRect();\r\n      const itemRect = item.getBoundingClientRect();\r\n\r\n      if (itemRect.bottom > listRect.bottom) {\r\n        list.scrollTop += (itemRect.bottom - listRect.bottom);\r\n      } else if (itemRect.top < listRect.top) {\r\n        list.scrollTop -= (listRect.top - itemRect.top);\r\n      }\r\n    }\r\n  }\r\n}\r\n"]}
|
|
@@ -1007,15 +1007,45 @@ class FvFileSelectorComponent {
|
|
|
1007
1007
|
}
|
|
1008
1008
|
// Subscribe to value changes
|
|
1009
1009
|
this.subscription = this.control.valueChanges.subscribe((value) => {
|
|
1010
|
+
this.handleValue(value);
|
|
1010
1011
|
this.validateValue(value);
|
|
1011
1012
|
this.valueChange.emit(value);
|
|
1012
1013
|
});
|
|
1013
1014
|
// Validate initial value
|
|
1014
|
-
|
|
1015
|
-
this.selectedFile = this.control.value;
|
|
1016
|
-
}
|
|
1015
|
+
this.handleValue(this.control.value);
|
|
1017
1016
|
this.validateValue(this.control.value);
|
|
1018
1017
|
}
|
|
1018
|
+
handleValue(value) {
|
|
1019
|
+
if (!value) {
|
|
1020
|
+
this.selectedFile = null;
|
|
1021
|
+
return;
|
|
1022
|
+
}
|
|
1023
|
+
if (typeof value === 'string') {
|
|
1024
|
+
// Handle API URL/String
|
|
1025
|
+
const name = value.split('/').pop() || 'Document';
|
|
1026
|
+
// Guess type from extension
|
|
1027
|
+
const ext = name.split('.').pop()?.toLowerCase() || '';
|
|
1028
|
+
let type = 'application/octet-stream';
|
|
1029
|
+
if (ext === 'pdf')
|
|
1030
|
+
type = 'application/pdf';
|
|
1031
|
+
if (['png', 'jpg', 'jpeg', 'gif'].includes(ext))
|
|
1032
|
+
type = `image/${ext}`;
|
|
1033
|
+
if (['xls', 'xlsx'].includes(ext))
|
|
1034
|
+
type = 'application/vnd.ms-excel';
|
|
1035
|
+
if (['doc', 'docx'].includes(ext))
|
|
1036
|
+
type = 'application/msword';
|
|
1037
|
+
this.selectedFile = {
|
|
1038
|
+
file: new File([''], name, { type }),
|
|
1039
|
+
name: name,
|
|
1040
|
+
size: 0,
|
|
1041
|
+
type: type
|
|
1042
|
+
};
|
|
1043
|
+
}
|
|
1044
|
+
else {
|
|
1045
|
+
// Assume it's a FileInfo object
|
|
1046
|
+
this.selectedFile = value;
|
|
1047
|
+
}
|
|
1048
|
+
}
|
|
1019
1049
|
ngOnDestroy() {
|
|
1020
1050
|
this.subscription?.unsubscribe();
|
|
1021
1051
|
}
|
|
@@ -1180,12 +1210,30 @@ class FvImageSelectorComponent {
|
|
|
1180
1210
|
if (!this.control)
|
|
1181
1211
|
return;
|
|
1182
1212
|
this.subscription = this.control.valueChanges.subscribe((value) => {
|
|
1213
|
+
this.handleValue(value);
|
|
1183
1214
|
this.validateValue(value);
|
|
1184
1215
|
this.valueChange.emit(value);
|
|
1185
1216
|
});
|
|
1186
|
-
|
|
1187
|
-
|
|
1188
|
-
|
|
1217
|
+
this.handleValue(this.control.value);
|
|
1218
|
+
this.validateValue(this.control.value);
|
|
1219
|
+
}
|
|
1220
|
+
handleValue(value) {
|
|
1221
|
+
if (!value) {
|
|
1222
|
+
this.selectedImage = null;
|
|
1223
|
+
return;
|
|
1224
|
+
}
|
|
1225
|
+
if (typeof value === 'string') {
|
|
1226
|
+
const name = value.split('/').pop() || 'Image';
|
|
1227
|
+
this.selectedImage = {
|
|
1228
|
+
file: new File([''], name, { type: 'image/jpeg' }),
|
|
1229
|
+
url: this.sanitizer.bypassSecurityTrustUrl(value),
|
|
1230
|
+
width: 0,
|
|
1231
|
+
height: 0,
|
|
1232
|
+
size: 0
|
|
1233
|
+
};
|
|
1234
|
+
}
|
|
1235
|
+
else {
|
|
1236
|
+
this.selectedImage = value;
|
|
1189
1237
|
}
|
|
1190
1238
|
}
|
|
1191
1239
|
ngOnDestroy() {
|
|
@@ -1636,13 +1684,18 @@ class FvNameCodeComponent {
|
|
|
1636
1684
|
this.highlightedIndex = -1;
|
|
1637
1685
|
}
|
|
1638
1686
|
writeValue(value) {
|
|
1687
|
+
if (value === this.value) {
|
|
1688
|
+
return;
|
|
1689
|
+
}
|
|
1639
1690
|
this.value = value;
|
|
1640
1691
|
const selectedOption = this.options.find(o => o.value === value);
|
|
1641
1692
|
if (selectedOption) {
|
|
1642
1693
|
this.searchControl.setValue(`${selectedOption.code} - ${selectedOption.name}`, { emitEvent: false });
|
|
1643
1694
|
}
|
|
1644
1695
|
else {
|
|
1645
|
-
|
|
1696
|
+
if (value === null || value === undefined || value === '') {
|
|
1697
|
+
this.searchControl.setValue('', { emitEvent: false });
|
|
1698
|
+
}
|
|
1646
1699
|
}
|
|
1647
1700
|
}
|
|
1648
1701
|
registerOnChange(fn) {
|
|
@@ -2731,6 +2784,7 @@ class FvDocumentFieldComponent {
|
|
|
2731
2784
|
showPreview = false;
|
|
2732
2785
|
previewUrl = null;
|
|
2733
2786
|
subscription;
|
|
2787
|
+
fileInfo = null;
|
|
2734
2788
|
constructor(sanitizer) {
|
|
2735
2789
|
this.sanitizer = sanitizer;
|
|
2736
2790
|
}
|
|
@@ -2740,14 +2794,45 @@ class FvDocumentFieldComponent {
|
|
|
2740
2794
|
return;
|
|
2741
2795
|
}
|
|
2742
2796
|
this.subscription = this.control.valueChanges.subscribe((value) => {
|
|
2797
|
+
this.handleValue(value);
|
|
2743
2798
|
this.validateValue(value);
|
|
2744
2799
|
this.valueChange.emit(value);
|
|
2745
2800
|
});
|
|
2746
2801
|
// Initial validation
|
|
2802
|
+
this.handleValue(this.control.value);
|
|
2747
2803
|
if (this.control.value) {
|
|
2748
2804
|
this.validateValue(this.control.value);
|
|
2749
2805
|
}
|
|
2750
2806
|
}
|
|
2807
|
+
handleValue(value) {
|
|
2808
|
+
if (!value) {
|
|
2809
|
+
this.fileInfo = null;
|
|
2810
|
+
return;
|
|
2811
|
+
}
|
|
2812
|
+
if (typeof value === 'string') {
|
|
2813
|
+
const name = value.split('/').pop() || 'Document';
|
|
2814
|
+
// Guess type from extension
|
|
2815
|
+
const ext = name.split('.').pop()?.toLowerCase() || '';
|
|
2816
|
+
let type = 'application/octet-stream';
|
|
2817
|
+
if (ext === 'pdf')
|
|
2818
|
+
type = 'application/pdf';
|
|
2819
|
+
if (['png', 'jpg', 'jpeg', 'gif'].includes(ext))
|
|
2820
|
+
type = `image/${ext}`;
|
|
2821
|
+
if (['xls', 'xlsx'].includes(ext))
|
|
2822
|
+
type = 'application/vnd.ms-excel';
|
|
2823
|
+
if (['doc', 'docx'].includes(ext))
|
|
2824
|
+
type = 'application/msword';
|
|
2825
|
+
this.fileInfo = {
|
|
2826
|
+
file: value, // Store the URL string as file property if strict typing allows, or cast. Interface says file: File | Blob | string. Perfect.
|
|
2827
|
+
name: name,
|
|
2828
|
+
size: 0,
|
|
2829
|
+
type: type
|
|
2830
|
+
};
|
|
2831
|
+
}
|
|
2832
|
+
else {
|
|
2833
|
+
this.fileInfo = value;
|
|
2834
|
+
}
|
|
2835
|
+
}
|
|
2751
2836
|
ngOnDestroy() {
|
|
2752
2837
|
this.subscription?.unsubscribe();
|
|
2753
2838
|
}
|
|
@@ -2807,7 +2892,7 @@ class FvDocumentFieldComponent {
|
|
|
2807
2892
|
}
|
|
2808
2893
|
}
|
|
2809
2894
|
viewFile() {
|
|
2810
|
-
const fileInfo = this.
|
|
2895
|
+
const fileInfo = this.fileInfo;
|
|
2811
2896
|
if (!fileInfo)
|
|
2812
2897
|
return;
|
|
2813
2898
|
const type = fileInfo.type || '';
|
|
@@ -2839,7 +2924,7 @@ class FvDocumentFieldComponent {
|
|
|
2839
2924
|
this.previewUrl = null;
|
|
2840
2925
|
}
|
|
2841
2926
|
downloadFile() {
|
|
2842
|
-
const fileInfo = this.
|
|
2927
|
+
const fileInfo = this.fileInfo;
|
|
2843
2928
|
if (!fileInfo)
|
|
2844
2929
|
return;
|
|
2845
2930
|
let url;
|
|
@@ -2862,7 +2947,7 @@ class FvDocumentFieldComponent {
|
|
|
2862
2947
|
}
|
|
2863
2948
|
}
|
|
2864
2949
|
getFileIcon() {
|
|
2865
|
-
const fileInfo = this.
|
|
2950
|
+
const fileInfo = this.fileInfo;
|
|
2866
2951
|
if (!fileInfo)
|
|
2867
2952
|
return 'description';
|
|
2868
2953
|
const type = fileInfo.type || '';
|
|
@@ -2891,11 +2976,11 @@ class FvDocumentFieldComponent {
|
|
|
2891
2976
|
return errorMessages[this.errorMessage] || this.errorMessage;
|
|
2892
2977
|
}
|
|
2893
2978
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FvDocumentFieldComponent, deps: [{ token: i1$1.DomSanitizer }], target: i0.ɵɵFactoryTarget.Component });
|
|
2894
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: FvDocumentFieldComponent, isStandalone: true, selector: "fv-document-field", inputs: { label: "label", placeholder: "placeholder", schema: "schema", control: "control", disabled: "disabled", accept: "accept", maxSize: "maxSize" }, outputs: { valueChange: "valueChange" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"fv-document-field-container\">\r\n <label *ngIf=\"label\" class=\"fv-document-field-label\">\r\n {{ label }}\r\n <span *ngIf=\"isRequired()\" class=\"required-asterisk\">*</span>\r\n </label>\r\n\r\n <input #fileInput type=\"file\" [accept]=\"accept\" (change)=\"onFileSelected($event)\" style=\"display: none\" />\r\n\r\n <div *ngIf=\"!
|
|
2979
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: FvDocumentFieldComponent, isStandalone: true, selector: "fv-document-field", inputs: { label: "label", placeholder: "placeholder", schema: "schema", control: "control", disabled: "disabled", accept: "accept", maxSize: "maxSize" }, outputs: { valueChange: "valueChange" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"fv-document-field-container\">\r\n <label *ngIf=\"label\" class=\"fv-document-field-label\">\r\n {{ label }}\r\n <span *ngIf=\"isRequired()\" class=\"required-asterisk\">*</span>\r\n </label>\r\n\r\n <input #fileInput type=\"file\" [accept]=\"accept\" (change)=\"onFileSelected($event)\" style=\"display: none\" />\r\n\r\n <div *ngIf=\"!fileInfo; else fileUploaded\" class=\"fv-upload-btn-wrapper\">\r\n <button type=\"button\" class=\"fv-upload-button\" [disabled]=\"disabled\" (click)=\"openFileDialog()\">\r\n <span class=\"fv-upload-text\">{{ placeholder }}</span>\r\n <span class=\"fv-upload-icon-section\">\r\n <svg viewBox=\"0 0 24 24\" class=\"upload-icon\">\r\n <path d=\"M9 16h6v-6h4l-7-7-7 7h4zm-4 2h14v2H5z\" fill=\"currentColor\" />\r\n </svg>\r\n </span>\r\n </button>\r\n </div>\r\n\r\n <ng-template #fileUploaded>\r\n <div class=\"fv-preview-row\">\r\n <div class=\"fv-preview-wrapper\" (click)=\"viewFile()\" style=\"cursor: pointer;\">\r\n <div class=\"fv-file-icon-preview\" [ngClass]=\"getFileIcon()\">\r\n <span class=\"material-icons\">{{ getFileIcon() }}</span>\r\n </div>\r\n </div>\r\n <span class=\"fv-filename\" (click)=\"viewFile()\" style=\"cursor: pointer;\">{{ fileInfo?.name }}</span>\r\n <button type=\"button\" class=\"fv-trash-btn\" (click)=\"removeFile()\" [disabled]=\"disabled\">\r\n <svg viewBox=\"0 0 24 24\" class=\"trash-icon\">\r\n <path d=\"M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\" />\r\n </svg>\r\n </button>\r\n </div>\r\n </ng-template>\r\n\r\n <div *ngIf=\"errorMessage && control?.touched\" class=\"fv-document-error-message\">\r\n \u26A0 {{ getErrorMessage() }}\r\n </div>\r\n</div>\r\n\r\n<!-- Preview Overlay/Modal for Image -->\r\n<div *ngIf=\"showPreview\" class=\"fv-preview-overlay\" (click)=\"closePreview()\">\r\n <div class=\"fv-preview-content\" (click)=\"$event.stopPropagation()\">\r\n <img [src]=\"previewUrl\" alt=\"Preview\" />\r\n <div class=\"fv-preview-actions\">\r\n <button type=\"button\" class=\"fv-download-btn\" (click)=\"downloadFile()\">Download</button>\r\n <button type=\"button\" class=\"fv-close-btn\" (click)=\"closePreview()\">Close</button>\r\n </div>\r\n </div>\r\n</div>", styles: ["@import\"https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;500;600;700&display=swap\";.fv-document-field-container{display:flex;flex-direction:column;margin-bottom:16px;width:fit-content;max-width:100%;font-family:Poppins,sans-serif}.fv-document-field-label{font-size:14px;font-weight:600;color:#151d48;margin-bottom:6px;display:block}.required-asterisk{color:#e74c3c;font-weight:700}.fv-upload-button{display:flex;align-items:center;border:1px solid #dcdcdc;border-radius:8px;background:#fff;padding:0;cursor:pointer;overflow:hidden;width:100px;height:34px;transition:all .2s ease}.fv-upload-button:hover:not(:disabled){border-color:#bbb;box-shadow:0 2px 4px #0000000d}.fv-upload-button:disabled{cursor:not-allowed;background-color:#f9f9f9;opacity:.7}.fv-upload-text{flex:1;padding:0 16px;font-size:14px;color:#1f2b41;text-align:left;font-weight:500}.fv-upload-icon-section{background:#f4f5f7;padding:0 10px;border-left:1px solid #dcdcdc;display:flex;align-items:center;justify-content:center;color:#333;height:100%}.upload-icon{width:20px;height:20px}.fv-preview-row{display:flex;align-items:center;gap:8px;padding:8px 0;width:100px;box-sizing:border-box}.fv-preview-wrapper{position:relative;width:40px;height:40px}.fv-file-icon-preview{width:40px;height:40px;border-radius:8px;background:#f4f5f7;display:flex;align-items:center;justify-content:center;color:#3498db;border:1px solid #dcdcdc}.fv-file-icon-preview.picture_as_pdf{color:#e74c3c;background:#fdf2f2}.fv-file-icon-preview.image{color:#3498db;background:#f1f8fe}.fv-file-icon-preview.table_view{color:#27ae60;background:#f1f9f4}.material-icons{font-size:24px}.fv-filename{flex:1;font-size:14px;color:#666;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:300px}.fv-trash-btn{background:none;border:none;cursor:pointer;padding:6px;border-radius:50%;transition:background .2s;display:flex}.fv-trash-btn:hover:not(:disabled){background:#fee}.trash-icon{width:22px;height:22px;fill:red}.fv-document-error-message{margin-top:4px;font-size:12px;color:#e74c3c}.fv-preview-overlay{position:fixed;top:0;left:0;width:100vw;height:100vh;background:#000c;display:flex;justify-content:center;align-items:center;z-index:10000}.fv-preview-content{position:relative;background:#fff;padding:24px;border-radius:12px;max-width:90vw;max-height:90vh;display:flex;flex-direction:column;align-items:center}.fv-preview-content img{max-width:100%;max-height:70vh;object-fit:contain;border-radius:4px;margin-bottom:20px}.fv-preview-actions{display:flex;gap:16px}.fv-preview-actions button{padding:10px 24px;border-radius:6px;border:none;cursor:pointer;font-weight:500;transition:background .2s}.fv-download-btn{background:#3498db;color:#fff}.fv-download-btn:hover{background:#2980b9}.fv-close-btn{background:#f0f0f0;color:#333}.fv-close-btn:hover{background:#e0e0e0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }] });
|
|
2895
2980
|
}
|
|
2896
2981
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FvDocumentFieldComponent, decorators: [{
|
|
2897
2982
|
type: Component,
|
|
2898
|
-
args: [{ standalone: true, imports: [CommonModule, ReactiveFormsModule], selector: 'fv-document-field', template: "<div class=\"fv-document-field-container\">\r\n <label *ngIf=\"label\" class=\"fv-document-field-label\">\r\n {{ label }}\r\n <span *ngIf=\"isRequired()\" class=\"required-asterisk\">*</span>\r\n </label>\r\n\r\n <input #fileInput type=\"file\" [accept]=\"accept\" (change)=\"onFileSelected($event)\" style=\"display: none\" />\r\n\r\n <div *ngIf=\"!
|
|
2983
|
+
args: [{ standalone: true, imports: [CommonModule, ReactiveFormsModule], selector: 'fv-document-field', template: "<div class=\"fv-document-field-container\">\r\n <label *ngIf=\"label\" class=\"fv-document-field-label\">\r\n {{ label }}\r\n <span *ngIf=\"isRequired()\" class=\"required-asterisk\">*</span>\r\n </label>\r\n\r\n <input #fileInput type=\"file\" [accept]=\"accept\" (change)=\"onFileSelected($event)\" style=\"display: none\" />\r\n\r\n <div *ngIf=\"!fileInfo; else fileUploaded\" class=\"fv-upload-btn-wrapper\">\r\n <button type=\"button\" class=\"fv-upload-button\" [disabled]=\"disabled\" (click)=\"openFileDialog()\">\r\n <span class=\"fv-upload-text\">{{ placeholder }}</span>\r\n <span class=\"fv-upload-icon-section\">\r\n <svg viewBox=\"0 0 24 24\" class=\"upload-icon\">\r\n <path d=\"M9 16h6v-6h4l-7-7-7 7h4zm-4 2h14v2H5z\" fill=\"currentColor\" />\r\n </svg>\r\n </span>\r\n </button>\r\n </div>\r\n\r\n <ng-template #fileUploaded>\r\n <div class=\"fv-preview-row\">\r\n <div class=\"fv-preview-wrapper\" (click)=\"viewFile()\" style=\"cursor: pointer;\">\r\n <div class=\"fv-file-icon-preview\" [ngClass]=\"getFileIcon()\">\r\n <span class=\"material-icons\">{{ getFileIcon() }}</span>\r\n </div>\r\n </div>\r\n <span class=\"fv-filename\" (click)=\"viewFile()\" style=\"cursor: pointer;\">{{ fileInfo?.name }}</span>\r\n <button type=\"button\" class=\"fv-trash-btn\" (click)=\"removeFile()\" [disabled]=\"disabled\">\r\n <svg viewBox=\"0 0 24 24\" class=\"trash-icon\">\r\n <path d=\"M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\" />\r\n </svg>\r\n </button>\r\n </div>\r\n </ng-template>\r\n\r\n <div *ngIf=\"errorMessage && control?.touched\" class=\"fv-document-error-message\">\r\n \u26A0 {{ getErrorMessage() }}\r\n </div>\r\n</div>\r\n\r\n<!-- Preview Overlay/Modal for Image -->\r\n<div *ngIf=\"showPreview\" class=\"fv-preview-overlay\" (click)=\"closePreview()\">\r\n <div class=\"fv-preview-content\" (click)=\"$event.stopPropagation()\">\r\n <img [src]=\"previewUrl\" alt=\"Preview\" />\r\n <div class=\"fv-preview-actions\">\r\n <button type=\"button\" class=\"fv-download-btn\" (click)=\"downloadFile()\">Download</button>\r\n <button type=\"button\" class=\"fv-close-btn\" (click)=\"closePreview()\">Close</button>\r\n </div>\r\n </div>\r\n</div>", styles: ["@import\"https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;500;600;700&display=swap\";.fv-document-field-container{display:flex;flex-direction:column;margin-bottom:16px;width:fit-content;max-width:100%;font-family:Poppins,sans-serif}.fv-document-field-label{font-size:14px;font-weight:600;color:#151d48;margin-bottom:6px;display:block}.required-asterisk{color:#e74c3c;font-weight:700}.fv-upload-button{display:flex;align-items:center;border:1px solid #dcdcdc;border-radius:8px;background:#fff;padding:0;cursor:pointer;overflow:hidden;width:100px;height:34px;transition:all .2s ease}.fv-upload-button:hover:not(:disabled){border-color:#bbb;box-shadow:0 2px 4px #0000000d}.fv-upload-button:disabled{cursor:not-allowed;background-color:#f9f9f9;opacity:.7}.fv-upload-text{flex:1;padding:0 16px;font-size:14px;color:#1f2b41;text-align:left;font-weight:500}.fv-upload-icon-section{background:#f4f5f7;padding:0 10px;border-left:1px solid #dcdcdc;display:flex;align-items:center;justify-content:center;color:#333;height:100%}.upload-icon{width:20px;height:20px}.fv-preview-row{display:flex;align-items:center;gap:8px;padding:8px 0;width:100px;box-sizing:border-box}.fv-preview-wrapper{position:relative;width:40px;height:40px}.fv-file-icon-preview{width:40px;height:40px;border-radius:8px;background:#f4f5f7;display:flex;align-items:center;justify-content:center;color:#3498db;border:1px solid #dcdcdc}.fv-file-icon-preview.picture_as_pdf{color:#e74c3c;background:#fdf2f2}.fv-file-icon-preview.image{color:#3498db;background:#f1f8fe}.fv-file-icon-preview.table_view{color:#27ae60;background:#f1f9f4}.material-icons{font-size:24px}.fv-filename{flex:1;font-size:14px;color:#666;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:300px}.fv-trash-btn{background:none;border:none;cursor:pointer;padding:6px;border-radius:50%;transition:background .2s;display:flex}.fv-trash-btn:hover:not(:disabled){background:#fee}.trash-icon{width:22px;height:22px;fill:red}.fv-document-error-message{margin-top:4px;font-size:12px;color:#e74c3c}.fv-preview-overlay{position:fixed;top:0;left:0;width:100vw;height:100vh;background:#000c;display:flex;justify-content:center;align-items:center;z-index:10000}.fv-preview-content{position:relative;background:#fff;padding:24px;border-radius:12px;max-width:90vw;max-height:90vh;display:flex;flex-direction:column;align-items:center}.fv-preview-content img{max-width:100%;max-height:70vh;object-fit:contain;border-radius:4px;margin-bottom:20px}.fv-preview-actions{display:flex;gap:16px}.fv-preview-actions button{padding:10px 24px;border-radius:6px;border:none;cursor:pointer;font-weight:500;transition:background .2s}.fv-download-btn{background:#3498db;color:#fff}.fv-download-btn:hover{background:#2980b9}.fv-close-btn{background:#f0f0f0;color:#333}.fv-close-btn:hover{background:#e0e0e0}\n"] }]
|
|
2899
2984
|
}], ctorParameters: () => [{ type: i1$1.DomSanitizer }], propDecorators: { label: [{
|
|
2900
2985
|
type: Input
|
|
2901
2986
|
}], placeholder: [{
|