ngx-file-required
Version:
Angular input file required directive
224 lines (223 loc) • 16.3 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
import { Directive, ElementRef, Input, HostListener } from '@angular/core';
import { NG_VALIDATORS } from '@angular/forms';
/** @typedef {?} */
var HTMLFileInputAttribute;
export { HTMLFileInputAttribute };
/**
* @record
*/
function FileInputEventTarget() { }
/** @type {?} */
FileInputEventTarget.prototype.files;
var NgxFileRequiredDirective = /** @class */ (function () {
function NgxFileRequiredDirective(element) {
this.requiredErrorMsg = 'File is required';
this._required = false;
this._multiple = false;
this._element = element;
}
Object.defineProperty(NgxFileRequiredDirective.prototype, "required", {
get: /**
* @return {?}
*/
function () {
return this._required || this._element.nativeElement.hasAttribute('required');
},
set: /**
* @param {?} value
* @return {?}
*/
function (value) {
this._required = value || this._element.nativeElement.hasAttribute('required');
},
enumerable: true,
configurable: true
});
Object.defineProperty(NgxFileRequiredDirective.prototype, "multiple", {
get: /**
* @return {?}
*/
function () {
return this._element.nativeElement.hasAttribute('multiple');
},
set: /**
* @param {?} value
* @return {?}
*/
function (value) {
this._multiple = value === '' || !!value;
},
enumerable: true,
configurable: true
});
/**
* @return {?}
*/
NgxFileRequiredDirective.prototype.ngOnInit = /**
* @return {?}
*/
function () {
var _this = this;
this._mutationObserver = new MutationObserver(function (mutations) {
_this._setValidity(_this._getInputValue(/** @type {?} */ (_this._element.nativeElement)));
});
this._mutationObserver.observe(this._element.nativeElement, {
attributes: true,
attributeOldValue: true,
attributeFilter: ['required']
});
};
/**
* @return {?}
*/
NgxFileRequiredDirective.prototype.ngOnDestroy = /**
* @return {?}
*/
function () {
this._mutationObserver.disconnect();
};
/**
* @param {?} changes
* @return {?}
*/
NgxFileRequiredDirective.prototype.ngOnChanges = /**
* @param {?} changes
* @return {?}
*/
function (changes) {
if (this.required &&
(changes["requiredErrorMsg"] && !changes["requiredErrorMsg"].firstChange)) {
this._setValidity(this._getInputValue(/** @type {?} */ (this._element.nativeElement)));
}
};
/**
* @param {?} control
* @return {?}
*/
NgxFileRequiredDirective.prototype.validate = /**
* @param {?} control
* @return {?}
*/
function (control) {
if (!this._control) {
this._control = control;
}
if (this._hasError(this._control.value)) {
return /** @type {?} */ ({
required: this.requiredErrorMsg
});
}
};
/**
* @param {?} eventTarget
* @return {?}
*/
NgxFileRequiredDirective.prototype.onChange = /**
* @param {?} eventTarget
* @return {?}
*/
function (eventTarget) {
/** @type {?} */
var value = this._getInputValue(/** @type {?} */ (eventTarget));
this._setValidity(value);
};
/**
* @param {?} value
* @return {?}
*/
NgxFileRequiredDirective.prototype._setValidity = /**
* @param {?} value
* @return {?}
*/
function (value) {
/** @type {?} */
var errors = Object.assign({}, this._control.errors);
if (this._hasError(value)) {
errors["required"] = this.requiredErrorMsg;
}
else {
if (this._control.hasError('required')) {
delete errors["required"];
}
}
this._control.setErrors(Object.keys(errors).length ? errors : null);
};
/**
* @param {?} value
* @return {?}
*/
NgxFileRequiredDirective.prototype._hasError = /**
* @param {?} value
* @return {?}
*/
function (value) {
return this.required && !this._hasValue(value);
};
/**
* @param {?} value
* @return {?}
*/
NgxFileRequiredDirective.prototype._hasValue = /**
* @param {?} value
* @return {?}
*/
function (value) {
return this.multiple ?
value instanceof FileList && !!value.length :
value instanceof File;
};
/**
* @param {?} eventTarget
* @return {?}
*/
NgxFileRequiredDirective.prototype._getInputValue = /**
* @param {?} eventTarget
* @return {?}
*/
function (eventTarget) {
return this.multiple ? eventTarget.files : eventTarget.files.item(0);
};
NgxFileRequiredDirective.decorators = [
{ type: Directive, args: [{
selector: "\n input[type=\"file\"][attr.required][formControlName],\n input[type=\"file\"][attr.required][formControl],\n input[type=\"file\"][attr.required][ngModel],\n input[type=\"file\"][required][formControlName],\n input[type=\"file\"][required][formControl],\n input[type=\"file\"][required][ngModel]\n ",
exportAs: 'ngxFileRequiredDirective',
providers: [
{
provide: NG_VALIDATORS,
useExisting: NgxFileRequiredDirective,
multi: true
}
]
},] },
];
/** @nocollapse */
NgxFileRequiredDirective.ctorParameters = function () { return [
{ type: ElementRef }
]; };
NgxFileRequiredDirective.propDecorators = {
requiredErrorMsg: [{ type: Input }],
multiple: [{ type: Input }],
onChange: [{ type: HostListener, args: ['change', ['$event.target'],] }]
};
return NgxFileRequiredDirective;
}());
export { NgxFileRequiredDirective };
if (false) {
/** @type {?} */
NgxFileRequiredDirective.prototype.requiredErrorMsg;
/** @type {?} */
NgxFileRequiredDirective.prototype._required;
/** @type {?} */
NgxFileRequiredDirective.prototype._multiple;
/** @type {?} */
NgxFileRequiredDirective.prototype._element;
/** @type {?} */
NgxFileRequiredDirective.prototype._control;
/** @type {?} */
NgxFileRequiredDirective.prototype._mutationObserver;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-file-required.directive.js","sourceRoot":"ng://ngx-file-required/","sources":["lib/ngx-file-required.directive.ts"],"names":[],"mappings":";;;;AAAA,OAAO,EAKL,SAAS,EACT,UAAU,EACV,KAAK,EACL,YAAY,EACb,MAAM,eAAe,CAAC;AACvB,OAAO,EACL,aAAa,EAId,MAAM,gBAAgB,CAAC;;;;;;;;;;;sCAyDH,OAAmB;gCArBZ,kBAAkB;yBAWxB,KAAK;yBAEL,KAAK;QASvB,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;;0BA/Bf,8CAAQ;;;;;YACjB,MAAM,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;;;;;;kBAG5D,KAA6B;YAC/C,IAAI,CAAC,SAAS,GAAG,KAAK,IAAI,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;;;;;IAMjF,sBACW,8CAAQ;;;;QADnB;YAEE,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;SAC7D;;;;;kBAEmB,KAA6B;YAC/C,IAAI,CAAC,SAAS,GAAG,KAAK,KAAK,EAAE,IAAI,CAAC,CAAC,KAAK,CAAC;;;;OAH1C;;;;IAoBM,2CAAQ;;;;;QACb,IAAI,CAAC,iBAAiB,GAAG,IAAI,gBAAgB,CAAC,UAAC,SAA2B;YACxE,KAAI,CAAC,YAAY,CAAC,KAAI,CAAC,cAAc,mBAAC,KAAI,CAAC,QAAQ,CAAC,aAAqC,EAAC,CAAC,CAAC;SAC7F,CAAC,CAAC;QAEH,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE;YAC1D,UAAU,EAAE,IAAI;YAChB,iBAAiB,EAAE,IAAI;YACvB,eAAe,EAAE,CAAC,UAAU,CAAC;SAC9B,CAAC,CAAC;;;;;IAGE,8CAAW;;;;QAChB,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,CAAC;;;;;;IAG/B,8CAAW;;;;cAAC,OAAsB;QACvC,EAAE,CAAC,CACD,IAAI,CAAC,QAAQ;YACb,CAAC,OAAO,wBAAqB,CAAC,OAAO,qBAAkB,WAAW,CACpE,CAAC,CAAC,CAAC;YACD,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,cAAc,mBAAC,IAAI,CAAC,QAAQ,CAAC,aAAqC,EAAC,CAAC,CAAC;SAC7F;;;;;;IAGI,2CAAQ;;;;cAAC,OAAwB;QACtC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;YACnB,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;SACzB;QAED,EAAE,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACxC,MAAM,mBAAC;gBACL,QAAQ,EAAE,IAAI,CAAC,gBAAgB;aACZ,EAAC;SACvB;;;;;;IAII,2CAAQ;;;;IADf,UACgB,WAAwB;;QACtC,IAAM,KAAK,GAA4B,IAAI,CAAC,cAAc,mBAAC,WAAmC,EAAC,CAAC;QAChG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;KAC1B;;;;;IAEO,+CAAY;;;;cAAC,KAA8B;;QACjD,IAAM,MAAM,GAAqB,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAEzE,EAAE,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC1B,MAAM,eAAY,IAAI,CAAC,gBAAgB,CAAC;SACzC;QAAC,IAAI,CAAC,CAAC;YACN,EAAE,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;gBACvC,OAAO,MAAM,YAAS,CAAC;aACxB;SACF;QAED,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;;;;;;IAG9D,4CAAS;;;;cAAC,KAA8B;QAC9C,MAAM,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;;;;;;IAGzC,4CAAS;;;;cAAC,KAA8B;QAC9C,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YACpB,KAAK,YAAY,QAAQ,IAAI,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAC7C,KAAK,YAAY,IAAI,CAAC;;;;;;IAGlB,iDAAc;;;;cAAC,WAAiC;QACtD,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;;;gBA1HxE,SAAS,SAAC;oBACT,QAAQ,EAAE,gUAOT;oBACD,QAAQ,EAAG,0BAA0B;oBACrC,SAAS,EAAE;wBACT;4BACE,OAAO,EAAM,aAAa;4BAC1B,WAAW,EAAE,wBAAwB;4BACrC,KAAK,EAAQ,IAAI;yBAClB;qBACF;iBACF;;;;gBAjCC,UAAU;;;mCA4CT,KAAK;2BAGL,KAAK;2BA4DL,YAAY,SAAC,QAAQ,EAAE,CAAC,eAAe,CAAC;;mCAjH3C;;SAwCa,wBAAwB","sourcesContent":["import {\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  Directive,\n  ElementRef,\n  Input,\n  HostListener\n} from '@angular/core';\nimport {\n  NG_VALIDATORS,\n  Validator,\n  AbstractControl,\n  ValidationErrors\n} from '@angular/forms';\n\nexport type HTMLFileInputAttribute = any | boolean;\ninterface FileInputEventTarget extends EventTarget {\n  files: FileList;\n}\n\n@Directive({\n  selector: `\n    input[type=\"file\"][attr.required][formControlName],\n    input[type=\"file\"][attr.required][formControl],\n    input[type=\"file\"][attr.required][ngModel],\n    input[type=\"file\"][required][formControlName],\n    input[type=\"file\"][required][formControl],\n    input[type=\"file\"][required][ngModel]\n  `,\n  exportAs:  'ngxFileRequiredDirective',\n  providers: [\n    {\n      provide:     NG_VALIDATORS,\n      useExisting: NgxFileRequiredDirective,\n      multi:       true\n    }\n  ]\n})\nexport class NgxFileRequiredDirective implements Validator, OnInit, OnDestroy, OnChanges {\n\n  public get required(): HTMLFileInputAttribute {\n    return this._required || this._element.nativeElement.hasAttribute('required');\n  }\n\n  public set required(value: HTMLFileInputAttribute) {\n    this._required = value || this._element.nativeElement.hasAttribute('required');\n  }\n\n  @Input()\n  public requiredErrorMsg = 'File is required';\n\n  @Input()\n  public get multiple(): HTMLFileInputAttribute {\n    return this._element.nativeElement.hasAttribute('multiple');\n  }\n\n  public set multiple(value: HTMLFileInputAttribute) {\n    this._multiple = value === '' || !!value;\n  }\n\n  private _required = false;\n\n  private _multiple = false;\n\n  private _element: ElementRef;\n\n  private _control: AbstractControl;\n\n  private _mutationObserver: MutationObserver;\n\n  public constructor(element: ElementRef) {\n    this._element = element;\n  }\n\n  public ngOnInit(): void {\n    this._mutationObserver = new MutationObserver((mutations: MutationRecord[]): void => {\n      this._setValidity(this._getInputValue(this._element.nativeElement as FileInputEventTarget));\n    });\n\n    this._mutationObserver.observe(this._element.nativeElement, {\n      attributes: true,\n      attributeOldValue: true,\n      attributeFilter: ['required']\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this._mutationObserver.disconnect();\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    if (\n      this.required &&\n      (changes.requiredErrorMsg && !changes.requiredErrorMsg.firstChange)\n    ) {\n      this._setValidity(this._getInputValue(this._element.nativeElement as FileInputEventTarget));\n    }\n  }\n\n  public validate(control: AbstractControl): ValidationErrors {\n    if (!this._control) {\n      this._control = control;\n    }\n\n    if (this._hasError(this._control.value)) {\n      return {\n        required: this.requiredErrorMsg\n      } as ValidationErrors;\n    }\n  }\n\n  @HostListener('change', ['$event.target'])\n  public onChange(eventTarget: EventTarget): void {\n    const value: File|FileList|undefined = this._getInputValue(eventTarget as FileInputEventTarget);\n    this._setValidity(value);\n  }\n\n  private _setValidity(value: File|FileList|undefined): void {\n    const errors: ValidationErrors = Object.assign({}, this._control.errors);\n\n    if (this._hasError(value)) {\n      errors.required = this.requiredErrorMsg;\n    } else {\n      if (this._control.hasError('required')) {\n        delete errors.required;\n      }\n    }\n\n    this._control.setErrors(Object.keys(errors).length ? errors : null);\n  }\n\n  private _hasError(value: File|FileList|undefined): boolean {\n    return this.required && !this._hasValue(value);\n  }\n\n  private _hasValue(value: File|FileList|undefined): boolean {\n    return this.multiple ?\n      value instanceof FileList && !!value.length :\n      value instanceof File;\n  }\n\n  private _getInputValue(eventTarget: FileInputEventTarget): File|FileList|undefined {\n    return this.multiple ? eventTarget.files : eventTarget.files.item(0);\n  }\n\n}\n"]}