2016-03-01 17:27:45 +01:00
|
|
|
/*
|
2015-06-25 19:44:28 +02:00
|
|
|
* Egroupware
|
2021-06-11 11:31:06 +02:00
|
|
|
*
|
|
|
|
* @license https://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
|
|
* @package calendar
|
|
|
|
* @subpackage etemplate
|
|
|
|
* @link https://www.egroupware.org
|
2015-06-25 19:44:28 +02:00
|
|
|
* @author Nathan Gray
|
|
|
|
*/
|
2016-03-01 17:27:45 +01:00
|
|
|
/*egw:uses
|
2020-02-27 21:37:36 +01:00
|
|
|
/calendar/js/et2_widget_view.js;
|
|
|
|
/calendar/js/et2_widget_daycol.js;
|
|
|
|
/calendar/js/et2_widget_event.js;
|
2016-03-01 17:27:45 +01:00
|
|
|
*/
|
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 { ClassWithAttributes } from "../../api/js/etemplate/et2_core_inheritance";
|
|
|
|
import { et2_action_object_impl } from "../../api/js/etemplate/et2_core_DOMWidget";
|
|
|
|
import { EGW_AI_DRAG_OUT, EGW_AI_DRAG_OVER, egw_getObjectManager, egwActionObject } from "../../api/js/egw_action/egw_action.js";
|
|
|
|
import { egw } from "../../api/js/jsapi/egw_global";
|
2015-06-25 19:44:28 +02:00
|
|
|
/**
|
|
|
|
* Class for one row of a planner
|
|
|
|
*
|
|
|
|
* This widget is responsible for the label on the side
|
|
|
|
*
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
export class et2_calendar_planner_row 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_planner_row._attributes, _child || {}));
|
|
|
|
this._row_height = 20;
|
2020-02-27 21:37:36 +01:00
|
|
|
// 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_plannerRowWidget")
|
2021-06-10 15:40:49 +02:00
|
|
|
.css('width', this.options.width);
|
|
|
|
this.title = jQuery(document.createElement('div'))
|
2020-02-27 21:37:36 +01:00
|
|
|
.addClass("calendar_plannerRowHeader")
|
2021-06-10 15:40:49 +02:00
|
|
|
.appendTo(this.div);
|
|
|
|
this.rows = jQuery(document.createElement('div'))
|
2020-02-27 21:37:36 +01:00
|
|
|
.addClass("calendar_eventRows")
|
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
|
2021-06-10 15:40:49 +02:00
|
|
|
this._date_helper = et2_createWidget('date-time', {}, null);
|
|
|
|
this._date_helper.loadingFinished();
|
|
|
|
this.set_start_date(this.options.start_date);
|
|
|
|
this.set_end_date(this.options.end_date);
|
|
|
|
this._cached_rows = [];
|
2020-02-27 21:37:36 +01:00
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
doLoadingFinished() {
|
|
|
|
super.doLoadingFinished();
|
2020-02-27 21:37:36 +01:00
|
|
|
this.set_label(this.options.label);
|
|
|
|
this._draw();
|
|
|
|
// Actions are set on the parent, so we need to explicitly get in here
|
|
|
|
// and get ours
|
|
|
|
this._link_actions(this.getParent().options.actions || []);
|
|
|
|
return true;
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
|
|
|
destroy() {
|
|
|
|
super.destroy();
|
2020-02-27 21:37:36 +01:00
|
|
|
// date_helper has no parent, so we must explicitly remove it
|
|
|
|
this._date_helper.destroy();
|
|
|
|
this._date_helper = null;
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
|
|
|
getDOMNode(_sender) {
|
2020-02-27 21:37:36 +01:00
|
|
|
if (_sender === this || !_sender) {
|
|
|
|
return this.div[0];
|
|
|
|
}
|
|
|
|
if (_sender._parent === this) {
|
|
|
|
return this.rows[0];
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Link the actions to the DOM nodes / widget bits.
|
|
|
|
*
|
|
|
|
* @param {object} actions {ID: {attributes..}+} map of egw action information
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
_link_actions(actions) {
|
2020-02-27 21:37:36 +01:00
|
|
|
// Get the parent? Might be a grid row, might not. Either way, it is
|
|
|
|
// just a container with no valid actions
|
2021-06-10 15:40:49 +02:00
|
|
|
let objectManager = egw_getObjectManager(this.getInstanceManager().app, true, 1);
|
2020-02-27 21:37:36 +01:00
|
|
|
objectManager = objectManager.getObjectById(this.getInstanceManager().uniqueId, 2) || objectManager;
|
2021-06-10 15:40:49 +02:00
|
|
|
let parent = objectManager.getObjectById(this.id, 1) || objectManager.getObjectById(this.getParent().id, 1) || objectManager;
|
2020-02-27 21:37:36 +01:00
|
|
|
if (!parent) {
|
|
|
|
egw.debug('error', 'No parent objectManager found');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// This binds into the egw action system. Most user interactions (drag to move, resize)
|
|
|
|
// are handled internally using jQuery directly.
|
2021-06-10 15:40:49 +02:00
|
|
|
let widget_object = this._actionObject || parent.getObjectById(this.id);
|
|
|
|
const aoi = new et2_action_object_impl(this, this.getDOMNode(this)).getAOI();
|
|
|
|
const planner = this.getParent();
|
|
|
|
for (let i = 0; i < parent.children.length; i++) {
|
|
|
|
const parent_finder = jQuery(parent.children[i].iface.doGetDOMNode()).find(this.div);
|
2020-02-27 21:37:36 +01:00
|
|
|
if (parent_finder.length > 0) {
|
|
|
|
parent = parent.children[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Determine if we allow a dropped event to use the invite/change actions
|
2021-06-10 15:40:49 +02:00
|
|
|
const _invite_enabled = function (action, event, target) {
|
2020-02-27 21:37:36 +01:00
|
|
|
var event = event.iface.getWidget();
|
2021-06-10 15:40:49 +02:00
|
|
|
const row = target.iface.getWidget() || false;
|
2020-02-27 21:37:36 +01:00
|
|
|
if (event === row || !event || !row ||
|
|
|
|
!event.options || !event.options.value.participants) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
let owner_match = false;
|
|
|
|
const own_row = event.getParent() === row;
|
|
|
|
for (let id in event.options.value.participants) {
|
2020-02-27 21:37:36 +01:00
|
|
|
owner_match = owner_match || row.node.dataset.participants === '' + id;
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
const enabled = !owner_match &&
|
2020-02-27 21:37:36 +01:00
|
|
|
// Not inside its own timegrid
|
|
|
|
!own_row;
|
|
|
|
widget_object.getActionLink('invite').enabled = enabled;
|
|
|
|
widget_object.getActionLink('change_participant').enabled = enabled;
|
|
|
|
// If invite or change participant are enabled, drag is not
|
|
|
|
widget_object.getActionLink('egw_link_drop').enabled = !enabled;
|
|
|
|
};
|
|
|
|
aoi.doTriggerEvent = function (_event, _data) {
|
|
|
|
// Determine target node
|
|
|
|
var event = _data.event || false;
|
|
|
|
if (!event)
|
|
|
|
return;
|
|
|
|
if (_data.ui.draggable.hasClass('rowNoEdit'))
|
|
|
|
return;
|
|
|
|
/*
|
|
|
|
We have to handle the drop in the normal event stream instead of waiting
|
|
|
|
for the egwAction system so we can get the helper, and destination
|
|
|
|
*/
|
|
|
|
if (event.type === 'drop' && widget_object.getActionLink('egw_link_drop').enabled) {
|
|
|
|
this.getWidget().getParent()._event_drop.call(jQuery('.calendar_d-n-d_timeCounter', _data.ui.helper)[0], this.getWidget().getParent(), event, _data.ui, this.getWidget());
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
const drag_listener = function (_event, ui) {
|
2020-02-27 21:37:36 +01:00
|
|
|
if (planner.options.group_by === 'month') {
|
|
|
|
var position = { left: _event.clientX, top: _event.clientY };
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
var position = { top: ui.position.top, left: ui.position.left - jQuery(this).parent().offset().left };
|
|
|
|
}
|
|
|
|
aoi.getWidget().getParent()._drag_helper(jQuery('.calendar_d-n-d_timeCounter', ui.helper)[0], position, 0);
|
2021-06-10 15:40:49 +02:00
|
|
|
let event = _data.ui.draggable.data('selected')[0];
|
2020-02-27 21:37:36 +01:00
|
|
|
if (!event || event.id && event.id.indexOf('calendar') !== 0) {
|
|
|
|
event = false;
|
|
|
|
}
|
|
|
|
if (event) {
|
|
|
|
_invite_enabled(widget_object.getActionLink('invite').actionObj, event, widget_object);
|
|
|
|
}
|
|
|
|
};
|
2021-06-10 15:40:49 +02:00
|
|
|
const time = jQuery('.calendar_d-n-d_timeCounter', _data.ui.helper);
|
2020-02-27 21:37:36 +01:00
|
|
|
switch (_event) {
|
|
|
|
// Triggered once, when something is dragged into the timegrid's div
|
|
|
|
case EGW_AI_DRAG_OVER:
|
|
|
|
// Listen to the drag and update the helper with the time
|
|
|
|
// This part lets us drag between different timegrids
|
|
|
|
_data.ui.draggable.on('drag.et2_timegrid_row' + widget_object.id, drag_listener);
|
|
|
|
_data.ui.draggable.on('dragend.et2_timegrid_row' + widget_object.id, function () {
|
|
|
|
_data.ui.draggable.off('drag.et2_timegrid_row' + widget_object.id);
|
|
|
|
});
|
|
|
|
widget_object.iface.getWidget().div.addClass('drop-hover');
|
|
|
|
// Disable invite / change actions for same calendar or already participant
|
|
|
|
var event = _data.ui.draggable.data('selected')[0];
|
|
|
|
if (!event || event.id && event.id.indexOf('calendar') !== 0) {
|
|
|
|
event = false;
|
|
|
|
}
|
|
|
|
if (event) {
|
|
|
|
_invite_enabled(widget_object.getActionLink('invite').actionObj, event, widget_object);
|
|
|
|
}
|
|
|
|
if (time.length) {
|
|
|
|
// The out will trigger after the over, so we count
|
|
|
|
time.data('count', time.data('count') + 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
_data.ui.helper.prepend('<div class="calendar_d-n-d_timeCounter" data-count="1"><span></span></div>');
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// Triggered once, when something is dragged out of the timegrid
|
|
|
|
case EGW_AI_DRAG_OUT:
|
|
|
|
// Stop listening
|
|
|
|
_data.ui.draggable.off('drag.et2_timegrid_row' + widget_object.id);
|
|
|
|
// Remove highlight
|
|
|
|
widget_object.iface.getWidget().div.removeClass('drop-hover');
|
|
|
|
// Out triggers after the over, count to not accidentally remove
|
|
|
|
time.data('count', time.data('count') - 1);
|
|
|
|
if (time.length && time.data('count') <= 0) {
|
|
|
|
time.remove();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
if (widget_object == null) {
|
|
|
|
// Add a new container to the object manager which will hold the widget
|
|
|
|
// objects
|
|
|
|
widget_object = parent.insertObject(false, new egwActionObject(this.id, parent, aoi, this._actionManager || parent.manager.getActionById(this.id) || parent.manager));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
widget_object.setAOI(aoi);
|
|
|
|
}
|
|
|
|
this._actionObject = widget_object;
|
|
|
|
// Delete all old objects
|
|
|
|
widget_object.clear();
|
|
|
|
widget_object.unregisterActions();
|
|
|
|
// Go over the widget & add links - this is where we decide which actions are
|
|
|
|
// 'allowed' for this widget at this time
|
2021-06-10 15:40:49 +02:00
|
|
|
const action_links = this._get_action_links(actions);
|
2020-02-27 21:37:36 +01:00
|
|
|
this.getParent()._init_links_dnd(widget_object.manager, action_links);
|
|
|
|
widget_object.updateActionLinks(action_links);
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Get all action-links / id's of 1.-level actions from a given action object
|
|
|
|
*
|
|
|
|
* Here we are only interested in drop events.
|
|
|
|
*
|
|
|
|
* @param actions
|
|
|
|
* @returns {Array}
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
_get_action_links(actions) {
|
|
|
|
const action_links = [];
|
2020-02-27 21:37:36 +01:00
|
|
|
// Only these actions are allowed without a selection (empty actions)
|
2021-06-10 15:40:49 +02:00
|
|
|
const empty_actions = ['add'];
|
|
|
|
for (let i in actions) {
|
|
|
|
const action = actions[i];
|
2020-02-27 21:37:36 +01:00
|
|
|
if (empty_actions.indexOf(action.id) !== -1 || action.type == 'drop') {
|
|
|
|
action_links.push(typeof action.id != 'undefined' ? action.id : i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return action_links;
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Draw the individual divs for weekends and events
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
_draw() {
|
2020-02-27 21:37:36 +01:00
|
|
|
// Remove any existing
|
|
|
|
this.rows.remove('.calendar_eventRowsMarkedDay,.calendar_eventRowsFiller').nextAll().remove();
|
2021-06-10 15:40:49 +02:00
|
|
|
let days = 31;
|
|
|
|
let width = '100';
|
2020-02-27 21:37:36 +01:00
|
|
|
if (this.getParent().options.group_by === 'month') {
|
|
|
|
days = this.options.end_date.getUTCDate();
|
|
|
|
if (days < 31) {
|
2021-06-10 15:40:49 +02:00
|
|
|
const diff = 31 - days;
|
2020-02-27 21:37:36 +01:00
|
|
|
width = 'calc(' + (diff * 3.23) + '% - ' + (diff * 7) + 'px)';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// mark weekends and other special days in yearly planner
|
|
|
|
if (this.getParent().options.group_by == 'month') {
|
|
|
|
this.rows.append(this._yearlyPlannerMarkDays(this.options.start_date, days));
|
|
|
|
}
|
|
|
|
if (this.getParent().options.group_by === 'month' && days < 31) {
|
|
|
|
// add a filler for non existing days in that month
|
|
|
|
this.rows.after('<div class="calendar_eventRowsFiller"' +
|
|
|
|
' style="width:' + width + ';" ></div>');
|
|
|
|
}
|
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);
|
|
|
|
if (this.getParent().options.group_by === 'month') {
|
|
|
|
this.title.attr('data-date', this.options.start_date.toJSON());
|
|
|
|
this.title.attr('data-sortby', 'user');
|
|
|
|
this.title.addClass('et2_clickable et2_link');
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.title.attr('data-date', '');
|
|
|
|
this.title.removeClass('et2_clickable');
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Change the start date
|
|
|
|
*
|
|
|
|
* @param {Date} new_date New end date
|
|
|
|
* @returns {undefined}
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
set_start_date(new_date) {
|
2020-02-27 21:37:36 +01:00
|
|
|
if (!new_date || new_date === null) {
|
|
|
|
throw new TypeError('Invalid end date. ' + new_date.toString());
|
|
|
|
}
|
|
|
|
this.options.start_date = new Date(typeof new_date == 'string' ? new_date : new_date.toJSON());
|
|
|
|
this.options.start_date.setUTCHours(0);
|
|
|
|
this.options.start_date.setUTCMinutes(0);
|
|
|
|
this.options.start_date.setUTCSeconds(0);
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Change the end date
|
|
|
|
*
|
|
|
|
* @param {string|number|Date} new_date New end date
|
|
|
|
* @returns {undefined}
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
set_end_date(new_date) {
|
2020-02-27 21:37:36 +01:00
|
|
|
if (!new_date || new_date === null) {
|
|
|
|
throw new TypeError('Invalid end date. ' + new_date.toString());
|
|
|
|
}
|
|
|
|
this.options.end_date = new Date(typeof new_date == 'string' ? new_date : new_date.toJSON());
|
|
|
|
this.options.end_date.setUTCHours(23);
|
|
|
|
this.options.end_date.setUTCMinutes(59);
|
|
|
|
this.options.end_date.setUTCSeconds(59);
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Mark special days (birthdays, holidays) on the planner
|
|
|
|
*
|
|
|
|
* @param {Date} start Start of the month
|
|
|
|
* @param {number} days How many days in the month
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
_yearlyPlannerMarkDays(start, days) {
|
|
|
|
const day_width = 3.23;
|
|
|
|
const t = new Date(start);
|
|
|
|
let content = '';
|
|
|
|
for (let i = 0; i < days; i++) {
|
|
|
|
const holidays = [];
|
2020-02-27 21:37:36 +01:00
|
|
|
// TODO: implement this, pull / copy data from et2_widget_timegrid
|
2021-06-10 15:40:49 +02:00
|
|
|
const day_class = this.getParent().day_class_holiday(t, holidays);
|
2020-02-27 21:37:36 +01:00
|
|
|
if (day_class) // no regular weekday
|
|
|
|
{
|
|
|
|
content += '<div class="calendar_eventRowsMarkedDay ' + day_class +
|
|
|
|
'" style="left: ' + (i * day_width) + '%; width:' + day_width + '%;"' +
|
|
|
|
(holidays ? ' title="' + holidays.join(',') + '"' : '') +
|
|
|
|
' ></div>';
|
|
|
|
}
|
|
|
|
t.setUTCDate(t.getUTCDate() + 1);
|
|
|
|
}
|
|
|
|
return content;
|
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 row.
|
|
|
|
*
|
|
|
|
* @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) {
|
|
|
|
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.getParent().disabled && event_ids.length > 0) {
|
|
|
|
this.resize();
|
|
|
|
this._update_events(events);
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
|
|
|
get date_helper() {
|
|
|
|
return this._date_helper;
|
|
|
|
}
|
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) {
|
2020-02-27 21:37:36 +01:00
|
|
|
// Remove all 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._cached_rows = [];
|
|
|
|
for (var c = 0; c < events.length; c++) {
|
|
|
|
// Create event
|
|
|
|
var event = et2_createWidget('calendar-event', {
|
|
|
|
id: 'event_' + events[c].row_id,
|
|
|
|
value: events[c]
|
|
|
|
}, this);
|
|
|
|
}
|
|
|
|
// Seperate loop so column sorting finds all children in the right place
|
|
|
|
for (var c = 0; c < events.length; c++) {
|
2021-06-10 15:40:49 +02:00
|
|
|
let event = this.getWidgetById('event_' + events[c].row_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
|
|
|
}
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Position the event according to it's time and how this widget is laid
|
|
|
|
* out.
|
|
|
|
*
|
|
|
|
* @param {undefined|Object|et2_calendar_event} event
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
position_event(event) {
|
|
|
|
const rows = this._spread_events();
|
|
|
|
const height = rows.length * this._row_height;
|
|
|
|
let row_width = this.rows.width();
|
2020-02-27 21:37:36 +01:00
|
|
|
if (row_width == 0) {
|
|
|
|
// Not rendered yet or something
|
|
|
|
row_width = this.getParent().gridHeader.width() - this.title.width();
|
|
|
|
}
|
|
|
|
row_width -= 15;
|
2021-06-10 15:40:49 +02:00
|
|
|
for (let c = 0; c < rows.length; c++) {
|
2020-02-27 21:37:36 +01:00
|
|
|
// Calculate vertical positioning
|
2021-06-10 15:40:49 +02:00
|
|
|
const top = c * (100.0 / rows.length);
|
|
|
|
for (let i = 0; (rows[c].indexOf(event) >= 0 || !event) && i < rows[c].length; i++) {
|
2020-02-27 21:37:36 +01:00
|
|
|
// Calculate horizontal positioning
|
2021-06-10 15:40:49 +02:00
|
|
|
const left = this._time_to_position(rows[c][i].options.value.start);
|
|
|
|
const width = this._time_to_position(rows[c][i].options.value.end) - left;
|
2020-02-27 21:37:36 +01:00
|
|
|
// Position the event
|
2021-06-10 15:40:49 +02:00
|
|
|
rows[c][i].div.css('top', top + '%');
|
2020-02-27 21:37:36 +01:00
|
|
|
rows[c][i].div.css('height', (100 / rows.length) + '%');
|
|
|
|
rows[c][i].div.css('left', left.toFixed(1) + '%');
|
|
|
|
rows[c][i].div.outerWidth((width / 100 * row_width) + 'px');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (height) {
|
|
|
|
this.div.height(height + 'px');
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Sort a day's events into non-overlapping rows
|
|
|
|
*
|
|
|
|
* @returns {Array[]} Events sorted into rows
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
_spread_events() {
|
2020-02-27 21:37:36 +01:00
|
|
|
// Keep it so we don't have to re-do it when the next event asks
|
2021-06-10 15:40:49 +02:00
|
|
|
let cached_length = 0;
|
2020-02-27 21:37:36 +01:00
|
|
|
this._cached_rows.map(function (row) { cached_length += row.length; });
|
|
|
|
if (cached_length === this._children.length) {
|
|
|
|
return this._cached_rows;
|
|
|
|
}
|
|
|
|
// sorting the events in non-overlapping rows
|
2021-06-10 15:40:49 +02:00
|
|
|
const rows = [];
|
|
|
|
const row_end = [0];
|
2020-02-27 21:37:36 +01:00
|
|
|
// Sort in chronological order, so earliest ones are at the top
|
|
|
|
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 n = 0; n < this._children.length; n++) {
|
|
|
|
const event = this._children[n].options.value || false;
|
|
|
|
if (typeof event.start !== 'object') {
|
|
|
|
this._date_helper.set_value(event.start);
|
|
|
|
event.start = new Date(this._date_helper.getValue());
|
2020-02-27 21:37:36 +01:00
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
if (typeof event.end !== 'object') {
|
|
|
|
this._date_helper.set_value(event.end);
|
|
|
|
event.end = new Date(this._date_helper.getValue());
|
2020-02-27 21:37:36 +01:00
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
if (typeof event['start_m'] === 'undefined') {
|
|
|
|
let day_start = event.start.valueOf() / 1000;
|
|
|
|
const dst_check = new Date(event.start);
|
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
|
|
|
event['start_m'] = event.start.getUTCHours() * 60 + event.start.getUTCMinutes();
|
|
|
|
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'] = event.end.getUTCHours() * 60 + event.end.getUTCMinutes();
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
// Skip events entirely on hidden weekends
|
2021-06-10 15:40:49 +02:00
|
|
|
if (this._hidden_weekend_event(event)) {
|
|
|
|
const node = this._children[n];
|
2020-02-27 21:37:36 +01:00
|
|
|
this.removeChild(n--);
|
|
|
|
node.destroy();
|
|
|
|
continue;
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
const event_start = new Date(event.start).valueOf();
|
2020-02-27 21:37:36 +01:00
|
|
|
for (var row = 0; row_end[row] > event_start; ++row)
|
|
|
|
; // find a "free" row (no other event)
|
|
|
|
if (typeof rows[row] === 'undefined')
|
|
|
|
rows[row] = [];
|
|
|
|
rows[row].push(this._children[n]);
|
2021-06-10 15:40:49 +02:00
|
|
|
row_end[row] = new Date(event['end']).valueOf();
|
2020-02-27 21:37:36 +01:00
|
|
|
}
|
|
|
|
this._cached_rows = rows;
|
|
|
|
return rows;
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Check to see if the event is entirely on a hidden weekend
|
|
|
|
*
|
|
|
|
* @param values Array of event values, not an et2_widget_event
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
_hidden_weekend_event(values) {
|
2020-02-27 21:37:36 +01:00
|
|
|
if (!this.getParent() || this.getParent().options.group_by == 'month' || this.getParent().options.show_weekend) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// Starts on Saturday or Sunday, ends Sat or Sun, less than 2 days long
|
|
|
|
else if ([0, 6].indexOf(values.start.getUTCDay()) !== -1 && [0, 6].indexOf(values.end.getUTCDay()) !== -1
|
|
|
|
&& values.end - values.start < 2 * 24 * 3600 * 1000) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
2020-02-27 21:37:36 +01:00
|
|
|
/**
|
|
|
|
* Calculates the horizontal position based on the time given, as a percentage
|
|
|
|
* between the start and end times
|
|
|
|
*
|
|
|
|
* @param {int|Date|string} time in minutes from midnight, or a Date in string or object form
|
|
|
|
* @param {int|Date|string} start Earliest possible time (0%)
|
|
|
|
* @param {int|Date|string} end Latest possible time (100%)
|
|
|
|
* @return {float} position in percent
|
|
|
|
*/
|
2021-06-10 15:40:49 +02:00
|
|
|
_time_to_position(time, start, end) {
|
|
|
|
let pos = 0.0;
|
2020-02-27 21:37:36 +01:00
|
|
|
// Handle the different value types
|
|
|
|
start = this.options.start_date;
|
|
|
|
end = this.options.end_date;
|
|
|
|
if (typeof start === 'string') {
|
|
|
|
start = new Date(start);
|
|
|
|
end = new Date(end);
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
const wd_start = 60 * (parseInt('' + egw.preference('workdaystarts', 'calendar')) || 9);
|
|
|
|
const wd_end = 60 * (parseInt('' + egw.preference('workdayends', 'calendar')) || 17);
|
|
|
|
let t = time;
|
2020-02-27 21:37:36 +01:00
|
|
|
if (typeof time === 'number' && time < 3600) {
|
|
|
|
t = new Date(start.valueOf() + wd_start * 3600 * 1000);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
t = new Date(time);
|
|
|
|
}
|
|
|
|
// Limits
|
|
|
|
if (t <= start)
|
|
|
|
return 0; // We are left of our scale
|
|
|
|
if (t >= end)
|
|
|
|
return 100; // We are right of our scale
|
|
|
|
// Remove space for weekends, if hidden
|
2021-06-10 15:40:49 +02:00
|
|
|
let weekend_count = 0;
|
|
|
|
let weekend_before = 0;
|
|
|
|
let partial_weekend = 0;
|
2020-02-27 21:37:36 +01:00
|
|
|
if (this.getParent().options.group_by !== 'month' && this.getParent() && !this.getParent().options.show_weekend) {
|
2021-06-10 15:40:49 +02:00
|
|
|
const counter_date = new Date(start);
|
2020-02-27 21:37:36 +01:00
|
|
|
do {
|
|
|
|
if ([0, 6].indexOf(counter_date.getUTCDay()) !== -1) {
|
|
|
|
if (counter_date.getUTCDate() === t.getUTCDate() && counter_date.getUTCMonth() === t.getUTCMonth()) {
|
|
|
|
// Event is partially on a weekend
|
|
|
|
partial_weekend += (t.getUTCHours() * 60 + t.getUTCMinutes()) * 60 * 1000;
|
|
|
|
}
|
|
|
|
else if (counter_date < t) {
|
|
|
|
weekend_before++;
|
|
|
|
}
|
|
|
|
weekend_count++;
|
|
|
|
}
|
|
|
|
counter_date.setUTCDate(counter_date.getUTCDate() + 1);
|
|
|
|
} while (counter_date < end);
|
|
|
|
// Put it in ms
|
|
|
|
weekend_before *= 24 * 3600 * 1000;
|
|
|
|
weekend_count *= 24 * 3600 * 1000;
|
|
|
|
}
|
|
|
|
// Basic scaling, doesn't consider working times
|
|
|
|
pos = (t - start - weekend_before - partial_weekend) / (end - start - weekend_count);
|
|
|
|
// Month view
|
|
|
|
if (this.getParent().options.group_by !== 'month') {
|
|
|
|
// Daywise scaling
|
|
|
|
/* Needs hourly scales that consider working hours
|
|
|
|
var start_date = new Date(start.getUTCFullYear(), start.getUTCMonth(),start.getUTCDate());
|
|
|
|
var end_date = new Date(end.getUTCFullYear(), end.getUTCMonth(),end.getUTCDate());
|
|
|
|
var t_date = new Date(t.getUTCFullYear(), t.getUTCMonth(),t.getUTCDate());
|
|
|
|
|
|
|
|
var days = Math.round((end_date - start_date) / (24 * 3600 * 1000))+1;
|
|
|
|
pos = 1 / days * Math.round((t_date - start_date) / (24*3600 * 1000));
|
|
|
|
|
|
|
|
var time_of_day = typeof t === 'object' ? 60 * t.getUTCHours() + t.getUTCMinutes() : t;
|
|
|
|
|
|
|
|
if (time_of_day >= wd_start)
|
|
|
|
{
|
|
|
|
var day_percentage = 0.1;
|
|
|
|
if (time_of_day > wd_end)
|
|
|
|
{
|
|
|
|
day_percentage = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var wd_length = wd_end - wd_start;
|
|
|
|
if (wd_length <= 0) wd_length = 24*60;
|
|
|
|
day_percentage = (time_of_day-wd_start) / wd_length; // between 0 and 1
|
|
|
|
}
|
|
|
|
pos += day_percentage / days;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// 2678400 is the number of seconds in 31 days
|
|
|
|
pos = (t - start) / 2678400000;
|
|
|
|
}
|
|
|
|
pos = 100 * pos;
|
|
|
|
return pos;
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
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;
|
|
|
|
}
|
2021-06-10 15:40:49 +02:00
|
|
|
const row = jQuery('<div class="calendar_plannerEventRowWidget"></div>').appendTo(this.rows);
|
2020-02-27 21:37:36 +01:00
|
|
|
this._row_height = (parseInt(window.getComputedStyle(row[0]).getPropertyValue("height")) || 20);
|
|
|
|
row.remove();
|
|
|
|
// Resize & position all events
|
|
|
|
this.position_event();
|
2021-06-10 15:40:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
et2_calendar_planner_row._attributes = {
|
|
|
|
start_date: {
|
|
|
|
name: "Start date",
|
|
|
|
type: "any"
|
|
|
|
},
|
|
|
|
end_date: {
|
|
|
|
name: "End date",
|
|
|
|
type: "any"
|
|
|
|
},
|
|
|
|
value: {
|
|
|
|
type: "any"
|
|
|
|
}
|
|
|
|
};
|
|
|
|
et2_register_widget(et2_calendar_planner_row, ["calendar-planner_row"]);
|
2020-02-27 21:37:36 +01:00
|
|
|
//# sourceMappingURL=et2_widget_planner_row.js.map
|