ngx-file-required
Version:
Angular input file required directive
236 lines (228 loc) • 18.4 kB
JavaScript
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('@angular/core'), require('@angular/forms')) :
typeof define === 'function' && define.amd ? define('ngx-file-required', ['exports', '@angular/core', '@angular/forms'], factory) :
(factory((global['ngx-file-required'] = {}),global.ng.core,global.ng.forms));
}(this, (function (exports,core,forms) { 'use strict';
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
var NgxFileRequiredDirective = (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: core.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: forms.NG_VALIDATORS,
useExisting: NgxFileRequiredDirective,
multi: true
}
]
},] },
];
/** @nocollapse */
NgxFileRequiredDirective.ctorParameters = function () {
return [
{ type: core.ElementRef }
];
};
NgxFileRequiredDirective.propDecorators = {
requiredErrorMsg: [{ type: core.Input }],
multiple: [{ type: core.Input }],
onChange: [{ type: core.HostListener, args: ['change', ['$event.target'],] }]
};
return NgxFileRequiredDirective;
}());
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
var NgxFileRequiredModule = (function () {
function NgxFileRequiredModule() {
}
NgxFileRequiredModule.decorators = [
{ type: core.NgModule, args: [{
imports: [],
declarations: [NgxFileRequiredDirective],
exports: [NgxFileRequiredDirective]
},] },
];
return NgxFileRequiredModule;
}());
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
exports.NgxFileRequiredDirective = NgxFileRequiredDirective;
exports.NgxFileRequiredModule = NgxFileRequiredModule;
Object.defineProperty(exports, '__esModule', { value: true });
})));
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"ngx-file-required.umd.js.map","sources":["ng://ngx-file-required/lib/ngx-file-required.directive.ts","ng://ngx-file-required/lib/ngx-file-required.module.ts"],"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","import { NgModule } from '@angular/core';\nimport { NgxFileRequiredDirective } from './ngx-file-required.directive';\n\n@NgModule({\n  imports: [\n  ],\n  declarations: [NgxFileRequiredDirective],\n  exports: [NgxFileRequiredDirective]\n})\nexport class NgxFileRequiredModule { }\n"],"names":["Directive","NG_VALIDATORS","ElementRef","Input","HostListener","NgModule"],"mappings":";;;;;;;;;;AAAA;0CAwEqB,OAAmB;oCArBZ,kBAAkB;6BAWxB,KAAK;6BAEL,KAAK;YASvB,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;;8BA/Bf,8CAAQ;;;;gBACjB,OAAO,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;;;;;0BAG5D,KAA6B;gBAC/C,IAAI,CAAC,SAAS,GAAG,KAAK,IAAI,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;;;;;QAMjF,sBACW,8CAAQ;;;gBADnB;gBAEE,OAAO,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;aAC7D;;;;0BAEmB,KAA6B;gBAC/C,IAAI,CAAC,SAAS,GAAG,KAAK,KAAK,EAAE,IAAI,CAAC,CAAC,KAAK,CAAC;;;;WAH1C;;;;QAoBM,2CAAQ;;;;;gBACb,IAAI,CAAC,iBAAiB,GAAG,IAAI,gBAAgB,CAAC,UAAC,SAA2B;oBACxE,KAAI,CAAC,YAAY,CAAC,KAAI,CAAC,cAAc,mBAAC,KAAI,CAAC,QAAQ,CAAC,aAAqC,EAAC,CAAC,CAAC;iBAC7F,CAAC,CAAC;gBAEH,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE;oBAC1D,UAAU,EAAE,IAAI;oBAChB,iBAAiB,EAAE,IAAI;oBACvB,eAAe,EAAE,CAAC,UAAU,CAAC;iBAC9B,CAAC,CAAC;;;;;QAGE,8CAAW;;;;gBAChB,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,CAAC;;;;;;QAG/B,8CAAW;;;;sBAAC,OAAsB;gBACvC,IACE,IAAI,CAAC,QAAQ;qBACZ,OAAO,wBAAqB,CAAC,OAAO,qBAAkB,WAAW,CACpE,EAAE;oBACA,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,cAAc,mBAAC,IAAI,CAAC,QAAQ,CAAC,aAAqC,EAAC,CAAC,CAAC;iBAC7F;;;;;;QAGI,2CAAQ;;;;sBAAC,OAAwB;gBACtC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;oBAClB,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;iBACzB;gBAED,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;oBACvC,yBAAO;wBACL,QAAQ,EAAE,IAAI,CAAC,gBAAgB;qBACZ,EAAC;iBACvB;;;;;;QAII,2CAAQ;;;;YADf,UACgB,WAAwB;;gBACtC,IAAM,KAAK,GAA4B,IAAI,CAAC,cAAc,mBAAC,WAAmC,EAAC,CAAC;gBAChG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;aAC1B;;;;;QAEO,+CAAY;;;;sBAAC,KAA8B;;gBACjD,IAAM,MAAM,GAAqB,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;gBAEzE,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;oBACzB,MAAM,eAAY,IAAI,CAAC,gBAAgB,CAAC;iBACzC;qBAAM;oBACL,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;wBACtC,OAAO,MAAM,YAAS,CAAC;qBACxB;iBACF;gBAED,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,GAAG,MAAM,GAAG,IAAI,CAAC,CAAC;;;;;;QAG9D,4CAAS;;;;sBAAC,KAA8B;gBAC9C,OAAO,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;;;;;;QAGzC,4CAAS;;;;sBAAC,KAA8B;gBAC9C,OAAO,IAAI,CAAC,QAAQ;oBAClB,KAAK,YAAY,QAAQ,IAAI,CAAC,CAAC,KAAK,CAAC,MAAM;oBAC3C,KAAK,YAAY,IAAI,CAAC;;;;;;QAGlB,iDAAc;;;;sBAAC,WAAiC;gBACtD,OAAO,IAAI,CAAC,QAAQ,GAAG,WAAW,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;;;oBA1HxEA,cAAS,SAAC;wBACT,QAAQ,EAAE,gUAOT;wBACD,QAAQ,EAAG,0BAA0B;wBACrC,SAAS,EAAE;4BACT;gCACE,OAAO,EAAMC,mBAAa;gCAC1B,WAAW,EAAE,wBAAwB;gCACrC,KAAK,EAAQ,IAAI;6BAClB;yBACF;qBACF;;;;;wBAjCCC,eAAU;;;;uCA4CTC,UAAK;+BAGLA,UAAK;+BA4DLC,iBAAY,SAAC,QAAQ,EAAE,CAAC,eAAe,CAAC;;uCAjH3C;;;;;;;ACAA;;;;oBAGCC,aAAQ,SAAC;wBACR,OAAO,EAAE,EACR;wBACD,YAAY,EAAE,CAAC,wBAAwB,CAAC;wBACxC,OAAO,EAAE,CAAC,wBAAwB,CAAC;qBACpC;;oCARD;;;;;;;;;;;;;;;;;;;;;;;;"}