UNPKG

@ngez/core

Version:

A collection of minimalistic, easy-to-use and fully customizable Angular components, directives and services

235 lines 25.2 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import { FormGroup } from '@angular/forms'; // @dynamic export class NgEzValidators { /** * @param {?} c1 * @param {?} c2 * @return {?} */ static equals(c1, c2) { return (control) => { const { value } = control; if (!value || !(control instanceof FormGroup)) return null; /** @type {?} */ const formControl1 = control.get(c1); /** @type {?} */ const formControl2 = control.get(c2); if (!formControl1 || !formControl2) return null; return formControl1.value == formControl2.value ? null : { equals: true }; }; } /** * @param {?} control * @return {?} */ static url(control) { if (!control.value) return null; /** @type {?} */ const urlRegex = new RegExp("^" + // protocol identifier (optional) // short syntax // still required "(?:(?:(?:https?|ftp):)?\\/\\/)" + // user:pass BasicAuth (optional) "(?:\\S+(?::\\S*)?@)?" + "(?:" + // IP address exclusion // private & local networks "(?!(?:10|127)(?:\\.\\d{1,3}){3})" + "(?!(?:169\\.254|192\\.168)(?:\\.\\d{1,3}){2})" + "(?!172\\.(?:1[6-9]|2\\d|3[0-1])(?:\\.\\d{1,3}){2})" + // IP address dotted notation octets // excludes loopback network 0.0.0.0 // excludes reserved space >= 224.0.0.0 // excludes network & broacast addresses // (first & last IP address of each class) "(?:[1-9]\\d?|1\\d\\d|2[01]\\d|22[0-3])" + "(?:\\.(?:1?\\d{1,2}|2[0-4]\\d|25[0-5])){2}" + "(?:\\.(?:[1-9]\\d?|1\\d\\d|2[0-4]\\d|25[0-4]))" + "|" + // host & domain names, may end with dot // can be replaced by a shortest alternative // (?![-_])(?:[-\\w\\u00a1-\\uffff]{0,63}[^-_]\\.)+ "(?:" + "(?:" + "[a-z0-9\\u00a1-\\uffff]" + "[a-z0-9\\u00a1-\\uffff_-]{0,62}" + ")?" + "[a-z0-9\\u00a1-\\uffff]\\." + ")+" + // TLD identifier name, may end with dot "(?:[a-z\\u00a1-\\uffff]{2,}\\.?)" + ")" + // port number (optional) "(?::\\d{2,5})?" + // resource path (optional) "(?:[/?#]\\S*)?" + "$", "i"); return urlRegex.test(control.value) ? null : { url: true }; } /** * @param {?} accept * @return {?} */ static fileType(accept) { if (!accept) return (control) => null; /** @type {?} */ const AUDIO_WILDCARD = "audio/*"; /** @type {?} */ const VIDEO_WILDCARD = "video/*"; /** @type {?} */ const IMAGE_WILDCARD = "image/*"; /** @type {?} */ const extensions = []; /** @type {?} */ const mimeTypes = []; /** @type {?} */ let anyAudio = false; /** @type {?} */ let anyVideo = false; /** @type {?} */ let anyImage = false; accept .split(",") .map(accept => accept.trim()) .filter(accept => accept) .forEach(accept => { //is an extension if (/^\./.test(accept)) return extensions.push(accept); //is a wildcard switch (accept) { case AUDIO_WILDCARD: return (anyAudio = true); case VIDEO_WILDCARD: return (anyVideo = true); case IMAGE_WILDCARD: return (anyImage = true); } //anything else is a mime-type return mimeTypes.push(accept); }); /** @type {?} */ const validate = (file) => { if ((anyImage && this.getMimeTypeRegex(IMAGE_WILDCARD).test(file.type)) || (anyAudio && this.getMimeTypeRegex(AUDIO_WILDCARD).test(file.type)) || (anyVideo && this.getMimeTypeRegex(VIDEO_WILDCARD).test(file.type)) || mimeTypes.some(type => file.type == type)) return null; /** @type {?} */ const name = file.name.split("."); /** @type {?} */ const fileExtension = name[name.length - 1]; if (extensions.some(extension => extension.toLowerCase().includes(fileExtension.toLowerCase()))) return null; return { fileType: { accept, actualFile: file } }; }; return (control) => { const { value } = control; if (!value || !(value instanceof File || (Array.isArray(value) && value.every(value => value instanceof File)) || value instanceof FileList)) return null; if (value instanceof File) return validate(value); /** @type {?} */ const files = value instanceof FileList ? Array.from(value) : value; /** @type {?} */ const errors = files.map(file => validate(file)).filter(file => file); return errors.length ? errors : null; }; } /** * @param {?} bytes * @return {?} */ static maxSize(bytes) { /** @type {?} */ const validate = (file) => { return file.size > bytes ? { maxSize: { requiredSize: bytes, actualFile: file } } : null; }; return (control) => { const { value } = control; if (!value || !bytes || !(value instanceof File || (Array.isArray(value) && value.every(value => value instanceof File)) || value instanceof FileList)) return null; if (value instanceof File) return validate(value); /** @type {?} */ const files = value instanceof FileList ? Array.from(value) : value; /** @type {?} */ const errors = files.map(file => validate(file)).filter(file => file); return errors.length ? errors : null; }; } /** * @param {?} bytes * @return {?} */ static totalSize(bytes) { return (control) => { const { value } = control; if (!value || !bytes || !(value instanceof File || (Array.isArray(value) && value.every(value => value instanceof File)) || value instanceof FileList)) return null; /** @type {?} */ const files = value instanceof FileList ? Array.from(value) : Array.isArray(value) ? value : [value]; /** @type {?} */ const totalBytes = files.reduce((total, file) => total + file.size, 0); return totalBytes <= bytes ? null : { totalSize: { requiredTotal: bytes, actualTotal: totalBytes } }; }; } /** * @private * @param {?} str * @return {?} */ static getMimeTypeRegex(str) { return new RegExp(str.replace("*", ".*"), "i"); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validators.js","sourceRoot":"ng://@ngez/core/","sources":["forms/validators.ts"],"names":[],"mappings":";;;;AAAA,OAAO,EAAmB,SAAS,EAAE,MAAM,gBAAgB,CAAC;;AAG5D,MAAM,OAAO,cAAc;;;;;;IACzB,MAAM,CAAC,MAAM,CAAC,EAAqB,EAAE,EAAqB;QACxD,OAAO,CAAC,OAAkB,EAAE,EAAE;kBACtB,EAAE,KAAK,EAAE,GAAG,OAAO;YAEzB,IAAI,CAAC,KAAK,IAAI,CAAC,CAAC,OAAO,YAAY,SAAS,CAAC;gBAAE,OAAO,IAAI,CAAC;;kBAErD,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;;kBAC9B,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;YAEpC,IAAI,CAAC,YAAY,IAAI,CAAC,YAAY;gBAAE,OAAO,IAAI,CAAC;YAEhD,OAAO,YAAY,CAAC,KAAK,IAAI,YAAY,CAAC,KAAK;gBAC7C,CAAC,CAAC,IAAI;gBACN,CAAC,CAAC;oBACE,MAAM,EAAE,IAAI;iBACb,CAAC;QACR,CAAC,CAAC;IACJ,CAAC;;;;;IAED,MAAM,CAAC,GAAG,CAAC,OAAwB;QACjC,IAAI,CAAC,OAAO,CAAC,KAAK;YAAE,OAAO,IAAI,CAAC;;cAE1B,QAAQ,GAAG,IAAI,MAAM,CACzB,GAAG;YACD,iCAAiC;YACjC,iCAAiC;YACjC,gCAAgC;YAChC,iCAAiC;YACjC,sBAAsB;YACtB,KAAK;YACL,uBAAuB;YACvB,2BAA2B;YAC3B,kCAAkC;YAClC,+CAA+C;YAC/C,oDAAoD;YACpD,oCAAoC;YACpC,oCAAoC;YACpC,uCAAuC;YACvC,wCAAwC;YACxC,0CAA0C;YAC1C,wCAAwC;YACxC,4CAA4C;YAC5C,gDAAgD;YAChD,GAAG;YACH,wCAAwC;YACxC,4CAA4C;YAC5C,mDAAmD;YACnD,KAAK;YACL,KAAK;YACL,yBAAyB;YACzB,iCAAiC;YACjC,IAAI;YACJ,4BAA4B;YAC5B,IAAI;YACJ,wCAAwC;YACxC,kCAAkC;YAClC,GAAG;YACH,yBAAyB;YACzB,gBAAgB;YAChB,2BAA2B;YAC3B,gBAAgB;YAChB,GAAG,EACL,GAAG,CACJ;QAED,OAAO,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,IAAI,EAAE,CAAC;IAC7D,CAAC;;;;;IAED,MAAM,CAAC,QAAQ,CAAC,MAAc;QAC5B,IAAI,CAAC,MAAM;YAAE,OAAO,CAAC,OAAwB,EAAE,EAAE,CAAC,IAAI,CAAC;;cAEjD,cAAc,GAAG,SAAS;;cAC1B,cAAc,GAAG,SAAS;;cAC1B,cAAc,GAAG,SAAS;;cAC1B,UAAU,GAAa,EAAE;;cACzB,SAAS,GAAa,EAAE;;YAC1B,QAAQ,GAAG,KAAK;;YAChB,QAAQ,GAAG,KAAK;;YAChB,QAAQ,GAAG,KAAK;QAEpB,MAAM;aACH,KAAK,CAAC,GAAG,CAAC;aACV,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;aAC5B,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC;aACxB,OAAO,CAAC,MAAM,CAAC,EAAE;YAChB,iBAAiB;YACjB,IAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC;gBAAE,OAAO,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAEvD,eAAe;YACf,QAAQ,MAAM,EAAE;gBACd,KAAK,cAAc;oBACjB,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,CAAC;gBAC3B,KAAK,cAAc;oBACjB,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,CAAC;gBAC3B,KAAK,cAAc;oBACjB,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,CAAC;aAC5B;YAED,8BAA8B;YAC9B,OAAO,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;;cAEC,QAAQ,GAAG,CAAC,IAAU,EAAE,EAAE;YAC9B,IACE,CAAC,QAAQ,IAAI,IAAI,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACnE,CAAC,QAAQ,IAAI,IAAI,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACnE,CAAC,QAAQ,IAAI,IAAI,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACnE,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC;gBAEzC,OAAO,IAAI,CAAC;;kBAER,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC;;kBAC3B,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC;YAE3C,IACE,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAC1B,SAAS,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC,CAC9D;gBAED,OAAO,IAAI,CAAC;YAEd,OAAO;gBACL,QAAQ,EAAE;oBACR,MAAM;oBACN,UAAU,EAAE,IAAI;iBACjB;aACF,CAAC;QACJ,CAAC;QAED,OAAO,CAAC,OAAwB,EAAE,EAAE;kBAC5B,EAAE,KAAK,EAAE,GAAG,OAAO;YAEzB,IACE,CAAC,KAAK;gBACN,CAAC,CACC,KAAK,YAAY,IAAI;oBACrB,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;wBACnB,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,YAAY,IAAI,CAAC,CAAC;oBAC9C,KAAK,YAAY,QAAQ,CAC1B;gBAED,OAAO,IAAI,CAAC;YAEd,IAAI,KAAK,YAAY,IAAI;gBAAE,OAAO,QAAQ,CAAC,KAAK,CAAC,CAAC;;kBAE5C,KAAK,GACT,KAAK,YAAY,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK;;kBAEjD,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC;YAErE,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC;QACvC,CAAC,CAAC;IACJ,CAAC;;;;;IAED,MAAM,CAAC,OAAO,CAAC,KAAa;;cACpB,QAAQ,GAAG,CAAC,IAAU,EAAE,EAAE;YAC9B,OAAO,IAAI,CAAC,IAAI,GAAG,KAAK;gBACtB,CAAC,CAAC;oBACE,OAAO,EAAE;wBACP,YAAY,EAAE,KAAK;wBACnB,UAAU,EAAE,IAAI;qBACjB;iBACF;gBACH,CAAC,CAAC,IAAI,CAAC;QACX,CAAC;QAED,OAAO,CAAC,OAAwB,EAAE,EAAE;kBAC5B,EAAE,KAAK,EAAE,GAAG,OAAO;YAEzB,IACE,CAAC,KAAK;gBACN,CAAC,KAAK;gBACN,CAAC,CACC,KAAK,YAAY,IAAI;oBACrB,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;wBACnB,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,YAAY,IAAI,CAAC,CAAC;oBAC9C,KAAK,YAAY,QAAQ,CAC1B;gBAED,OAAO,IAAI,CAAC;YAEd,IAAI,KAAK,YAAY,IAAI;gBAAE,OAAO,QAAQ,CAAC,KAAK,CAAC,CAAC;;kBAE5C,KAAK,GAAG,KAAK,YAAY,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK;;kBAE7D,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC;YAErE,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC;QACvC,CAAC,CAAC;IACJ,CAAC;;;;;IAED,MAAM,CAAC,SAAS,CAAC,KAAa;QAC5B,OAAO,CAAC,OAAwB,EAAE,EAAE;kBAC5B,EAAE,KAAK,EAAE,GAAG,OAAO;YAEzB,IACE,CAAC,KAAK;gBACN,CAAC,KAAK;gBACN,CAAC,CACC,KAAK,YAAY,IAAI;oBACrB,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;wBACnB,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,YAAY,IAAI,CAAC,CAAC;oBAC9C,KAAK,YAAY,QAAQ,CAC1B;gBAED,OAAO,IAAI,CAAC;;kBAER,KAAK,GACT,KAAK,YAAY,QAAQ;gBACvB,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC;gBACnB,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;oBACtB,CAAC,CAAC,KAAK;oBACP,CAAC,CAAC,CAAC,KAAK,CAAC;;kBAEP,UAAU,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;YAEtE,OAAO,UAAU,IAAI,KAAK;gBACxB,CAAC,CAAC,IAAI;gBACN,CAAC,CAAC;oBACE,SAAS,EAAE;wBACT,aAAa,EAAE,KAAK;wBACpB,WAAW,EAAE,UAAU;qBACxB;iBACF,CAAC;QACR,CAAC,CAAC;IACJ,CAAC;;;;;;IAEO,MAAM,CAAC,gBAAgB,CAAC,GAAW;QACzC,OAAO,IAAI,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC;IACjD,CAAC;CACF","sourcesContent":["import { AbstractControl, FormGroup } from '@angular/forms';\r\n\r\n// @dynamic\r\nexport class NgEzValidators {\r\n  static equals(c1: string | string[], c2: string | string[]) {\r\n    return (control: FormGroup) => {\r\n      const { value } = control;\r\n\r\n      if (!value || !(control instanceof FormGroup)) return null;\r\n\r\n      const formControl1 = control.get(c1);\r\n      const formControl2 = control.get(c2);\r\n\r\n      if (!formControl1 || !formControl2) return null;\r\n\r\n      return formControl1.value == formControl2.value\r\n        ? null\r\n        : {\r\n            equals: true\r\n          };\r\n    };\r\n  }\r\n\r\n  static url(control: AbstractControl) {\r\n    if (!control.value) return null;\r\n\r\n    const urlRegex = new RegExp(\r\n      \"^\" +\r\n        // protocol identifier (optional)\r\n        // short syntax // still required\r\n        \"(?:(?:(?:https?|ftp):)?\\\\/\\\\/)\" +\r\n        // user:pass BasicAuth (optional)\r\n        \"(?:\\\\S+(?::\\\\S*)?@)?\" +\r\n        \"(?:\" +\r\n        // IP address exclusion\r\n        // private & local networks\r\n        \"(?!(?:10|127)(?:\\\\.\\\\d{1,3}){3})\" +\r\n        \"(?!(?:169\\\\.254|192\\\\.168)(?:\\\\.\\\\d{1,3}){2})\" +\r\n        \"(?!172\\\\.(?:1[6-9]|2\\\\d|3[0-1])(?:\\\\.\\\\d{1,3}){2})\" +\r\n        // IP address dotted notation octets\r\n        // excludes loopback network 0.0.0.0\r\n        // excludes reserved space >= 224.0.0.0\r\n        // excludes network & broacast addresses\r\n        // (first & last IP address of each class)\r\n        \"(?:[1-9]\\\\d?|1\\\\d\\\\d|2[01]\\\\d|22[0-3])\" +\r\n        \"(?:\\\\.(?:1?\\\\d{1,2}|2[0-4]\\\\d|25[0-5])){2}\" +\r\n        \"(?:\\\\.(?:[1-9]\\\\d?|1\\\\d\\\\d|2[0-4]\\\\d|25[0-4]))\" +\r\n        \"|\" +\r\n        // host & domain names, may end with dot\r\n        // can be replaced by a shortest alternative\r\n        // (?![-_])(?:[-\\\\w\\\\u00a1-\\\\uffff]{0,63}[^-_]\\\\.)+\r\n        \"(?:\" +\r\n        \"(?:\" +\r\n        \"[a-z0-9\\\\u00a1-\\\\uffff]\" +\r\n        \"[a-z0-9\\\\u00a1-\\\\uffff_-]{0,62}\" +\r\n        \")?\" +\r\n        \"[a-z0-9\\\\u00a1-\\\\uffff]\\\\.\" +\r\n        \")+\" +\r\n        // TLD identifier name, may end with dot\r\n        \"(?:[a-z\\\\u00a1-\\\\uffff]{2,}\\\\.?)\" +\r\n        \")\" +\r\n        // port number (optional)\r\n        \"(?::\\\\d{2,5})?\" +\r\n        // resource path (optional)\r\n        \"(?:[/?#]\\\\S*)?\" +\r\n        \"$\",\r\n      \"i\"\r\n    );\r\n\r\n    return urlRegex.test(control.value) ? null : { url: true };\r\n  }\r\n\r\n  static fileType(accept: string) {\r\n    if (!accept) return (control: AbstractControl) => null;\r\n\r\n    const AUDIO_WILDCARD = \"audio/*\";\r\n    const VIDEO_WILDCARD = \"video/*\";\r\n    const IMAGE_WILDCARD = \"image/*\";\r\n    const extensions: string[] = [];\r\n    const mimeTypes: string[] = [];\r\n    let anyAudio = false;\r\n    let anyVideo = false;\r\n    let anyImage = false;\r\n\r\n    accept\r\n      .split(\",\")\r\n      .map(accept => accept.trim())\r\n      .filter(accept => accept)\r\n      .forEach(accept => {\r\n        //is an extension\r\n        if (/^\\./.test(accept)) return extensions.push(accept);\r\n\r\n        //is a wildcard\r\n        switch (accept) {\r\n          case AUDIO_WILDCARD:\r\n            return (anyAudio = true);\r\n          case VIDEO_WILDCARD:\r\n            return (anyVideo = true);\r\n          case IMAGE_WILDCARD:\r\n            return (anyImage = true);\r\n        }\r\n\r\n        //anything else is a mime-type\r\n        return mimeTypes.push(accept);\r\n      });\r\n\r\n    const validate = (file: File) => {\r\n      if (\r\n        (anyImage && this.getMimeTypeRegex(IMAGE_WILDCARD).test(file.type)) ||\r\n        (anyAudio && this.getMimeTypeRegex(AUDIO_WILDCARD).test(file.type)) ||\r\n        (anyVideo && this.getMimeTypeRegex(VIDEO_WILDCARD).test(file.type)) ||\r\n        mimeTypes.some(type => file.type == type)\r\n      )\r\n        return null;\r\n\r\n      const name = file.name.split(\".\");\r\n      const fileExtension = name[name.length - 1];\r\n\r\n      if (\r\n        extensions.some(extension =>\r\n          extension.toLowerCase().includes(fileExtension.toLowerCase())\r\n        )\r\n      )\r\n        return null;\r\n\r\n      return {\r\n        fileType: {\r\n          accept,\r\n          actualFile: file\r\n        }\r\n      };\r\n    };\r\n\r\n    return (control: AbstractControl) => {\r\n      const { value } = control;\r\n\r\n      if (\r\n        !value ||\r\n        !(\r\n          value instanceof File ||\r\n          (Array.isArray(value) &&\r\n            value.every(value => value instanceof File)) ||\r\n          value instanceof FileList\r\n        )\r\n      )\r\n        return null;\r\n\r\n      if (value instanceof File) return validate(value);\r\n\r\n      const files: File[] =\r\n        value instanceof FileList ? Array.from(value) : value;\r\n\r\n      const errors = files.map(file => validate(file)).filter(file => file);\r\n\r\n      return errors.length ? errors : null;\r\n    };\r\n  }\r\n\r\n  static maxSize(bytes: number) {\r\n    const validate = (file: File) => {\r\n      return file.size > bytes\r\n        ? {\r\n            maxSize: {\r\n              requiredSize: bytes,\r\n              actualFile: file\r\n            }\r\n          }\r\n        : null;\r\n    };\r\n\r\n    return (control: AbstractControl) => {\r\n      const { value } = control;\r\n\r\n      if (\r\n        !value ||\r\n        !bytes ||\r\n        !(\r\n          value instanceof File ||\r\n          (Array.isArray(value) &&\r\n            value.every(value => value instanceof File)) ||\r\n          value instanceof FileList\r\n        )\r\n      )\r\n        return null;\r\n\r\n      if (value instanceof File) return validate(value);\r\n\r\n      const files = value instanceof FileList ? Array.from(value) : value;\r\n\r\n      const errors = files.map(file => validate(file)).filter(file => file);\r\n\r\n      return errors.length ? errors : null;\r\n    };\r\n  }\r\n\r\n  static totalSize(bytes: number) {\r\n    return (control: AbstractControl) => {\r\n      const { value } = control;\r\n\r\n      if (\r\n        !value ||\r\n        !bytes ||\r\n        !(\r\n          value instanceof File ||\r\n          (Array.isArray(value) &&\r\n            value.every(value => value instanceof File)) ||\r\n          value instanceof FileList\r\n        )\r\n      )\r\n        return null;\r\n\r\n      const files: File[] =\r\n        value instanceof FileList\r\n          ? Array.from(value)\r\n          : Array.isArray(value)\r\n          ? value\r\n          : [value];\r\n\r\n      const totalBytes = files.reduce((total, file) => total + file.size, 0);\r\n\r\n      return totalBytes <= bytes\r\n        ? null\r\n        : {\r\n            totalSize: {\r\n              requiredTotal: bytes,\r\n              actualTotal: totalBytes\r\n            }\r\n          };\r\n    };\r\n  }\r\n\r\n  private static getMimeTypeRegex(str: string) {\r\n    return new RegExp(str.replace(\"*\", \".*\"), \"i\");\r\n  }\r\n}\r\n"]}