* Calendar: Add line indicating current time

This commit is contained in:
nathangray 2020-05-21 14:07:29 -06:00
parent 8bcbc184fb
commit c27522b00f
10 changed files with 232 additions and 8 deletions

View File

@ -234,9 +234,9 @@ var weekN = /** @class */ (function (_super_1) {
}; };
weekN.end_date = function (state) { weekN.end_date = function (state) {
state.days = '' + (state.days >= 5 ? state.days : egw.preference('days_in_weekview', 'calendar') || 7); state.days = '' + (state.days >= 5 ? state.days : egw.preference('days_in_weekview', 'calendar') || 7);
var d = app.calendar.date.start_of_week(app.calendar.View.start_date.call(this, state)); var d = app.calendar.date.start_of_week(_super_1.start_date.call(this, state));
// Always 7 days, we just turn weekends on or off // Always 7 days, we just turn weekends on or off
d.setUTCHours(24 * 7 * (parseInt(this.egw.preference('multiple_weeks', 'calendar')) || 3) - 1); d.setUTCHours(24 * 7 * (parseInt(app.calendar.egw.preference('multiple_weeks', 'calendar')) || 3) - 1);
return d; return d;
}; };
return weekN; return weekN;

View File

@ -257,9 +257,9 @@ export class weekN extends View
{ {
state.days = '' + (state.days >= 5 ? state.days : egw.preference('days_in_weekview', 'calendar') || 7); state.days = '' + (state.days >= 5 ? state.days : egw.preference('days_in_weekview', 'calendar') || 7);
var d = app.calendar.date.start_of_week(app.calendar.View.start_date.call(this, state)); var d = app.calendar.date.start_of_week(super.start_date(state));
// Always 7 days, we just turn weekends on or off // Always 7 days, we just turn weekends on or off
d.setUTCHours(24 * 7 * (parseInt(this.egw.preference('multiple_weeks', 'calendar')) || 3) - 1); d.setUTCHours(24 * 7 * (parseInt(app.calendar.egw.preference('multiple_weeks', 'calendar')) || 3) - 1);
return d; return d;
} }
} }

View File

@ -668,6 +668,7 @@ var et2_calendar_planner = /** @class */ (function (_super) {
} }
this.widget.update_timer = null; this.widget.update_timer = null;
this.widget.doInvalidate = true; this.widget.doInvalidate = true;
this.widget._updateNow();
window.setTimeout(jQuery.proxy(function () { if (this.loader) window.setTimeout(jQuery.proxy(function () { if (this.loader)
this.loader.hide(); }, this.widget), 500); this.loader.hide(); }, this.widget), 500);
}, { widget: this, "trigger": trigger }), et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT); }, { widget: this, "trigger": trigger }), et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT);
@ -819,6 +820,31 @@ var et2_calendar_planner = /** @class */ (function (_super) {
content += "</div>\n"; content += "</div>\n";
return content; return content;
}; };
/**
* Update the 'now' line
* @private
*/
et2_calendar_planner.prototype._updateNow = function () {
var now = _super.prototype._updateNow.call(this);
if (now === false || this.grouper == this.groupers.month) {
this.now_div.hide();
return false;
}
// Planner uses the dates, not just the times so need things right
now = new Date(now.valueOf() - now.getTimezoneOffset() * 60000);
var row = null;
for (var i = 0; i < this._children.length && row == null; i++) {
if (this._children[i].instanceOf(et2_widget_planner_row_1.et2_calendar_planner_row)) {
row = this._children[i];
}
}
if (!row) {
this.now_div.hide();
return false;
}
this.now_div.appendTo(this.grid)
.css('left', row._time_to_position(now) + '%');
};
/** /**
* Make a header showing the months * Make a header showing the months
* @param {Date} start * @param {Date} start

View File

@ -855,6 +855,7 @@ export class et2_calendar_planner extends et2_calendar_view implements et2_IDeta
this.widget.update_timer = null; this.widget.update_timer = null;
this.widget.doInvalidate = true; this.widget.doInvalidate = true;
this.widget._updateNow();
window.setTimeout(jQuery.proxy(function() {if(this.loader) this.loader.hide();},this.widget),500); window.setTimeout(jQuery.proxy(function() {if(this.loader) this.loader.hide();},this.widget),500);
},{widget:this,"trigger":trigger}),et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT); },{widget:this,"trigger":trigger}),et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT);
} }
@ -1054,6 +1055,37 @@ export class et2_calendar_planner extends et2_calendar_view implements et2_IDeta
return content; return content;
} }
/**
* Update the 'now' line
* @private
*/
public _updateNow()
{
let now = super._updateNow();
if(now === false || this.grouper == this.groupers.month)
{
this.now_div.hide();
return false;
}
// Planner uses the dates, not just the times so need things right
now = new Date(now.valueOf() - now.getTimezoneOffset()*60000);
let row = null;
for(let i = 0; i < this._children.length && row == null; i++)
{
if(this._children[i].instanceOf(et2_calendar_planner_row))
{
row = this._children[i];
}
}
if(!row)
{
this.now_div.hide();
return false;
}
this.now_div.appendTo(this.grid)
.css('left', row._time_to_position(now) + '%');
}
/** /**
* Make a header showing the months * Make a header showing the months

View File

@ -30,6 +30,7 @@ var et2_core_inheritance_1 = require("../../api/js/etemplate/et2_core_inheritanc
var et2_widget_view_1 = require("./et2_widget_view"); var et2_widget_view_1 = require("./et2_widget_view");
var et2_core_DOMWidget_1 = require("../../api/js/etemplate/et2_core_DOMWidget"); var et2_core_DOMWidget_1 = require("../../api/js/etemplate/et2_core_DOMWidget");
var et2_dataview_view_grid_1 = require("../../api/js/etemplate/et2_dataview_view_grid"); var et2_dataview_view_grid_1 = require("../../api/js/etemplate/et2_dataview_view_grid");
var et2_widget_daycol_1 = require("./et2_widget_daycol");
/** /**
* Class which implements the "calendar-timegrid" XET-Tag for displaying a span of days * Class which implements the "calendar-timegrid" XET-Tag for displaying a span of days
* *
@ -450,6 +451,7 @@ var et2_calendar_timegrid = /** @class */ (function (_super) {
if (this.trigger) { if (this.trigger) {
this.widget.change(); this.widget.change();
} }
this.widget._updateNow();
// Hide loader // Hide loader
window.setTimeout(jQuery.proxy(function () { this.loader.hide(); }, this.widget), 200); window.setTimeout(jQuery.proxy(function () { this.loader.hide(); }, this.widget), 200);
}, { widget: this, "trigger": trigger }), et2_dataview_view_grid_1.et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT); }, { widget: this, "trigger": trigger }), et2_dataview_view_grid_1.et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT);
@ -484,7 +486,7 @@ var et2_calendar_timegrid = /** @class */ (function (_super) {
if (_sender === this || !_sender) { if (_sender === this || !_sender) {
return this.div ? this.div[0] : null; return this.div ? this.div[0] : null;
} }
else if (_sender.instanceOf(et2_calendar_daycol)) { else if (_sender.instanceOf(et2_widget_daycol_1.et2_calendar_daycol)) {
return this.days ? this.days[0] : null; return this.days ? this.days[0] : null;
} }
else if (_sender) { else if (_sender) {
@ -504,6 +506,29 @@ var et2_calendar_timegrid = /** @class */ (function (_super) {
this.scrolling.scrollTop(this._top_time + 2); this.scrolling.scrollTop(this._top_time + 2);
} }
}; };
/**
* Update the 'now' line
* @private
*/
et2_calendar_timegrid.prototype._updateNow = function () {
var now = _super.prototype._updateNow.call(this);
if (now === false || this.options.granularity == 0) {
this.now_div.hide();
return false;
}
for (var i = 0; i < this.day_widgets.length; i++) {
var day = this.day_widgets[i];
if (day.getDate() >= now) {
day = this.day_widgets[i - 1];
this.now_div.appendTo(day.getDOMNode()).show();
var pos = day._time_to_position(now.getHours() * 60 + now.getMinutes());
//this.now_div.position({my: 'left', at: 'left', of: day.getDOMNode()});
this.now_div.css('top', pos + '%');
break;
}
}
return true;
};
/** /**
* Clear everything, and redraw the whole grid * Clear everything, and redraw the whole grid
*/ */
@ -541,7 +566,7 @@ var et2_calendar_timegrid = /** @class */ (function (_super) {
this.days.css('height', '100%'); this.days.css('height', '100%');
this.iterateOver(function (day) { this.iterateOver(function (day) {
day.resize(); day.resize();
}, this, et2_calendar_daycol); }, this, et2_widget_daycol_1.et2_calendar_daycol);
return; return;
} }
var wd_start = 60 * this.options.day_start; var wd_start = 60 * this.options.day_start;
@ -768,6 +793,8 @@ var et2_calendar_timegrid = /** @class */ (function (_super) {
} }
// Handle not fully visible elements // Handle not fully visible elements
this._scroll(); this._scroll();
// Set 'now' line
this._updateNow();
// TODO: Figure out how to do this with detached nodes // TODO: Figure out how to do this with detached nodes
/* /*
var nodes = this.day_col.getDetachedNodes(); var nodes = this.day_col.getDetachedNodes();
@ -893,7 +920,7 @@ var et2_calendar_timegrid = /** @class */ (function (_super) {
owner_match = owner_match || col.options.owner.indexOf(id) !== -1; owner_match = owner_match || col.options.owner.indexOf(id) !== -1;
own_timegrid = (col === event.getParent()); own_timegrid = (col === event.getParent());
} }
}, this, et2_calendar_daycol); }, this, et2_widget_daycol_1.et2_calendar_daycol);
} }
} }
var enabled = !owner_match && var enabled = !owner_match &&
@ -1112,7 +1139,7 @@ var et2_calendar_timegrid = /** @class */ (function (_super) {
if (col.div.has(timegrid.gridHover).length || col.header.has(timegrid.gridHover).length) { if (col.div.has(timegrid.gridHover).length || col.header.has(timegrid.gridHover).length) {
add_owner = col.options.owner; add_owner = col.options.owner;
} }
}, this, et2_calendar_daycol); }, this, et2_widget_daycol_1.et2_calendar_daycol);
} }
egw().json('calendar.calendar_uiforms.ajax_invite', [ egw().json('calendar.calendar_uiforms.ajax_invite', [
button_id === 'series' ? event_data.id : event_data.app_id, button_id === 'series' ? event_data.id : event_data.app_id,

View File

@ -18,6 +18,7 @@ import {ClassWithAttributes} from "../../api/js/etemplate/et2_core_inheritance";
import {et2_calendar_view} from "./et2_widget_view"; import {et2_calendar_view} from "./et2_widget_view";
import {et2_action_object_impl} from "../../api/js/etemplate/et2_core_DOMWidget"; import {et2_action_object_impl} from "../../api/js/etemplate/et2_core_DOMWidget";
import {et2_dataview_grid} from "../../api/js/etemplate/et2_dataview_view_grid"; import {et2_dataview_grid} from "../../api/js/etemplate/et2_dataview_view_grid";
import {et2_calendar_daycol} from "./et2_widget_daycol";
/** /**
* Class which implements the "calendar-timegrid" XET-Tag for displaying a span of days * Class which implements the "calendar-timegrid" XET-Tag for displaying a span of days
@ -618,6 +619,8 @@ export class et2_calendar_timegrid extends et2_calendar_view implements et2_IDet
{ {
this.widget.change(); this.widget.change();
} }
this.widget._updateNow();
// Hide loader // Hide loader
window.setTimeout(jQuery.proxy(function() {this.loader.hide();},this.widget),200); window.setTimeout(jQuery.proxy(function() {this.loader.hide();},this.widget),200);
},{widget:this,"trigger":trigger}),et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT); },{widget:this,"trigger":trigger}),et2_dataview_grid.ET2_GRID_INVALIDATE_TIMEOUT);
@ -692,6 +695,34 @@ export class et2_calendar_timegrid extends et2_calendar_view implements et2_IDet
} }
/**
* Update the 'now' line
* @private
*/
public _updateNow()
{
let now = super._updateNow();
if(now === false || this.options.granularity == 0)
{
this.now_div.hide();
return false;
}
for(var i = 0; i < this.day_widgets.length; i++)
{
let day = this.day_widgets[i];
if(day.getDate() >= now)
{
day = this.day_widgets[i-1];
this.now_div.appendTo(day.getDOMNode()).show();
let pos = day._time_to_position(now.getHours() * 60 + now.getMinutes());
//this.now_div.position({my: 'left', at: 'left', of: day.getDOMNode()});
this.now_div.css('top', pos + '%');
break;
}
}
return true;
}
/** /**
* Clear everything, and redraw the whole grid * Clear everything, and redraw the whole grid
*/ */
@ -1032,6 +1063,9 @@ export class et2_calendar_timegrid extends et2_calendar_view implements et2_IDet
// Handle not fully visible elements // Handle not fully visible elements
this._scroll(); this._scroll();
// Set 'now' line
this._updateNow();
// TODO: Figure out how to do this with detached nodes // TODO: Figure out how to do this with detached nodes
/* /*
var nodes = this.day_col.getDetachedNodes(); var nodes = this.day_col.getDetachedNodes();

View File

@ -45,11 +45,14 @@ var et2_calendar_view = /** @class */ (function (_super) {
_super.call(this, _parent, _attrs, et2_core_inheritance_1.ClassWithAttributes.extendAttributes(et2_calendar_view._attributes, _child || {})) || this; _super.call(this, _parent, _attrs, et2_core_inheritance_1.ClassWithAttributes.extendAttributes(et2_calendar_view._attributes, _child || {})) || this;
_this.dataStorePrefix = 'calendar'; _this.dataStorePrefix = 'calendar';
_this.update_timer = null; _this.update_timer = null;
_this.now_timer = null;
// Used for its date calculations // Used for its date calculations
_this._date_helper = et2_createWidget('date-time', {}, null); _this._date_helper = et2_createWidget('date-time', {}, null);
_this._date_helper.loadingFinished(); _this._date_helper.loadingFinished();
_this.loader = jQuery('<div class="egw-loading-prompt-container ui-front loading"></div>'); _this.loader = jQuery('<div class="egw-loading-prompt-container ui-front loading"></div>');
_this.now_div = jQuery('<div class="calendar_now"/>');
_this.update_timer = null; _this.update_timer = null;
_this.now_timer = null;
// Used to support dragging on empty space to create an event // Used to support dragging on empty space to create an event
_this.drag_create = { _this.drag_create = {
start: null, start: null,
@ -68,12 +71,19 @@ var et2_calendar_view = /** @class */ (function (_super) {
if (this.update_timer) { if (this.update_timer) {
window.clearTimeout(this.update_timer); window.clearTimeout(this.update_timer);
} }
// Stop the 'now' line
if (this.now_timer) {
window.clearInterval(this.now_timer);
}
}; };
et2_calendar_view.prototype.doLoadingFinished = function () { et2_calendar_view.prototype.doLoadingFinished = function () {
_super.prototype.doLoadingFinished.call(this); _super.prototype.doLoadingFinished.call(this);
this.loader.hide(0).prependTo(this.div); this.loader.hide(0).prependTo(this.div);
this.div.append(this.now_div);
if (this.options.owner) if (this.options.owner)
this.set_owner(this.options.owner); this.set_owner(this.options.owner);
// Start moving 'now' line
this.now_timer = window.setInterval(this._updateNow.bind(this), 60000);
return true; return true;
}; };
/** /**
@ -274,6 +284,25 @@ var et2_calendar_view = /** @class */ (function (_super) {
et2_calendar_view.prototype._createNamespace = function () { et2_calendar_view.prototype._createNamespace = function () {
return true; return true;
}; };
/**
* Update the 'now' line
*
* Here we just do some limit checks and return the current date/time.
* Extending widgets should handle position.
*
* @private
*/
et2_calendar_view.prototype._updateNow = function () {
var now = new Date();
// Use date widget's existing functions to deal
this._date_helper.set_value(now);
now = new Date(this._date_helper.getValue());
if (this.get_start_date() <= now && this.get_end_date() >= now) {
return now;
}
this.now_div.hide();
return false;
};
/** /**
* Calendar supports many different owner types, including users & resources. * Calendar supports many different owner types, including users & resources.
* This translates an ID to a user-friendly name. * This translates an ID to a user-friendly name.

View File

@ -16,6 +16,7 @@ import {et2_widget, WidgetConfig} from "../../api/js/etemplate/et2_core_widget";
import {et2_valueWidget} from "../../api/js/etemplate/et2_core_valueWidget"; import {et2_valueWidget} from "../../api/js/etemplate/et2_core_valueWidget";
import {ClassWithAttributes} from "../../api/js/etemplate/et2_core_inheritance"; import {ClassWithAttributes} from "../../api/js/etemplate/et2_core_inheritance";
import {et2_date} from "../../api/js/etemplate/et2_widget_date"; import {et2_date} from "../../api/js/etemplate/et2_widget_date";
import {et2_calendar_event} from "./et2_widget_event";
/** /**
* Parent class for the various calendar views to reduce copied code * Parent class for the various calendar views to reduce copied code
@ -48,8 +49,10 @@ export class et2_calendar_view extends et2_valueWidget
protected _date_helper: et2_date; protected _date_helper: et2_date;
protected loader: JQuery; protected loader: JQuery;
protected div: JQuery; protected div: JQuery;
protected now_div: JQuery;
protected update_timer: number = null; protected update_timer: number = null;
protected now_timer: number = null;
protected drag_create: { parent: et2_widget; start: any; end: any; event: et2_calendar_event }; protected drag_create: { parent: et2_widget; start: any; end: any; event: et2_calendar_event };
protected value: any; protected value: any;
@ -70,7 +73,9 @@ export class et2_calendar_view extends et2_valueWidget
this._date_helper.loadingFinished(); this._date_helper.loadingFinished();
this.loader = jQuery('<div class="egw-loading-prompt-container ui-front loading"></div>'); this.loader = jQuery('<div class="egw-loading-prompt-container ui-front loading"></div>');
this.now_div = jQuery('<div class="calendar_now"/>');
this.update_timer = null; this.update_timer = null;
this.now_timer = null;
// Used to support dragging on empty space to create an event // Used to support dragging on empty space to create an event
this.drag_create = { this.drag_create = {
@ -93,14 +98,23 @@ export class et2_calendar_view extends et2_valueWidget
{ {
window.clearTimeout(this.update_timer); window.clearTimeout(this.update_timer);
} }
// Stop the 'now' line
if(this.now_timer)
{
window.clearInterval(this.now_timer);
}
} }
doLoadingFinished( ) doLoadingFinished( )
{ {
super.doLoadingFinished(); super.doLoadingFinished();
this.loader.hide(0).prependTo(this.div); this.loader.hide(0).prependTo(this.div);
this.div.append(this.now_div);
if(this.options.owner) this.set_owner(this.options.owner); if(this.options.owner) this.set_owner(this.options.owner);
// Start moving 'now' line
this.now_timer = window.setInterval(this._updateNow.bind(this), 60000);
return true; return true;
} }
@ -341,6 +355,30 @@ export class et2_calendar_view extends et2_valueWidget
return true; return true;
} }
/**
* Update the 'now' line
*
* Here we just do some limit checks and return the current date/time.
* Extending widgets should handle position.
*
* @private
*/
public _updateNow()
{
let now = new Date();
// Use date widget's existing functions to deal
this._date_helper.set_value(now);
now = new Date(this._date_helper.getValue());
if(this.get_start_date() <= now && this.get_end_date() >= now)
{
return now;
}
this.now_div.hide();
return false;
}
/** /**
* Calendar supports many different owner types, including users & resources. * Calendar supports many different owner types, including users & resources.
* This translates an ID to a user-friendly name. * This translates an ID to a user-friendly name.

View File

@ -241,6 +241,26 @@ e.g. the div with class calendar_calTimeGrid is generated by the timeGridWidget
box-shadow: 0 0 5px; box-shadow: 0 0 5px;
} }
/* Current time indicator */
.calendar_now {
position: absolute;
background: red;
}
.calendar_calTimeGrid .calendar_now {
height: 2px;
width: 100%;
left: 0px
}
.calendar_calTimeGridList .calendar_now {
display: none;
}
.calendar_plannerWidget .calendar_now {
height: 100%;
width: 2px;
top: 0px;
}
.calendar_size120b { font-size: 120%; font-weight: bold; } .calendar_size120b { font-size: 120%; font-weight: bold; }
/* marks a day in the colum-header as today /* marks a day in the colum-header as today

View File

@ -245,6 +245,24 @@ e.g. the div with class calendar_calTimeGrid is generated by the timeGridWidget
-moz-box-shadow: 0 0 5px; -moz-box-shadow: 0 0 5px;
box-shadow: 0 0 5px; box-shadow: 0 0 5px;
} }
/* Current time indicator */
.calendar_now {
position: absolute;
background: red;
}
.calendar_calTimeGrid .calendar_now {
height: 2px;
width: 100%;
left: 0px;
}
.calendar_calTimeGridList .calendar_now {
display: none;
}
.calendar_plannerWidget .calendar_now {
height: 100%;
width: 2px;
top: 0px;
}
.calendar_size120b { .calendar_size120b {
font-size: 120%; font-size: 120%;
font-weight: bold; font-weight: bold;