@ngez/core
Version:
A collection of minimalistic, easy-to-use and fully customizable Angular components, directives and services
235 lines • 25.2 kB
JavaScript
/**
* @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":"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","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"]}