| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975 |
- /*
- Input Mask plugin extensions
- http://github.com/RobinHerbots/inputmask
- Copyright (c) Robin Herbots
- Licensed under the MIT license
- */
- import escapeRegex from "../escapeRegex";
- import Inputmask from "../inputmask";
- import { keyCode, keys } from "../keycode.js";
- import { seekNext } from "../positioning";
- import { getMaskTemplate, getTest } from "../validation-tests";
- import "./inputmask.date.i18n";
- const $ = Inputmask.dependencyLib;
- class DateObject {
- constructor(mask, format, opts, inputmask) {
- this.mask = mask;
- this.format = format;
- this.opts = opts;
- this.inputmask = inputmask;
- this._date = new Date(1, 0, 1);
- this.initDateObject(mask, this.opts, this.inputmask);
- }
- get date() {
- if (this._date === undefined) {
- this._date = new Date(1, 0, 1);
- this.initDateObject(undefined, this.opts, this.inputmask);
- }
- return this._date;
- }
- initDateObject(mask, opts, inputmask) {
- let match;
- getTokenizer(opts).lastIndex = 0;
- while ((match = getTokenizer(opts).exec(this.format))) {
- let dynMatches = /\d+$/.exec(match[0]),
- fcode = dynMatches ? match[0][0] + "x" : match[0],
- value;
- if (mask !== undefined) {
- // console.log("mask", mask);
- if (dynMatches) {
- const lastIndex = getTokenizer(opts).lastIndex,
- tokenMatch = getTokenMatch.call(
- inputmask,
- match.index,
- opts,
- inputmask && inputmask.maskset
- );
- getTokenizer(opts).lastIndex = lastIndex;
- value = mask.slice(0, mask.indexOf(tokenMatch.nextMatch[0]));
- } else {
- let targetSymbol = match[0][0],
- ndx = match.index;
- while (
- inputmask &&
- (opts.placeholder[getTest.call(inputmask, ndx).match.placeholder] ||
- getTest.call(inputmask, ndx).match.placeholder) === targetSymbol
- ) {
- ndx++;
- }
- const targetMatchLength = ndx - match.index;
- value = mask.slice(
- 0,
- targetMatchLength ||
- (formatCode[fcode] && formatCode[fcode][4]) ||
- fcode.length
- );
- }
- mask = mask.slice(value.length);
- }
- if (Object.prototype.hasOwnProperty.call(formatCode, fcode)) {
- this.setValue(
- this,
- value,
- fcode,
- formatCode[fcode][2],
- formatCode[fcode][1]
- );
- }
- }
- }
- setValue(dateObj, value, fcode, targetProp, dateOperation) {
- if (value !== undefined) {
- switch (targetProp) {
- case "ampm":
- dateObj[targetProp] = value;
- dateObj["raw" + targetProp] = value.replace(/\s/g, "_");
- break;
- case "month":
- if (fcode === "mmm" || fcode === "mmmm") {
- fcode === "mmm"
- ? (dateObj[targetProp] = pad(
- i18n.monthNames
- .slice(0, 12)
- .findIndex(
- (item) => value.toLowerCase() === item.toLowerCase()
- ) + 1,
- 2
- ))
- : (dateObj[targetProp] = pad(
- i18n.monthNames
- .slice(12, 24)
- .findIndex(
- (item) => value.toLowerCase() === item.toLowerCase()
- ) + 1,
- 2
- ));
- dateObj[targetProp] =
- dateObj[targetProp] === "00"
- ? ""
- : dateObj[targetProp].toString();
- dateObj["raw" + targetProp] = dateObj[targetProp];
- break;
- }
- // eslint-disable-next-line no-fallthrough
- default:
- dateObj[targetProp] = value.replace(/[^0-9]/g, "0");
- dateObj["raw" + targetProp] = value.replace(/\s/g, "_");
- }
- }
- if (dateOperation !== undefined) {
- let datavalue = dateObj[targetProp];
- if (
- (targetProp === "day" && parseInt(datavalue) === 29) ||
- (targetProp === "month" && parseInt(datavalue) === 2)
- ) {
- if (
- parseInt(dateObj.day) === 29 &&
- parseInt(dateObj.month) === 2 &&
- (dateObj.year === "" || dateObj.year === undefined)
- ) {
- // set temporary leap year in dateObj
- dateObj._date.setFullYear(2012, 1, 29);
- }
- }
- if (targetProp === "day") {
- useDateObject = true;
- if (parseInt(datavalue) === 0) datavalue = 1;
- }
- if (targetProp === "month") useDateObject = true;
- if (targetProp === "year") {
- useDateObject = true;
- if (datavalue.length < formatCode[fcode][4])
- datavalue = pad(datavalue, formatCode[fcode][4], true);
- }
- if ((datavalue !== "" && !isNaN(datavalue)) || targetProp === "ampm")
- dateOperation.call(dateObj._date, datavalue);
- }
- }
- reset() {
- this._date = new Date(1, 0, 1);
- }
- reInit() {
- this._date = undefined;
- // eslint-disable-next-line no-unused-expressions
- this.date;
- }
- }
- let currentYear = new Date().getFullYear(),
- i18n = Inputmask.prototype.i18n,
- useDateObject = false, // supported codes for formatting
- // http://blog.stevenlevithan.com/archives/date-time-format
- // https://docs.microsoft.com/en-us/dotnet/standard/base-types/custom-date-and-time-format-strings?view=netframework-4.7
- formatCode = {
- // regex, valueSetter, type, displayformatter, #entries (optional)
- d: [
- "[1-9]|[12][0-9]|3[01]",
- Date.prototype.setDate,
- "day",
- Date.prototype.getDate
- ], // Day of the month as digits; no leading zero for single-digit days.
- dd: [
- "0[1-9]|[12][0-9]|3[01]",
- Date.prototype.setDate,
- "day",
- function () {
- return pad(Date.prototype.getDate.call(this), 2);
- }
- ], // Day of the month as digits; leading zero for single-digit days.
- ddd: [""], // Day of the week as a three-letter abbreviation.
- dddd: [""], // Day of the week as its full name.
- m: [
- "[1-9]|1[012]",
- function (val) {
- let mval = val ? parseInt(val) : 0;
- if (mval > 0) mval--;
- return Date.prototype.setMonth.call(this, mval);
- },
- "month",
- function () {
- return Date.prototype.getMonth.call(this) + 1;
- }
- ], // Month as digits; no leading zero for single-digit months.
- mm: [
- "0[1-9]|1[012]",
- function (val) {
- let mval = val ? parseInt(val) : 0;
- if (mval > 0) mval--;
- return Date.prototype.setMonth.call(this, mval);
- },
- "month",
- function () {
- return pad(Date.prototype.getMonth.call(this) + 1, 2);
- }
- ], // Month as digits; leading zero for single-digit months.
- mmm: [
- i18n.monthNames.slice(0, 12).join("|"),
- function (val) {
- const mval = i18n.monthNames
- .slice(0, 12)
- .findIndex((item) => val.toLowerCase() === item.toLowerCase());
- return mval !== -1 ? Date.prototype.setMonth.call(this, mval) : false;
- },
- "month",
- function () {
- return i18n.monthNames.slice(0, 12)[Date.prototype.getMonth.call(this)];
- }
- ], // Month as a three-letter abbreviation.
- mmmm: [
- i18n.monthNames.slice(12, 24).join("|"),
- function (val) {
- const mval = i18n.monthNames
- .slice(12, 24)
- .findIndex((item) => val.toLowerCase() === item.toLowerCase());
- return mval !== -1 ? Date.prototype.setMonth.call(this, mval) : false;
- },
- "month",
- function () {
- return i18n.monthNames.slice(12, 24)[
- Date.prototype.getMonth.call(this)
- ];
- }
- ], // Month as its full name.
- yy: [
- "[0-9]{2}",
- function (val) {
- const centuryPart = new Date().getFullYear().toString().slice(0, 2);
- Date.prototype.setFullYear.call(this, `${centuryPart}${val}`);
- },
- "year",
- function () {
- return pad(Date.prototype.getFullYear.call(this), 2);
- },
- 2
- ], // Year as last two digits; leading zero for years less than 10.
- yyyy: [
- "[0-9]{4}",
- Date.prototype.setFullYear,
- "year",
- function () {
- return pad(Date.prototype.getFullYear.call(this), 4);
- },
- 4
- ],
- h: [
- "[1-9]|1[0-2]",
- Date.prototype.setHours,
- "hours",
- Date.prototype.getHours
- ], // Hours; no leading zero for single-digit hours (12-hour clock).
- hh: [
- "0[1-9]|1[0-2]",
- Date.prototype.setHours,
- "hours",
- function () {
- return pad(Date.prototype.getHours.call(this), 2);
- }
- ], // Hours; leading zero for single-digit hours (12-hour clock).
- hx: [
- function (x) {
- return `[0-9]{${x}}`;
- },
- Date.prototype.setHours,
- "hours",
- function (x) {
- return Date.prototype.getHours;
- }
- ], // Hours; no limit; set maximum digits
- H: [
- "1?[0-9]|2[0-3]",
- Date.prototype.setHours,
- "hours",
- Date.prototype.getHours
- ], // Hours; no leading zero for single-digit hours (24-hour clock).
- HH: [
- "0[0-9]|1[0-9]|2[0-3]",
- Date.prototype.setHours,
- "hours",
- function () {
- return pad(Date.prototype.getHours.call(this), 2);
- }
- ], // Hours; leading zero for single-digit hours (24-hour clock).
- Hx: [
- function (x) {
- return `[0-9]{${x}}`;
- },
- Date.prototype.setHours,
- "hours",
- function (x) {
- return function () {
- return pad(Date.prototype.getHours.call(this), x);
- };
- }
- ], // Hours; no limit; set maximum digits
- M: [
- "[1-5]?[0-9]",
- Date.prototype.setMinutes,
- "minutes",
- Date.prototype.getMinutes
- ], // Minutes; no leading zero for single-digit minutes. Uppercase M unlike CF timeFormat's m to avoid conflict with months.
- MM: [
- "0[0-9]|1[0-9]|2[0-9]|3[0-9]|4[0-9]|5[0-9]",
- Date.prototype.setMinutes,
- "minutes",
- function () {
- return pad(Date.prototype.getMinutes.call(this), 2);
- }
- ], // Minutes; leading zero for single-digit minutes. Uppercase MM unlike CF timeFormat's mm to avoid conflict with months.
- s: [
- "[1-5]?[0-9]",
- Date.prototype.setSeconds,
- "seconds",
- Date.prototype.getSeconds
- ], // Seconds; no leading zero for single-digit seconds.
- ss: [
- "0[0-9]|1[0-9]|2[0-9]|3[0-9]|4[0-9]|5[0-9]",
- Date.prototype.setSeconds,
- "seconds",
- function () {
- return pad(Date.prototype.getSeconds.call(this), 2);
- }
- ], // Seconds; leading zero for single-digit seconds.
- l: [
- "[0-9]{3}",
- Date.prototype.setMilliseconds,
- "milliseconds",
- function () {
- return pad(Date.prototype.getMilliseconds.call(this), 3);
- },
- 3
- ], // Milliseconds. 3 digits.
- L: [
- "[0-9]{2}",
- Date.prototype.setMilliseconds,
- "milliseconds",
- function () {
- return pad(Date.prototype.getMilliseconds.call(this), 2);
- },
- 2
- ], // Milliseconds. 2 digits.
- t: ["[ap]", setAMPM, "ampm", getAMPM, 1], // Lowercase, single-character time marker string: a or p.
- tt: ["[ap]m", setAMPM, "ampm", getAMPM, 2], // two-character time marker string: am or pm.
- T: ["[AP]", setAMPM, "ampm", getAMPM, 1], // single-character time marker string: A or P.
- TT: ["[AP]M", setAMPM, "ampm", getAMPM, 2], // two-character time marker string: AM or PM.
- Z: [".*", undefined, "Z", getTimeZoneAbbreviated], // US timezone abbreviation, e.g. EST or MDT. With non-US timezones or in the Opera browser, the GMT/UTC offset is returned, e.g. GMT-0500
- o: [""], // GMT/UTC timezone offset, e.g. -0500 or +0230.
- S: [""] // The date's ordinal suffix (st, nd, rd, or th).
- },
- formatAlias = {
- isoDate: "yyyy-mm-dd", // 2007-06-09
- isoTime: "HH:MM:ss", // 17:46:21
- isoDateTime: "yyyy-mm-dd'T'HH:MM:ss", // 2007-06-09T17:46:21
- isoUtcDateTime: "UTC:yyyy-mm-dd'T'HH:MM:ss'Z'" // 2007-06-09T22:46:21Z
- };
- function setAMPM(value) {
- const hours = this.getHours();
- if (value.toLowerCase().includes("p")) {
- this.setHours(hours + 12);
- // console.log("setAMPM + 12");
- } else if (value.toLowerCase().includes("a") && hours >= 12) {
- this.setHours(hours - 12);
- }
- }
- function getAMPM() {
- let date = this,
- hours = date.getHours();
- hours = hours || 12;
- return hours >= 12 ? "PM" : "AM";
- }
- function getTimeZoneAbbreviated() {
- // not perfect, but ok for now
- let date = this,
- { 1: tz } = date.toString().match(/\((.+)\)/);
- if (tz.includes(" ")) {
- tz = tz.replace("-", " ").toUpperCase();
- tz = tz
- .split(" ")
- .map(([first]) => first)
- .join("");
- }
- return tz;
- }
- function formatcode(match) {
- const dynMatches = /\d+$/.exec(match[0]);
- if (dynMatches && dynMatches[0] !== undefined) {
- const fcode = formatCode[match[0][0] + "x"].slice("");
- fcode[0] = fcode[0](dynMatches[0]);
- fcode[3] = fcode[3](dynMatches[0]);
- return fcode;
- } else if (formatCode[match[0]]) {
- return formatCode[match[0]];
- }
- }
- function getTokenizer(opts) {
- if (!opts.tokenizer) {
- const tokens = [],
- dyntokens = [];
- for (const ndx in formatCode) {
- if (/\.*x$/.test(ndx)) {
- const dynToken = ndx[0] + "\\d+";
- if (dyntokens.indexOf(dynToken) === -1) {
- dyntokens.push(dynToken);
- }
- } else if (tokens.indexOf(ndx[0]) === -1) {
- tokens.push(ndx[0]);
- }
- }
- opts.tokenizer =
- "(" +
- (dyntokens.length > 0 ? dyntokens.join("|") + "|" : "") +
- tokens.join("+|") +
- ")+?|.";
- opts.tokenizer = new RegExp(opts.tokenizer, "g");
- }
- return opts.tokenizer;
- }
- function prefillYear(dateParts, currentResult, opts) {
- if (dateParts.year !== dateParts.rawyear) {
- const crrntyear = currentYear.toString(),
- enteredPart = dateParts.rawyear.replace(/[^0-9]/g, ""),
- currentYearPart = crrntyear.slice(0, enteredPart.length),
- currentYearNextPart = crrntyear.slice(enteredPart.length);
- if (enteredPart.length === 2 && enteredPart === currentYearPart) {
- const entryCurrentYear = new Date(
- currentYear,
- dateParts.month - 1,
- dateParts.day
- );
- if (
- dateParts.day == entryCurrentYear.getDate() &&
- (!opts.max || opts.max.date.getTime() >= entryCurrentYear.getTime())
- ) {
- // update dateParts
- dateParts.date.setFullYear(currentYear);
- dateParts.year = crrntyear;
- // update result
- currentResult.insert = [
- {
- pos: currentResult.pos + 1,
- c: currentYearNextPart[0]
- },
- {
- pos: currentResult.pos + 2,
- c: currentYearNextPart[1]
- }
- ];
- }
- }
- }
- return currentResult;
- }
- function isValidDate(dateParts, currentResult, opts) {
- const inputmask = this;
- if (!useDateObject) return true;
- if (
- dateParts.rawday === undefined ||
- (!isFinite(dateParts.rawday) &&
- new Date(
- dateParts.date.getFullYear(),
- isFinite(dateParts.rawmonth)
- ? dateParts.month
- : dateParts.date.getMonth() + 1,
- 0
- ).getDate() >= dateParts.day) ||
- (dateParts.day == "29" &&
- (!isFinite(dateParts.rawyear) ||
- dateParts.rawyear === undefined ||
- dateParts.rawyear === "")) ||
- new Date(
- dateParts.date.getFullYear(),
- isFinite(dateParts.rawmonth)
- ? dateParts.month
- : dateParts.date.getMonth() + 1,
- 0
- ).getDate() >= dateParts.day
- ) {
- return currentResult;
- } else {
- // take corrective action if possible
- if (dateParts.day == "29") {
- const tokenMatch = getTokenMatch.call(
- inputmask,
- currentResult.pos,
- opts,
- inputmask.maskset
- );
- if (
- tokenMatch.targetMatch &&
- tokenMatch.targetMatch[0] === "yyyy" &&
- currentResult.pos - tokenMatch.targetMatchIndex === 2
- ) {
- currentResult.remove = currentResult.pos + 1;
- return currentResult;
- }
- } else if (
- dateParts.date.getMonth() == 2 &&
- dateParts.day == "30" &&
- currentResult.c !== undefined
- ) {
- dateParts.day = "03";
- dateParts.date.setDate(3);
- dateParts.date.setMonth(1);
- currentResult.insert = [
- { pos: currentResult.pos, c: "0" },
- { pos: currentResult.pos + 1, c: currentResult.c }
- ];
- currentResult.caret = seekNext.call(this, currentResult.pos + 1);
- return currentResult;
- }
- return false;
- }
- }
- function isDateInRange(dateParts, result, opts, maskset, fromCheckval) {
- if (!result) return result;
- if (result && opts.min) {
- if (
- /* useDateObject && (dateParts["year"] === undefined || dateParts["yearSet"]) && */ !isNaN(
- opts.min.date.getTime()
- )
- ) {
- let match;
- dateParts.reset();
- getTokenizer(opts).lastIndex = 0;
- while ((match = getTokenizer(opts).exec(opts.inputFormat))) {
- var fcode;
- if ((fcode = formatcode(match))) {
- if (fcode[3]) {
- let setFn = fcode[1],
- current = dateParts[fcode[2]],
- minVal = opts.min[fcode[2]],
- maxVal = opts.max ? opts.max[fcode[2]] : minVal + 1,
- curVal = [],
- forceCurrentValue = false;
- for (let i = 0; i < minVal.length; i++) {
- if (
- maskset.validPositions[i + match.index] === undefined &&
- !forceCurrentValue
- ) {
- if (i + match.index == 0 && current[i] < minVal[i]) {
- curVal[i] = current[i];
- forceCurrentValue = true;
- } else {
- curVal[i] = minVal[i];
- }
- // ADD +1 to whole
- if (
- fcode[2] === "year" &&
- current.length - 1 == i &&
- minVal != maxVal
- )
- curVal = (parseInt(curVal.join("")) + 1).toString().split("");
- if (
- fcode[2] === "ampm" &&
- minVal != maxVal &&
- opts.min.date.getTime() > dateParts.date.getTime()
- )
- curVal[i] = maxVal[i];
- } else {
- curVal[i] = current[i];
- forceCurrentValue = forceCurrentValue || current[i] > minVal[i];
- }
- }
- setFn.call(dateParts._date, curVal.join(""));
- }
- }
- }
- result = opts.min.date.getTime() <= dateParts.date.getTime();
- dateParts.reInit();
- }
- }
- if (result && opts.max) {
- if (!isNaN(opts.max.date.getTime())) {
- result = opts.max.date.getTime() >= dateParts.date.getTime();
- }
- }
- return result;
- }
- // parse the given format and return a mask pattern
- // when a dateObjValue is passed a datestring in the requested format is returned
- function parse(format, dateObjValue, opts, raw) {
- // parse format to regex string
- let mask = "",
- match,
- fcode,
- ndx = 0,
- placeHolder = {};
- getTokenizer(opts).lastIndex = 0;
- while ((match = getTokenizer(opts).exec(format))) {
- if (dateObjValue === undefined) {
- if ((fcode = formatcode(match))) {
- mask += "(" + fcode[0] + ")";
- // map placeholder to placeholder object and set placeholder mappings
- if (opts.placeholder && opts.placeholder !== "") {
- placeHolder[ndx] =
- opts.placeholder[match.index % opts.placeholder.length];
- placeHolder[opts.placeholder[match.index % opts.placeholder.length]] =
- match[0].charAt(0);
- } else {
- placeHolder[ndx] = match[0].charAt(0);
- }
- } else {
- switch (match[0]) {
- case "[":
- mask += "(";
- break;
- case "]":
- mask += ")?";
- break;
- default:
- mask += escapeRegex(match[0]);
- placeHolder[ndx] = match[0].charAt(0);
- }
- }
- } else {
- if ((fcode = formatcode(match))) {
- if (raw !== true && fcode[3]) {
- const getFn = fcode[3];
- mask += getFn.call(dateObjValue.date);
- } else if (fcode[2]) {
- mask += dateObjValue["raw" + fcode[2]];
- } else {
- mask += match[0];
- }
- } else {
- mask += match[0];
- }
- }
- ndx++;
- }
- if (dateObjValue === undefined) {
- opts.placeholder = placeHolder;
- }
- return mask;
- }
- // padding function
- function pad(val, len, right) {
- val = String(val);
- len = len || 2;
- while (val.length < len) val = right ? val + "0" : "0" + val;
- return val;
- }
- function analyseMask(mask, format, opts) {
- const inputmask = this;
- if (typeof mask === "string") {
- return new DateObject(mask, format, opts, inputmask);
- } else if (
- mask &&
- typeof mask === "object" &&
- Object.prototype.hasOwnProperty.call(mask, "date")
- ) {
- return mask;
- }
- return undefined;
- }
- function importDate(dateObj, opts) {
- return parse(opts.inputFormat, { date: dateObj }, opts);
- }
- function getTokenMatch(pos, opts, maskset) {
- let inputmask = this,
- masksetHint =
- maskset && maskset.tests[pos]
- ? opts.placeholder[maskset.tests[pos][0].match.placeholder] ||
- maskset.tests[pos][0].match.placeholder
- : "",
- calcPos = 0,
- targetMatch,
- match,
- matchLength = 0;
- getTokenizer(opts).lastIndex = 0;
- while ((match = getTokenizer(opts).exec(opts.inputFormat))) {
- const dynMatches = /\d+$/.exec(match[0]);
- if (dynMatches) {
- matchLength = parseInt(dynMatches[0]);
- } else {
- let targetSymbol = match[0][0],
- ndx = calcPos;
- while (
- inputmask &&
- (opts.placeholder[getTest.call(inputmask, ndx).match.placeholder] ||
- getTest.call(inputmask, ndx).match.placeholder) === targetSymbol
- ) {
- ndx++;
- }
- matchLength = ndx - calcPos;
- if (matchLength === 0) matchLength = match[0].length;
- }
- calcPos += matchLength;
- if (match[0].indexOf(masksetHint) != -1 || calcPos >= pos + 1) {
- // console.log("gettokenmatch " + match[0] + " ~ " + (maskset ? maskset.tests[pos][0].match.placeholder : ""));
- targetMatch = match;
- match = getTokenizer(opts).exec(opts.inputFormat);
- break;
- }
- }
- return {
- targetMatchIndex: calcPos - matchLength,
- nextMatch: match,
- targetMatch
- };
- }
- Inputmask.extendAliases({
- datetime: {
- mask: function (opts) {
- // do not allow numeric input in datetime alias
- opts.numericInput = false;
- // localize
- formatCode.S = i18n.ordinalSuffix.join("|");
- opts.inputFormat = formatAlias[opts.inputFormat] || opts.inputFormat; // resolve possible formatAlias
- opts.displayFormat =
- formatAlias[opts.displayFormat] ||
- opts.displayFormat ||
- opts.inputFormat; // resolve possible formatAlias
- opts.outputFormat =
- formatAlias[opts.outputFormat] || opts.outputFormat || opts.inputFormat; // resolve possible formatAlias
- // opts.placeholder = opts.placeholder !== "" ? opts.placeholder : opts.inputFormat.replace(/[[\]]/, "");
- opts.regex = parse(opts.inputFormat, undefined, opts);
- opts.min = analyseMask(opts.min, opts.inputFormat, opts);
- opts.max = analyseMask(opts.max, opts.inputFormat, opts);
- return null; // migrate to regex mask
- },
- placeholder: "", // set default as none (~ auto); when a custom placeholder is passed it will be used
- inputFormat: "isoDateTime", // format used to input the date
- displayFormat: null, // visual format when the input looses focus
- outputFormat: null, // unmasking format
- min: null, // needs to be in the same format as the inputfornat
- max: null, // needs to be in the same format as the inputfornat,
- skipOptionalPartCharacter: "",
- preValidation: function (
- buffer,
- pos,
- c,
- isSelection,
- opts,
- maskset,
- caretPos,
- strict
- ) {
- const inputmask = this;
- if (strict) return true;
- if (isNaN(c) && buffer[pos] !== c) {
- const tokenMatch = getTokenMatch.call(inputmask, pos, opts, maskset);
- if (
- tokenMatch.nextMatch &&
- tokenMatch.nextMatch[0] === c &&
- tokenMatch.targetMatch[0].length > 1
- ) {
- const validator = formatcode(tokenMatch.targetMatch)[0];
- if (new RegExp(validator).test("0" + buffer[pos - 1])) {
- buffer[pos] = buffer[pos - 1];
- buffer[pos - 1] = "0";
- return {
- fuzzy: true,
- buffer,
- refreshFromBuffer: { start: pos - 1, end: pos + 1 },
- pos: pos + 1
- };
- }
- }
- }
- return true;
- },
- postValidation: function (
- buffer,
- pos,
- c,
- currentResult,
- opts,
- maskset,
- strict,
- fromCheckval
- ) {
- const inputmask = this;
- if (strict) return true;
- let tokenMatch, validator;
- if (currentResult === false) {
- // try some shifting
- tokenMatch = getTokenMatch.call(inputmask, pos + 1, opts, maskset);
- if (
- tokenMatch.targetMatch &&
- tokenMatch.targetMatchIndex === pos &&
- tokenMatch.targetMatch[0].length > 1 &&
- formatCode[tokenMatch.targetMatch[0]] !== undefined
- ) {
- validator = formatcode(tokenMatch.targetMatch)[0];
- } else {
- tokenMatch = getTokenMatch.call(inputmask, pos + 2, opts, maskset);
- if (
- tokenMatch.targetMatch &&
- tokenMatch.targetMatchIndex === pos + 1 &&
- tokenMatch.targetMatch[0].length > 1 &&
- formatCode[tokenMatch.targetMatch[0]] !== undefined
- ) {
- validator = formatcode(tokenMatch.targetMatch)[0];
- }
- }
- if (validator !== undefined) {
- if (
- maskset.validPositions[pos + 1] !== undefined &&
- new RegExp(validator).test(c + "0")
- ) {
- buffer[pos] = c;
- buffer[pos + 1] = "0";
- currentResult = {
- // insert: [{pos: pos, c: "0"}, {pos: pos + 1, c: c}],
- pos: pos + 2, // this will triggeer a refreshfrombuffer
- caret: pos
- };
- } else if (new RegExp(validator).test("0" + c)) {
- buffer[pos] = "0";
- buffer[pos + 1] = c;
- currentResult = {
- // insert: [{pos: pos, c: "0"}, {pos: pos + 1, c: c}],
- pos: pos + 2 // this will triggeer a refreshfrombuffer
- };
- }
- }
- if (currentResult === false) return currentResult;
- }
- if (currentResult.fuzzy) {
- buffer = currentResult.buffer;
- pos = currentResult.pos;
- }
- // full validate target
- tokenMatch = getTokenMatch.call(inputmask, pos, opts, maskset);
- if (
- tokenMatch.targetMatch &&
- tokenMatch.targetMatch[0] &&
- formatCode[tokenMatch.targetMatch[0]] !== undefined
- ) {
- const fcode = formatcode(tokenMatch.targetMatch);
- validator = fcode[0];
- const part = buffer.slice(
- tokenMatch.targetMatchIndex,
- tokenMatch.targetMatchIndex + tokenMatch.targetMatch[0].length
- );
- if (
- new RegExp(validator).test(part.join("")) === false &&
- tokenMatch.targetMatch[0].length === 2 &&
- maskset.validPositions[tokenMatch.targetMatchIndex] &&
- maskset.validPositions[tokenMatch.targetMatchIndex + 1]
- ) {
- maskset.validPositions[tokenMatch.targetMatchIndex + 1].input = "0";
- }
- if (fcode[2] == "year") {
- const _buffer = getMaskTemplate.call(
- inputmask,
- false,
- 1,
- undefined,
- true
- );
- for (let i = pos + 1; i < buffer.length; i++) {
- buffer[i] = _buffer[i];
- maskset.validPositions.splice(pos + 1, 1);
- }
- }
- }
- let result = currentResult,
- dateParts = analyseMask.call(
- inputmask,
- buffer.join(""),
- opts.inputFormat,
- opts
- );
- if (result && !isNaN(dateParts.date.getTime())) {
- // check for a valid date ~ an invalid date returns NaN which isn't equal
- if (opts.prefillYear) result = prefillYear(dateParts, result, opts);
- result = isValidDate.call(inputmask, dateParts, result, opts);
- result = isDateInRange(dateParts, result, opts, maskset, fromCheckval);
- }
- if (pos !== undefined && result && currentResult.pos !== pos) {
- return {
- buffer: parse(opts.inputFormat, dateParts, opts).split(""),
- refreshFromBuffer: { start: pos, end: currentResult.pos },
- pos: currentResult.caret || currentResult.pos // correct caret position
- };
- }
- return result;
- },
- onKeyDown: function (e, buffer, caretPos, opts) {
- const input = this;
- if (e.ctrlKey && e.key === keys.ArrowRight) {
- input.inputmask._valueSet(importDate(new Date(), opts));
- $(input).trigger("setvalue");
- }
- },
- onUnMask: function (maskedValue, unmaskedValue, opts) {
- const inputmask = this;
- return unmaskedValue
- ? parse(
- opts.outputFormat,
- analyseMask.call(inputmask, maskedValue, opts.inputFormat, opts),
- opts,
- true
- )
- : unmaskedValue;
- },
- casing: function (elem, test, pos, validPositions) {
- if (test.nativeDef.indexOf("[ap]") == 0) return elem.toLowerCase();
- if (test.nativeDef.indexOf("[AP]") == 0) return elem.toUpperCase();
- const posBefore = getTest.call(this, [pos - 1]);
- if (
- pos === 0 ||
- (posBefore && posBefore.input === String.fromCharCode(keyCode.Space)) ||
- (posBefore &&
- posBefore.match.def === String.fromCharCode(keyCode.Space))
- ) {
- return elem.toUpperCase();
- }
- return elem.toLowerCase();
- },
- onBeforeMask: function (initialValue, opts) {
- if (Object.prototype.toString.call(initialValue) === "[object Date]") {
- initialValue = importDate(initialValue, opts);
- }
- return initialValue;
- },
- insertMode: false,
- insertModeVisual: false,
- shiftPositions: false,
- keepStatic: false,
- inputmode: "numeric",
- prefillYear: true // Allows to disable prefill for datetime year.
- }
- });
|