egroupware/calendar/js/et2_widget_daycol.js

988 lines
43 KiB
JavaScript
Raw Normal View History

/*
* Egroupware
*
* @license https://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package calendar
* @subpackage etemplate
* @link https://www.egroupware.org
* @author Nathan Gray
*/
/*egw:uses
2020-02-27 21:37:36 +01:00
et2_core_valueWidget;
/calendar/js/et2_widget_event.js;
*/
2021-06-10 15:40:49 +02:00
import { et2_createWidget, et2_register_widget } from "../../api/js/etemplate/et2_core_widget";
import { et2_valueWidget } from "../../api/js/etemplate/et2_core_valueWidget";
import { et2_calendar_timegrid } from "./et2_widget_timegrid";
import { et2_calendar_view } from "./et2_widget_view";
import { et2_calendar_event } from "./et2_widget_event";
import { ClassWithAttributes } from "../../api/js/etemplate/et2_core_inheritance";
import { et2_no_init } from "../../api/js/etemplate/et2_core_common";
import { egw } from "../../api/js/jsapi/egw_global";
import { egwIsMobile } from "../../api/js/egw_action/egw_action_common.js";
import { CalendarApp } from "./app";
import { sprintf } from "../../api/js/egw_action/egw_action_common.js";
/**
2016-01-13 23:07:09 +01:00
* Class which implements the "calendar-timegrid" XET-Tag for displaying a single days
*
2016-01-13 23:07:09 +01:00
* This widget is responsible mostly for positioning its events
*
*/
2021-06-10 15:40:49 +02:00
export class et2_calendar_daycol extends et2_valueWidget {
2020-02-27 21:37:36 +01:00
/**
* Constructor
*/
2021-06-10 15:40:49 +02:00
constructor(_parent, _attrs, _child) {
2020-02-27 21:37:36 +01:00
// Call the inherited constructor
2021-06-10 15:40:49 +02:00
super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_calendar_daycol._attributes, _child || {}));
this.registeredUID = null;
2020-02-27 21:37:36 +01:00
// Init to defaults, just in case - they will be updated from parent
2021-06-10 15:40:49 +02:00
this.display_settings = {
2020-02-27 21:37:36 +01:00
wd_start: 60 * 9,
wd_end: 60 * 17,
granularity: 30,
rowsToDisplay: 10,
rowHeight: 20,
// Percentage; not yet available
titleHeight: 2.0
};
// Main container
2021-06-10 15:40:49 +02:00
this.div = jQuery(document.createElement("div"))
2020-02-27 21:37:36 +01:00
.addClass("calendar_calDayCol")
2021-06-10 15:40:49 +02:00
.css('width', this.options.width)
.css('left', this.options.left);
this.header = jQuery(document.createElement('div'))
2020-02-27 21:37:36 +01:00
.addClass("calendar_calDayColHeader")
2021-06-10 15:40:49 +02:00
.css('width', this.options.width)
.css('left', this.options.left);
this.title = jQuery(document.createElement('div'))
2020-02-27 21:37:36 +01:00
.addClass('et2_clickable et2_link')
2021-06-10 15:40:49 +02:00
.appendTo(this.header);
this.user_spacer = jQuery(document.createElement('div'))
2020-02-27 21:37:36 +01:00
.addClass("calendar_calDayColHeader_spacer")
2021-06-10 15:40:49 +02:00
.appendTo(this.header);
this.all_day = jQuery(document.createElement('div'))
2020-02-27 21:37:36 +01:00
.addClass("calendar_calDayColAllDay")
.css('max-height', (egw.preference('limit_all_day_lines', 'calendar') || 3) * 1.4 + 'em')
2021-06-10 15:40:49 +02:00
.appendTo(this.header);
this.event_wrapper = jQuery(document.createElement('div'))
2020-02-27 21:37:36 +01:00
.addClass("event_wrapper")
2021-06-10 15:40:49 +02:00
.appendTo(this.div);
this.setDOMNode(this.div[0]);
2020-02-27 21:37:36 +01:00
// Used for its date calculations - note this is a datetime, parent
// uses just a date
2021-06-10 15:40:49 +02:00
this._date_helper = et2_createWidget('date-time', {}, null);
this._date_helper.loadingFinished();
2020-02-27 21:37:36 +01:00
}
2021-06-10 15:40:49 +02:00
doLoadingFinished() {
let result = super.doLoadingFinished();
2020-02-27 21:37:36 +01:00
// Parent will have everything we need, just load it from there
if (this.getParent() && this.getParent().options.owner) {
this.set_owner(this.getParent().options.owner);
}
if (this.title.text() === '' && this.options.date &&
2021-06-10 15:40:49 +02:00
this.getParent() && this.getParent().instanceOf(et2_calendar_timegrid)) {
2020-02-27 21:37:36 +01:00
// Forces an update
2021-06-10 15:40:49 +02:00
const date = this.options.date;
2020-02-27 21:37:36 +01:00
this.options.date = '';
this.set_date(date);
2020-02-27 21:37:36 +01:00
}
return result;
2021-06-10 15:40:49 +02:00
}
destroy() {
super.destroy();
2020-02-27 21:37:36 +01:00
this.div.off();
this.header.off().remove();
this.title.off();
this.div = null;
this.header = null;
this.title = null;
this.user_spacer = null;
// date_helper has no parent, so we must explicitly remove it
this._date_helper.destroy();
this._date_helper = null;
egw.dataUnregisterUID(this.registeredUID, null, this);
2021-06-10 15:40:49 +02:00
}
getDOMNode(sender) {
2020-02-27 21:37:36 +01:00
if (!sender || sender === this)
return this.div[0];
2021-06-10 15:40:49 +02:00
if (sender.instanceOf && sender.instanceOf(et2_calendar_event)) {
2020-02-27 21:37:36 +01:00
if (this.display_settings.granularity === 0) {
return this.event_wrapper[0];
}
if (sender.options.value.whole_day_on_top ||
sender.options.value.whole_day && sender.options.value.non_blocking === true) {
return this.all_day[0];
}
return this.div[0];
}
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Draw the individual divs for clicking to add an event
*/
2021-06-10 15:40:49 +02:00
_draw() {
2020-02-27 21:37:36 +01:00
// Remove any existing
jQuery('.calendar_calAddEvent', this.div).remove();
// Grab real values from parent
2021-06-10 15:40:49 +02:00
if (this.getParent() && this.getParent().instanceOf(et2_calendar_timegrid)) {
2020-02-27 21:37:36 +01:00
this.display_settings.wd_start = 60 * this.getParent().options.day_start;
this.display_settings.wd_end = 60 * this.getParent().options.day_end;
this.display_settings.granularity = this.getParent().options.granularity;
2021-06-10 15:40:49 +02:00
const header = this.getParent().dayHeader.children();
2020-02-27 21:37:36 +01:00
// Figure out insert index
2021-06-10 15:40:49 +02:00
let idx = 0;
const siblings = this.getParent().getDOMNode(this).childNodes;
2020-02-27 21:37:36 +01:00
while (idx < siblings.length && siblings[idx] != this.getDOMNode()) {
idx++;
}
// Stick header in the right place
if (idx == 0) {
this.getParent().dayHeader.prepend(this.header);
}
else if (header.length) {
header.eq(Math.min(header.length, idx) - 1).after(this.header);
}
}
this.div.attr('data-date', this.options.date);
2021-06-10 15:40:49 +02:00
}
getDate() {
2020-02-27 21:37:36 +01:00
return this.date;
2021-06-10 15:40:49 +02:00
}
get date_helper() {
return this._date_helper;
}
2020-02-27 21:37:36 +01:00
/**
* Set the date
*
* @param {string|Date} _date New date
* @param {Object[]} events =false List of event data to be displayed, or false to
* automatically fetch data from content array
* @param {boolean} force_redraw =false Redraw even if the date is the same.
* Used for when new data is available.
*/
2021-06-10 15:40:49 +02:00
set_date(_date, events, force_redraw) {
2020-02-27 21:37:36 +01:00
if (typeof events === 'undefined' || !events) {
events = false;
}
if (typeof force_redraw === 'undefined' || !force_redraw) {
force_redraw = false;
}
if (!this.getParent() || !this.getParent().date_helper) {
egw.debug('warn', 'Day col widget "' + this.id + '" is missing its parent.');
return false;
}
if (typeof _date === "object") {
this.getParent().date_helper.set_value(_date);
}
else if (typeof _date === "string") {
// Need a new date to avoid invalid month/date combinations when setting
// month then day. Use a string to avoid browser timezone.
this.getParent().date_helper.set_value(_date.substring(0, 4) + '-' + (_date.substring(4, 6)) + '-' + _date.substring(6, 8) + 'T00:00:00Z');
}
this.date = new Date(this.getParent().date_helper.getValue());
// Keep internal option in Ymd format, it gets passed around in this format
2021-06-10 15:40:49 +02:00
const new_date = "" + this.getParent().date_helper.get_year() +
2020-02-27 21:37:36 +01:00
sprintf("%02d", this.getParent().date_helper.get_month()) +
sprintf("%02d", this.getParent().date_helper.get_date());
// Set label
if (!this.options.label) {
// Add timezone offset back in, or formatDate will lose those hours
2021-06-10 15:40:49 +02:00
const formatDate = new Date(this.date.valueOf() + this.date.getTimezoneOffset() * 60 * 1000);
2020-02-27 21:37:36 +01:00
this.title.html('<span class="long_date">' + jQuery.datepicker.formatDate('DD', formatDate) +
'</span><span class="short_date">' + jQuery.datepicker.formatDate('D', formatDate) + '</span>' +
jQuery.datepicker.formatDate('d', formatDate));
}
this.title
.attr("data-date", new_date)
.toggleClass('et2_label', !!this.options.label);
this.header
.attr('data-date', new_date)
.attr('data-whole_day', true);
// Avoid redrawing if date is the same
if (new_date === this.options.date &&
this.display_settings.granularity === this.getParent().options.granularity &&
!force_redraw) {
return;
}
2021-06-10 15:40:49 +02:00
const cache_id = CalendarApp._daywise_cache_id(new_date, this.options.owner);
2020-02-27 21:37:36 +01:00
if (this.options.date && this.registeredUID &&
cache_id !== this.registeredUID) {
egw.dataUnregisterUID(this.registeredUID, null, this);
// Remove existing events
while (this._children.length > 0) {
2021-06-10 15:40:49 +02:00
const node = this._children[this._children.length - 1];
2020-02-27 21:37:36 +01:00
this.removeChild(node);
node.destroy();
}
}
this.options.date = new_date;
// Set holiday and today classes
this.day_class_holiday();
// Update all the little boxes
this._draw();
// Register for updates on events for this day
if (this.registeredUID !== cache_id) {
this.registeredUID = cache_id;
egw.dataRegisterUID(this.registeredUID, this._data_callback, this, this.getInstanceManager().execId, this.id);
}
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Set the owner of this day
*
* @param {number|number[]|string|string[]} _owner - Owner ID, which can
* be an account ID, a resource ID (as defined in calendar_bo, not
* necessarily an entry from the resource app), or a list containing a
* combination of both.
*/
2021-06-10 15:40:49 +02:00
set_owner(_owner) {
2020-02-27 21:37:36 +01:00
this.title
.attr("data-owner", _owner);
this.header.attr('data-owner', _owner);
this.div.attr('data-owner', _owner);
// Simple comparison, both numbers
if (_owner === this.options.owner)
return;
// More complicated comparison, one or the other is an array
if ((typeof _owner == 'object' || typeof this.options.owner == 'object') &&
_owner.toString() == this.options.owner.toString()) {
return;
}
this.options.owner = typeof _owner !== 'object' ? [_owner] : _owner;
2021-06-10 15:40:49 +02:00
const cache_id = CalendarApp._daywise_cache_id(this.options.date, _owner);
2020-02-27 21:37:36 +01:00
if (this.options.date && this.registeredUID &&
cache_id !== this.registeredUID) {
egw.dataUnregisterUID(this.registeredUID, null, this);
}
if (this.registeredUID !== cache_id) {
this.registeredUID = cache_id;
egw.dataRegisterUID(this.registeredUID, this._data_callback, this, this.getInstanceManager().execId, this.id);
}
2021-06-10 15:40:49 +02:00
}
set_class(classnames) {
2020-02-27 21:37:36 +01:00
this.header.removeClass(this.class);
2021-06-10 15:40:49 +02:00
super.set_class(classnames);
2020-02-27 21:37:36 +01:00
this.header.addClass(classnames);
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Callback used when the daywise data changes
*
* Events should update themselves when their data changes, here we are
* dealing with a change in which events are displayed on this day.
*
* @param {String[]} event_ids
* @returns {undefined}
*/
2021-06-10 15:40:49 +02:00
_data_callback(event_ids) {
const events = [];
2020-02-27 21:37:36 +01:00
if (event_ids == null || typeof event_ids.length == 'undefined')
event_ids = [];
2021-06-10 15:40:49 +02:00
for (let i = 0; i < event_ids.length; i++) {
let event = egw.dataGetUIDdata('calendar::' + event_ids[i]);
event = event && event.data || false;
if (event && event.date && et2_calendar_event.owner_check(event, this) && (event.date === this.options.date ||
2020-02-27 21:37:36 +01:00
// Accept multi-day events
2021-06-10 15:40:49 +02:00
new Date(event.start) <= this.date //&& new Date(event.end) >= this.date
2020-02-27 21:37:36 +01:00
)) {
2021-06-10 15:40:49 +02:00
events.push(event);
2020-02-27 21:37:36 +01:00
}
2021-06-10 15:40:49 +02:00
else if (event) {
2020-02-27 21:37:36 +01:00
// Got an ID that doesn't belong
event_ids.splice(i--, 1);
}
}
if (!this.div.is(":visible")) {
// Not visible, defer the layout or it all winds up at the top
// Cancel any existing listener & bind
jQuery(this.getInstanceManager().DOMContainer.parentNode)
.off('show.' + CalendarApp._daywise_cache_id(this.options.date, this.options.owner))
.one('show.' + CalendarApp._daywise_cache_id(this.options.date, this.options.owner), function () {
this._update_events(events);
}.bind(this));
return;
}
2020-02-27 21:37:36 +01:00
if (!this.getParent().disabled)
this._update_events(events);
2021-06-10 15:40:49 +02:00
}
set_label(label) {
2020-02-27 21:37:36 +01:00
this.options.label = label;
this.title.text(label);
this.title.toggleClass('et2_clickable et2_link', label === '');
2021-06-10 15:40:49 +02:00
}
set_left(left) {
2020-02-27 21:37:36 +01:00
if (this.div) {
this.div.css('left', left);
}
2021-06-10 15:40:49 +02:00
}
set_width(width) {
2020-02-27 21:37:36 +01:00
this.options.width = width;
if (this.div) {
this.div.outerWidth(this.options.width);
this.header.outerWidth(this.options.width);
}
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Applies class for today, and any holidays for current day
*/
2021-06-10 15:40:49 +02:00
day_class_holiday() {
2020-02-27 21:37:36 +01:00
this.title
// Remove all special day classes
.removeClass('calendar_calToday calendar_calBirthday calendar_calHoliday')
// Except this one...
.addClass("et2_clickable et2_link");
this.title.attr('data-holiday', '');
// Set today class - note +1 when dealing with today, as months in JS are 0-11
2021-06-10 15:40:49 +02:00
const today = new Date();
2020-02-27 21:37:36 +01:00
today.setUTCMinutes(today.getUTCMinutes() - today.getTimezoneOffset());
this.title.toggleClass("calendar_calToday", this.options.date === '' + today.getUTCFullYear() +
sprintf("%02d", today.getUTCMonth() + 1) +
sprintf("%02d", today.getUTCDate()));
// Holidays and birthdays
2021-06-10 15:40:49 +02:00
let holidays = et2_calendar_view.get_holidays(this, this.options.date.substring(0, 4));
const holiday_list = [];
let holiday_pref = (egw.preference('birthdays_as_events', 'calendar') || []);
2020-02-27 21:37:36 +01:00
if (typeof holiday_pref === 'string') {
holiday_pref = holiday_pref.split(',');
}
else {
holiday_pref = jQuery.extend([], holiday_pref);
}
// Show holidays as events on mobile or by preference
2021-06-10 15:40:49 +02:00
const holidays_as_events = egwIsMobile() || egw.preference('birthdays_as_events', 'calendar') === true ||
2020-02-27 21:37:36 +01:00
holiday_pref.indexOf('holiday') >= 0;
2021-06-10 15:40:49 +02:00
const birthdays_as_events = egwIsMobile() || holiday_pref.indexOf('birthday') >= 0;
2020-02-27 21:37:36 +01:00
if (holidays && holidays[this.options.date]) {
holidays = holidays[this.options.date];
2021-06-10 15:40:49 +02:00
for (let i = 0; i < holidays.length; i++) {
2020-02-27 21:37:36 +01:00
if (typeof holidays[i]['birthyear'] !== 'undefined') {
// Show birthdays as events on mobile or by preference
if (birthdays_as_events) {
// Create event
this.getParent().date_helper.set_value(this.options.date.substring(0, 4) + '-' +
(this.options.date.substring(4, 6)) + '-' + this.options.date.substring(6, 8) +
'T00:00:00Z');
var event = et2_createWidget('calendar-event', {
id: 'event_' + holidays[i].name,
value: {
title: holidays[i].name,
whole_day: true,
whole_day_on_top: true,
start: new Date(this.getParent().date_helper.get_value()),
end: this.options.date,
owner: this.options.owner,
participants: this.options.owner,
app: 'calendar',
class: 'calendar_calBirthday'
},
readonly: true,
class: 'calendar_calBirthday'
}, this);
event.doLoadingFinished();
event._update();
}
if (!egwIsMobile()) {
//If the birthdays are already displayed as event, don't
//show them in the caption
this.title.addClass('calendar_calBirthday');
holiday_list.push(holidays[i]['name']);
}
}
else {
// Show holidays as events on mobile
if (holidays_as_events) {
// Create event
this.getParent().date_helper.set_value(this.options.date.substring(0, 4) + '-' +
(this.options.date.substring(4, 6)) + '-' + this.options.date.substring(6, 8) +
'T00:00:00Z');
var event = et2_createWidget('calendar-event', {
id: 'event_' + holidays[i].name,
value: {
title: holidays[i].name,
whole_day: true,
whole_day_on_top: true,
start: new Date(this.getParent().date_helper.get_value()),
end: this.options.date,
owner: this.options.owner,
participants: this.options.owner,
app: 'calendar',
class: 'calendar_calHoliday'
},
readonly: true,
class: 'calendar_calHoliday'
}, this);
event.doLoadingFinished();
event._update();
}
else {
this.title.addClass('calendar_calHoliday');
this.title.attr('data-holiday', holidays[i]['name']);
//If the birthdays are already displayed as event, don't
//show them in the caption
if (!this.options.display_holiday_as_event) {
holiday_list.push(holidays[i]['name']);
}
}
}
}
}
this.title.attr('title', holiday_list.join(', '));
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Load the event data for this day and create event widgets for each.
*
* If event information is not provided, it will be pulled from the content array.
*
* @param {Object[]} [_events] Array of event information, one per event.
*/
2021-06-10 15:40:49 +02:00
_update_events(_events) {
let c;
const events = _events || this.getArrayMgr('content').getEntry(this.options.date) || [];
2020-02-27 21:37:36 +01:00
// Remove extra events
while (this._children.length > 0) {
2021-06-10 15:40:49 +02:00
const node = this._children[this._children.length - 1];
2020-02-27 21:37:36 +01:00
this.removeChild(node);
node.destroy();
}
// Make sure children are in cronological order, or columns are backwards
events.sort(function (a, b) {
2021-06-10 15:40:49 +02:00
const start = new Date(a.start) - new Date(b.start);
const end = new Date(a.end) - new Date(b.end);
2020-02-27 21:37:36 +01:00
// Whole day events sorted by ID, normal events by start / end time
if (a.whole_day && b.whole_day) {
return (a.app_id - b.app_id);
}
else if (a.whole_day || b.whole_day) {
return a.whole_day ? -1 : 1;
}
return start ? start : end;
});
for (c = 0; c < events.length; c++) {
// Create event
var event = et2_createWidget('calendar-event', {
id: 'event_' + events[c].id,
value: events[c]
}, this);
}
// Seperate loop so column sorting finds all children in the right place
2021-06-10 15:40:49 +02:00
let child_length = this._children.length;
for (c = 0; c < events.length && c < child_length; c++) {
2021-06-10 15:40:49 +02:00
let event = this.getWidgetById('event_' + events[c].id);
if (!event)
2020-02-27 21:37:36 +01:00
continue;
if (this.isInTree()) {
2021-06-10 15:40:49 +02:00
event.doLoadingFinished();
2020-02-27 21:37:36 +01:00
}
}
// Show holidays as events on mobile or by preference
if (egwIsMobile() || egw.preference('birthdays_as_events', 'calendar')) {
this.day_class_holiday();
}
// Apply styles to hidden events
this._out_of_view();
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Apply styles for out-of-view and partially hidden events
*
* There are 3 different states or modes of display:
*
* - 'Normal' - When showing events positioned by time, the indicator is just
* a bar colored by the last category color. On hover it shows either the
* title of a single event or "x event(s)" if more than one are hidden.
* Clicking adjusts the current view to show the earliest / latest hidden
* event
*
* - Fixed - When showing events positioned by time but in a fixed-height
* week (not auto-sized to fit screen) the indicator is the same as sized.
* On hover it shows the titles of the hidden events, clicking changes
* the view to the selected day.
*
* - GridList - When showing just a list, the indicator shows "x event(s)",
* and on hover shows the category color, title & time. Clicking changes
* the view to the selected day, and opens the event for editing.
*/
2021-06-10 15:40:49 +02:00
_out_of_view() {
2020-02-27 21:37:36 +01:00
// Reset
this.header.children('.hiddenEventBefore').remove();
this.div.children('.hiddenEventAfter').remove();
this.event_wrapper.css('overflow', 'visible');
this.all_day.removeClass('overflown');
jQuery('.calendar_calEventBody', this.div).css({ 'padding-top': '', 'margin-top': '' });
2021-06-10 15:40:49 +02:00
const timegrid = this.getParent();
2020-02-27 21:37:36 +01:00
// elem is jquery div of event
function isHidden(elem) {
// Add an extra 5px top and bottom to include events just on the
// edge of visibility
2021-06-10 15:40:49 +02:00
const docViewTop = timegrid.scrolling.scrollTop() + 5, docViewBottom = docViewTop + (this.display_settings.granularity === 0 ?
2020-02-27 21:37:36 +01:00
this.event_wrapper.height() :
timegrid.scrolling.height() - 10), elemTop = elem.position().top, elemBottom = elemTop + elem.outerHeight(true);
if ((elemBottom <= docViewBottom) && (elemTop >= docViewTop)) {
// Entirely visible
return false;
}
2021-06-10 15:40:49 +02:00
const visible = {
2020-02-27 21:37:36 +01:00
hidden: elemTop > docViewTop ? 'bottom' : 'top',
completely: false
};
visible.completely = visible.hidden == 'top' ? elemBottom < docViewTop : elemTop > docViewBottom;
return visible;
}
// In gridlist view, we can quickly check if we need it at all
if (this.display_settings.granularity === 0 && this._children.length) {
jQuery('div.calendar_calEvent', this.div).show(0);
if (Math.ceil(this.div.height() / this._children[0].div.height()) > this._children.length) {
return;
}
}
// Check all day overflow
this.all_day.toggleClass('overflown', this.all_day[0].scrollHeight - this.all_day.innerHeight() > 5);
// Check each event
this.iterateOver(function (event) {
// Skip whole day events and events missing value
if (this.display_settings.granularity && ((!event.options || !event.options.value || event.options.value.whole_day_on_top))) {
return;
}
// Reset
event.title.css({ 'top': '', 'background-color': '' });
event.body.css({ 'padding-top': '', 'margin-top': '' });
2021-06-10 15:40:49 +02:00
const hidden = isHidden.call(this, event.div);
const day = this;
2020-02-27 21:37:36 +01:00
if (!hidden) {
return;
}
// Only top is hidden, move label
// Bottom hidden is fine
if (hidden.hidden === 'top' && !hidden.completely && !event.div.hasClass('calendar_calEventSmall')) {
2021-06-10 15:40:49 +02:00
const title_height = event.title.outerHeight();
2020-02-27 21:37:36 +01:00
event.title.css({
'top': timegrid.scrolling.scrollTop() - event.div.position().top,
'background-color': 'transparent'
});
event.body.css({
'padding-top': timegrid.scrolling.scrollTop() - event.div.position().top + title_height,
'margin-top': -title_height
});
}
// Too many in gridlist view, show indicator
else if (this.display_settings.granularity === 0 && hidden) {
if (jQuery('.hiddenEventAfter', this.div).length == 0) {
this.event_wrapper.css('overflow', 'hidden');
}
this._hidden_indicator(event, false, function () {
app.calendar.update_state({ view: 'day', date: day.date });
});
// Avoid partially visible events
// We need to hide all, or the next row will be visible
event.div.hide(0);
}
// Completely out of view, show indicator
else if (hidden.completely) {
this._hidden_indicator(event, hidden.hidden == 'top', false);
}
2021-06-10 15:40:49 +02:00
}, this, et2_calendar_event);
}
2020-02-27 21:37:36 +01:00
/**
* Show an indicator that there are hidden events
*
* The indicator works 3 different ways, depending on if the day can be
* scrolled, is fixed, or if in gridview.
*
* @see _out_of_view()
*
* @param {et2_calendar_event} event Event we're creating the indicator for
* @param {boolean} top Events hidden at the top (true) or bottom (false)
* @param {function} [onclick] Callback for when user clicks on the indicator
*/
2021-06-10 15:40:49 +02:00
_hidden_indicator(event, top, onclick) {
let indicator = null;
const day = this;
const timegrid = this.getParent();
const fixed_height = timegrid.div.hasClass('calendar_calTimeGridFixed');
2020-02-27 21:37:36 +01:00
// Event is before the displayed times
if (top) {
// Create if not already there
if (jQuery('.hiddenEventBefore', this.header).length === 0) {
indicator = jQuery('<div class="hiddenEventBefore"></div>')
.appendTo(this.header)
.attr('data-hidden_count', 1);
if (!fixed_height) {
indicator
.text(event.options.value.title)
.on('click', typeof onclick === 'function' ? onclick : function () {
jQuery('.calendar_calEvent', day.div).first()[0].scrollIntoView();
return false;
});
}
}
else {
indicator = jQuery('.hiddenEventBefore', this.header);
indicator.attr('data-hidden_count', parseInt(indicator.attr('data-hidden_count')) + 1);
if (!fixed_height) {
indicator.text(day.egw().lang('%1 event(s) %2', indicator.attr('data-hidden_count'), ''));
}
}
}
// Event is after displayed times
else {
indicator = jQuery('.hiddenEventAfter', this.div);
// Create if not already there
if (indicator.length === 0) {
indicator = jQuery('<div class="hiddenEventAfter"></div>')
.attr('data-hidden_count', 0)
.appendTo(this.div);
if (!fixed_height) {
indicator
.on('click', typeof onclick === 'function' ? onclick : function () {
jQuery('.calendar_calEvent', day.div).last()[0].scrollIntoView(false);
// Better re-run this to clean up
day._out_of_view();
return false;
});
}
else {
indicator
.on('mouseover', function () {
indicator.css({
'height': (indicator.attr('data-hidden_count') * 1.2) + 'em',
'margin-top': -(indicator.attr('data-hidden_count') * 1.2) + 'em'
});
})
.on('mouseout', function () {
indicator.css({
'height': '',
'margin-top': ''
});
});
}
}
2021-06-10 15:40:49 +02:00
const count = parseInt(indicator.attr('data-hidden_count')) + 1;
2020-02-27 21:37:36 +01:00
indicator.attr('data-hidden_count', count);
if (this.display_settings.granularity === 0) {
indicator.append(event.div.clone());
indicator.attr('data-hidden_label', day.egw().lang('%1 event(s) %2', indicator.attr('data-hidden_count'), ''));
}
else if (!fixed_height) {
indicator.text(day.egw().lang('%1 event(s) %2', indicator.attr('data-hidden_count'), ''));
}
indicator.css('top', timegrid.scrolling.height() + timegrid.scrolling.scrollTop() - indicator.innerHeight());
}
// Show different stuff for fixed height
if (fixed_height) {
indicator
.append("<div id='" + event.dom_id +
"' data-id='" + event.options.value.id + "'>" +
event.options.value.title +
"</div>");
}
// Match color to the event
if (indicator !== null) {
// Avoid white, which is hard to see
// Use border-bottom-color, Firefox doesn't give a value with border-color
2021-06-10 15:40:49 +02:00
const color = jQuery.Color(event.div.css('background-color')).toString() !== jQuery.Color('white').toString() ?
2020-02-27 21:37:36 +01:00
event.div.css('background-color') : event.div.css('border-bottom-color');
if (color !== 'rgba(0, 0, 0, 0)') {
indicator.css('border-color', color);
}
}
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Sort a day's events into minimally overlapping columns
*
* @returns {Array[]} Events sorted into columns
*/
2021-06-10 15:40:49 +02:00
_spread_events() {
2020-02-27 21:37:36 +01:00
if (!this.date)
return [];
2021-06-10 15:40:49 +02:00
let day_start = this.date.valueOf() / 1000;
const dst_check = new Date(this.date);
2020-02-27 21:37:36 +01:00
dst_check.setUTCHours(12);
// if daylight saving is switched on or off, correct $day_start
// gives correct times after 2am, times between 0am and 2am are wrong
2021-06-10 15:40:49 +02:00
const daylight_diff = day_start + 12 * 60 * 60 - (dst_check.valueOf() / 1000);
2020-02-27 21:37:36 +01:00
if (daylight_diff) {
day_start -= daylight_diff;
}
2021-06-10 15:40:49 +02:00
const eventCols = [], col_ends = [];
2020-02-27 21:37:36 +01:00
// Make sure children are in cronological order, or columns are backwards
this._children.sort(function (a, b) {
2021-06-10 15:40:49 +02:00
const start = new Date(a.options.value.start) - new Date(b.options.value.start);
const end = new Date(a.options.value.end) - new Date(b.options.value.end);
2020-02-27 21:37:36 +01:00
// Whole day events sorted by ID, normal events by start / end time
if (a.options.value.whole_day && b.options.value.whole_day) {
// Longer duration comes first so we have nicer bars across the top
2021-06-10 15:40:49 +02:00
const duration = (new Date(b.options.value.end) - new Date(b.options.value.start)) -
2020-02-27 21:37:36 +01:00
(new Date(a.options.value.end) - new Date(a.options.value.start));
return duration ? duration : (a.options.value.app_id - b.options.value.app_id);
}
else if (a.options.value.whole_day || b.options.value.whole_day) {
return a.options.value.whole_day ? -1 : 1;
}
return start ? start : end;
});
2021-06-10 15:40:49 +02:00
for (let i = 0; i < this._children.length; i++) {
const event = this._children[i].options.value || false;
if (!event)
2020-02-27 21:37:36 +01:00
continue;
2021-06-10 15:40:49 +02:00
if (event.date && event.date != this.options.date &&
2020-02-27 21:37:36 +01:00
// Multi-day events date may be different
2021-06-10 15:40:49 +02:00
(new Date(event.start) >= this.date || new Date(event.end) < this.date)) {
2020-02-27 21:37:36 +01:00
// Still have a child event that has changed date (DnD)
this._children[i].destroy();
this.removeChild(this._children[i]);
continue;
}
2021-06-10 15:40:49 +02:00
let c = 0;
event['multiday'] = false;
if (typeof event.start !== 'object') {
event.start = new Date(event.start);
2020-02-27 21:37:36 +01:00
}
2021-06-10 15:40:49 +02:00
if (typeof event.end !== 'object') {
event.end = new Date(event.end);
2020-02-27 21:37:36 +01:00
}
2021-06-10 15:40:49 +02:00
event['start_m'] = parseInt(String((event.start.valueOf() / 1000 - day_start) / 60), 10);
if (event['start_m'] < 0) {
event['start_m'] = 0;
event['multiday'] = true;
2020-02-27 21:37:36 +01:00
}
2021-06-10 15:40:49 +02:00
event['end_m'] = parseInt(String((event.end.valueOf() / 1000 - day_start) / 60), 10);
if (event['end_m'] >= 24 * 60) {
event['end_m'] = 24 * 60 - 1;
event['multiday'] = true;
2020-02-27 21:37:36 +01:00
}
2021-06-10 15:40:49 +02:00
if (!event.start.getUTCHours() && !event.start.getUTCMinutes() && event.end.getUTCHours() == 23 && event.end.getUTCMinutes() == 59) {
event.whole_day_on_top = (event.non_blocking && event.non_blocking != '0');
2020-02-27 21:37:36 +01:00
}
2021-06-10 15:40:49 +02:00
if (!event['whole_day_on_top']) {
for (c = 0; event['start_m'] < col_ends[c]; ++c)
2020-02-27 21:37:36 +01:00
;
2021-06-10 15:40:49 +02:00
col_ends[c] = event['end_m'];
2020-02-27 21:37:36 +01:00
}
if (typeof eventCols[c] === 'undefined') {
eventCols[c] = [];
}
eventCols[c].push(this._children[i]);
}
return eventCols;
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Position the event according to its time and how this widget is laid
* out.
*
* @param {et2_calendar_event} [event] - Event to be updated
* If a single event is not provided, all events are repositioned.
*/
2021-06-10 15:40:49 +02:00
position_event(event) {
2020-02-27 21:37:36 +01:00
// If hidden, skip it - it takes too long
if (!this.div.is(':visible'))
return;
// Sort events into minimally-overlapping columns
2021-06-10 15:40:49 +02:00
const columns = this._spread_events();
for (let c = 0; c < columns.length; c++) {
2020-02-27 21:37:36 +01:00
// Calculate horizontal positioning
2021-06-10 15:40:49 +02:00
let left = Math.ceil(5 + (1.5 * 100 / (parseFloat(this.options.width) || 100)));
let right = 2;
2020-02-27 21:37:36 +01:00
if (columns.length !== 1) {
right = !c ? 30 : 2;
left += c * (100.0 - left) / columns.length;
}
2021-06-10 15:40:49 +02:00
for (let i = 0; (columns[c].indexOf(event) >= 0 || !event) && i < columns[c].length; i++) {
2020-02-27 21:37:36 +01:00
// Calculate vertical positioning
2021-06-10 15:40:49 +02:00
let top = 0;
let height = 0;
2020-02-27 21:37:36 +01:00
// Position the event
if (this.display_settings.granularity === 0) {
if (this.all_day.has(columns[c][i].div).length) {
columns[c][i].div.prependTo(this.event_wrapper);
}
columns[c][i].div.css('top', '');
columns[c][i].div.css('height', '');
columns[c][i].div.css('left', '');
columns[c][i].div.css('right', '');
// Strip out of view padding
columns[c][i].body.css('padding-top', '');
continue;
}
if (columns[c][i].options.value.whole_day_on_top) {
if (!this.all_day.has(columns[c][i].div).length) {
columns[c][i].div.css('top', '');
columns[c][i].div.css('height', '');
columns[c][i].div.css('left', '');
columns[c][i].div.css('right', '');
columns[c][i].body.css('padding-top', '');
columns[c][i].div
.appendTo(this.all_day);
this.getParent().resizeTimes();
}
continue;
}
else {
if (this.all_day.has(columns[c][i].div).length) {
columns[c][i].div.appendTo(this.event_wrapper);
this.getParent().resizeTimes();
}
2021-06-10 15:40:49 +02:00
top = this._time_to_position(columns[c][i].options.value.start_m);
height = this._time_to_position(columns[c][i].options.value.end_m) - top;
2020-02-27 21:37:36 +01:00
}
// Position the event
if (event && columns[c].indexOf(event) >= 0 || !event) {
2021-06-10 15:40:49 +02:00
columns[c][i].div.css('top', top + '%');
2020-02-27 21:37:36 +01:00
columns[c][i].div.css('height', height + '%');
// Remove spacing from border, but only if visible or the height will be wrong
if (columns[c][i].div.is(':visible')) {
2021-06-10 15:40:49 +02:00
const border_diff = columns[c][i].div.outerHeight() - columns[c][i].div.height();
2020-02-27 21:37:36 +01:00
columns[c][i].div.css('height', 'calc(' + height + '% - ' + border_diff + ')');
}
// This gives the wrong height
//columns[c][i].div.outerHeight(height+'%');
columns[c][i].div.css('left', left.toFixed(1) + '%');
columns[c][i].div.css('right', right.toFixed(1) + '%');
columns[c][i].div.css('z-index', parseInt(20) + c);
columns[c][i]._small_size();
}
}
// Only wanted to position this event, leave the other columns alone
if (event && columns[c].indexOf(event) >= 0) {
return;
}
}
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Calculates the vertical position based on the time
*
* This calculation is a percentage from 00:00 to 23:59
*
* @param {int} time in minutes from midnight
* @return {float} position in percent
*/
2021-06-10 15:40:49 +02:00
_time_to_position(time) {
let pos = 0.0;
2020-02-27 21:37:36 +01:00
// 24h
pos = ((time / 60) / 24) * 100;
return pos.toFixed(1);
2021-06-10 15:40:49 +02:00
}
attachToDOM() {
let result = super.attachToDOM();
2020-02-27 21:37:36 +01:00
// Remove the binding for the click handler, unless there's something
// custom here.
if (!this.onclick) {
jQuery(this.node).off("click");
}
// But we do want to listen to certain clicks, and handle them internally
jQuery(this.node).on('click.et2_daycol', '.calendar_calDayColHeader,.calendar_calAddEvent', jQuery.proxy(this.click, this));
return result;
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Click handler calling custom handler set via onclick attribute to this.onclick,
* or the default which is to open a new event at that time.
*
* Normally, you don't bind to this one, but the attribute is supported if you
* can get a reference to the widget.
*
* @param {Event} _ev
* @returns {boolean}
*/
2021-06-10 15:40:49 +02:00
click(_ev) {
2020-02-27 21:37:36 +01:00
if (this.getParent().options.readonly)
return;
// Drag to create in progress
if (this.getParent().drag_create.start !== null)
return;
// Click on the title
if (jQuery(_ev.target).hasClass('calendar_calAddEvent')) {
if (this.header.has(_ev.target).length == 0 && !_ev.target.dataset.whole_day) {
// Default handler to open a new event at the selected time
var options = {
date: _ev.target.dataset.date || this.options.date,
hour: _ev.target.dataset.hour || this.getParent().options.day_start,
minute: _ev.target.dataset.minute || 0,
owner: this.options.owner
};
app.calendar.add(options);
return false;
}
// Header, all day non-blocking
else if (this.header.has(_ev.target).length && !jQuery('.hiddenEventBefore', this.header).has(_ev.target).length ||
this.header.is(_ev.target)) {
// Click on the header, but not the title. That's an all-day non-blocking
2021-06-10 15:40:49 +02:00
const end = this.date.getFullYear() + '-' + (this.date.getUTCMonth() + 1) + '-' + this.date.getUTCDate() + 'T23:59';
let options = {
2020-02-27 21:37:36 +01:00
start: this.date.toJSON(),
end: end,
non_blocking: true,
owner: this.options.owner
};
2021-06-10 15:40:49 +02:00
app.calendar.add(options);
2020-02-27 21:37:36 +01:00
return false;
}
}
// Day label
else if (this.title.is(_ev.target) || this.title.has(_ev.target).length) {
app.calendar.update_state({ view: 'day', date: this.date.toJSON() });
return false;
}
2021-06-10 15:40:49 +02:00
}
2020-02-27 21:37:36 +01:00
/**
* Code for implementing et2_IDetachedDOM
*
* @param {array} _attrs array to add further attributes to
*/
2021-06-10 15:40:49 +02:00
getDetachedAttributes(_attrs) {
}
getDetachedNodes() {
2020-02-27 21:37:36 +01:00
return [this.getDOMNode(this)];
2021-06-10 15:40:49 +02:00
}
setDetachedAttributes(_nodes, _values) {
}
2020-02-27 21:37:36 +01:00
// Resizable interface
/**
* Resize
*
* Parent takes care of setting proper width & height for the containing div
* here we just need to adjust the events to fit the new size.
*/
2021-06-10 15:40:49 +02:00
resize() {
2020-02-27 21:37:36 +01:00
if (this.disabled || !this.div.is(':visible') || this.getParent().disabled) {
return;
}
if (this.display_settings.granularity !== this.getParent().options.granularity) {
// Layout has changed
this._draw();
// Resize & position all events
this.position_event();
}
else {
// Don't need to resize & reposition, just clear some stuff
// to reset for _out_of_view()
this.iterateOver(function (widget) {
widget._small_size();
2021-06-10 15:40:49 +02:00
}, this, et2_calendar_event);
2020-02-27 21:37:36 +01:00
}
this._out_of_view();
2021-06-10 15:40:49 +02:00
}
}
et2_calendar_daycol._attributes = {
date: {
name: "Date",
type: "any",
description: "What date is this daycol for. YYYYMMDD or Date",
default: et2_no_init
},
owner: {
name: "Owner",
type: "any",
default: et2_no_init,
description: "Account ID number of the calendar owner, if not the current user"
},
display_birthday_as_event: {
name: "Birthdays",
type: "boolean",
default: false,
description: "Display birthdays as events"
},
display_holiday_as_event: {
name: "Holidays",
type: "boolean",
default: false,
description: "Display holidays as events"
}
};
et2_register_widget(et2_calendar_daycol, ["calendar-daycol"]);
2020-02-27 21:37:36 +01:00
//# sourceMappingURL=et2_widget_daycol.js.map