You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
716 lines
21 KiB
716 lines
21 KiB
1 year ago
|
// JavaScript Document
|
||
|
/*
|
||
|
** Zabbix
|
||
|
** Copyright (C) 2001-2023 Zabbix SIA
|
||
|
**
|
||
|
** This program is free software; you can redistribute it and/or modify
|
||
|
** it under the terms of the GNU General Public License as published by
|
||
|
** the Free Software Foundation; either version 2 of the License, or
|
||
|
** (at your option) any later version.
|
||
|
**
|
||
|
** This program is distributed in the hope that it will be useful,
|
||
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
** GNU General Public License for more details.
|
||
|
**
|
||
|
** You should have received a copy of the GNU General Public License
|
||
|
** along with this program; if not, write to the Free Software
|
||
|
** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||
|
**
|
||
|
*/
|
||
|
|
||
|
|
||
|
var CLNDR = null,
|
||
|
calendar = function (timeobject, trigger_elmnt, date_time_format) {
|
||
|
if (!this.checkOuterObj(timeobject)) {
|
||
|
throw 'Calendar: constructor expects second parameter to be input form field DOM node.';
|
||
|
}
|
||
|
|
||
|
this.id = jQuery(trigger_elmnt).attr('id');
|
||
|
this.trigger_elmnt = trigger_elmnt;
|
||
|
this.date_time_format = date_time_format;
|
||
|
this.sdt = new CDate();
|
||
|
};
|
||
|
|
||
|
function toggleCalendar(trigger_elmnt, time_input, date_time_format) {
|
||
|
if (CLNDR && jQuery(trigger_elmnt).is(CLNDR.trigger_elmnt) && CLNDR.is_visible) {
|
||
|
CLNDR.clndrhide();
|
||
|
}
|
||
|
else {
|
||
|
CLNDR && CLNDR.clndrhide();
|
||
|
CLNDR = new calendar(time_input, trigger_elmnt, date_time_format);
|
||
|
CLNDR.clndrshow();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
calendar.prototype = {
|
||
|
id: null, // Calendar ID. Should be unique on page.
|
||
|
sdt: null, // Date object of a selected date.
|
||
|
month: 0, // Represents currently opened month number.
|
||
|
year: 2008, // Represents currently opened year.
|
||
|
day: 1, // Represents currently opened day.
|
||
|
clndr_calendar: null, // html obj of calendar
|
||
|
clndr_month_div: null, // html obj
|
||
|
clndr_year_div: null, // html obj
|
||
|
clndr_days: null, // html obj
|
||
|
clndr_month: null, // html obj
|
||
|
clndr_year: null, // html obj
|
||
|
clndr_year_wrap: null, // html obj
|
||
|
clndr_month_wrap: null, // html obj
|
||
|
clndr_monthup: null, // html bttn obj
|
||
|
clndr_monthdown: null, // html bttn obj
|
||
|
clndr_yearup: null, // html bttn obj
|
||
|
clndr_yeardown: null, // html bttn obj
|
||
|
timeobject: null, // Input field with selected time. Source and destination of selected date.
|
||
|
is_visible: false, // State of calendar visibility.
|
||
|
has_user_time: false, // Confirms, if time was selected from input field.
|
||
|
hl_month: null, // highlighted month number
|
||
|
hl_year: null, // highlighted year number
|
||
|
hl_day: null, // highlighted days number
|
||
|
active_section: null, // Active calendar section. See 'sections' array. Default value set in method clndrshow.
|
||
|
monthname: new Array(t('S_JANUARY'), t('S_FEBRUARY'), t('S_MARCH'), t('S_APRIL'), t('S_MAY'), t('S_JUNE'),
|
||
|
t('S_JULY'), t('S_AUGUST'), t('S_SEPTEMBER'), t('S_OCTOBER'), t('S_NOVEMBER'), t('S_DECEMBER')
|
||
|
),
|
||
|
dayname: new Array(t('S_SUNDAY'), t('S_MONDAY'), t('S_TUESDAY'), t('S_WEDNESDAY'), t('S_THURSDAY'), t('S_FRIDAY'),
|
||
|
t('S_SATURDAY')
|
||
|
),
|
||
|
sections: new Array('.calendar-year', '.calendar-month', '.calendar-date'),
|
||
|
date_time_format: PHP_ZBX_FULL_DATE_TIME,
|
||
|
trigger_elmnt: null, // Calendar visibility trigger element.
|
||
|
|
||
|
ondateselected: function() {
|
||
|
this.setDateToOuterObj();
|
||
|
this.clndrhide();
|
||
|
},
|
||
|
|
||
|
clndrhide: function() {
|
||
|
if (this.is_visible) {
|
||
|
this.calendardelete();
|
||
|
this.is_visible = false;
|
||
|
|
||
|
jQuery(window).off('resize', this.calendarPositionHandler);
|
||
|
|
||
|
jQuery(document)
|
||
|
.off('click', this.calendarDocumentClickHandler)
|
||
|
.off('keydown', this.calendarKeyDownHandler)
|
||
|
.off('keyup', this.calendarKeyUpHandler);
|
||
|
|
||
|
removeFromOverlaysStack(this.id);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
clndrshow: function() {
|
||
|
this.calendarcreate();
|
||
|
this.setSDateFromOuterObj();
|
||
|
this.syncBSDateBySDT();
|
||
|
this.syncHlDate();
|
||
|
this.setCDate();
|
||
|
|
||
|
this.calendarPositionHandler();
|
||
|
this.clndr_calendar.style.display = (this.clndr_calendar.tagName === 'span') ? 'inline' : 'block';
|
||
|
this.is_visible = true;
|
||
|
|
||
|
jQuery(window).on('resize', jQuery.proxy(this.calendarPositionHandler, this));
|
||
|
jQuery(this.trigger_elmnt).on('remove', jQuery.proxy(this.clndrhide, this));
|
||
|
|
||
|
jQuery(document)
|
||
|
.on('keydown', jQuery.proxy(this.calendarKeyDownHandler, this))
|
||
|
.on('keyup', jQuery.proxy(this.calendarKeyUpHandler, this))
|
||
|
.on('click', jQuery.proxy(this.calendarDocumentClickHandler, this));
|
||
|
|
||
|
addToOverlaysStack(this.id, this.trigger_elmnt, 'clndr');
|
||
|
|
||
|
this.active_section = this.sections.indexOf('.calendar-date');
|
||
|
this.focusSection();
|
||
|
},
|
||
|
|
||
|
setPosition: function(top, left) {
|
||
|
jQuery(this.clndr_calendar).css({
|
||
|
top: Math.min(top, jQuery(window).height() - jQuery(this.clndr_calendar).outerHeight()) + 'px',
|
||
|
left: Math.min(left, jQuery(window).width() - jQuery(this.clndr_calendar).outerWidth()) + 'px'
|
||
|
});
|
||
|
},
|
||
|
|
||
|
calendarDocumentClickHandler: function(e) {
|
||
|
var $target = jQuery(e.target);
|
||
|
|
||
|
if (!$target.is(this.trigger_elmnt) && !$target.closest('.overlay-dialogue.calendar').length) {
|
||
|
this.clndrhide();
|
||
|
}
|
||
|
},
|
||
|
|
||
|
calendarPositionHandler: function () {
|
||
|
var $anchor = jQuery(this.trigger_elmnt),
|
||
|
offset = $anchor.offset();
|
||
|
|
||
|
this.setPosition(offset.top + $anchor.height(), offset.left + $anchor.width());
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* This function is workaround for Firefox bug.
|
||
|
*
|
||
|
* When triggering keydown event on [space] button, event is called for both, the actual element as well as calendar
|
||
|
* icon elemnet, so the calendar is first closed (by handler of actually focused element) and immediately opened
|
||
|
* again (by calendar icon element's handler).
|
||
|
*
|
||
|
* Workaround works as follow - it separates [enter] and [space] button in 2 handlers with similar functionality
|
||
|
* (since pressing [space] and [enter] does the same thing in calendar). Keyup handles the [space] click event,
|
||
|
* while other keyboard events are handled by keydown event.
|
||
|
*/
|
||
|
calendarKeyUpHandler: function(event) {
|
||
|
if (event.which == 32) { // Space
|
||
|
// Enter has special meaning for each Calendar section.
|
||
|
var active_section = this.sections[this.active_section];
|
||
|
if (active_section === '.calendar-year' || active_section === '.calendar-month') {
|
||
|
this.active_section++;
|
||
|
this.focusSection();
|
||
|
}
|
||
|
else if (active_section === '.calendar-date') {
|
||
|
this.setday(event, this.hl_day, this.hl_month, this.hl_year);
|
||
|
}
|
||
|
|
||
|
return false; // Prevent page scrolling when pressing Space.
|
||
|
}
|
||
|
},
|
||
|
|
||
|
calendarKeyDownHandler: function(event) {
|
||
|
var hl_date;
|
||
|
|
||
|
if (this.active_section < 0 || this.active_section > this.sections.length) {
|
||
|
this.active_section = 0;
|
||
|
}
|
||
|
|
||
|
switch (event.which) {
|
||
|
case 37: // arrow left
|
||
|
case 38: // arrow up
|
||
|
case 39: // arrow right
|
||
|
case 40: // arrow down
|
||
|
switch (this.sections[this.active_section]) {
|
||
|
case '.calendar-date':
|
||
|
hl_date = new Date(this.hl_year, this.hl_month, this.hl_day, 0, 0, 0, 0);
|
||
|
|
||
|
switch (event.which) {
|
||
|
case 37: // arrow left
|
||
|
hl_date.setDate(hl_date.getDate() - 1);
|
||
|
break;
|
||
|
|
||
|
case 38: // arrow up
|
||
|
hl_date.setDate(hl_date.getDate() - 7);
|
||
|
break;
|
||
|
|
||
|
case 39: // arrow right
|
||
|
hl_date.setDate(hl_date.getDate() + 1);
|
||
|
break;
|
||
|
|
||
|
case 40: // arrow down
|
||
|
hl_date.setDate(hl_date.getDate() + 7);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
this.hl_year = hl_date.getFullYear();
|
||
|
this.hl_month = hl_date.getMonth();
|
||
|
this.hl_day = hl_date.getDate();
|
||
|
|
||
|
jQuery('td.highlighted', this.clndr_calendar)
|
||
|
.removeClass('highlighted')
|
||
|
.attr('tabindex', '-1');
|
||
|
|
||
|
if (this.hl_year != this.year || this.hl_month != this.month) {
|
||
|
this.year = this.hl_year;
|
||
|
this.month = this.hl_month;
|
||
|
this.day = this.hl_day;
|
||
|
this.setCDate();
|
||
|
}
|
||
|
|
||
|
jQuery('td[data-date='+this.hl_day+']', this.clndr_calendar)
|
||
|
.addClass('highlighted')
|
||
|
.attr('tabindex', '0')
|
||
|
.focus();
|
||
|
break;
|
||
|
|
||
|
case '.calendar-year':
|
||
|
// Arrow left or arrow down.
|
||
|
if (event.which == 37 || event.which == 40) {
|
||
|
this.yeardown();
|
||
|
}
|
||
|
// Arrow right or arrow up.
|
||
|
else if (event.which == 38 || event.which == 39) {
|
||
|
this.yearup();
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case '.calendar-month':
|
||
|
// Arrow left or arrow down.
|
||
|
if (event.which == 37 || event.which == 40) {
|
||
|
this.monthdown();
|
||
|
}
|
||
|
// Arrow right or arrow up.
|
||
|
else if (event.which == 38 || event.which == 39) {
|
||
|
this.monthup();
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// Prevent page scrolling.
|
||
|
event.preventDefault();
|
||
|
|
||
|
break;
|
||
|
|
||
|
case 9: // Tab
|
||
|
event.preventDefault();
|
||
|
|
||
|
if (event.shiftKey) {
|
||
|
this.active_section--;
|
||
|
if (this.active_section < 0) {
|
||
|
this.active_section = this.sections.length - 1;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
this.active_section++;
|
||
|
if (this.active_section >= this.sections.length) {
|
||
|
this.active_section = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.focusSection();
|
||
|
break;
|
||
|
|
||
|
case 13: // Enter
|
||
|
// Enter has special meaning for each Calendar section.
|
||
|
var active_section = this.sections[this.active_section];
|
||
|
if (active_section === '.calendar-year' || active_section === '.calendar-month') {
|
||
|
this.active_section++;
|
||
|
this.focusSection();
|
||
|
}
|
||
|
else if (active_section === '.calendar-date') {
|
||
|
this.setday(event, this.hl_day, this.hl_month, this.hl_year);
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
|
||
|
case 32: // Prevent page scrolling when pressing Space.
|
||
|
return false;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
focusSection: function() {
|
||
|
var section_to_focus = this.sections[this.active_section];
|
||
|
|
||
|
jQuery('.highlighted', this.clndr_calendar).removeClass('highlighted').blur();
|
||
|
if (section_to_focus === '.calendar-year' || section_to_focus === '.calendar-month') {
|
||
|
jQuery(section_to_focus, this.clndr_calendar).addClass('highlighted').focus();
|
||
|
}
|
||
|
else if (section_to_focus === '.calendar-date') {
|
||
|
/**
|
||
|
* Switching between months and years, date picker will highlight previously selected date. If
|
||
|
* selected date is in different year or month, the first date of displayed year is highlighted.
|
||
|
* Same happens also if the number of dates in selected month is smaller than selected date in different
|
||
|
* month.
|
||
|
*/
|
||
|
if (this.hl_year != this.year || this.hl_month != this.month
|
||
|
|| new Date(this.year, this.month + 1, 0).getDate() < this.hl_day) {
|
||
|
this.hl_day = 1;
|
||
|
}
|
||
|
|
||
|
jQuery('td[data-date='+this.hl_day+']', this.clndr_calendar)
|
||
|
.addClass('highlighted')
|
||
|
.attr('tabindex', '0')
|
||
|
.focus();
|
||
|
}
|
||
|
},
|
||
|
|
||
|
checkOuterObj: function(timeobject) {
|
||
|
if (typeof(timeobject) === 'undefined' || empty(timeobject)) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
this.timeobject = document.getElementById(timeobject);
|
||
|
|
||
|
if (this.timeobject === null || this.timeobject.tagName.toLowerCase() !== 'input') {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
},
|
||
|
|
||
|
setSDateFromOuterObj: function() {
|
||
|
var val = this.timeobject.value,
|
||
|
/**
|
||
|
* Date and time format separators must be synced with ZBX_FULL_DATE_TIME, ZBX_DATE_TIME and ZBX_DATE in
|
||
|
* defines.inc.php.
|
||
|
*/
|
||
|
datetime = val.split(' '),
|
||
|
date = datetime[0].split('-'),
|
||
|
time = (datetime.length > 1) ? datetime[1].split(':') : new Array();
|
||
|
|
||
|
// Open calendar with current time by default.
|
||
|
this.sdt = new CDate();
|
||
|
this.has_user_time = false;
|
||
|
|
||
|
if (date.length === 3 && this.setSDateDMY(date[2], date[1], date[0])) {
|
||
|
if (!time.length) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
this.sdt.setTimeObject(null, null, null, 0, 0, 0);
|
||
|
|
||
|
// Set time to calendar, so time doesn't change when selecting different date.
|
||
|
if ((time.length === 2 || time.length === 3)
|
||
|
&& time[0] > -1 && time[0] < 24
|
||
|
&& time[1] > -1 && time[1] < 60) {
|
||
|
this.has_user_time = true;
|
||
|
this.sdt.setHours(time[0]);
|
||
|
this.sdt.setMinutes(time[1]);
|
||
|
|
||
|
if (time.length === 3 && time[2] > -1 && time[2] < 60) {
|
||
|
this.sdt.setSeconds(time[2]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
setSDateDMY: function(d, m, y) {
|
||
|
var dateHolder = new Date(y, m - 1, d, 0, 0, 0);
|
||
|
|
||
|
if (y >= 1970 && dateHolder.getFullYear() == y && dateHolder.getMonth() == m - 1 && dateHolder.getDate() == d) {
|
||
|
this.sdt.setTimeObject(y, m - 1, d);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
},
|
||
|
|
||
|
setDateToOuterObj: function() {
|
||
|
var $input = jQuery(this.timeobject),
|
||
|
new_val = this.sdt.format(this.date_time_format);
|
||
|
|
||
|
if ($input.val() != new_val) {
|
||
|
$input.val(new_val).trigger('change');
|
||
|
}
|
||
|
},
|
||
|
|
||
|
setday: function(e, day, month, year) {
|
||
|
this.day = day;
|
||
|
this.month = month;
|
||
|
this.year = year;
|
||
|
this.syncSDT();
|
||
|
this.syncBSDateBySDT();
|
||
|
this.ondateselected();
|
||
|
},
|
||
|
|
||
|
monthup: function() {
|
||
|
this.month++;
|
||
|
|
||
|
if (this.month > 11) {
|
||
|
// prevent months from running in loop in year 2038
|
||
|
if (this.year < 2038) {
|
||
|
this.month = 0;
|
||
|
this.yearup();
|
||
|
}
|
||
|
else {
|
||
|
this.month = 11;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
this.setCDate();
|
||
|
}
|
||
|
|
||
|
this.hl_month = this.month;
|
||
|
this.hl_year = this.year;
|
||
|
},
|
||
|
|
||
|
monthdown: function() {
|
||
|
this.month--;
|
||
|
|
||
|
if (this.month < 0) {
|
||
|
// prevent months from running in loop in year 1970
|
||
|
if (this.year > 1970) {
|
||
|
this.month = 11;
|
||
|
this.yeardown();
|
||
|
}
|
||
|
else {
|
||
|
this.month = 0;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
this.setCDate();
|
||
|
}
|
||
|
|
||
|
this.hl_month = this.month;
|
||
|
this.hl_year = this.year;
|
||
|
},
|
||
|
|
||
|
yearup: function() {
|
||
|
if (this.year >= 2038) {
|
||
|
return ;
|
||
|
}
|
||
|
this.year++;
|
||
|
this.setCDate();
|
||
|
this.hl_year = this.year;
|
||
|
},
|
||
|
|
||
|
yeardown: function() {
|
||
|
if (this.year <= 1970) {
|
||
|
return ;
|
||
|
}
|
||
|
this.year--;
|
||
|
this.setCDate();
|
||
|
this.hl_year = this.year;
|
||
|
},
|
||
|
|
||
|
syncBSDateBySDT: function() {
|
||
|
this.day = this.sdt.getDate();
|
||
|
this.month = this.sdt.getMonth();
|
||
|
this.year = this.sdt.getFullYear();
|
||
|
},
|
||
|
|
||
|
syncHlDate: function() {
|
||
|
this.hl_day = this.day;
|
||
|
this.hl_month = this.month;
|
||
|
this.hl_year = this.year;
|
||
|
},
|
||
|
|
||
|
syncSDT: function() {
|
||
|
var hours = 0,
|
||
|
minutes = 0,
|
||
|
seconds = 0;
|
||
|
|
||
|
if (this.has_user_time) {
|
||
|
// If time was present in input field, use it.
|
||
|
hours = this.sdt.getHours();
|
||
|
minutes = this.sdt.getMinutes();
|
||
|
seconds = this.sdt.getSeconds();
|
||
|
}
|
||
|
else {
|
||
|
var cdt = new CDate();
|
||
|
|
||
|
// If today was selected, use current time. Otherwise use 00:00:00.
|
||
|
if (cdt.getFullYear() === this.year && cdt.getMonth() === this.month && cdt.getDate() === this.day) {
|
||
|
hours = cdt.getHours();
|
||
|
minutes = cdt.getMinutes();
|
||
|
seconds = cdt.getSeconds();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.sdt.setTimeObject(this.year, this.month, this.day, hours, minutes, seconds);
|
||
|
},
|
||
|
|
||
|
setCDate: function() {
|
||
|
this.clndr_month.textContent = this.monthname[this.month];
|
||
|
this.clndr_year.textContent = this.year;
|
||
|
this.createDaysTab();
|
||
|
},
|
||
|
|
||
|
createDaysTab: function() {
|
||
|
var tbody = this.clndr_days;
|
||
|
tbody.innerHTML = '';
|
||
|
|
||
|
var cdt = new CDate();
|
||
|
|
||
|
// Start drawing days from first week of the month.
|
||
|
cdt.setTimeObject(this.year, this.month, 1);
|
||
|
|
||
|
// make 0 - Monday, not Sunday (as default)
|
||
|
var prev_days = cdt.getDay() - 1;
|
||
|
if (prev_days < 0) {
|
||
|
prev_days = 6;
|
||
|
}
|
||
|
|
||
|
// Set to first day of the week.
|
||
|
if (prev_days > 0) {
|
||
|
cdt.setTime(cdt.getTime() - prev_days * 86400000);
|
||
|
}
|
||
|
|
||
|
for (var y = 0; y < 6; y++) {
|
||
|
var tr = document.createElement('tr');
|
||
|
tr.setAttribute('role', 'presentation');
|
||
|
|
||
|
tbody.appendChild(tr);
|
||
|
for (var x = 0; x < 7; x++) {
|
||
|
var td = document.createElement('td');
|
||
|
tr.appendChild(td);
|
||
|
|
||
|
if (this.month != cdt.getMonth()) {
|
||
|
$(td).addClass('grey');
|
||
|
}
|
||
|
else {
|
||
|
td.setAttribute('data-date', cdt.getDate());
|
||
|
}
|
||
|
|
||
|
if (this.sdt.getFullYear() == cdt.getFullYear()
|
||
|
&& this.sdt.getMonth() == cdt.getMonth()
|
||
|
&& this.sdt.getDate() == cdt.getDate()) {
|
||
|
$(td).addClass('selected');
|
||
|
}
|
||
|
|
||
|
td.setAttribute('aria-label', this.calendarGetReadableDate(cdt));
|
||
|
td.setAttribute('tabindex', '-1');
|
||
|
td.setAttribute('role', 'button');
|
||
|
|
||
|
var span = document.createElement('span');
|
||
|
span.setAttribute('aria-hidden', 'true');
|
||
|
span.appendChild(document.createTextNode(cdt.getDate()));
|
||
|
td.appendChild(span);
|
||
|
|
||
|
addListener(td, 'click', this.setday.bindAsEventListener(
|
||
|
this, cdt.getDate(), cdt.getMonth(), cdt.getFullYear()
|
||
|
));
|
||
|
|
||
|
cdt.setTime(cdt.getTime() + 86400000); // + 1day
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
calendarGetReadableDate: function(cdt) {
|
||
|
return cdt.getDate() + ', ' + this.dayname[cdt.getDay()] + ' ' + this.monthname[cdt.getMonth()] + ' ' +
|
||
|
cdt.getFullYear();
|
||
|
},
|
||
|
/**
|
||
|
* Create and append calendar DOM element to body.
|
||
|
*/
|
||
|
calendarcreate: function() {
|
||
|
this.clndr_calendar = document.createElement('div');
|
||
|
this.clndr_calendar.className = 'overlay-dialogue calendar';
|
||
|
this.clndr_calendar.setAttribute('aria-label', t('S_CALENDAR'));
|
||
|
this.clndr_calendar.setAttribute('role', 'application');
|
||
|
this.clndr_calendar.setAttribute('tabindex', '0');
|
||
|
this.clndr_calendar.style.display = 'none';
|
||
|
|
||
|
document.body.appendChild(this.clndr_calendar);
|
||
|
|
||
|
/*
|
||
|
* Calendar header
|
||
|
*/
|
||
|
var header = document.createElement('div');
|
||
|
this.clndr_calendar.appendChild(header);
|
||
|
header.className = 'calendar-header';
|
||
|
|
||
|
// year
|
||
|
this.clndr_year_div = document.createElement('div');
|
||
|
this.clndr_year_div.setAttribute('role', 'presentation');
|
||
|
this.clndr_year_div.className = 'calendar-year';
|
||
|
header.appendChild(this.clndr_year_div);
|
||
|
|
||
|
var arrow_left = document.createElement('span');
|
||
|
arrow_left.className = 'arrow-left';
|
||
|
var arrow_right = document.createElement('span');
|
||
|
arrow_right.className = 'arrow-right';
|
||
|
|
||
|
this.clndr_yeardown = document.createElement('button');
|
||
|
this.clndr_yeardown.setAttribute('type', 'button');
|
||
|
this.clndr_yeardown.setAttribute('tabindex', '-1');
|
||
|
this.clndr_yeardown.className = 'btn-grey';
|
||
|
this.clndr_yeardown.appendChild(arrow_left);
|
||
|
this.clndr_year_div.appendChild(this.clndr_yeardown);
|
||
|
|
||
|
this.clndr_year = document.createTextNode('');
|
||
|
|
||
|
this.clndr_year_wrap = document.createElement('span');
|
||
|
this.clndr_year_wrap.appendChild(this.clndr_year);
|
||
|
this.clndr_year_wrap.setAttribute('aria-live', 'assertive');
|
||
|
this.clndr_year_wrap.setAttribute('id', 'current-year'+this.id);
|
||
|
this.clndr_year_wrap.setAttribute('aria-atomic', 'true');
|
||
|
this.clndr_year_div.appendChild(this.clndr_year_wrap);
|
||
|
this.clndr_year_div.setAttribute('aria-labelledby', this.clndr_year_wrap.id);
|
||
|
|
||
|
this.clndr_yearup = document.createElement('button');
|
||
|
this.clndr_yearup.setAttribute('type', 'button');
|
||
|
this.clndr_yearup.setAttribute('tabindex', '-1');
|
||
|
this.clndr_yearup.className = 'btn-grey';
|
||
|
this.clndr_yearup.appendChild(arrow_right);
|
||
|
this.clndr_year_div.appendChild(this.clndr_yearup);
|
||
|
|
||
|
// month
|
||
|
this.clndr_month_div = document.createElement('div');
|
||
|
this.clndr_month_div.className = 'calendar-month';
|
||
|
this.clndr_month_div.setAttribute('role', 'presentation');
|
||
|
header.appendChild(this.clndr_month_div);
|
||
|
|
||
|
var arrow_left = document.createElement('span');
|
||
|
arrow_left.className = 'arrow-left';
|
||
|
var arrow_right = document.createElement('span');
|
||
|
arrow_right.className = 'arrow-right';
|
||
|
|
||
|
this.clndr_monthdown = document.createElement('button');
|
||
|
this.clndr_monthdown.setAttribute('type', 'button');
|
||
|
this.clndr_monthdown.setAttribute('tabindex', '-1');
|
||
|
this.clndr_monthdown.className = 'btn-grey';
|
||
|
this.clndr_monthdown.appendChild(arrow_left);
|
||
|
this.clndr_month_div.appendChild(this.clndr_monthdown);
|
||
|
|
||
|
this.clndr_month = document.createTextNode('');
|
||
|
this.clndr_month_wrap = document.createElement('span');
|
||
|
this.clndr_month_wrap.setAttribute('aria-live', 'assertive');
|
||
|
this.clndr_month_wrap.setAttribute('aria-atomic', 'true');
|
||
|
this.clndr_month_wrap.setAttribute('id', 'current-month'+this.id);
|
||
|
this.clndr_month_wrap.appendChild(this.clndr_month);
|
||
|
this.clndr_month_div.appendChild(this.clndr_month_wrap);
|
||
|
this.clndr_month_div.setAttribute('aria-labelledby', this.clndr_month_wrap.id);
|
||
|
|
||
|
this.clndr_monthup = document.createElement('button');
|
||
|
this.clndr_monthup.setAttribute('type', 'button');
|
||
|
this.clndr_monthup.setAttribute('tabindex', '-1');
|
||
|
this.clndr_monthup.className = 'btn-grey';
|
||
|
this.clndr_monthup.appendChild(arrow_right);
|
||
|
this.clndr_month_div.appendChild(this.clndr_monthup);
|
||
|
|
||
|
// days heading
|
||
|
var table = document.createElement('table');
|
||
|
this.clndr_calendar.appendChild(table);
|
||
|
|
||
|
var thead = document.createElement('thead');
|
||
|
thead.setAttribute('role', 'presentation');
|
||
|
table.appendChild(thead);
|
||
|
|
||
|
var tr = document.createElement('tr');
|
||
|
thead.appendChild(tr);
|
||
|
|
||
|
[
|
||
|
t('S_MONDAY_SHORT_BIG'),
|
||
|
t('S_TUESDAY_SHORT_BIG'),
|
||
|
t('S_WEDNESDAY_SHORT_BIG'),
|
||
|
t('S_THURSDAY_SHORT_BIG'),
|
||
|
t('S_FRIDAY_SHORT_BIG'),
|
||
|
t('S_SATURDAY_SHORT_BIG'),
|
||
|
t('S_SUNDAY_SHORT_BIG')
|
||
|
].forEach(function(str) {
|
||
|
var td = document.createElement('th');
|
||
|
td.appendChild(document.createTextNode(str));
|
||
|
tr.appendChild(td);
|
||
|
});
|
||
|
|
||
|
/*
|
||
|
* Days calendar
|
||
|
*/
|
||
|
this.clndr_days = document.createElement('tbody');
|
||
|
this.clndr_days.setAttribute('class', 'calendar-date');
|
||
|
table.appendChild(this.clndr_days);
|
||
|
|
||
|
addListener(this.clndr_monthdown, 'click', this.monthdown.bindAsEventListener(this));
|
||
|
addListener(this.clndr_monthup, 'click', this.monthup.bindAsEventListener(this));
|
||
|
addListener(this.clndr_yeardown, 'click', this.yeardown.bindAsEventListener(this));
|
||
|
addListener(this.clndr_yearup, 'click', this.yearup.bindAsEventListener(this));
|
||
|
|
||
|
// Active section setter.
|
||
|
var cal_obj = this;
|
||
|
jQuery(this.sections).each(function(index, item) {
|
||
|
jQuery(item, cal_obj.clndr_calendar)
|
||
|
.attr({'tabindex': '0'})
|
||
|
.on('click', function() {
|
||
|
cal_obj.active_section = index;
|
||
|
cal_obj.focusSection();
|
||
|
});
|
||
|
});
|
||
|
},
|
||
|
/**
|
||
|
* Remove calendar DOM element. Detach event listeners.
|
||
|
*/
|
||
|
calendardelete: function() {
|
||
|
removeListener(this.clndr_monthdown, 'click', this.monthdown);
|
||
|
removeListener(this.clndr_monthup, 'click', this.monthup);
|
||
|
removeListener(this.clndr_yeardown, 'click', this.yeardown);
|
||
|
removeListener(this.clndr_yearup, 'click', this.yearup);
|
||
|
jQuery(this.clndr_calendar).remove();
|
||
|
}
|
||
|
};
|