2009-09-30 10:02:08 -04:00
|
|
|
/* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */
|
|
|
|
|
2009-10-01 15:18:20 -04:00
|
|
|
const Clutter = imports.gi.Clutter;
|
2010-05-05 17:05:42 -04:00
|
|
|
const Gio = imports.gi.Gio;
|
2009-09-30 10:02:08 -04:00
|
|
|
const Lang = imports.lang;
|
|
|
|
const St = imports.gi.St;
|
2011-01-28 16:35:46 -05:00
|
|
|
const Signals = imports.signals;
|
2010-05-27 03:31:46 -04:00
|
|
|
const Pango = imports.gi.Pango;
|
2010-11-29 16:41:05 -05:00
|
|
|
const Gettext_gtk30 = imports.gettext.domain('gtk30');
|
2011-01-28 16:35:46 -05:00
|
|
|
const Gettext = imports.gettext.domain('gnome-shell');
|
|
|
|
const _ = Gettext.gettext;
|
|
|
|
const Mainloop = imports.mainloop;
|
|
|
|
const Shell = imports.gi.Shell;
|
2009-09-30 10:02:08 -04:00
|
|
|
|
|
|
|
const MSECS_IN_DAY = 24 * 60 * 60 * 1000;
|
2010-05-27 03:31:46 -04:00
|
|
|
const WEEKDATE_HEADER_WIDTH_DIGITS = 3;
|
2010-05-05 17:05:42 -04:00
|
|
|
const SHOW_WEEKDATE_KEY = 'show-weekdate';
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
// in org.gnome.desktop.interface
|
|
|
|
const CLOCK_FORMAT_KEY = 'clock-format';
|
|
|
|
|
2009-09-30 10:02:08 -04:00
|
|
|
function _sameDay(dateA, dateB) {
|
|
|
|
return (dateA.getDate() == dateB.getDate() &&
|
|
|
|
dateA.getMonth() == dateB.getMonth() &&
|
|
|
|
dateA.getYear() == dateB.getYear());
|
|
|
|
}
|
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
function _sameYear(dateA, dateB) {
|
|
|
|
return (dateA.getYear() == dateB.getYear());
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: maybe needs config - right now we assume that Saturday and
|
|
|
|
* Sunday are non-work days (not true in e.g. Israel, it's Sunday and
|
|
|
|
* Monday there)
|
|
|
|
*/
|
|
|
|
function _isWorkDay(date) {
|
|
|
|
return date.getDay() != 0 && date.getDay() != 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _getBeginningOfDay(date) {
|
|
|
|
let ret = new Date(date.getTime());
|
|
|
|
ret.setHours(0);
|
|
|
|
ret.setMinutes(0);
|
|
|
|
ret.setSeconds(0);
|
|
|
|
ret.setMilliseconds(0);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _getEndOfDay(date) {
|
|
|
|
let ret = new Date(date.getTime());
|
|
|
|
ret.setHours(23);
|
|
|
|
ret.setMinutes(59);
|
|
|
|
ret.setSeconds(59);
|
|
|
|
ret.setMilliseconds(999);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _formatEventTime(event, clockFormat) {
|
|
|
|
let ret;
|
|
|
|
if (event.allDay) {
|
|
|
|
/* Translators: Shown in calendar event list for all day events */
|
|
|
|
ret = _("All Day");
|
|
|
|
} else {
|
|
|
|
switch (clockFormat) {
|
|
|
|
case '24h':
|
|
|
|
ret = event.date.toLocaleFormat('%H:%M');
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* explicit fall-through */
|
|
|
|
case '12h':
|
|
|
|
ret = event.date.toLocaleFormat('%l:%M %p');
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-05-27 03:31:46 -04:00
|
|
|
function _getCalendarWeekForDate(date) {
|
2010-05-30 17:17:40 -04:00
|
|
|
// Based on the algorithms found here:
|
|
|
|
// http://en.wikipedia.org/wiki/Talk:ISO_week_date
|
|
|
|
let midnightDate = new Date(date.getFullYear(), date.getMonth(), date.getDate());
|
|
|
|
// Need to get Monday to be 1 ... Sunday to be 7
|
|
|
|
let dayOfWeek = 1 + ((midnightDate.getDay() + 6) % 7);
|
|
|
|
let nearestThursday = new Date(midnightDate.getFullYear(), midnightDate.getMonth(),
|
|
|
|
midnightDate.getDate() + (4 - dayOfWeek));
|
|
|
|
|
|
|
|
let jan1st = new Date(nearestThursday.getFullYear(), 0, 1);
|
|
|
|
let diffDate = nearestThursday - jan1st;
|
|
|
|
let dayNumber = Math.floor(Math.abs(diffDate) / MSECS_IN_DAY);
|
|
|
|
let weekNumber = Math.floor(dayNumber / 7) + 1;
|
|
|
|
|
|
|
|
return weekNumber;
|
2010-05-27 03:31:46 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function _getDigitWidth(actor){
|
|
|
|
let context = actor.get_pango_context();
|
|
|
|
let themeNode = actor.get_theme_node();
|
|
|
|
let font = themeNode.get_font();
|
|
|
|
let metrics = context.get_metrics(font, context.get_language());
|
|
|
|
let width = metrics.get_approximate_digit_width();
|
|
|
|
return width;
|
|
|
|
}
|
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
function _getCalendarDayAbbreviation(dayNumber) {
|
|
|
|
let abbreviations = [
|
|
|
|
/* Translators: Calendar grid abbreviation for Sunday.
|
|
|
|
*
|
|
|
|
* NOTE: These abbreviations are always shown together and in
|
|
|
|
* order, e.g. "S M T W T F S".
|
|
|
|
*/
|
|
|
|
_("S"),
|
|
|
|
/* Translators: Calendar grid abbreviation for Monday */
|
|
|
|
_("M"),
|
|
|
|
/* Translators: Calendar grid abbreviation for Tuesday */
|
|
|
|
_("T"),
|
|
|
|
/* Translators: Calendar grid abbreviation for Wednesday */
|
|
|
|
_("W"),
|
|
|
|
/* Translators: Calendar grid abbreviation for Thursday */
|
|
|
|
_("T"),
|
|
|
|
/* Translators: Calendar grid abbreviation for Friday */
|
|
|
|
_("F"),
|
|
|
|
/* Translators: Calendar grid abbreviation for Saturday */
|
|
|
|
_("S")
|
|
|
|
];
|
|
|
|
return abbreviations[dayNumber];
|
|
|
|
}
|
|
|
|
|
|
|
|
function _getEventDayAbbreviation(dayNumber) {
|
|
|
|
let abbreviations = [
|
|
|
|
/* Translators: Event list abbreviation for Sunday.
|
|
|
|
*
|
|
|
|
* NOTE: These abbreviations are normally not shown together
|
|
|
|
* so they need to be unique (e.g. Tuesday and Thursday cannot
|
|
|
|
* both be 'T').
|
|
|
|
*/
|
|
|
|
_("Su"),
|
|
|
|
/* Translators: Event list abbreviation for Monday */
|
|
|
|
_("M"),
|
|
|
|
/* Translators: Event list abbreviation for Tuesday */
|
|
|
|
_("T"),
|
|
|
|
/* Translators: Event list abbreviation for Wednesday */
|
|
|
|
_("W"),
|
|
|
|
/* Translators: Event list abbreviation for Thursday */
|
|
|
|
_("Th"),
|
|
|
|
/* Translators: Event list abbreviation for Friday */
|
|
|
|
_("F"),
|
|
|
|
/* Translators: Event list abbreviation for Saturday */
|
|
|
|
_("S")
|
|
|
|
];
|
|
|
|
return abbreviations[dayNumber];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Abstraction for an appointment/event in a calendar
|
|
|
|
|
|
|
|
function CalendarEvent(date, summary, allDay) {
|
|
|
|
this._init(date, summary, allDay);
|
|
|
|
}
|
|
|
|
|
|
|
|
CalendarEvent.prototype = {
|
|
|
|
_init: function(date, summary, allDay) {
|
|
|
|
this.date = date;
|
|
|
|
this.summary = summary;
|
|
|
|
this.allDay = allDay;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Interface for appointments/events - e.g. the contents of a calendar
|
|
|
|
//
|
|
|
|
|
|
|
|
// First, an implementation with no events
|
|
|
|
function EmptyEventSource() {
|
2009-09-30 10:02:08 -04:00
|
|
|
this._init();
|
2010-03-15 09:50:05 -04:00
|
|
|
}
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
EmptyEventSource.prototype = {
|
|
|
|
_init: function() {
|
|
|
|
},
|
|
|
|
|
|
|
|
requestRange: function(begin, end) {
|
|
|
|
},
|
|
|
|
|
|
|
|
getEvents: function(begin, end) {
|
|
|
|
let result = [];
|
|
|
|
return result;
|
|
|
|
},
|
|
|
|
|
|
|
|
hasEvents: function(day) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Signals.addSignalMethods(EmptyEventSource.prototype);
|
|
|
|
|
|
|
|
// Second, wrap native Evolution event source
|
|
|
|
function EvolutionEventSource() {
|
|
|
|
this._init();
|
|
|
|
}
|
|
|
|
|
|
|
|
EvolutionEventSource.prototype = {
|
|
|
|
_init: function() {
|
|
|
|
this._native = new Shell.EvolutionEventSource();
|
|
|
|
this._native.connect('changed', Lang.bind(this, function() {
|
|
|
|
this.emit('changed');
|
|
|
|
}));
|
|
|
|
},
|
|
|
|
|
|
|
|
requestRange: function(begin, end) {
|
|
|
|
this._native.request_range(begin.getTime(), end.getTime());
|
|
|
|
},
|
|
|
|
|
|
|
|
getEvents: function(begin, end) {
|
|
|
|
let result = [];
|
|
|
|
let nativeEvents = this._native.get_events(begin.getTime(), end.getTime());
|
|
|
|
for (let n = 0; n < nativeEvents.length; n++) {
|
|
|
|
let nativeEvent = nativeEvents[n];
|
|
|
|
result.push(new CalendarEvent(new Date(nativeEvent.msec_begin), nativeEvent.summary, nativeEvent.all_day));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
},
|
|
|
|
|
|
|
|
hasEvents: function(day) {
|
|
|
|
let dayBegin = _getBeginningOfDay(day);
|
|
|
|
let dayEnd = _getEndOfDay(day);
|
|
|
|
|
|
|
|
let events = this.getEvents(dayBegin, dayEnd);
|
|
|
|
|
|
|
|
if (events.length == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Signals.addSignalMethods(EvolutionEventSource.prototype);
|
|
|
|
|
|
|
|
// Finally, an implementation with fake events
|
|
|
|
function FakeEventSource() {
|
|
|
|
this._init();
|
|
|
|
}
|
|
|
|
|
|
|
|
FakeEventSource.prototype = {
|
2009-09-30 10:02:08 -04:00
|
|
|
_init: function() {
|
2011-01-28 16:35:46 -05:00
|
|
|
|
|
|
|
this._fakeEvents = [];
|
|
|
|
|
|
|
|
// Generate fake events
|
|
|
|
//
|
|
|
|
let midnightToday = _getBeginningOfDay(new Date());
|
|
|
|
let summary = '';
|
|
|
|
|
|
|
|
// '10-oclock pow-wow' is an event occuring IN THE PAST every four days at 10am
|
|
|
|
for (let n = 0; n < 10; n++) {
|
|
|
|
let t = new Date(midnightToday.getTime() - n * 4 * 86400 * 1000);
|
|
|
|
t.setHours(10);
|
|
|
|
summary = '10-oclock pow-wow (n=' + n + ')';
|
|
|
|
this._fakeEvents.push(new CalendarEvent(t, summary, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
// '11-oclock thing' is an event occuring every three days at 11am
|
|
|
|
for (let n = 0; n < 10; n++) {
|
|
|
|
let t = new Date(midnightToday.getTime() + n * 3 * 86400 * 1000);
|
|
|
|
t.setHours(11);
|
|
|
|
summary = '11-oclock thing (n=' + n + ')';
|
|
|
|
this._fakeEvents.push(new CalendarEvent(t, summary, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
// 'Weekly Meeting' is an event occuring every seven days at 1:45pm (two days displaced)
|
|
|
|
for (let n = 0; n < 5; n++) {
|
|
|
|
let t = new Date(midnightToday.getTime() + (n * 7 + 2) * 86400 * 1000);
|
|
|
|
t.setHours(13);
|
|
|
|
t.setMinutes(45);
|
|
|
|
summary = 'Weekly Meeting (n=' + n + ')';
|
|
|
|
this._fakeEvents.push(new CalendarEvent(t, summary, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
// 'Fun All Day' is an all-day event occuring every fortnight (three days displayed)
|
|
|
|
for (let n = 0; n < 10; n++) {
|
|
|
|
let t = new Date(midnightToday.getTime() + (n * 14 + 3) * 86400 * 1000);
|
|
|
|
summary = 'Fun All Day (n=' + n + ')';
|
|
|
|
this._fakeEvents.push(new CalendarEvent(t, summary, true));
|
|
|
|
}
|
|
|
|
|
|
|
|
// 'Get Married' is an event that actually reflects reality (Dec 4, 2010) :-)
|
|
|
|
this._fakeEvents.push(new CalendarEvent(new Date(2010, 11, 4, 16, 0), 'Get Married', false));
|
|
|
|
|
|
|
|
// ditto for 'NE Patriots vs NY Jets'
|
|
|
|
this._fakeEvents.push(new CalendarEvent(new Date(2010, 11, 6, 20, 30), 'NE Patriots vs NY Jets', false));
|
|
|
|
|
|
|
|
// An event for tomorrow @6:30pm that is added/removed every five
|
|
|
|
// seconds (to check that the ::changed signal works)
|
|
|
|
let transientEventDate = new Date(midnightToday.getTime() + 86400 * 1000);
|
|
|
|
transientEventDate.setHours(18);
|
|
|
|
transientEventDate.setMinutes(30);
|
|
|
|
transientEventDate.setSeconds(0);
|
|
|
|
Mainloop.timeout_add(5000, Lang.bind(this, this._updateTransientEvent));
|
|
|
|
this._includeTransientEvent = false;
|
|
|
|
this._transientEvent = new CalendarEvent(transientEventDate, 'A Transient Event', false);
|
|
|
|
this._transientEventCounter = 1;
|
|
|
|
},
|
|
|
|
|
|
|
|
_updateTransientEvent: function() {
|
|
|
|
this._includeTransientEvent = !this._includeTransientEvent;
|
|
|
|
this._transientEventCounter = this._transientEventCounter + 1;
|
|
|
|
this._transientEvent.summary = 'A Transient Event (' + this._transientEventCounter + ')';
|
|
|
|
this.emit('changed');
|
|
|
|
Mainloop.timeout_add(5000, Lang.bind(this, this._updateTransientEvent));
|
|
|
|
},
|
|
|
|
|
|
|
|
requestRange: function(begin, end) {
|
|
|
|
},
|
|
|
|
|
|
|
|
getEvents: function(begin, end) {
|
|
|
|
let result = [];
|
|
|
|
//log('begin:' + begin);
|
|
|
|
//log('end: ' + end);
|
|
|
|
for(let n = 0; n < this._fakeEvents.length; n++) {
|
|
|
|
let event = this._fakeEvents[n];
|
|
|
|
if (event.date >= begin && event.date <= end) {
|
|
|
|
result.push(event);
|
|
|
|
}
|
|
|
|
//log('when:' + event.date + ' summary:' + event.summary);
|
|
|
|
}
|
|
|
|
if (this._includeTransientEvent && this._transientEvent.date >= begin && this._transientEvent.date <= end)
|
|
|
|
result.push(this._transientEvent);
|
|
|
|
result.sort(function(event1, event2) {
|
|
|
|
return event1.date.getTime() - event2.date.getTime();
|
|
|
|
});
|
|
|
|
return result;
|
|
|
|
},
|
|
|
|
|
|
|
|
hasEvents: function(day) {
|
|
|
|
let dayBegin = _getBeginningOfDay(day);
|
|
|
|
let dayEnd = _getEndOfDay(day);
|
|
|
|
|
|
|
|
let events = this.getEvents(dayBegin, dayEnd);
|
|
|
|
|
|
|
|
if (events.length == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
Signals.addSignalMethods(FakeEventSource.prototype);
|
|
|
|
|
|
|
|
// Calendar:
|
|
|
|
// @eventSource: is an object implementing the EventSource API, e.g. the
|
|
|
|
// requestRange(), getEvents(), hasEvents() methods and the ::changed signal.
|
|
|
|
function Calendar(eventSource) {
|
|
|
|
this._init(eventSource);
|
|
|
|
}
|
|
|
|
|
|
|
|
Calendar.prototype = {
|
|
|
|
_init: function(eventSource) {
|
|
|
|
this._eventSource = eventSource;
|
|
|
|
|
|
|
|
this._eventSource.connect('changed', Lang.bind(this, this._update));
|
|
|
|
|
2009-09-30 10:02:08 -04:00
|
|
|
// FIXME: This is actually the fallback method for GTK+ for the week start;
|
|
|
|
// GTK+ by preference uses nl_langinfo (NL_TIME_FIRST_WEEKDAY). We probably
|
|
|
|
// should add a C function so we can do the full handling.
|
|
|
|
this._weekStart = NaN;
|
2010-05-27 03:31:46 -04:00
|
|
|
this._weekdate = NaN;
|
|
|
|
this._digitWidth = NaN;
|
2010-05-05 17:05:42 -04:00
|
|
|
this._settings = new Gio.Settings({ schema: 'org.gnome.shell.calendar' });
|
2010-05-27 03:31:46 -04:00
|
|
|
|
2010-05-05 17:05:42 -04:00
|
|
|
this._settings.connect('changed::' + SHOW_WEEKDATE_KEY, Lang.bind(this, this._onSettingsChange));
|
|
|
|
this._useWeekdate = this._settings.get_boolean(SHOW_WEEKDATE_KEY);
|
2010-05-27 03:31:46 -04:00
|
|
|
|
2010-11-29 16:41:05 -05:00
|
|
|
let weekStartString = Gettext_gtk30.gettext('calendar:week_start:0');
|
2010-05-13 15:46:04 -04:00
|
|
|
if (weekStartString.indexOf('calendar:week_start:') == 0) {
|
2009-09-30 10:02:08 -04:00
|
|
|
this._weekStart = parseInt(weekStartString.substring(20));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isNaN(this._weekStart) || this._weekStart < 0 || this._weekStart > 6) {
|
2010-05-13 15:46:04 -04:00
|
|
|
log('Translation of "calendar:week_start:0" in GTK+ is not correct');
|
2009-12-25 03:34:40 -05:00
|
|
|
this._weekStart = 0;
|
2009-09-30 10:02:08 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find the ordering for month/year in the calendar heading
|
2011-01-28 16:35:46 -05:00
|
|
|
this._headerFormatWithoutYear = '%B';
|
2010-11-29 16:41:05 -05:00
|
|
|
switch (Gettext_gtk30.gettext('calendar:MY')) {
|
2010-05-13 15:46:04 -04:00
|
|
|
case 'calendar:MY':
|
|
|
|
this._headerFormat = '%B %Y';
|
2009-09-30 10:02:08 -04:00
|
|
|
break;
|
2010-05-13 15:46:04 -04:00
|
|
|
case 'calendar:YM':
|
|
|
|
this._headerFormat = '%Y %B';
|
2009-09-30 10:02:08 -04:00
|
|
|
break;
|
|
|
|
default:
|
2010-05-13 15:46:04 -04:00
|
|
|
log('Translation of "calendar:MY" in GTK+ is not correct');
|
|
|
|
this._headerFormat = '%B %Y';
|
2009-09-30 10:02:08 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start off with the current date
|
2011-01-28 16:35:46 -05:00
|
|
|
this._selectedDate = new Date();
|
2009-09-30 10:02:08 -04:00
|
|
|
|
|
|
|
this.actor = new St.Table({ homogeneous: false,
|
2010-05-13 15:46:04 -04:00
|
|
|
style_class: 'calendar',
|
2009-10-01 15:18:20 -04:00
|
|
|
reactive: true });
|
|
|
|
|
|
|
|
this.actor.connect('scroll-event',
|
|
|
|
Lang.bind(this, this._onScroll));
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2010-05-27 03:31:46 -04:00
|
|
|
this._buildHeader ();
|
|
|
|
this._update();
|
|
|
|
},
|
|
|
|
|
|
|
|
// Sets the calendar to show a specific date
|
|
|
|
setDate: function(date) {
|
2011-01-28 16:35:46 -05:00
|
|
|
if (!_sameDay(date, this._selectedDate)) {
|
|
|
|
this._selectedDate = date;
|
2010-05-27 03:31:46 -04:00
|
|
|
this._update();
|
2011-01-28 16:35:46 -05:00
|
|
|
this.emit('selected-date-changed', new Date(this._selectedDate));
|
2010-05-27 03:31:46 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_buildHeader: function() {
|
|
|
|
let offsetCols = this._useWeekdate ? 1 : 0;
|
|
|
|
this.actor.destroy_children();
|
|
|
|
|
2009-09-30 10:02:08 -04:00
|
|
|
// Top line of the calendar '<| September 2009 |>'
|
|
|
|
this._topBox = new St.BoxLayout();
|
|
|
|
this.actor.add(this._topBox,
|
2010-05-27 03:31:46 -04:00
|
|
|
{ row: 0, col: 0, col_span: offsetCols + 7 });
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2010-05-27 03:31:46 -04:00
|
|
|
this.actor.connect('style-changed', Lang.bind(this, this._onStyleChange));
|
2009-12-25 03:34:40 -05:00
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
let back = new St.Button({ style_class: 'calendar-change-month-back' });
|
2009-09-30 10:02:08 -04:00
|
|
|
this._topBox.add(back);
|
2011-01-28 16:35:46 -05:00
|
|
|
back.connect('clicked', Lang.bind(this, this._onPrevMonthButtonClicked));
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
this._monthLabel = new St.Label({style_class: 'calendar-month-label'});
|
|
|
|
this._topBox.add(this._monthLabel, { expand: true, x_fill: false, x_align: St.Align.MIDDLE });
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
let forward = new St.Button({ style_class: 'calendar-change-month-forward' });
|
2009-09-30 10:02:08 -04:00
|
|
|
this._topBox.add(forward);
|
2011-01-28 16:35:46 -05:00
|
|
|
forward.connect('clicked', Lang.bind(this, this._onNextMonthButtonClicked));
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
// Add weekday labels...
|
|
|
|
//
|
2009-09-30 10:02:08 -04:00
|
|
|
// We need to figure out the abbreviated localized names for the days of the week;
|
|
|
|
// we do this by just getting the next 7 days starting from right now and then putting
|
|
|
|
// them in the right cell in the table. It doesn't matter if we add them in order
|
2011-01-28 16:35:46 -05:00
|
|
|
let iter = new Date(this._selectedDate);
|
2009-09-30 10:02:08 -04:00
|
|
|
iter.setSeconds(0); // Leap second protection. Hah!
|
2009-10-01 16:14:50 -04:00
|
|
|
iter.setHours(12);
|
2009-09-30 10:02:08 -04:00
|
|
|
for (let i = 0; i < 7; i++) {
|
2011-01-28 16:35:46 -05:00
|
|
|
// Could use iter.toLocaleFormat('%a') but that normally gives three characters
|
|
|
|
// and we want, ideally, a single character for e.g. S M T W T F S
|
|
|
|
let customDayAbbrev = _getCalendarDayAbbreviation(iter.getDay());
|
|
|
|
let label = new St.Label({ style_class: 'calendar-day-base calendar-day-heading',
|
|
|
|
text: customDayAbbrev });
|
|
|
|
this.actor.add(label,
|
2009-09-30 10:02:08 -04:00
|
|
|
{ row: 1,
|
2010-05-27 03:31:46 -04:00
|
|
|
col: offsetCols + (7 + iter.getDay() - this._weekStart) % 7,
|
2011-01-28 16:35:46 -05:00
|
|
|
x_fill: false, x_align: St.Align.MIDDLE });
|
2009-09-30 10:02:08 -04:00
|
|
|
iter.setTime(iter.getTime() + MSECS_IN_DAY);
|
|
|
|
}
|
|
|
|
|
2009-10-01 16:14:50 -04:00
|
|
|
// All the children after this are days, and get removed when we update the calendar
|
|
|
|
this._firstDayIndex = this.actor.get_children().length;
|
2010-05-27 03:31:46 -04:00
|
|
|
},
|
2009-10-01 16:14:50 -04:00
|
|
|
|
2010-05-27 03:31:46 -04:00
|
|
|
_onStyleChange: function(actor, event) {
|
|
|
|
// width of a digit in pango units
|
|
|
|
this._digitWidth = _getDigitWidth(this.actor) / Pango.SCALE;
|
|
|
|
this._setWeekdateHeaderWidth();
|
2009-09-30 10:02:08 -04:00
|
|
|
},
|
|
|
|
|
2010-05-27 03:31:46 -04:00
|
|
|
_setWeekdateHeaderWidth: function() {
|
|
|
|
if (this.digitWidth != NaN && this._useWeekdate && this._weekdateHeader) {
|
|
|
|
this._weekdateHeader.set_width (this._digitWidth * WEEKDATE_HEADER_WIDTH_DIGITS);
|
2009-09-30 10:02:08 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2009-10-01 15:18:20 -04:00
|
|
|
_onScroll : function(actor, event) {
|
|
|
|
switch (event.get_scroll_direction()) {
|
|
|
|
case Clutter.ScrollDirection.UP:
|
|
|
|
case Clutter.ScrollDirection.LEFT:
|
2011-01-28 16:35:46 -05:00
|
|
|
this._onPrevMonthButtonClicked();
|
2009-10-01 15:18:20 -04:00
|
|
|
break;
|
|
|
|
case Clutter.ScrollDirection.DOWN:
|
|
|
|
case Clutter.ScrollDirection.RIGHT:
|
2011-01-28 16:35:46 -05:00
|
|
|
this._onNextMonthButtonClicked();
|
2009-10-01 15:18:20 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
_onPrevMonthButtonClicked: function() {
|
|
|
|
let newDate = new Date(this._selectedDate);
|
|
|
|
if (newDate.getMonth() == 0) {
|
|
|
|
newDate.setMonth(11);
|
|
|
|
newDate.setFullYear(newDate.getFullYear() - 1);
|
2009-09-30 10:02:08 -04:00
|
|
|
} else {
|
2011-01-28 16:35:46 -05:00
|
|
|
newDate.setMonth(newDate.getMonth() - 1);
|
2009-09-30 10:02:08 -04:00
|
|
|
}
|
2011-01-28 16:35:46 -05:00
|
|
|
this.setDate(newDate);
|
2009-09-30 10:02:08 -04:00
|
|
|
},
|
|
|
|
|
2011-01-28 16:35:46 -05:00
|
|
|
_onNextMonthButtonClicked: function() {
|
|
|
|
let newDate = new Date(this._selectedDate);
|
|
|
|
if (newDate.getMonth() == 11) {
|
|
|
|
newDate.setMonth(0);
|
|
|
|
newDate.setFullYear(newDate.getFullYear() + 1);
|
2009-09-30 10:02:08 -04:00
|
|
|
} else {
|
2011-01-28 16:35:46 -05:00
|
|
|
newDate.setMonth(newDate.getMonth() + 1);
|
2009-09-30 10:02:08 -04:00
|
|
|
}
|
2011-01-28 16:35:46 -05:00
|
|
|
this.setDate(newDate);
|
2009-09-30 10:02:08 -04:00
|
|
|
},
|
|
|
|
|
2010-05-27 03:31:46 -04:00
|
|
|
_onSettingsChange: function() {
|
2010-05-05 17:05:42 -04:00
|
|
|
this._useWeekdate = this._settings.get_boolean(SHOW_WEEKDATE_KEY);
|
2010-05-27 03:31:46 -04:00
|
|
|
this._buildHeader();
|
|
|
|
this._update();
|
|
|
|
},
|
|
|
|
|
2009-09-30 10:02:08 -04:00
|
|
|
_update: function() {
|
2011-01-28 16:35:46 -05:00
|
|
|
let now = new Date();
|
|
|
|
|
|
|
|
if (_sameYear(this._selectedDate, now))
|
|
|
|
this._monthLabel.text = this._selectedDate.toLocaleFormat(this._headerFormatWithoutYear);
|
|
|
|
else
|
|
|
|
this._monthLabel.text = this._selectedDate.toLocaleFormat(this._headerFormat);
|
2009-09-30 10:02:08 -04:00
|
|
|
|
|
|
|
// Remove everything but the topBox and the weekday labels
|
|
|
|
let children = this.actor.get_children();
|
2009-10-01 16:14:50 -04:00
|
|
|
for (let i = this._firstDayIndex; i < children.length; i++)
|
2009-09-30 10:02:08 -04:00
|
|
|
children[i].destroy();
|
|
|
|
|
|
|
|
// Start at the beginning of the week before the start of the month
|
2011-01-28 16:35:46 -05:00
|
|
|
let beginDate = new Date(this._selectedDate);
|
|
|
|
beginDate.setDate(1);
|
|
|
|
beginDate.setSeconds(0);
|
|
|
|
beginDate.setHours(12);
|
|
|
|
let daysToWeekStart = (7 + beginDate.getDay() - this._weekStart) % 7;
|
|
|
|
beginDate.setTime(beginDate.getTime() - daysToWeekStart * MSECS_IN_DAY);
|
|
|
|
|
|
|
|
let iter = new Date(beginDate);
|
2009-09-30 10:02:08 -04:00
|
|
|
let row = 2;
|
|
|
|
while (true) {
|
2011-01-28 16:35:46 -05:00
|
|
|
let button = new St.Button({ label: iter.getDate().toString() });
|
|
|
|
|
|
|
|
let iterStr = iter.toUTCString();
|
|
|
|
button.connect('clicked', Lang.bind(this, function() {
|
|
|
|
let newlySelectedDate = new Date(iterStr);
|
|
|
|
this.setDate(newlySelectedDate);
|
|
|
|
}));
|
|
|
|
|
|
|
|
let hasEvents = this._eventSource.hasEvents(iter);
|
|
|
|
let styleClass = 'calendar-day-base calendar-day';
|
|
|
|
if (_isWorkDay(iter))
|
|
|
|
styleClass += ' calendar-work-day'
|
2009-09-30 10:02:08 -04:00
|
|
|
else
|
2011-01-28 16:35:46 -05:00
|
|
|
styleClass += ' calendar-nonwork-day'
|
|
|
|
|
|
|
|
// Hack used in lieu of border-collapse - see gnome-shell.css
|
|
|
|
if (row == 2)
|
|
|
|
styleClass = 'calendar-day-top ' + styleClass;
|
|
|
|
if (iter.getDay() == 0)
|
|
|
|
styleClass = 'calendar-day-left ' + styleClass;
|
|
|
|
|
|
|
|
if (_sameDay(now, iter))
|
|
|
|
styleClass += ' calendar-today';
|
|
|
|
else if (iter.getMonth() != this._selectedDate.getMonth())
|
|
|
|
styleClass += ' calendar-other-month-day';
|
|
|
|
|
|
|
|
if (_sameDay(this._selectedDate, iter))
|
|
|
|
button.add_style_pseudo_class('active');
|
|
|
|
|
|
|
|
if (hasEvents)
|
|
|
|
styleClass += ' calendar-day-with-events'
|
|
|
|
|
|
|
|
button.style_class = styleClass;
|
2010-05-27 03:31:46 -04:00
|
|
|
|
|
|
|
let offsetCols = this._useWeekdate ? 1 : 0;
|
2011-01-28 16:35:46 -05:00
|
|
|
this.actor.add(button,
|
|
|
|
{ row: row, col: offsetCols + (7 + iter.getDay() - this._weekStart) % 7 });
|
2009-09-30 10:02:08 -04:00
|
|
|
|
2010-05-27 03:31:46 -04:00
|
|
|
if (this._useWeekdate && iter.getDay() == 4) {
|
|
|
|
let label = new St.Label({ text: _getCalendarWeekForDate(iter).toString(),
|
2011-01-28 16:35:46 -05:00
|
|
|
style_class: 'calendar-day-base calendar-week-number'});
|
2010-05-27 03:31:46 -04:00
|
|
|
this.actor.add(label,
|
2011-01-28 16:35:46 -05:00
|
|
|
{ row: row, col: 0, y_align: St.Align.MIDDLE });
|
2010-05-27 03:31:46 -04:00
|
|
|
}
|
|
|
|
|
2009-09-30 10:02:08 -04:00
|
|
|
iter.setTime(iter.getTime() + MSECS_IN_DAY);
|
|
|
|
if (iter.getDay() == this._weekStart) {
|
|
|
|
// We stop on the first "first day of the week" after the month we are displaying
|
2011-01-28 16:35:46 -05:00
|
|
|
if (iter.getMonth() > this._selectedDate.getMonth() || iter.getYear() > this._selectedDate.getYear())
|
2009-09-30 10:02:08 -04:00
|
|
|
break;
|
|
|
|
row++;
|
|
|
|
}
|
|
|
|
}
|
2011-01-28 16:35:46 -05:00
|
|
|
// Signal to the event source that we are interested in events
|
|
|
|
// only from this date range
|
|
|
|
this._eventSource.requestRange(beginDate, iter);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Signals.addSignalMethods(Calendar.prototype);
|
|
|
|
|
|
|
|
function EventsList(eventSource) {
|
|
|
|
this._init(eventSource);
|
|
|
|
}
|
|
|
|
|
|
|
|
EventsList.prototype = {
|
|
|
|
_init: function(eventSource) {
|
|
|
|
this.actor = new St.BoxLayout({ vertical: true, style_class: 'events-header-vbox'});
|
|
|
|
this._date = new Date();
|
|
|
|
this._eventSource = eventSource;
|
|
|
|
this._eventSource.connect('changed', Lang.bind(this, this._update));
|
|
|
|
this._desktopSettings = new Gio.Settings({ schema: 'org.gnome.desktop.interface' });
|
|
|
|
this._desktopSettings.connect('changed', Lang.bind(this, this._update));
|
|
|
|
this._update();
|
|
|
|
},
|
|
|
|
|
|
|
|
_addEvent: function(dayNameBox, timeBox, eventTitleBox, includeDayName, day, time, desc) {
|
|
|
|
if (includeDayName) {
|
|
|
|
dayNameBox.add(new St.Label( { style_class: 'events-day-dayname',
|
|
|
|
text: day } ),
|
|
|
|
{ x_fill: true } );
|
|
|
|
}
|
|
|
|
timeBox.add(new St.Label( { style_class: 'events-day-time',
|
|
|
|
text: time} ),
|
|
|
|
{ x_fill: true } );
|
|
|
|
eventTitleBox.add(new St.Label( { style_class: 'events-day-task',
|
|
|
|
text: desc} ));
|
|
|
|
},
|
|
|
|
|
|
|
|
_addPeriod: function(header, begin, end, includeDayName, showNothingScheduled) {
|
|
|
|
let events = this._eventSource.getEvents(begin, end);
|
|
|
|
|
|
|
|
let clockFormat = this._desktopSettings.get_string(CLOCK_FORMAT_KEY);;
|
|
|
|
|
|
|
|
if (events.length == 0 && !showNothingScheduled)
|
|
|
|
return;
|
|
|
|
|
|
|
|
let vbox = new St.BoxLayout( {vertical: true} );
|
|
|
|
this.actor.add(vbox);
|
|
|
|
|
|
|
|
vbox.add(new St.Label({ style_class: 'events-day-header', text: header }));
|
|
|
|
let box = new St.BoxLayout({style_class: 'events-header-hbox'});
|
|
|
|
let dayNameBox = new St.BoxLayout({ vertical: true, style_class: 'events-day-name-box' });
|
|
|
|
let timeBox = new St.BoxLayout({ vertical: true, style_class: 'events-time-box' });
|
|
|
|
let eventTitleBox = new St.BoxLayout({ vertical: true, style_class: 'events-event-box' });
|
|
|
|
box.add(dayNameBox, {x_fill: false});
|
|
|
|
box.add(timeBox, {x_fill: false});
|
|
|
|
box.add(eventTitleBox, {expand: true});
|
|
|
|
vbox.add(box);
|
|
|
|
|
|
|
|
for (let n = 0; n < events.length; n++) {
|
|
|
|
let event = events[n];
|
|
|
|
let dayString = _getEventDayAbbreviation(event.date.getDay());
|
|
|
|
let timeString = _formatEventTime(event, clockFormat);
|
|
|
|
let summaryString = event.summary;
|
|
|
|
this._addEvent(dayNameBox, timeBox, eventTitleBox, includeDayName, dayString, timeString, summaryString);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (events.length == 0 && showNothingScheduled) {
|
|
|
|
let now = new Date();
|
|
|
|
/* Translators: Text to show if there are no events */
|
|
|
|
let nothingEvent = new CalendarEvent(now, _("Nothing Scheduled"), true);
|
|
|
|
let timeString = _formatEventTime(nothingEvent, clockFormat);
|
|
|
|
this._addEvent(dayNameBox, timeBox, eventTitleBox, false, "", timeString, nothingEvent.summary);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_showOtherDay: function(day) {
|
|
|
|
this.actor.destroy_children();
|
|
|
|
|
|
|
|
let dayBegin = _getBeginningOfDay(day);
|
|
|
|
let dayEnd = _getEndOfDay(day);
|
|
|
|
|
|
|
|
let dayString;
|
|
|
|
let now = new Date();
|
|
|
|
if (_sameYear(day, now))
|
|
|
|
dayString = day.toLocaleFormat('%A, %B %d');
|
|
|
|
else
|
|
|
|
dayString = day.toLocaleFormat('%A, %B %d, %Y');
|
|
|
|
this._addPeriod(dayString, dayBegin, dayEnd, false, true);
|
|
|
|
},
|
|
|
|
|
|
|
|
_showToday: function() {
|
|
|
|
this.actor.destroy_children();
|
|
|
|
|
|
|
|
let now = new Date();
|
|
|
|
let dayBegin = _getBeginningOfDay(now);
|
|
|
|
let dayEnd = _getEndOfDay(now);
|
|
|
|
this._addPeriod(_("Today"), dayBegin, dayEnd, false, true);
|
|
|
|
|
|
|
|
let tomorrowBegin = new Date(dayBegin.getTime() + 86400 * 1000);
|
|
|
|
let tomorrowEnd = new Date(dayEnd.getTime() + 86400 * 1000);
|
|
|
|
this._addPeriod(_("Tomorrow"), tomorrowBegin, tomorrowEnd, false, true);
|
|
|
|
|
|
|
|
if (dayEnd.getDay() <= 4) {
|
|
|
|
/* if now is Sunday through Thursday show "This week" and include events up until
|
|
|
|
* and including Saturday
|
|
|
|
*/
|
|
|
|
let thisWeekBegin = new Date(dayBegin.getTime() + 2 * 86400 * 1000);
|
|
|
|
let thisWeekEnd = new Date(dayEnd.getTime() + (6 - dayEnd.getDay()) * 86400 * 1000);
|
|
|
|
this._addPeriod(_("This week"), thisWeekBegin, thisWeekEnd, true, false);
|
|
|
|
} else {
|
|
|
|
/* otherwise it's a Friday or Saturday... show "Next week" and include events up
|
|
|
|
* until and including *next* Saturday
|
|
|
|
*/
|
|
|
|
let nextWeekBegin = new Date(dayBegin.getTime() + 2 * 86400 * 1000);
|
|
|
|
let nextWeekEnd = new Date(dayEnd.getTime() + (13 - dayEnd.getDay()) * 86400 * 1000);
|
|
|
|
this._addPeriod(_("Next week"), nextWeekBegin, nextWeekEnd, true, false);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
// Sets the event list to show events from a specific date
|
|
|
|
setDate: function(date) {
|
|
|
|
if (!_sameDay(date, this._date)) {
|
|
|
|
this._date = date;
|
|
|
|
this._update();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_update: function() {
|
|
|
|
let today = new Date();
|
|
|
|
if (_sameDay (this._date, today)) {
|
|
|
|
this._showToday();
|
|
|
|
} else {
|
|
|
|
this._showOtherDay(this._date);
|
|
|
|
}
|
2009-09-30 10:02:08 -04:00
|
|
|
}
|
|
|
|
};
|