Current Path : /var/www/www-root/data/www/monolith-realty.ru/bitrix/js/main/date/src/ |
Current File : /var/www/www-root/data/www/monolith-realty.ru/bitrix/js/main/date/src/date-time-format.js |
import { Type, Loc } from 'main.core'; import { convertBitrixFormat } from './internal/convert-bitrix-format'; import { getFormat } from './internal/get-format'; /** * @memberOf BX.Main * @alias Date */ export class DateTimeFormat { static AM_PM_MODE = { UPPER: 1, LOWER: 2, NONE: false }; static convertBitrixFormat = convertBitrixFormat; static getFormat = getFormat; static isAmPmMode(returnConst: any) { if (returnConst === true) { return this._getMessage('AMPM_MODE'); } return this._getMessage('AMPM_MODE') !== false; } static convertToUTC(date: any): ?Date { if (!Type.isDate(date)) { return null; } return new Date( Date.UTC( date.getFullYear(), date.getMonth(), date.getDate(), date.getHours(), date.getMinutes(), date.getSeconds(), date.getMilliseconds(), ) ); } /** * Function creates and returns Javascript Date() object from server timestamp regardless of local browser (system) timezone. * For example can be used to convert timestamp from some exact date on server to the JS Date object with the same value. * * @param timestamp - timestamp in seconds * @returns {Date} */ static getNewDate(timestamp: any): Date { return new Date(this.getBrowserTimestamp(timestamp)); } /** * Function transforms server timestamp (in sec) to javascript timestamp (calculated depend on local browser timezone offset). Returns timestamp in milliseconds. * Also see BX.Main.Date.getNewDate description. * * @param timestamp - timestamp in seconds * @returns {number} */ static getBrowserTimestamp(timestamp: any): number { timestamp = parseInt(timestamp, 10); const browserOffset = new Date(timestamp * 1000).getTimezoneOffset() * 60; return (parseInt(timestamp, 10) + parseInt(this._getMessage('SERVER_TZ_OFFSET')) + browserOffset) * 1000; } /** * Function transforms local browser timestamp (in ms) to server timestamp (calculated depend on local browser timezone offset). Returns timestamp in seconds. * * @param timestamp - timestamp in milliseconds * @returns {number} */ static getServerTimestamp(timestamp: any): number { timestamp = parseInt(timestamp, 10); const browserOffset = new Date(timestamp).getTimezoneOffset() * 60; return Math.round(timestamp / 1000 - (parseInt(this._getMessage('SERVER_TZ_OFFSET'), 10) + parseInt(browserOffset, 10))); } static formatLastActivityDate(timestamp, now, utc) { const ampm = this.isAmPmMode(true); const timeFormat = (ampm === this.AM_PM_MODE.LOWER ? 'g:i a' : (ampm === this.AM_PM_MODE.UPPER ? 'g:i A' : 'H:i')); const format = [ ['tomorrow', '#01#' + timeFormat], ['now', '#02#'], ['todayFuture', '#03#' + timeFormat], ['yesterday', '#04#' + timeFormat], ['-', this.convertBitrixFormat(this._getMessage('FORMAT_DATETIME')).replace(/:s/g, '')], ['s60', 'sago'], ['i60', 'iago'], ['H5', 'Hago'], ['H24', '#03#' + timeFormat], ['d31', 'dago'], ['m12>1', 'mago'], ['m12>0', 'dago'], ['', '#05#'] ]; let formattedDate = this.format(format, timestamp, now, utc); let match = null; if ((match = /^#(\d+)#(.*)/.exec(formattedDate)) != null) { switch (match[1]) { case '01': formattedDate = this._getMessage('FD_LAST_SEEN_TOMORROW').replace('#TIME#', match[2]); break; case '02': formattedDate = this._getMessage('FD_LAST_SEEN_NOW'); break; case '03': formattedDate = this._getMessage('FD_LAST_SEEN_TODAY').replace('#TIME#', match[2]); break; case '04': formattedDate = this._getMessage('FD_LAST_SEEN_YESTERDAY').replace('#TIME#', match[2]); break; case '05': formattedDate = this._getMessage('FD_LAST_SEEN_MORE_YEAR'); break; default: formattedDate = match[2]; break; } } return formattedDate; } /** * The method is designed to replace the localization storage on sites without Bitrix Framework. * It gets overloaded with custom implementation: * * const CustomDate = Object.create(BX.Main.Date); * CustomDate._getMessage = () => ...new implementation...; * * This class should get messages only via this method. * Otherwise, the class won't work on sites without Bitrix Framework. * * @param message * @returns {*} * @private */ static _getMessage(message) { return Loc.getMessage(message); } /** * The method used to parse date from string by given format. * * @param {string} str - date in given format * @param {boolean} isUTC - is date in UTC * @param {string} formatDate - format of the date without time * @param {string} formatDatetime - format of the date with time * @returns {Date|null} - returns Date object if string was parsed or null */ static parse(str, isUTC, formatDate, formatDatetime): ?Date { if (Type.isStringFilled(str)) { if (!formatDate) { formatDate = this._getMessage('FORMAT_DATE'); } if (!formatDatetime) { formatDatetime = this._getMessage('FORMAT_DATETIME'); } let regMonths = ''; for (let i = 1; i <= 12; i++) { regMonths = regMonths + '|' + this._getMessage('MON_' + i); } const expr = new RegExp('([0-9]+|[a-z]+' + regMonths + ')', 'ig'); const aDate = str.match(expr); let aFormat = formatDate.match(/(DD|MI|MMMM|MM|M|YYYY)/ig); const aDateArgs = []; const aFormatArgs = []; const aResult = {}; if (!aDate) { return null; } if (aDate.length > aFormat.length) { aFormat = formatDatetime.match(/(DD|MI|MMMM|MM|M|YYYY|HH|H|SS|TT|T|GG|G)/ig); } for (let i = 0, cnt = aDate.length; i < cnt; i++) { if (aDate[i].trim() !== '') { aDateArgs[aDateArgs.length] = aDate[i]; } } for (let i = 0, cnt = aFormat.length; i < cnt; i++) { if (aFormat[i].trim() !== '') { aFormatArgs[aFormatArgs.length] = aFormat[i]; } } let m = aFormatArgs.findIndex(item => item === 'MMMM'); if (m > 0) { aDateArgs[m] = this.getMonthIndex(aDateArgs[m]); aFormatArgs[m] = 'MM'; } else { m = aFormatArgs.findIndex(item => item === 'M'); if (m > 0) { aDateArgs[m] = this.getMonthIndex(aDateArgs[m]); aFormatArgs[m] = 'MM'; } } for (let i = 0, cnt = aFormatArgs.length; i < cnt; i++) { const k = aFormatArgs[i].toUpperCase(); aResult[k] = k === 'T' || k === 'TT' ? aDateArgs[i] : parseInt(aDateArgs[i], 10); } if (aResult['DD'] > 0 && aResult['MM'] > 0 && aResult['YYYY'] > 0) { const d = new Date(); if (isUTC) { d.setUTCDate(1); d.setUTCFullYear(aResult['YYYY']); d.setUTCMonth(aResult['MM'] - 1); d.setUTCDate(aResult['DD']); d.setUTCHours(0, 0, 0, 0); } else { d.setDate(1); d.setFullYear(aResult['YYYY']); d.setMonth(aResult['MM'] - 1); d.setDate(aResult['DD']); d.setHours(0, 0, 0, 0); } if ( (!isNaN(aResult['HH']) || !isNaN(aResult['GG']) || !isNaN(aResult['H']) || !isNaN(aResult['G'])) && !isNaN(aResult['MI']) ) { if (!isNaN(aResult['H']) || !isNaN(aResult['G'])) { const bPM = (aResult['T'] || aResult['TT'] || 'am').toUpperCase() === 'PM', h = parseInt(aResult['H'] || aResult['G'] || 0, 10); if (bPM) { aResult['HH'] = h + (h === 12 ? 0 : 12); } else { aResult['HH'] = h < 12 ? h : 0; } } else { aResult['HH'] = parseInt(aResult['HH'] || aResult['GG'] || 0, 10); } if (isNaN(aResult['SS'])) { aResult['SS'] = 0; } if (isUTC) { d.setUTCHours(aResult['HH'], aResult['MI'], aResult['SS']); } else { d.setHours(aResult['HH'], aResult['MI'], aResult['SS']); } } return d; } } return null; } static getMonthIndex(month: string): string | number { const q = month.toUpperCase(); const wordMonthCut = ['jan', 'feb', 'mar', 'apr', 'may', 'jun', 'jul', 'aug', 'sep', 'oct', 'nov', 'dec']; const wordMonth = ['january', 'february', 'march', 'april', 'may', 'june', 'july', 'august', 'september', 'october', 'november', 'december']; for (let i = 1; i <= 12; i++) { if (q === this._getMessage('MON_' + i).toUpperCase() || q === this._getMessage('MONTH_' + i).toUpperCase() || q === wordMonthCut[i - 1].toUpperCase() || q === wordMonth[i - 1].toUpperCase()) { return i; } } return month; } static format(format, timestamp, now, utc): string { /* PHP to Javascript: time() = new Date() mktime(...) = new Date(...) gmmktime(...) = new Date(Date.UTC(...)) mktime(0,0,0, 1, 1, 1970) != 0 new Date(1970,0,1).getTime() != 0 gmmktime(0,0,0, 1, 1, 1970) == 0 new Date(Date.UTC(1970,0,1)).getTime() == 0 date('d.m.Y H:i:s') = BX.Main.Date.format('d.m.Y H:i:s') gmdate('d.m.Y H:i:s') = BX.Main.Date.format('d.m.Y H:i:s', null, null, true); */ const date = Type.isDate(timestamp) ? new Date(timestamp.getTime()) : Type.isNumber(timestamp) ? new Date(timestamp * 1000) : new Date(); const nowDate = Type.isDate(now) ? new Date(now.getTime()) : Type.isNumber(now) ? new Date(now * 1000) : new Date(); const isUTC = !!utc; // used in hoisting inner functions, like _formatDateInterval const thisDateTimeFormat = this; if (Type.isArray(format)) { return _formatDateInterval(format, date, nowDate, isUTC); } else { if (!Type.isStringFilled(format)) { return ''; } } const replaceMap = (format.match(/{{([^{}]*)}}/g) || []).map((x) => { return (x.match(/[^{}]+/) || [''])[0]; }); if (replaceMap.length > 0) { replaceMap.forEach((element, index) => { format = format.replace('{{' + element + '}}', '{{' + index + '}}'); }); } const formatRegex = /\\?(sago|iago|isago|Hago|dago|mago|Yago|sdiff|idiff|Hdiff|ddiff|mdiff|Ydiff|sshort|ishort|Hshort|dshort|mhort|Yshort|yesterday|today|tommorow|tomorrow|[a-z])/gi; const dateFormats = { d: () => { // Day of the month 01 to 31 return getDate(date).toString().padStart(2, '0'); }, D: () => { //Mon through Sun return this._getMessage('DOW_' + getDay(date)); }, j: () => { //Day of the month 1 to 31 return getDate(date); }, l: () => { //Sunday through Saturday return this._getMessage('DAY_OF_WEEK_' + getDay(date)); }, N: () => { //1 (for Monday) through 7 (for Sunday) return getDay(date) || 7; }, S: () => { //st, nd, rd or th. Works well with j if (getDate(date) % 10 == 1 && getDate(date) != 11) { return 'st'; } else if (getDate(date) % 10 == 2 && getDate(date) != 12) { return 'nd'; } else if (getDate(date) % 10 == 3 && getDate(date) != 13) { return 'rd'; } else { return 'th'; } }, w: () => { //0 (for Sunday) through 6 (for Saturday) return getDay(date); }, z: () => { //0 through 365 const firstDay = new Date(getFullYear(date), 0, 1); const currentDay = new Date(getFullYear(date), getMonth(date), getDate(date)); return Math.ceil((currentDay - firstDay) / (24 * 3600 * 1000)); }, W: () => { //ISO-8601 week number of year const newDate = new Date(date.getTime()); const dayNumber = (getDay(date) + 6) % 7; setDate(newDate, getDate(newDate) - dayNumber + 3); const firstThursday = newDate.getTime(); setMonth(newDate, 0, 1); if (getDay(newDate) != 4) { setMonth(newDate, 0, 1 + ((4 - getDay(newDate)) + 7) % 7); } const weekNumber = 1 + Math.ceil((firstThursday - newDate) / (7 * 24 * 3600 * 1000)); return weekNumber.toString().padStart(2, '0'); }, F: () => { //January through December return this._getMessage('MONTH_' + (getMonth(date) + 1) + '_S'); }, f: () => { //January through December return this._getMessage('MONTH_' + (getMonth(date) + 1)); }, m: () => { //Numeric representation of a month 01 through 12 return (getMonth(date) + 1).toString().padStart(2, '0'); }, M: () => { //A short textual representation of a month, three letters Jan through Dec return this._getMessage('MON_' + (getMonth(date) + 1)); }, n: () => { //Numeric representation of a month 1 through 12 return getMonth(date) + 1; }, t: () => { //Number of days in the given month 28 through 31 const lastMonthDay = isUTC ? new Date(Date.UTC(getFullYear(date), getMonth(date) + 1, 0)) : new Date(getFullYear(date), getMonth(date) + 1, 0); return getDate(lastMonthDay); }, L: () => { //1 if it is a leap year, 0 otherwise. const year = getFullYear(date); return (year % 4 == 0 && year % 100 != 0 || year % 400 == 0 ? 1 : 0); }, o: () => { //ISO-8601 year number const correctDate = new Date(date.getTime()); setDate(correctDate, getDate(correctDate) - ((getDay(date) + 6) % 7) + 3); return getFullYear(correctDate); }, Y: () => { //A full numeric representation of a year, 4 digits return getFullYear(date); }, y: () => { //A two digit representation of a year return getFullYear(date).toString().slice(2); }, a: () => { //am or pm return getHours(date) > 11 ? 'pm' : 'am'; }, A: () => { //AM or PM return getHours(date) > 11 ? 'PM' : 'AM'; }, B: () => { //000 through 999 const swatch = ((date.getUTCHours() + 1) % 24) + date.getUTCMinutes() / 60 + date.getUTCSeconds() / 3600; return Math.floor(swatch * 1000 / 24).toString().padStart(3, '0'); }, g: () => { //12-hour format of an hour without leading zeros 1 through 12 return getHours(date) % 12 || 12; }, G: () => { //24-hour format of an hour without leading zeros 0 through 23 return getHours(date); }, h: () => { //12-hour format of an hour with leading zeros 01 through 12 return (getHours(date) % 12 || 12).toString().padStart(2, '0'); }, H: () => { //24-hour format of an hour with leading zeros 00 through 23 return getHours(date).toString().padStart(2, '0'); }, i: () => { //Minutes with leading zeros 00 to 59 return getMinutes(date).toString().padStart(2, '0'); }, s: () => { //Seconds, with leading zeros 00 through 59 return getSeconds(date).toString().padStart(2, '0'); }, u: () => { //Microseconds return (getMilliseconds(date) * 1000).toString().padStart(6, '0'); }, e: () => { if (isUTC) { return 'UTC'; } return ''; }, I: () => { if (isUTC) { return 0; } //Whether or not the date is in daylight saving time 1 if Daylight Saving Time, 0 otherwise const firstJanuary = new Date(getFullYear(date), 0, 1); const firstJanuaryUTC = Date.UTC(getFullYear(date), 0, 1); const firstJuly = new Date(getFullYear(date), 6, 0); const firstJulyUTC = Date.UTC(getFullYear(date), 6, 0); return 0 + ((firstJanuary - firstJanuaryUTC) !== (firstJuly - firstJulyUTC)); }, O: () => { if (isUTC) { return '+0000'; } //Difference to Greenwich time (GMT) in hours +0200 const timezoneOffset = date.getTimezoneOffset(); const timezoneOffsetAbs = Math.abs(timezoneOffset); return (timezoneOffset > 0 ? '-' : '+') + (Math.floor(timezoneOffsetAbs / 60) * 100 + timezoneOffsetAbs % 60).toString().padStart(4, '0'); }, //this method references 'O' method of the same object, arrow function is not suitable here P: function() { if (isUTC) { return '+00:00'; } //Difference to Greenwich time (GMT) with colon between hours and minutes +02:00 const difference = this.O(); return difference.substr(0, 3) + ':' + difference.substr(3); }, Z: () => { if (isUTC) { return 0; } //Timezone offset in seconds. The offset for timezones west of UTC is always negative, //and for those east of UTC is always positive. return -date.getTimezoneOffset() * 60; }, c: () => { //ISO 8601 date return 'Y-m-d\\TH:i:sP'.replace(formatRegex, _replaceDateFormat); }, r: () => { //RFC 2822 formatted date return 'D, d M Y H:i:s O'.replace(formatRegex, _replaceDateFormat); }, U: () => { //Seconds since the Unix Epoch return Math.floor(date.getTime() / 1000); }, sago: () => { return _formatDateMessage(intval((nowDate - date) / 1000), { '0': 'FD_SECOND_AGO_0', '1': 'FD_SECOND_AGO_1', '10_20': 'FD_SECOND_AGO_10_20', 'MOD_1': 'FD_SECOND_AGO_MOD_1', 'MOD_2_4': 'FD_SECOND_AGO_MOD_2_4', 'MOD_OTHER': 'FD_SECOND_AGO_MOD_OTHER' }); }, sdiff: () => { return _formatDateMessage(intval((nowDate - date) / 1000), { '0': 'FD_SECOND_DIFF_0', '1': 'FD_SECOND_DIFF_1', '10_20': 'FD_SECOND_DIFF_10_20', 'MOD_1': 'FD_SECOND_DIFF_MOD_1', 'MOD_2_4': 'FD_SECOND_DIFF_MOD_2_4', 'MOD_OTHER': 'FD_SECOND_DIFF_MOD_OTHER' }); }, sshort: () => { return this._getMessage('FD_SECOND_SHORT').replace(/#VALUE#/g, intval((nowDate - date) / 1000)); }, iago: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 1000), { '0': 'FD_MINUTE_AGO_0', '1': 'FD_MINUTE_AGO_1', '10_20': 'FD_MINUTE_AGO_10_20', 'MOD_1': 'FD_MINUTE_AGO_MOD_1', 'MOD_2_4': 'FD_MINUTE_AGO_MOD_2_4', 'MOD_OTHER': 'FD_MINUTE_AGO_MOD_OTHER' }); }, idiff: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 1000), { '0': 'FD_MINUTE_DIFF_0', '1': 'FD_MINUTE_DIFF_1', '10_20': 'FD_MINUTE_DIFF_10_20', 'MOD_1': 'FD_MINUTE_DIFF_MOD_1', 'MOD_2_4': 'FD_MINUTE_DIFF_MOD_2_4', 'MOD_OTHER': 'FD_MINUTE_DIFF_MOD_OTHER' }); }, isago: () => { const minutesAgo = intval((nowDate - date) / 60 / 1000); let result = _formatDateMessage(minutesAgo, { '0': 'FD_MINUTE_0', '1': 'FD_MINUTE_1', '10_20': 'FD_MINUTE_10_20', 'MOD_1': 'FD_MINUTE_MOD_1', 'MOD_2_4': 'FD_MINUTE_MOD_2_4', 'MOD_OTHER': 'FD_MINUTE_MOD_OTHER' }); result += ' '; const secondsAgo = intval((nowDate - date) / 1000) - (minutesAgo * 60); result += _formatDateMessage(secondsAgo, { '0': 'FD_SECOND_AGO_0', '1': 'FD_SECOND_AGO_1', '10_20': 'FD_SECOND_AGO_10_20', 'MOD_1': 'FD_SECOND_AGO_MOD_1', 'MOD_2_4': 'FD_SECOND_AGO_MOD_2_4', 'MOD_OTHER': 'FD_SECOND_AGO_MOD_OTHER' }); return result; }, ishort: () => { return this._getMessage('FD_MINUTE_SHORT').replace(/#VALUE#/g, intval((nowDate - date) / 60 / 1000)); }, Hago: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 1000), { '0': 'FD_HOUR_AGO_0', '1': 'FD_HOUR_AGO_1', '10_20': 'FD_HOUR_AGO_10_20', 'MOD_1': 'FD_HOUR_AGO_MOD_1', 'MOD_2_4': 'FD_HOUR_AGO_MOD_2_4', 'MOD_OTHER': 'FD_HOUR_AGO_MOD_OTHER' }); }, Hdiff: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 1000), { '0': 'FD_HOUR_DIFF_0', '1': 'FD_HOUR_DIFF_1', '10_20': 'FD_HOUR_DIFF_10_20', 'MOD_1': 'FD_HOUR_DIFF_MOD_1', 'MOD_2_4': 'FD_HOUR_DIFF_MOD_2_4', 'MOD_OTHER': 'FD_HOUR_DIFF_MOD_OTHER' }); }, Hshort: () => { return this._getMessage('FD_HOUR_SHORT').replace(/#VALUE#/g, intval((nowDate - date) / 60 / 60 / 1000)); }, yesterday: () => { return this._getMessage('FD_YESTERDAY'); }, today: () => { return this._getMessage('FD_TODAY'); }, tommorow: () => { return this._getMessage('FD_TOMORROW'); }, tomorrow: () => { return this._getMessage('FD_TOMORROW'); }, dago: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 24 / 1000), { '0': 'FD_DAY_AGO_0', '1': 'FD_DAY_AGO_1', '10_20': 'FD_DAY_AGO_10_20', 'MOD_1': 'FD_DAY_AGO_MOD_1', 'MOD_2_4': 'FD_DAY_AGO_MOD_2_4', 'MOD_OTHER': 'FD_DAY_AGO_MOD_OTHER' }); }, ddiff: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 24 / 1000), { '0': 'FD_DAY_DIFF_0', '1': 'FD_DAY_DIFF_1', '10_20': 'FD_DAY_DIFF_10_20', 'MOD_1': 'FD_DAY_DIFF_MOD_1', 'MOD_2_4': 'FD_DAY_DIFF_MOD_2_4', 'MOD_OTHER': 'FD_DAY_DIFF_MOD_OTHER' }); }, dshort: () => { return this._getMessage('FD_DAY_SHORT').replace(/#VALUE#/g, intval((nowDate - date) / 60 / 60 / 24 / 1000)); }, mago: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 24 / 31 / 1000), { '0': 'FD_MONTH_AGO_0', '1': 'FD_MONTH_AGO_1', '10_20': 'FD_MONTH_AGO_10_20', 'MOD_1': 'FD_MONTH_AGO_MOD_1', 'MOD_2_4': 'FD_MONTH_AGO_MOD_2_4', 'MOD_OTHER': 'FD_MONTH_AGO_MOD_OTHER' }); }, mdiff: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 24 / 31 / 1000), { '0': 'FD_MONTH_DIFF_0', '1': 'FD_MONTH_DIFF_1', '10_20': 'FD_MONTH_DIFF_10_20', 'MOD_1': 'FD_MONTH_DIFF_MOD_1', 'MOD_2_4': 'FD_MONTH_DIFF_MOD_2_4', 'MOD_OTHER': 'FD_MONTH_DIFF_MOD_OTHER' }); }, mshort: () => { return this._getMessage('FD_MONTH_SHORT').replace(/#VALUE#/g, intval((nowDate - date) / 60 / 60 / 24 / 31 / 1000)); }, Yago: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 24 / 365 / 1000), { '0': 'FD_YEARS_AGO_0', '1': 'FD_YEARS_AGO_1', '10_20': 'FD_YEARS_AGO_10_20', 'MOD_1': 'FD_YEARS_AGO_MOD_1', 'MOD_2_4': 'FD_YEARS_AGO_MOD_2_4', 'MOD_OTHER': 'FD_YEARS_AGO_MOD_OTHER' }); }, Ydiff: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 24 / 365 / 1000), { '0': 'FD_YEARS_DIFF_0', '1': 'FD_YEARS_DIFF_1', '10_20': 'FD_YEARS_DIFF_10_20', 'MOD_1': 'FD_YEARS_DIFF_MOD_1', 'MOD_2_4': 'FD_YEARS_DIFF_MOD_2_4', 'MOD_OTHER': 'FD_YEARS_DIFF_MOD_OTHER' }); }, Yshort: () => { return _formatDateMessage(intval((nowDate - date) / 60 / 60 / 24 / 365 / 1000), { '0': 'FD_YEARS_SHORT_0', '1': 'FD_YEARS_SHORT_1', '10_20': 'FD_YEARS_SHORT_10_20', 'MOD_1': 'FD_YEARS_SHORT_MOD_1', 'MOD_2_4': 'FD_YEARS_SHORT_MOD_2_4', 'MOD_OTHER': 'FD_YEARS_SHORT_MOD_OTHER' }); }, x: () => { const ampm = this.isAmPmMode(true); const timeFormat = (ampm === this.AM_PM_MODE.LOWER ? 'g:i a' : (ampm === this.AM_PM_MODE.UPPER ? 'g:i A' : 'H:i')); return this.format([ ['tomorrow', 'tomorrow, ' + timeFormat], ['-', this.convertBitrixFormat(this._getMessage('FORMAT_DATETIME')).replace(/:s/g, '')], ['s', 'sago'], ['i', 'iago'], ['today', 'today, ' + timeFormat], ['yesterday', 'yesterday, ' + timeFormat], ['', this.convertBitrixFormat(this._getMessage('FORMAT_DATETIME')).replace(/:s/g, '')] ], date, nowDate, isUTC); }, X: () => { const ampm = this.isAmPmMode(true); const timeFormat = (ampm === this.AM_PM_MODE.LOWER ? 'g:i a' : (ampm === this.AM_PM_MODE.UPPER ? 'g:i A' : 'H:i')); const day = this.format([ ['tomorrow', 'tomorrow'], ['-', this.convertBitrixFormat(this._getMessage('FORMAT_DATE'))], ['today', 'today'], ['yesterday', 'yesterday'], ['', this.convertBitrixFormat(this._getMessage('FORMAT_DATE'))] ], date, nowDate, isUTC); const time = this.format([ ['tomorrow', timeFormat], ['today', timeFormat], ['yesterday', timeFormat], ['', ''] ], date, nowDate, isUTC); if (time.length > 0) { return this._getMessage('FD_DAY_AT_TIME').replace(/#DAY#/g, day).replace(/#TIME#/g, time); } else { return day; } }, Q: () => { const daysAgo = intval((nowDate - date) / 60 / 60 / 24 / 1000); if (daysAgo == 0) { return this._getMessage('FD_DAY_DIFF_1').replace(/#VALUE#/g, 1); } else { return this.format([['d', 'ddiff'], ['m', 'mdiff'], ['', 'Ydiff']], date, nowDate); } } }; let cutZeroTime = false; if (format[0] && format[0] == '^') { cutZeroTime = true; format = format.substr(1); } let result = format.replace(formatRegex, _replaceDateFormat); if (cutZeroTime) { /* 15.04.12 13:00:00 => 15.04.12 13:00 00:01:00 => 00:01 4 may 00:00:00 => 4 may 01-01-12 00:00 => 01-01-12 */ result = result.replace(/\s*00:00:00\s*/g, '').replace(/(\d\d:\d\d)(:00)/g, '$1').replace(/(\s*00:00\s*)(?!:)/g, ''); } if (replaceMap.length > 0) { replaceMap.forEach(function(element, index) { result = result.replace('{{' + index + '}}', element); }); } return result; function _formatDateInterval(formats, date, nowDate, isUTC) { const secondsAgo = intval((nowDate - date) / 1000); for (let i = 0; i < formats.length; i++) { const formatInterval = formats[i][0]; const formatValue = formats[i][1]; let match = null; if (formatInterval == 's') { if (secondsAgo < 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if ((match = /^s(\d+)\>?(\d+)?/.exec(formatInterval)) != null) { if (match[1] && match[2]) { if ( secondsAgo < match[1] && secondsAgo > match[2] ) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (secondsAgo < match[1]) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'i') { if (secondsAgo < 60 * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if ((match = /^i(\d+)\>?(\d+)?/.exec(formatInterval)) != null) { if (match[1] && match[2]) { if ( secondsAgo < match[1] * 60 && secondsAgo > match[2] * 60 ) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (secondsAgo < match[1] * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'H') { if (secondsAgo < 24 * 60 * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if ((match = /^H(\d+)\>?(\d+)?/.exec(formatInterval)) != null) { if (match[1] && match[2]) { if ( secondsAgo < match[1] * 60 * 60 && secondsAgo > match[2] * 60 * 60 ) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (secondsAgo < match[1] * 60 * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'd') { if (secondsAgo < 31 * 24 * 60 * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if ((match = /^d(\d+)\>?(\d+)?/.exec(formatInterval)) != null) { if (match[1] && match[2]) { if ( secondsAgo < match[1] * 24 * 60 * 60 && secondsAgo > match[2] * 24 * 60 * 60 ) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (secondsAgo < match[1] * 24 * 60 * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'm') { if (secondsAgo < 365 * 24 * 60 * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if ((match = /^m(\d+)\>?(\d+)?/.exec(formatInterval)) != null) { if (match[1] && match[2]) { if ( secondsAgo < match[1] * 31 * 24 * 60 * 60 && secondsAgo > match[2] * 31 * 24 * 60 * 60 ) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (secondsAgo < match[1] * 31 * 24 * 60 * 60) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'now') { if (date.getTime() == nowDate.getTime()) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'today') { const year = getFullYear(nowDate); const month = getMonth(nowDate); const day = getDate(nowDate); const todayStart = isUTC ? new Date(Date.UTC(year, month, day, 0, 0, 0, 0)) : new Date(year, month, day, 0, 0, 0, 0); const todayEnd = isUTC ? new Date(Date.UTC(year, month, day + 1, 0, 0, 0, 0)) : new Date(year, month, day + 1, 0, 0, 0, 0); if (date >= todayStart && date < todayEnd) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'todayFuture') { const year = getFullYear(nowDate); const month = getMonth(nowDate); const day = getDate(nowDate); const todayStart = nowDate.getTime(); const todayEnd = isUTC ? new Date(Date.UTC(year, month, day + 1, 0, 0, 0, 0)) : new Date(year, month, day + 1, 0, 0, 0, 0); if (date >= todayStart && date < todayEnd) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'yesterday') { const year = getFullYear(nowDate); const month = getMonth(nowDate); const day = getDate(nowDate); const yesterdayStart = isUTC ? new Date(Date.UTC(year, month, day - 1, 0, 0, 0, 0)) : new Date(year, month, day - 1, 0, 0, 0, 0); const yesterdayEnd = isUTC ? new Date(Date.UTC(year, month, day, 0, 0, 0, 0)) : new Date(year, month, day, 0, 0, 0, 0); if (date >= yesterdayStart && date < yesterdayEnd) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == 'tommorow' || formatInterval == 'tomorrow') { const year = getFullYear(nowDate); const month = getMonth(nowDate); const day = getDate(nowDate); const tomorrowStart = isUTC ? new Date(Date.UTC(year, month, day + 1, 0, 0, 0, 0)) : new Date(year, month, day + 1, 0, 0, 0, 0); const tomorrowEnd = isUTC ? new Date(Date.UTC(year, month, day + 2, 0, 0, 0, 0)) : new Date(year, month, day + 2, 0, 0, 0, 0); if (date >= tomorrowStart && date < tomorrowEnd) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } else if (formatInterval == '-') { if (secondsAgo < 0) { return thisDateTimeFormat.format(formatValue, date, nowDate, isUTC); } } } //return formats.length > 0 ? thisDateTimeFormat.format(formats.pop()[1], date, nowDate, isUTC) : ''; return formats.length > 0 ? thisDateTimeFormat.format(formats[formats.length - 1][1], date, nowDate, isUTC) : ''; } function getFullYear(date) { return isUTC ? date.getUTCFullYear() : date.getFullYear(); } function getDate(date) { return isUTC ? date.getUTCDate() : date.getDate(); } function getMonth(date) { return isUTC ? date.getUTCMonth() : date.getMonth(); } function getHours(date) { return isUTC ? date.getUTCHours() : date.getHours(); } function getMinutes(date) { return isUTC ? date.getUTCMinutes() : date.getMinutes(); } function getSeconds(date) { return isUTC ? date.getUTCSeconds() : date.getSeconds(); } function getMilliseconds(date) { return isUTC ? date.getUTCMilliseconds() : date.getMilliseconds(); } function getDay(date) { return isUTC ? date.getUTCDay() : date.getDay(); } function setDate(date, dayValue) { return isUTC ? date.setUTCDate(dayValue) : date.setDate(dayValue); } function setMonth(date, monthValue, dayValue) { return isUTC ? date.setUTCMonth(monthValue, dayValue) : date.setMonth(monthValue, dayValue); } function _formatDateMessage(value, messages) { const val = value < 100 ? Math.abs(value) : Math.abs(value % 100); const dec = val % 10; let message = ''; if (val == 0) { message = thisDateTimeFormat._getMessage(messages['0']); } else if (val == 1) { message = thisDateTimeFormat._getMessage(messages['1']); } else if (val >= 10 && val <= 20) { message = thisDateTimeFormat._getMessage(messages['10_20']); } else if (dec == 1) { message = thisDateTimeFormat._getMessage(messages['MOD_1']); } else if (2 <= dec && dec <= 4) { message = thisDateTimeFormat._getMessage(messages['MOD_2_4']); } else { message = thisDateTimeFormat._getMessage(messages['MOD_OTHER']); } return message.replace(/#VALUE#/g, value); } function _replaceDateFormat(match, matchFull) { if (dateFormats[match]) { return dateFormats[match](); } else { return matchFull; } } function intval(number) { return number >= 0 ? Math.floor(number) : Math.ceil(number); } } }