Finish alarm feature: display notifications on keep-alive request and allow to either dismiss or snooze them

This commit is contained in:
Thomas 2011-05-25 23:16:13 +02:00
parent 1ccb8f198a
commit 59511fd581
9 changed files with 213 additions and 77 deletions

View file

@ -39,10 +39,15 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
// Roundcube calendar client class // Roundcube calendar client class
function rcube_calendar(settings) function rcube_calendar(settings)
{ {
// member vars
this.settings = settings; this.settings = settings;
var me = this; this.alarm_ids = [];
this.alarm_dialog = null;
this.snooze_popup = null;
this.dismiss_link = null
// private vars // private vars
var me = this;
var day_clicked = day_clicked_ts = 0; var day_clicked = day_clicked_ts = 0;
var ignore_click = false; var ignore_click = false;
@ -116,8 +121,7 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
resizable: true, resizable: true,
title: null, title: null,
close: function() { close: function() {
$dialog.dialog('destroy'); $dialog.dialog('destroy').hide();
$dialog.hide();
}, },
buttons: buttons, buttons: buttons,
minWidth: 320, minWidth: 320,
@ -178,15 +182,16 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
for (var alarm, i=0; i < event.alarms.length; i++) { for (var alarm, i=0; i < event.alarms.length; i++) {
alarm = String(event.alarms[i]).split(':'); alarm = String(event.alarms[i]).split(':');
$('select.edit-alarm-type').val(alarm[0]); if (!alarm[1] && alarm[0]) alarm[1] = 'DISPLAY';
$('select.edit-alarm-type').val(alarm[1]);
if (alarm[1].match(/@(\d+)/)) { if (alarm[0].match(/@(\d+)/)) {
var ondate = new Date(parseInt(RegExp.$1) * 1000); var ondate = new Date(parseInt(RegExp.$1) * 1000);
$('select.edit-alarm-offset').val('@'); $('select.edit-alarm-offset').val('@');
$('input.edit-alarm-date').val($.fullCalendar.formatDate(ondate, settings['date_format'])); $('input.edit-alarm-date').val($.fullCalendar.formatDate(ondate, settings['date_format']));
$('input.edit-alarm-time').val($.fullCalendar.formatDate(ondate, settings['time_format'])); $('input.edit-alarm-time').val($.fullCalendar.formatDate(ondate, settings['time_format']));
} }
else if (alarm[1].match(/([-+])(\d+)([MHD])/)) { else if (alarm[0].match(/([-+])(\d+)([MHD])/)) {
$('input.edit-alarm-value').val(RegExp.$2); $('input.edit-alarm-value').val(RegExp.$2);
$('select.edit-alarm-offset').val(''+RegExp.$1+RegExp.$3); $('select.edit-alarm-offset').val(''+RegExp.$1+RegExp.$3);
} }
@ -252,7 +257,6 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
// post data to server // post data to server
var data = { var data = {
action: action,
start: start.getTime()/1000, start: start.getTime()/1000,
end: end.getTime()/1000, end: end.getTime()/1000,
allday: allday.checked?1:0, allday: allday.checked?1:0,
@ -272,9 +276,9 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
if (alarm) { if (alarm) {
var val, offset = $('select.edit-alarm-offset').val(); var val, offset = $('select.edit-alarm-offset').val();
if (offset == '@') if (offset == '@')
data.alarms = alarm + ':@' + (me.parse_datetime($('input.edit-alarm-time').val(), $('input.edit-alarm-date').val()).getTime()/1000); data.alarms = '@' + (me.parse_datetime($('input.edit-alarm-time').val(), $('input.edit-alarm-date').val()).getTime()/1000) + ':' + alarm;
else if ((val = parseInt($('input.edit-alarm-value').val())) && !isNaN(val) && val >= 0) else if ((val = parseInt($('input.edit-alarm-value').val())) && !isNaN(val) && val >= 0)
data.alarms = alarm + ':' + offset[0] + val + offset[1]; data.alarms = offset[0] + val + offset[1] + ':' + alarm;
} }
// gather recurrence settings // gather recurrence settings
@ -319,7 +323,7 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
else else
data.calendar = calendars.val(); data.calendar = calendars.val();
rcmail.http_post('plugin.event', { e:data }); rcmail.http_post('plugin.event', { action:action, e:data });
$dialog.dialog("close"); $dialog.dialog("close");
}; };
@ -347,8 +351,7 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
resizable: true, resizable: true,
title: rcmail.gettext((action == 'edit' ? 'edit_event' : 'new_event'), 'calendar'), title: rcmail.gettext((action == 'edit' ? 'edit_event' : 'new_event'), 'calendar'),
close: function() { close: function() {
$dialog.dialog("destroy"); $dialog.dialog("destroy").hide();
$dialog.hide();
}, },
buttons: buttons, buttons: buttons,
minWidth: 440, minWidth: 440,
@ -416,7 +419,7 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
this.delete_event = function(event) { this.delete_event = function(event) {
// send remove request to plugin // send remove request to plugin
if (confirm(rcmail.gettext('deleteventconfirm', 'calendar'))) { if (confirm(rcmail.gettext('deleteventconfirm', 'calendar'))) {
rcmail.http_post('plugin.event', { e:{ action:'remove', id:event.id } }); rcmail.http_post('plugin.event', { action:'remove', e:{ id:event.id } });
return true; return true;
} }
@ -426,42 +429,103 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
// display a notification for the given pending alarms // display a notification for the given pending alarms
this.display_alarms = function(alarms) { this.display_alarms = function(alarms) {
// clear old alert first // clear old alert first
$('#alarm-display').dialog('destroy'); if (this.alarm_dialog)
this.alarm_dialog.dialog('destroy');
var event_ids = []; this.alarm_dialog = $('<div>').attr('id', 'alarm-display');
var display = $('<div>').attr('id', 'alarm-display');
for (var html, alarm, i=0; i < alarms.length; i++) { var actions, adismiss, asnooze, alarm, html, event_ids = [];
for (var actions, html, alarm, i=0; i < alarms.length; i++) {
alarm = alarms[i]; alarm = alarms[i];
alarm.start = new Date(alarm.start); alarm.start = new Date(alarm.start * 1000);
alarm.end = new Date(alarm.end); alarm.end = new Date(alarm.end * 1000);
event_ids.push(alarm.id); event_ids.push(alarm.id);
html = '<h3 class="event-title">' + Q(alarm.title) + '</h3>'; html = '<h3 class="event-title">' + Q(alarm.title) + '</h3>';
html += '<div class="event-section">' + Q(alarm.location) + '</div>'; html += '<div class="event-section">' + Q(alarm.location) + '</div>';
html += '<div class="event-section">' + Q(event_date_text(alarm)) + '</div>'; html += '<div class="event-section">' + Q(event_date_text(alarm)) + '</div>';
$('<div>').addClass('alarm-item').html(html).appendTo(display);
adismiss = $('<a href="#" class="alarm-action-dismiss"></a>').html(rcmail.gettext('dismiss','calendar')).click(function(){
me.dismiss_link = $(this);
me.dismiss_alarm(me.dismiss_link.data('id'), 0);
});
asnooze = $('<a href="#" class="alarm-action-snooze"></a>').html(rcmail.gettext('snooze','calendar')).click(function(){
me.snooze_dropdown($(this));
});
actions = $('<div>').addClass('alarm-actions').append(adismiss.data('id', alarm.id)).append(asnooze.data('id', alarm.id));
$('<div>').addClass('alarm-item').html(html).append(actions).appendTo(this.alarm_dialog);
} }
display.appendTo(document.body).dialog({ var buttons = {};
modal: true, buttons[rcmail.gettext('dismissall','calendar')] = function() {
// submit dismissed event_ids to server
me.dismiss_alarm(me.alarm_ids.join(','), 0);
$(this).dialog('close');
};
this.alarm_dialog.appendTo(document.body).dialog({
modal: false,
resizable: true, resizable: true,
closeOnEscape: false, closeOnEscape: false,
dialogClass: 'alert', dialogClass: 'alarm',
title: rcmail.gettext('alarmtitle', 'calendar'), title: rcmail.gettext('alarmtitle', 'calendar'),
buttons: { buttons: buttons,
"Snooze": function() {
$(this).dialog('close');
},
"Dismiss": function() {
$(this).dialog('close');
}
},
close: function() { close: function() {
$(this).dialog('destroy'); $('#alarm-snooze-dropdown').hide();
// TODO: submit dismissed event_ids to server $(this).dialog('destroy').remove();
$(this).remove(); me.alarm_dialog = null;
me.alarm_ids = null;
},
drag: function(event, ui) {
$('#alarm-snooze-dropdown').hide();
} }
}).data('event_ids', event_ids.join(',')); });
this.alarm_ids = event_ids;
};
// show a drop-down menu with a selection of snooze times
this.snooze_dropdown = function(link)
{
if (!this.snooze_popup) {
this.snooze_popup = $('#alarm-snooze-dropdown');
$('#alarm-snooze-dropdown a').click(function(e){
var time = String(this.href).replace(/.+#/, '');
me.dismiss_alarm($('#alarm-snooze-dropdown').data('id'), time);
return false;
});
}
// hide visible popup
if (this.snooze_popup.is(':visible') && this.snooze_popup.data('id') == link.data('id')) {
this.snooze_popup.hide();
this.dismiss_link = null;
}
else { // open popup below the clicked link
var pos = link.offset();
pos.top += link.height() + 2;
this.snooze_popup.data('id', link.data('id')).css({ top:Math.floor(pos.top)+'px', left:Math.floor(pos.left)+'px' }).show();
this.dismiss_link = link;
}
};
// dismiss or snooze alarms for the given event
this.dismiss_alarm = function(id, snooze)
{
$('#alarm-snooze-dropdown').hide();
rcmail.http_post('plugin.event', { action:'dismiss', e:{ id:id, snooze:snooze } });
// remove dismissed alarm from list
if (this.dismiss_link) {
this.dismiss_link.closest('div.alarm-item').hide();
var new_ids = jQuery.grep(this.alarm_ids, function(v){ return v != id; });
if (new_ids.length)
this.alarm_ids = new_ids;
else
this.alarm_dialog.dialog('close');
}
this.dismiss_link = null;
}; };
@ -588,24 +652,22 @@ window.rcmail && rcmail.addEventListener('init', function(evt) {
} }
// send move request to server // send move request to server
var data = { var data = {
action: 'move',
id: event.id, id: event.id,
start: event.start.getTime()/1000, start: event.start.getTime()/1000,
end: event.end.getTime()/1000, end: event.end.getTime()/1000,
allday: allDay?1:0 allday: allDay?1:0
}; };
rcmail.http_post('plugin.event', { e:data }); rcmail.http_post('plugin.event', { action:'move', e:data });
}, },
// callback for event resizing // callback for event resizing
eventResize : function(event, delta) { eventResize : function(event, delta) {
// send resize request to server // send resize request to server
var data = { var data = {
action: 'resize',
id: event.id, id: event.id,
start: event.start.getTime()/1000, start: event.start.getTime()/1000,
end: event.end.getTime()/1000, end: event.end.getTime()/1000,
}; };
rcmail.http_post('plugin.event', { e:data }); rcmail.http_post('plugin.event', { action:'resize', e:data });
} }
}); });

View file

@ -111,6 +111,9 @@ class calendar extends rcube_plugin
} }
} }
/**
* Render the main calendar view from skin template
*/
function calendar_view() function calendar_view()
{ {
$this->rc->output->set_pagetitle($this->gettext('calendar')); $this->rc->output->set_pagetitle($this->gettext('calendar'));
@ -128,6 +131,7 @@ class calendar extends rcube_plugin
$this->register_handler('plugin.freebusy_select', array($this->ui, 'freebusy_select')); $this->register_handler('plugin.freebusy_select', array($this->ui, 'freebusy_select'));
$this->register_handler('plugin.priority_select', array($this->ui, 'priority_select')); $this->register_handler('plugin.priority_select', array($this->ui, 'priority_select'));
$this->register_handler('plugin.alarm_select', array($this->ui, 'alarm_select')); $this->register_handler('plugin.alarm_select', array($this->ui, 'alarm_select'));
$this->register_handler('plugin.snooze_select', array($this->ui, 'snooze_select'));
$this->register_handler('plugin.recurrence_form', array($this->ui, 'recurrence_form')); $this->register_handler('plugin.recurrence_form', array($this->ui, 'recurrence_form'));
$this->rc->output->set_env('calendar_settings', $this->load_settings()); $this->rc->output->set_env('calendar_settings', $this->load_settings());
@ -321,39 +325,56 @@ class calendar extends rcube_plugin
return $p; return $p;
} }
/**
* Dispatcher for event actions initiated by the client
*/
function event() function event()
{ {
$action = get_input_value('action', RCUBE_INPUT_POST);
$event = get_input_value('e', RCUBE_INPUT_POST); $event = get_input_value('e', RCUBE_INPUT_POST);
$success = false; $success = $reload = false;
switch ($event['action']) { switch ($action) {
case "new": case "new":
// create UID for new event // create UID for new event
$events['uid'] = strtoupper(md5(time() . uniqid(rand())) . '-' . substr(md5($this->rc->user->get_username()), 0, 16)); $events['uid'] = strtoupper(md5(time() . uniqid(rand())) . '-' . substr(md5($this->rc->user->get_username()), 0, 16));
$success = $this->driver->new_event($event); $success = $this->driver->new_event($event);
break; $reload = true;
break;
case "edit": case "edit":
$success = $this->driver->edit_event($event); $success = $this->driver->edit_event($event);
break; $reload = true;
break;
case "resize": case "resize":
$success = $this->driver->resize_event($event); $success = $this->driver->resize_event($event);
break; $reload = true;
break;
case "move": case "move":
$success = $this->driver->move_event($event); $success = $this->driver->move_event($event);
break; $reload = true;
break;
case "remove": case "remove":
$success = $this->driver->remove_event($event); $success = $this->driver->remove_event($event);
break; $reload = true;
break;
case "dismiss":
foreach (explode(',', $event['id']) as $id)
$success |= $this->driver->dismiss_alarm($id, $event['snooze']);
break;
} }
if ($success) { if (!$success) {
$this->rc->output->command('plugin.reload_calendar', array());
}
else {
$this->rc->output->show_message('calendar.errorsaving', 'error'); $this->rc->output->show_message('calendar.errorsaving', 'error');
} }
else if ($reload) {
$this->rc->output->command('plugin.reload_calendar', array());
}
} }
/**
* Handler for load-requests from fullcalendar
* This will return pure JSON formatted output
*/
function load_events() function load_events()
{ {
$events = $this->driver->load_events(get_input_value('start', RCUBE_INPUT_GET), get_input_value('end', RCUBE_INPUT_GET), get_input_value('source', RCUBE_INPUT_GET)); $events = $this->driver->load_events(get_input_value('start', RCUBE_INPUT_GET), get_input_value('end', RCUBE_INPUT_GET), get_input_value('source', RCUBE_INPUT_GET));
@ -368,8 +389,8 @@ class calendar extends rcube_plugin
function keep_alive($attr) function keep_alive($attr)
{ {
$alarms = $this->driver->pending_alarms(time()); $alarms = $this->driver->pending_alarms(time());
#if ($alarms) if ($alarms)
# $this->rc->output->command('plugin.display_alarms', $this->_alarms_output($alarms)); $this->rc->output->command('plugin.display_alarms', $this->_alarms_output($alarms));
} }
/** /**
@ -523,7 +544,7 @@ class calendar extends rcube_plugin
*/ */
private function _alarms_text($alarm) private function _alarms_text($alarm)
{ {
list($action, $trigger) = explode(':', $alarm); list($trigger, $action) = explode(':', $alarm);
$text = ''; $text = '';
switch ($action) { switch ($action) {

View file

@ -28,6 +28,7 @@ abstract class calendar_driver
public $alarms = false; public $alarms = false;
public $attendees = false; public $attendees = false;
public $attachments = false; public $attachments = false;
public $alarm_types = array('DISPLAY');
/** /**
* Get a list of available calendars from this source * Get a list of available calendars from this source
@ -150,7 +151,7 @@ abstract class calendar_driver
* @param string Event identifier * @param string Event identifier
* @param integer Suspend the alarm for this number of seconds * @param integer Suspend the alarm for this number of seconds
*/ */
abstract function confirm_alarm($event_id, $snooze = 0); abstract function dismiss_alarm($event_id, $snooze = 0);
/** /**
* Save an attachment related to the given event * Save an attachment related to the given event

View file

@ -29,6 +29,7 @@ class database_driver extends calendar_driver
public $alarms = true; public $alarms = true;
public $attendees = true; public $attendees = true;
public $attachments = true; public $attachments = true;
public $alarm_types = array('DISPLAY','EMAIL');
private $rc; private $rc;
private $cal; private $cal;
@ -229,7 +230,7 @@ class database_driver extends calendar_driver
// compute absolute time to notify the user // compute absolute time to notify the user
if ($event['alarms']) { if ($event['alarms']) {
list($action, $trigger) = explode(':', $event['alarms']); list($trigger, $action) = explode(':', $event['alarms']);
$notify = calendar::parse_alaram_value($trigger); $notify = calendar::parse_alaram_value($trigger);
if (!empty($notify[1])){ // offset if (!empty($notify[1])){ // offset
$mult = 1; $mult = 1;
@ -249,7 +250,8 @@ class database_driver extends calendar_driver
$notify_at = $notify[0]; $notify_at = $notify[0];
} }
$event['notifyat'] = date('Y-m-d H:i:s', $notify_at); if ($notify_at > time())
$event['notifyat'] = date('Y-m-d H:i:s', $notify_at);
} }
else else
$event['notifyat'] = null; $event['notifyat'] = null;
@ -466,17 +468,12 @@ class database_driver extends calendar_driver
/** /**
* Feedback after showing/sending an alarm notification * Feedback after showing/sending an alarm notification
* *
* @see Driver:confirm_alarm() * @see Driver:dismiss_alarm()
*/ */
public function confirm_alarm($event_id, $snooze = 0) public function dismiss_alarm($event_id, $snooze = 0)
{ {
// set new notifyat time // set new notifyat time or unset if not snoozed
if ($snooze > 0) { $notify_at = $snooze > 0 ? date('Y-m-d H:i:s', time() + $snooze) : null;
$event = $this->get_event($event_id);
$notify_at = date('Y-m-d H:i:s', strtotime($event['notifyat']) + $snooze);
}
else // unset notifyat value
$notify_at = null;
$query = $this->rc->db->query(sprintf( $query = $this->rc->db->query(sprintf(
"UPDATE " . $this->db_events . " "UPDATE " . $this->db_events . "
@ -487,6 +484,7 @@ class database_driver extends calendar_driver
$notify_at, $notify_at,
$event_id $event_id
); );
return $this->rc->db->affected_rows($query); return $this->rc->db->affected_rows($query);
} }

View file

@ -236,9 +236,9 @@ class kolab_driver extends calendar_driver
/** /**
* Feedback after showing/sending an alarm notification * Feedback after showing/sending an alarm notification
* *
* @see Driver:confirm_alarm() * @see Driver:dismiss_alarm()
*/ */
public function confirm_alarm($event_id, $snooze = 0) public function dismiss_alarm($event_id, $snooze = 0)
{ {
// TBD. // TBD.
return false; return false;

View file

@ -209,9 +209,9 @@ class calendar_ui
{ {
unset($attrib['name']); unset($attrib['name']);
$select_type = new html_select(array('name' => 'alarmtype[]', 'class' => 'edit-alarm-type')); $select_type = new html_select(array('name' => 'alarmtype[]', 'class' => 'edit-alarm-type'));
$select_type->add( $select_type->add($this->calendar->gettext('none'), '');
array($this->calendar->gettext('none'), $this->calendar->gettext('alarmdisplayoption'), $this->calendar->gettext('alarmemailoption')), foreach ($this->calendar->driver->alarm_types as $type)
array('','DISPLAY','EMAIL')); $select_type->add($this->calendar->gettext(strtolower("alarm{$type}option")), $type);
$input_value = new html_inputfield(array('name' => 'alarmvalue[]', 'class' => 'edit-alarm-value', 'size' => 3)); $input_value = new html_inputfield(array('name' => 'alarmvalue[]', 'class' => 'edit-alarm-value', 'size' => 3));
$input_date = new html_inputfield(array('name' => 'alarmdate[]', 'class' => 'edit-alarm-date', 'size' => 10)); $input_date = new html_inputfield(array('name' => 'alarmdate[]', 'class' => 'edit-alarm-date', 'size' => 10));
@ -241,6 +241,29 @@ class calendar_ui
return $html; return $html;
} }
function snooze_select($attrib = array())
{
$steps = array(
5 => 'repeatinmin',
10 => 'repeatinmin',
15 => 'repeatinmin',
20 => 'repeatinmin',
30 => 'repeatinmin',
60 => 'repeatinhr',
120 => 'repeatinhrs',
1440 => 'repeattomorrow',
10080 => 'repeatinweek',
);
$items = array();
foreach ($steps as $n => $label) {
$items[] = html::tag('li', null, html::a(array('href' => "#" . ($n * 60), 'class' => 'active'),
$this->calendar->gettext(array('name' => $label, 'vars' => array('min' => $n % 60, 'hrs' => intval($n / 60))))));
}
return html::tag('ul', $attrib, join("\n", $items));
}
/** /**
* Generate the form for recurrence settings * Generate the form for recurrence settings
*/ */

View file

@ -65,6 +65,14 @@ $labels['trigger+D'] = 'days after';
$labels['addalarm'] = 'add alarm'; $labels['addalarm'] = 'add alarm';
$labels['defaultalarmtype'] = 'Default reminder setting'; $labels['defaultalarmtype'] = 'Default reminder setting';
$labels['defaultalarmoffset'] = 'Default reminder time'; $labels['defaultalarmoffset'] = 'Default reminder time';
$labels['dismissall'] = 'Dismiss all';
$labels['dismiss'] = 'Dismiss';
$labels['snooze'] = 'Snooze';
$labels['repeatinmin'] = 'Repeat in $min minutes';
$labels['repeatinhr'] = 'Repeat in 1 hour';
$labels['repeatinhrs'] = 'Repeat in $hrs hours';
$labels['repeattomorrow'] = 'Repeat tomorrow';
$labels['repeatinweek'] = 'Repeat in a week';
$labels['alarmtitle'] = 'Upcoming events'; $labels['alarmtitle'] = 'Upcoming events';
// event dialog tabs // event dialog tabs

View file

@ -355,6 +355,26 @@ a.dropdown-link:after {
margin-bottom: 0.3em; margin-bottom: 0.3em;
} }
.alarm-item .alarm-actions {
margin-top: 0.4em;
}
.alarm-item div.alarm-actions a {
color: #CC0000;
margin-right: 0.8em;
text-decoration: none;
}
a.alarm-action-snooze:after {
content: ' ▼';
font-size: 10px;
color: #666;
}
#alarm-snooze-dropdown {
z-index: 5000;
}
.ui-dialog-buttonset a.dropdown-link { .ui-dialog-buttonset a.dropdown-link {
margin-right: 1em; margin-right: 1em;
} }

View file

@ -153,6 +153,9 @@
</div> </div>
</form> </form>
</div> </div>
<div id="alarm-snooze-dropdown" class="popupmenu">
<roundcube:object name="plugin.snooze_select" type="ul" />
</div>
<div id="calendartoolbar"> <div id="calendartoolbar">
<roundcube:button command="plugin.addevent" type="link" class="buttonPas addevent" classAct="button addevent" classSel="button addeventSel" title="calendar.new_event" content=" " /> <roundcube:button command="plugin.addevent" type="link" class="buttonPas addevent" classAct="button addevent" classSel="button addeventSel" title="calendar.new_event" content=" " />
<roundcube:button command="plugin.print" type="link" class="buttonPas print" classAct="button print" classSel="button printSel" title="calendar.print" content=" " /> <roundcube:button command="plugin.print" type="link" class="buttonPas print" classAct="button print" classSel="button printSel" title="calendar.print" content=" " />