summaryrefslogtreecommitdiffstats
path: root/comm/calendar/base/src/CalAlarmService.jsm
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--comm/calendar/base/src/CalAlarmService.jsm827
1 files changed, 827 insertions, 0 deletions
diff --git a/comm/calendar/base/src/CalAlarmService.jsm b/comm/calendar/base/src/CalAlarmService.jsm
new file mode 100644
index 0000000000..916582a9af
--- /dev/null
+++ b/comm/calendar/base/src/CalAlarmService.jsm
@@ -0,0 +1,827 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+var EXPORTED_SYMBOLS = ["CalAlarmService"];
+
+var { cal } = ChromeUtils.import("resource:///modules/calendar/calUtils.jsm");
+var { XPCOMUtils } = ChromeUtils.importESModule("resource://gre/modules/XPCOMUtils.sys.mjs");
+
+const lazy = {};
+ChromeUtils.defineModuleGetter(lazy, "CalDateTime", "resource:///modules/CalDateTime.jsm");
+
+var kHoursBetweenUpdates = 6;
+
+function nowUTC() {
+ return cal.dtz.jsDateToDateTime(new Date()).getInTimezone(cal.dtz.UTC);
+}
+
+function newTimerWithCallback(aCallback, aDelay, aRepeating) {
+ let timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
+
+ timer.initWithCallback(
+ aCallback,
+ aDelay,
+ aRepeating ? timer.TYPE_REPEATING_PRECISE : timer.TYPE_ONE_SHOT
+ );
+ return timer;
+}
+
+/**
+ * Keeps track of seemingly immutable items with alarms that we can't dismiss.
+ * Some servers quietly discard our modifications to repeating events which will
+ * cause dismissed alarms to re-appear if we do not keep track.
+ *
+ * We track the items by their "hashId" property storing it in the calendar
+ * property "alarms.ignored".
+ */
+const IgnoredAlarmsStore = {
+ /**
+ * @type {number}
+ */
+ maxItemsPerCalendar: 1500,
+
+ _getCache(item) {
+ return item.calendar.getProperty("alarms.ignored")?.split(",") || [];
+ },
+
+ /**
+ * Adds an item to the store. No alarms will be created for this item again.
+ *
+ * @param {calIItemBase} item
+ */
+ add(item) {
+ let cache = this._getCache(item);
+ let id = item.parentItem.hashId;
+ if (!cache.includes(id)) {
+ if (cache.length >= this.maxItemsPerCalendar) {
+ cache[0] = id;
+ } else {
+ cache.push(id);
+ }
+ }
+ item.calendar.setProperty("alarms.ignored", cache.join(","));
+ },
+
+ /**
+ * Returns true if the item's hashId is in the store.
+ *
+ * @param {calIItemBase} item
+ * @returns {boolean}
+ */
+ has(item) {
+ return this._getCache(item).includes(item.parentItem.hashId);
+ },
+};
+
+function CalAlarmService() {
+ this.wrappedJSObject = this;
+
+ this.mLoadedCalendars = {};
+ this.mTimerMap = {};
+ this.mNotificationTimerMap = {};
+ this.mObservers = new cal.data.ListenerSet(Ci.calIAlarmServiceObserver);
+
+ XPCOMUtils.defineLazyPreferenceGetter(
+ this,
+ "gNotificationsTimes",
+ "calendar.notifications.times",
+ "",
+ () => this.initAlarms(cal.manager.getCalendars())
+ );
+
+ this.calendarObserver = {
+ QueryInterface: ChromeUtils.generateQI(["calIObserver"]),
+ alarmService: this,
+
+ calendarsInBatch: new Set(),
+
+ // calIObserver:
+ onStartBatch(calendar) {
+ this.calendarsInBatch.add(calendar);
+ },
+ onEndBatch(calendar) {
+ this.calendarsInBatch.delete(calendar);
+ },
+ onLoad(calendar) {
+ // ignore any onLoad events until initial getItems() call of startup has finished:
+ if (calendar && this.alarmService.mLoadedCalendars[calendar.id]) {
+ // a refreshed calendar signals that it has been reloaded
+ // (and cannot notify detailed changes), thus reget all alarms of it:
+ this.alarmService.initAlarms([calendar]);
+ }
+ },
+
+ onAddItem(aItem) {
+ // If we're in a batch, ignore this notification. We're going to reload anyway.
+ if (!this.calendarsInBatch.has(aItem.calendar)) {
+ this.alarmService.addAlarmsForOccurrences(aItem);
+ }
+ },
+ onModifyItem(aNewItem, aOldItem) {
+ // If we're in a batch, ignore this notification. We're going to reload anyway.
+ if (this.calendarsInBatch.has(aNewItem.calendar)) {
+ return;
+ }
+
+ if (!aNewItem.recurrenceId) {
+ // deleting an occurrence currently calls modifyItem(newParent, *oldOccurrence*)
+ aOldItem = aOldItem.parentItem;
+ }
+
+ this.onDeleteItem(aOldItem);
+ this.onAddItem(aNewItem);
+ },
+ onDeleteItem(aDeletedItem) {
+ this.alarmService.removeAlarmsForOccurrences(aDeletedItem);
+ },
+ onError(aCalendar, aErrNo, aMessage) {},
+ onPropertyChanged(aCalendar, aName, aValue, aOldValue) {
+ switch (aName) {
+ case "suppressAlarms":
+ case "disabled":
+ case "notifications.times":
+ this.alarmService.initAlarms([aCalendar]);
+ break;
+ }
+ },
+ onPropertyDeleting(aCalendar, aName) {
+ this.onPropertyChanged(aCalendar, aName);
+ },
+ };
+
+ this.calendarManagerObserver = {
+ QueryInterface: ChromeUtils.generateQI(["calICalendarManagerObserver"]),
+ alarmService: this,
+
+ onCalendarRegistered(aCalendar) {
+ this.alarmService.observeCalendar(aCalendar);
+ // initial refresh of alarms for new calendar:
+ this.alarmService.initAlarms([aCalendar]);
+ },
+ onCalendarUnregistering(aCalendar) {
+ // XXX todo: we need to think about calendar unregistration;
+ // there may still be dangling items (-> alarm dialog),
+ // dismissing those alarms may write data...
+ this.alarmService.unobserveCalendar(aCalendar);
+ delete this.alarmService.mLoadedCalendars[aCalendar.id];
+ },
+ onCalendarDeleting(aCalendar) {
+ this.alarmService.unobserveCalendar(aCalendar);
+ delete this.alarmService.mLoadedCalendars[aCalendar.id];
+ },
+ };
+}
+
+var calAlarmServiceClassID = Components.ID("{7a9200dd-6a64-4fff-a798-c5802186e2cc}");
+var calAlarmServiceInterfaces = [Ci.calIAlarmService, Ci.nsIObserver];
+CalAlarmService.prototype = {
+ mRangeStart: null,
+ mRangeEnd: null,
+ mUpdateTimer: null,
+ mStarted: false,
+ mTimerMap: null,
+ mObservers: null,
+ mTimezone: null,
+
+ classID: calAlarmServiceClassID,
+ QueryInterface: cal.generateQI(["calIAlarmService", "nsIObserver"]),
+ classInfo: cal.generateCI({
+ classID: calAlarmServiceClassID,
+ contractID: "@mozilla.org/calendar/alarm-service;1",
+ classDescription: "Calendar Alarm Service",
+ interfaces: calAlarmServiceInterfaces,
+ flags: Ci.nsIClassInfo.SINGLETON,
+ }),
+
+ _logger: console.createInstance({
+ prefix: "calendar.alarms",
+ maxLogLevel: "Warn",
+ maxLogLevelPref: "calendar.alarms.loglevel",
+ }),
+
+ /**
+ * nsIObserver
+ */
+ observe(aSubject, aTopic, aData) {
+ // This will also be called on app-startup, but nothing is done yet, to
+ // prevent unwanted dialogs etc. See bug 325476 and 413296
+ if (aTopic == "profile-after-change" || aTopic == "wake_notification") {
+ this.shutdown();
+ this.startup();
+ }
+ if (aTopic == "xpcom-shutdown") {
+ this.shutdown();
+ }
+ },
+
+ /**
+ * calIAlarmService APIs
+ */
+ get timezone() {
+ // TODO Do we really need this? Do we ever set the timezone to something
+ // different than the default timezone?
+ return this.mTimezone || cal.dtz.defaultTimezone;
+ },
+
+ set timezone(aTimezone) {
+ this.mTimezone = aTimezone;
+ },
+
+ async snoozeAlarm(aItem, aAlarm, aDuration) {
+ // Right now we only support snoozing all alarms for the given item for
+ // aDuration.
+
+ // Make sure we're working with the parent, otherwise we'll accidentally
+ // create an exception
+ let newEvent = aItem.parentItem.clone();
+ let alarmTime = nowUTC();
+
+ // Set the last acknowledged time to now.
+ newEvent.alarmLastAck = alarmTime;
+
+ alarmTime = alarmTime.clone();
+ alarmTime.addDuration(aDuration);
+
+ let propName = "X-MOZ-SNOOZE-TIME";
+ if (aItem.parentItem != aItem) {
+ // This is the *really* hard case where we've snoozed a single
+ // instance of a recurring event. We need to not only know that
+ // there was a snooze, but also which occurrence was snoozed. Part
+ // of me just wants to create a local db of snoozes here...
+ propName = "X-MOZ-SNOOZE-TIME-" + aItem.recurrenceId.nativeTime;
+ }
+ newEvent.setProperty(propName, alarmTime.icalString);
+
+ // calling modifyItem will cause us to get the right callback
+ // and update the alarm properly
+ let modifiedItem = await newEvent.calendar.modifyItem(newEvent, aItem.parentItem);
+
+ if (modifiedItem.getProperty(propName) == alarmTime.icalString) {
+ return;
+ }
+
+ // The server did not persist our changes for some reason.
+ // Include the item in the ignored list so we skip displaying alarms for
+ // this item in the future.
+ IgnoredAlarmsStore.add(aItem);
+ },
+
+ async dismissAlarm(aItem, aAlarm) {
+ if (cal.acl.isCalendarWritable(aItem.calendar) && cal.acl.userCanModifyItem(aItem)) {
+ let now = nowUTC();
+ // We want the parent item, otherwise we're going to accidentally
+ // create an exception. We've relnoted (for 0.1) the slightly odd
+ // behavior this can cause if you move an event after dismissing an
+ // alarm
+ let oldParent = aItem.parentItem;
+ let newParent = oldParent.clone();
+ newParent.alarmLastAck = now;
+ // Make sure to clear out any snoozes that were here.
+ if (aItem.recurrenceId) {
+ newParent.deleteProperty("X-MOZ-SNOOZE-TIME-" + aItem.recurrenceId.nativeTime);
+ } else {
+ newParent.deleteProperty("X-MOZ-SNOOZE-TIME");
+ }
+
+ let modifiedItem = await newParent.calendar.modifyItem(newParent, oldParent);
+ if (modifiedItem.alarmLastAck && now.compare(modifiedItem.alarmLastAck) == 0) {
+ return;
+ }
+
+ // The server did not persist our changes for some reason.
+ // Include the item in the ignored list so we skip displaying alarms for
+ // this item in the future.
+ IgnoredAlarmsStore.add(aItem);
+ }
+ // if the calendar of the item is r/o, we simple remove the alarm
+ // from the list without modifying the item, so this works like
+ // effectively dismissing from a user's pov, since the alarm neither
+ // popups again in the current user session nor will be added after
+ // next restart, since it is missed then already
+ this.removeAlarmsForItem(aItem);
+ },
+
+ addObserver(aObserver) {
+ this.mObservers.add(aObserver);
+ },
+
+ removeObserver(aObserver) {
+ this.mObservers.delete(aObserver);
+ },
+
+ startup() {
+ if (this.mStarted) {
+ return;
+ }
+
+ Services.obs.addObserver(this, "profile-after-change");
+ Services.obs.addObserver(this, "xpcom-shutdown");
+ Services.obs.addObserver(this, "wake_notification");
+
+ // Make sure the alarm monitor is alive so it's observing the notification.
+ Cc["@mozilla.org/calendar/alarm-monitor;1"].getService(Ci.calIAlarmServiceObserver);
+ // Tell people that we're alive so they can start monitoring alarms.
+ Services.obs.notifyObservers(null, "alarm-service-startup");
+
+ cal.manager.addObserver(this.calendarManagerObserver);
+
+ for (let calendar of cal.manager.getCalendars()) {
+ this.observeCalendar(calendar);
+ }
+
+ /* set up a timer to update alarms every N hours */
+ let timerCallback = {
+ alarmService: this,
+ notify() {
+ let now = nowUTC();
+ let start;
+ if (this.alarmService.mRangeEnd) {
+ // This is a subsequent search, so we got all the past alarms before
+ start = this.alarmService.mRangeEnd.clone();
+ } else {
+ // This is our first search for alarms. We're going to look for
+ // alarms +/- 1 month from now. If someone sets an alarm more than
+ // a month ahead of an event, or doesn't start Thunderbird
+ // for a month, they'll miss some, but that's a slim chance
+ start = now.clone();
+ start.month -= Ci.calIAlarmService.MAX_SNOOZE_MONTHS;
+ this.alarmService.mRangeStart = start.clone();
+ }
+ let until = now.clone();
+ until.month += Ci.calIAlarmService.MAX_SNOOZE_MONTHS;
+
+ // We don't set timers for every future alarm, only those within 6 hours
+ let end = now.clone();
+ end.hour += kHoursBetweenUpdates;
+ this.alarmService.mRangeEnd = end.getInTimezone(cal.dtz.UTC);
+
+ this.alarmService.findAlarms(cal.manager.getCalendars(), start, until);
+ },
+ };
+ timerCallback.notify();
+
+ this.mUpdateTimer = newTimerWithCallback(timerCallback, kHoursBetweenUpdates * 3600000, true);
+
+ this.mStarted = true;
+ },
+
+ shutdown() {
+ if (!this.mStarted) {
+ return;
+ }
+
+ // Tell people that we're no longer running.
+ Services.obs.notifyObservers(null, "alarm-service-shutdown");
+
+ if (this.mUpdateTimer) {
+ this.mUpdateTimer.cancel();
+ this.mUpdateTimer = null;
+ }
+
+ cal.manager.removeObserver(this.calendarManagerObserver);
+
+ // Stop observing all calendars. This will also clear the timers.
+ for (let calendar of cal.manager.getCalendars()) {
+ this.unobserveCalendar(calendar);
+ }
+
+ this.mRangeEnd = null;
+
+ Services.obs.removeObserver(this, "profile-after-change");
+ Services.obs.removeObserver(this, "xpcom-shutdown");
+ Services.obs.removeObserver(this, "wake_notification");
+
+ this.mStarted = false;
+ },
+
+ observeCalendar(calendar) {
+ calendar.addObserver(this.calendarObserver);
+ },
+
+ unobserveCalendar(calendar) {
+ calendar.removeObserver(this.calendarObserver);
+ this.disposeCalendarTimers([calendar]);
+ this.mObservers.notify("onRemoveAlarmsByCalendar", [calendar]);
+ },
+
+ addAlarmsForItem(aItem) {
+ if ((aItem.isTodo() && aItem.isCompleted) || IgnoredAlarmsStore.has(aItem)) {
+ // If this is a task and it is completed or the id is in the ignored list,
+ // don't add the alarm.
+ return;
+ }
+
+ let showMissed = Services.prefs.getBoolPref("calendar.alarms.showmissed", true);
+
+ let alarms = aItem.getAlarms();
+ for (let alarm of alarms) {
+ let alarmDate = cal.alarms.calculateAlarmDate(aItem, alarm);
+
+ if (!alarmDate || alarm.action != "DISPLAY") {
+ // Only take care of DISPLAY alarms with an alarm date.
+ continue;
+ }
+
+ // Handle all day events. This is kinda weird, because they don't have
+ // a well defined startTime. We just consider the start/end to be
+ // midnight in the user's timezone.
+ if (alarmDate.isDate) {
+ alarmDate = alarmDate.getInTimezone(this.timezone);
+ alarmDate.isDate = false;
+ }
+ alarmDate = alarmDate.getInTimezone(cal.dtz.UTC);
+
+ // Check for snooze
+ let snoozeDate;
+ if (aItem.parentItem == aItem) {
+ snoozeDate = aItem.getProperty("X-MOZ-SNOOZE-TIME");
+ } else {
+ snoozeDate = aItem.parentItem.getProperty(
+ "X-MOZ-SNOOZE-TIME-" + aItem.recurrenceId.nativeTime
+ );
+ }
+
+ if (
+ snoozeDate &&
+ !(snoozeDate instanceof lazy.CalDateTime) &&
+ !(snoozeDate instanceof Ci.calIDateTime)
+ ) {
+ snoozeDate = cal.createDateTime(snoozeDate);
+ }
+
+ // an alarm can only be snoozed to a later time, if earlier it's from another alarm.
+ if (snoozeDate && snoozeDate.compare(alarmDate) > 0) {
+ // If the alarm was snoozed, the snooze time is more important.
+ alarmDate = snoozeDate;
+ }
+
+ let now = nowUTC();
+ if (alarmDate.timezone.isFloating) {
+ now = cal.dtz.now();
+ now.timezone = cal.dtz.floating;
+ }
+
+ if (alarmDate.compare(now) >= 0) {
+ // We assume that future alarms haven't been acknowledged
+ // Delay is in msec, so don't forget to multiply
+ let timeout = alarmDate.subtractDate(now).inSeconds * 1000;
+
+ // No sense in keeping an extra timeout for an alarm that's past
+ // our range.
+ let timeUntilRefresh = this.mRangeEnd.subtractDate(now).inSeconds * 1000;
+ if (timeUntilRefresh < timeout) {
+ continue;
+ }
+
+ this.addTimer(aItem, alarm, timeout);
+ } else if (
+ showMissed &&
+ cal.acl.isCalendarWritable(aItem.calendar) &&
+ cal.acl.userCanModifyItem(aItem)
+ ) {
+ // This alarm is in the past and the calendar is writable, so we
+ // could snooze or dismiss alarms. See if it has been previously
+ // ack'd.
+ let lastAck = aItem.parentItem.alarmLastAck;
+ if (lastAck && lastAck.compare(alarmDate) >= 0) {
+ // The alarm was previously dismissed or snoozed, no further
+ // action required.
+ continue;
+ } else {
+ // The alarm was not snoozed or dismissed, fire it now.
+ this.alarmFired(aItem, alarm);
+ }
+ }
+ }
+
+ this.addNotificationForItem(aItem);
+ },
+
+ removeAlarmsForItem(aItem) {
+ // make sure already fired alarms are purged out of the alarm window:
+ this.mObservers.notify("onRemoveAlarmsByItem", [aItem]);
+ // Purge alarms specifically for this item (i.e exception)
+ for (let alarm of aItem.getAlarms()) {
+ this.removeTimer(aItem, alarm);
+ }
+
+ this.removeNotificationForItem(aItem);
+ },
+
+ /**
+ * Get the timeouts before notifications are fired for an item.
+ *
+ * @param {calIItemBase} item - A calendar item instance.
+ * @returns {number[]} Timeouts of notifications in milliseconds in ascending order.
+ */
+ calculateNotificationTimeouts(item) {
+ let now = nowUTC();
+ let until = now.clone();
+ until.month += 1;
+ // We only care about items no more than a month ahead.
+ if (!cal.item.checkIfInRange(item, now, until)) {
+ return [];
+ }
+ let startDate = item[cal.dtz.startDateProp(item)];
+ let endDate = item[cal.dtz.endDateProp(item)];
+ let timeouts = [];
+ // The calendar level notifications setting overrides the global setting.
+ let prefValue = (
+ item.calendar.getProperty("notifications.times") || this.gNotificationsTimes
+ ).split(",");
+ for (let entry of prefValue) {
+ entry = entry.trim();
+ if (!entry) {
+ continue;
+ }
+ let [tag, value] = entry.split(":");
+ if (!value) {
+ value = tag;
+ tag = "";
+ }
+ let duration;
+ try {
+ duration = cal.createDuration(value);
+ } catch (e) {
+ this._logger.error(`Failed to parse ${entry}`, e);
+ continue;
+ }
+ let fireDate;
+ if (tag == "END" && endDate) {
+ fireDate = endDate.clone();
+ } else if (startDate) {
+ fireDate = startDate.clone();
+ } else {
+ continue;
+ }
+ fireDate.addDuration(duration);
+ let timeout = fireDate.subtractDate(now).inSeconds * 1000;
+ if (timeout > 0) {
+ timeouts.push(timeout);
+ }
+ }
+ return timeouts.sort((x, y) => x - y);
+ },
+
+ /**
+ * Set up notification timers for an item.
+ *
+ * @param {calIItemBase} item - A calendar item instance.
+ */
+ addNotificationForItem(item) {
+ let alarmTimerCallback = {
+ notify: () => {
+ this.mObservers.notify("onNotification", [item]);
+ this.removeFiredNotificationTimer(item);
+ },
+ };
+ let timeouts = this.calculateNotificationTimeouts(item);
+ let timers = timeouts.map(timeout => newTimerWithCallback(alarmTimerCallback, timeout, false));
+
+ if (timers.length > 0) {
+ this._logger.debug(
+ `addNotificationForItem hashId=${item.hashId}: adding ${timers.length} timers, timeouts=${timeouts}`
+ );
+ this.mNotificationTimerMap[item.calendar.id] =
+ this.mNotificationTimerMap[item.calendar.id] || {};
+ this.mNotificationTimerMap[item.calendar.id][item.hashId] = timers;
+ }
+ },
+
+ /**
+ * Remove notification timers for an item.
+ *
+ * @param {calIItemBase} item - A calendar item instance.
+ */
+ removeNotificationForItem(item) {
+ if (
+ !this.mNotificationTimerMap[item.calendar.id] ||
+ !this.mNotificationTimerMap[item.calendar.id][item.hashId]
+ ) {
+ return;
+ }
+
+ for (let timer of this.mNotificationTimerMap[item.calendar.id][item.hashId]) {
+ timer.cancel();
+ }
+
+ delete this.mNotificationTimerMap[item.calendar.id][item.hashId];
+
+ // If the calendar map is empty, remove it from the timer map
+ if (Object.keys(this.mNotificationTimerMap[item.calendar.id]).length == 0) {
+ delete this.mNotificationTimerMap[item.calendar.id];
+ }
+ },
+
+ /**
+ * Remove the first notification timers for an item to release some memory.
+ *
+ * @param {calIItemBase} item - A calendar item instance.
+ */
+ removeFiredNotificationTimer(item) {
+ // The first timer is fired first.
+ let removed = this.mNotificationTimerMap[item.calendar.id][item.hashId].shift();
+
+ let remainingTimersCount = this.mNotificationTimerMap[item.calendar.id][item.hashId].length;
+ this._logger.debug(
+ `removeFiredNotificationTimer hashId=${item.hashId}: removed=${removed.delay}, remaining ${remainingTimersCount} timers`
+ );
+ if (remainingTimersCount == 0) {
+ delete this.mNotificationTimerMap[item.calendar.id][item.hashId];
+ }
+
+ // If the calendar map is empty, remove it from the timer map
+ if (Object.keys(this.mNotificationTimerMap[item.calendar.id]).length == 0) {
+ delete this.mNotificationTimerMap[item.calendar.id];
+ }
+ },
+
+ getOccurrencesInRange(aItem) {
+ // We search 1 month in each direction for alarms. Therefore,
+ // we need occurrences between initial start date and 1 month from now
+ let until = nowUTC();
+ until.month += 1;
+
+ if (aItem && aItem.recurrenceInfo) {
+ return aItem.recurrenceInfo.getOccurrences(this.mRangeStart, until, 0);
+ }
+ return cal.item.checkIfInRange(aItem, this.mRangeStart, until) ? [aItem] : [];
+ },
+
+ addAlarmsForOccurrences(aParentItem) {
+ let occs = this.getOccurrencesInRange(aParentItem);
+
+ // Add an alarm for each occurrence
+ occs.forEach(this.addAlarmsForItem, this);
+ },
+
+ removeAlarmsForOccurrences(aParentItem) {
+ let occs = this.getOccurrencesInRange(aParentItem);
+
+ // Remove alarm for each occurrence
+ occs.forEach(this.removeAlarmsForItem, this);
+ },
+
+ addTimer(aItem, aAlarm, aTimeout) {
+ this.mTimerMap[aItem.calendar.id] = this.mTimerMap[aItem.calendar.id] || {};
+ this.mTimerMap[aItem.calendar.id][aItem.hashId] =
+ this.mTimerMap[aItem.calendar.id][aItem.hashId] || {};
+
+ let self = this;
+ let alarmTimerCallback = {
+ notify() {
+ self.alarmFired(aItem, aAlarm);
+ },
+ };
+
+ let timer = newTimerWithCallback(alarmTimerCallback, aTimeout, false);
+ this.mTimerMap[aItem.calendar.id][aItem.hashId][aAlarm.icalString] = timer;
+ },
+
+ removeTimer(aItem, aAlarm) {
+ /* Is the calendar in the timer map */
+ if (
+ aItem.calendar.id in this.mTimerMap &&
+ /* ...and is the item in the calendar map */
+ aItem.hashId in this.mTimerMap[aItem.calendar.id] &&
+ /* ...and is the alarm in the item map ? */
+ aAlarm.icalString in this.mTimerMap[aItem.calendar.id][aItem.hashId]
+ ) {
+ // First cancel the existing timer
+ let timer = this.mTimerMap[aItem.calendar.id][aItem.hashId][aAlarm.icalString];
+ timer.cancel();
+
+ // Remove the alarm from the item map
+ delete this.mTimerMap[aItem.calendar.id][aItem.hashId][aAlarm.icalString];
+
+ // If the item map is empty, remove it from the calendar map
+ if (this.mTimerMap[aItem.calendar.id][aItem.hashId].toSource() == "({})") {
+ delete this.mTimerMap[aItem.calendar.id][aItem.hashId];
+ }
+
+ // If the calendar map is empty, remove it from the timer map
+ if (this.mTimerMap[aItem.calendar.id].toSource() == "({})") {
+ delete this.mTimerMap[aItem.calendar.id];
+ }
+ }
+ },
+
+ disposeCalendarTimers(aCalendars) {
+ for (let calendar of aCalendars) {
+ if (calendar.id in this.mTimerMap) {
+ for (let hashId in this.mTimerMap[calendar.id]) {
+ let itemTimerMap = this.mTimerMap[calendar.id][hashId];
+ for (let icalString in itemTimerMap) {
+ let timer = itemTimerMap[icalString];
+ timer.cancel();
+ }
+ }
+ delete this.mTimerMap[calendar.id];
+ }
+ if (calendar.id in this.mNotificationTimerMap) {
+ for (let timers of Object.values(this.mNotificationTimerMap[calendar.id])) {
+ for (let timer of timers) {
+ timer.cancel();
+ }
+ }
+ delete this.mNotificationTimerMap[calendar.id];
+ }
+ }
+ },
+
+ async findAlarms(aCalendars, aStart, aUntil) {
+ const calICalendar = Ci.calICalendar;
+ let filter =
+ calICalendar.ITEM_FILTER_COMPLETED_ALL |
+ calICalendar.ITEM_FILTER_CLASS_OCCURRENCES |
+ calICalendar.ITEM_FILTER_TYPE_ALL;
+
+ await Promise.all(
+ aCalendars.map(async calendar => {
+ if (calendar.getProperty("suppressAlarms") && calendar.getProperty("disabled")) {
+ this.mLoadedCalendars[calendar.id] = true;
+ this.mObservers.notify("onAlarmsLoaded", [calendar]);
+ return;
+ }
+
+ // Assuming that suppressAlarms does not change anymore until next refresh.
+ this.mLoadedCalendars[calendar.id] = false;
+
+ for await (let items of cal.iterate.streamValues(
+ calendar.getItems(filter, 0, aStart, aUntil)
+ )) {
+ await new Promise((resolve, reject) => {
+ cal.iterate.forEach(
+ items,
+ item => {
+ try {
+ this.removeAlarmsForItem(item);
+ this.addAlarmsForItem(item);
+ } catch (e) {
+ console.error("Promise was rejected: " + e);
+ this.mLoadedCalendars[calendar.id] = true;
+ this.mObservers.notify("onAlarmsLoaded", [calendar]);
+ reject(e);
+ }
+ },
+ () => {
+ resolve();
+ }
+ );
+ });
+ }
+
+ // The calendar has been loaded, so until now, onLoad events can be ignored.
+ this.mLoadedCalendars[calendar.id] = true;
+
+ // Notify observers that the alarms for the calendar have been loaded.
+ this.mObservers.notify("onAlarmsLoaded", [calendar]);
+ })
+ );
+ },
+
+ initAlarms(aCalendars) {
+ // Purge out all alarm timers belonging to the refreshed/loaded calendars
+ this.disposeCalendarTimers(aCalendars);
+
+ // Purge out all alarms from dialog belonging to the refreshed/loaded calendars
+ for (let calendar of aCalendars) {
+ this.mLoadedCalendars[calendar.id] = false;
+ this.mObservers.notify("onRemoveAlarmsByCalendar", [calendar]);
+ }
+
+ // Total refresh similar to startup. We're going to look for
+ // alarms +/- 1 month from now. If someone sets an alarm more than
+ // a month ahead of an event, or doesn't start Thunderbird
+ // for a month, they'll miss some, but that's a slim chance
+ let start = nowUTC();
+ let until = start.clone();
+ start.month -= Ci.calIAlarmService.MAX_SNOOZE_MONTHS;
+ until.month += Ci.calIAlarmService.MAX_SNOOZE_MONTHS;
+ this.findAlarms(aCalendars, start, until);
+ },
+
+ alarmFired(aItem, aAlarm) {
+ if (
+ !aItem.calendar.getProperty("suppressAlarms") &&
+ !aItem.calendar.getProperty("disabled") &&
+ aItem.getProperty("STATUS") != "CANCELLED"
+ ) {
+ this.mObservers.notify("onAlarm", [aItem, aAlarm]);
+ }
+ },
+
+ get isLoading() {
+ for (let calId in this.mLoadedCalendars) {
+ // we need to exclude calendars which failed to load explicitly to
+ // prevent the alaram dialog to stay opened after dismissing all
+ // alarms if there is a network calendar that failed to load
+ let currentStatus = cal.manager.getCalendarById(calId).getProperty("currentStatus");
+ if (!this.mLoadedCalendars[calId] && Components.isSuccessCode(currentStatus)) {
+ return true;
+ }
+ }
+ return false;
+ },
+};