@xui/components
Version:
xUI Components for Angular
92 lines • 15.7 kB
JavaScript
import { booleanAttribute, ChangeDetectionStrategy, Component, computed, effect, ElementRef, input, model, Optional, Self, signal, ViewChild } from '@angular/core';
import { Dialog } from '@angular/cdk/dialog';
import { NgControl } from '@angular/forms';
import { XuiImageUploadCropper } from './image-upload-cropper';
import * as i0 from "@angular/core";
import * as i1 from "@angular/cdk/dialog";
import * as i2 from "@angular/forms";
import * as i3 from "@angular/common";
import * as i4 from "../icon/icon";
import * as i5 from "@ngx-translate/core";
export class XuiImageUpload {
constructor(dialog, control) {
this.dialog = dialog;
this.control = control;
this.croppedImage = null;
this._disabled = signal(false);
this.value = model(null);
this.type = input('square');
this.aspectRatio = input(1);
this.hoverLabel = input('xui.image_upload.change_image');
this.disabled = input(undefined, {
transform: booleanAttribute
});
this._borderRadius = computed(() => (this.type() === 'round' ? 50 : 4));
this._backgroundImageUrl = computed(() => (this.value() ? `url(${this.value()})` : null));
this.imageCropped = (event) => {
this.croppedImage = event.base64 ?? null;
};
this.save = () => {
this.onChange?.(this.croppedImage);
this.value.set(this.croppedImage);
this.dialogRef?.close();
};
if (this.control) {
this.control.valueAccessor = this;
}
effect(() => this.disabled() && this._disabled.set(this.disabled()), { allowSignalWrites: true });
}
handleFileInput(event) {
this.dialogRef = this.dialog.open(XuiImageUploadCropper, {
data: {
type: this.type,
aspectRatio: this.aspectRatio,
save: this.save,
imageChangedEvent: event,
imageCropped: this.imageCropped
}
});
}
_keyPress(event) {
event?.preventDefault();
this.inputElm.nativeElement.click();
}
writeValue(source) {
this.croppedImage = source;
this.value.set(source);
}
registerOnChange(onChange) {
this.onChange = onChange;
}
registerOnTouched(onTouched) {
this._onTouched = onTouched;
}
setDisabledState(isDisabled) {
this._disabled.set(isDisabled);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiImageUpload, deps: [{ token: i1.Dialog }, { token: i2.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.0.1", type: XuiImageUpload, selector: "xui-image-upload", inputs: { value: { classPropertyName: "value", publicName: "value", isSignal: true, isRequired: false, transformFunction: null }, type: { classPropertyName: "type", publicName: "type", isSignal: true, isRequired: false, transformFunction: null }, aspectRatio: { classPropertyName: "aspectRatio", publicName: "aspectRatio", isSignal: true, isRequired: false, transformFunction: null }, hoverLabel: { classPropertyName: "hoverLabel", publicName: "hoverLabel", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { value: "valueChange" }, host: { listeners: { "focusout": "_onTouched?.()", "keydown.enter": "_keyPress($event)", "keydown.space": "_keyPress($event)" }, properties: { "class.x-image-upload-disabled": "disabled()", "class.x-image-upload-square": "type() === \"square\"", "style.border-radius.%": "_borderRadius()", "style.aspect-ratio": "aspectRatio()", "style.background-image": "_backgroundImageUrl()", "tabindex": "_disabled() ? -1 : 0" }, classAttribute: "x-image-upload" }, viewQueries: [{ propertyName: "inputElm", first: true, predicate: ["input"], descendants: true }], ngImport: i0, template: "<input #input type=\"file\" [disabled]=\"_disabled()\" (change)=\"handleFileInput($event)\" accept=\"image/*\" tabindex=\"-1\" />\n<div class=\"x-image-upload-hover\" [style.border-radius.%]=\"_borderRadius()\">{{ hoverLabel() | translate }}</div>\n<div *ngIf=\"!_backgroundImageUrl()\" class=\"x-image-upload-no-image\">{{ 'xui.image_upload.no_image' | translate }}</div>\n\n<div class=\"x-image-upload-icon\">\n <xui-icon icon=\"add_photo_alternate\"></xui-icon>\n</div>\n", dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.XuiIcon, selector: "xui-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "icon"], exportAs: ["xuiIcon"] }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: XuiImageUpload, decorators: [{
type: Component,
args: [{ selector: 'xui-image-upload', changeDetection: ChangeDetectionStrategy.OnPush, host: {
class: 'x-image-upload',
'[class.x-image-upload-disabled]': 'disabled()',
'[class.x-image-upload-square]': 'type() === "square"',
'[style.border-radius.%]': '_borderRadius()',
'[style.aspect-ratio]': 'aspectRatio()',
'[style.background-image]': '_backgroundImageUrl()',
'[tabindex]': '_disabled() ? -1 : 0',
'(focusout)': '_onTouched?.()',
'(keydown.enter)': '_keyPress($event)',
'(keydown.space)': '_keyPress($event)'
}, template: "<input #input type=\"file\" [disabled]=\"_disabled()\" (change)=\"handleFileInput($event)\" accept=\"image/*\" tabindex=\"-1\" />\n<div class=\"x-image-upload-hover\" [style.border-radius.%]=\"_borderRadius()\">{{ hoverLabel() | translate }}</div>\n<div *ngIf=\"!_backgroundImageUrl()\" class=\"x-image-upload-no-image\">{{ 'xui.image_upload.no_image' | translate }}</div>\n\n<div class=\"x-image-upload-icon\">\n <xui-icon icon=\"add_photo_alternate\"></xui-icon>\n</div>\n" }]
}], ctorParameters: () => [{ type: i1.Dialog }, { type: i2.NgControl, decorators: [{
type: Self
}, {
type: Optional
}] }], propDecorators: { inputElm: [{
type: ViewChild,
args: ['input']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-upload.js","sourceRoot":"","sources":["../../../../../libs/xui/src/image-upload/image-upload.ts","../../../../../libs/xui/src/image-upload/image-upload.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  booleanAttribute,\n  ChangeDetectionStrategy,\n  Component,\n  computed,\n  effect,\n  ElementRef,\n  input,\n  model,\n  Optional,\n  Self,\n  signal,\n  ViewChild\n} from '@angular/core';\nimport { ImageCroppedEvent } from 'ngx-image-cropper';\nimport { Dialog, DialogRef } from '@angular/cdk/dialog';\nimport { ControlValueAccessor, NgControl } from '@angular/forms';\nimport { XuiImageUploadCropper } from './image-upload-cropper';\nimport { ImageUploadType } from './image-upload.types';\n\n@Component({\n  selector: 'xui-image-upload',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  templateUrl: 'image-upload.html',\n  host: {\n    class: 'x-image-upload',\n    '[class.x-image-upload-disabled]': 'disabled()',\n    '[class.x-image-upload-square]': 'type() === \"square\"',\n    '[style.border-radius.%]': '_borderRadius()',\n    '[style.aspect-ratio]': 'aspectRatio()',\n    '[style.background-image]': '_backgroundImageUrl()',\n    '[tabindex]': '_disabled() ? -1 : 0',\n    '(focusout)': '_onTouched?.()',\n    '(keydown.enter)': '_keyPress($event)',\n    '(keydown.space)': '_keyPress($event)'\n  }\n})\nexport class XuiImageUpload implements ControlValueAccessor {\n  private dialogRef?: DialogRef<unknown, XuiImageUploadCropper>;\n  private croppedImage: string | null = null;\n  private onChange?: (source: string | null) => void;\n  _onTouched?: () => void;\n  _disabled = signal(false);\n\n  value = model<string | null>(null);\n  type = input<ImageUploadType>('square');\n  aspectRatio = input(1);\n  hoverLabel = input('xui.image_upload.change_image');\n  disabled = input<boolean | undefined, string | boolean>(undefined, {\n    transform: booleanAttribute\n  });\n\n  _borderRadius = computed(() => (this.type() === 'round' ? 50 : 4));\n  _backgroundImageUrl = computed(() => (this.value() ? `url(${this.value()})` : null));\n\n  @ViewChild('input') private inputElm!: ElementRef;\n\n  constructor(\n    private dialog: Dialog,\n    @Self() @Optional() public control?: NgControl\n  ) {\n    if (this.control) {\n      this.control.valueAccessor = this;\n    }\n\n    effect(() => this.disabled() && this._disabled.set(this.disabled()!), { allowSignalWrites: true });\n  }\n\n  handleFileInput(event: unknown) {\n    this.dialogRef = this.dialog.open(XuiImageUploadCropper, {\n      data: {\n        type: this.type,\n        aspectRatio: this.aspectRatio,\n        save: this.save,\n        imageChangedEvent: event,\n        imageCropped: this.imageCropped\n      }\n    });\n  }\n\n  _keyPress(event: KeyboardEvent) {\n    event?.preventDefault();\n    this.inputElm.nativeElement.click();\n  }\n\n  private imageCropped = (event: ImageCroppedEvent) => {\n    this.croppedImage = event.base64 ?? null;\n  };\n\n  private save = () => {\n    this.onChange?.(this.croppedImage);\n    this.value.set(this.croppedImage);\n    this.dialogRef?.close();\n  };\n\n  writeValue(source: string) {\n    this.croppedImage = source;\n    this.value.set(source);\n  }\n\n  registerOnChange(onChange: (source: string | null) => void) {\n    this.onChange = onChange;\n  }\n\n  registerOnTouched(onTouched: () => void) {\n    this._onTouched = onTouched;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this._disabled.set(isDisabled);\n  }\n}\n","<input #input type=\"file\" [disabled]=\"_disabled()\" (change)=\"handleFileInput($event)\" accept=\"image/*\" tabindex=\"-1\" />\n<div class=\"x-image-upload-hover\" [style.border-radius.%]=\"_borderRadius()\">{{ hoverLabel() | translate }}</div>\n<div *ngIf=\"!_backgroundImageUrl()\" class=\"x-image-upload-no-image\">{{ 'xui.image_upload.no_image' | translate }}</div>\n\n<div class=\"x-image-upload-icon\">\n  <xui-icon icon=\"add_photo_alternate\"></xui-icon>\n</div>\n"]}