1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
|
# Copyright (C) 2008 Jimmy Do <jimmydo@users.sourceforge.net>
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
import gobject
import pynotify
class Notifier(object):
_NOTIFICATION_REDISPLAY_INTERVAL_SECONDS = 60
def __init__(self, app_name, icon, attach):
self._icon = icon
self._attach = attach
self._notify = None
self._handler_id = None
self._timeout_id = None
if not pynotify.is_initted():
pynotify.init(app_name)
def begin(self, summary, body, get_reminder_message_func):
# NOTE: This callback wrapper is to workaround an API-breaking change present in
# the version of libmatenotify used by Fedora 10. The API break adds an additional
# 'reason' parameter to the callback signature. This was fixed before
# the latest official release of libmatenotify, but it looks like Fedora 10
# is using an unofficial libmatenotify build that still contains the API-breaking change.
def closed_callback_wrapper(notification, reason_UNUSED=None):
self._on_notification_closed(notification, get_reminder_message_func)
self.end()
self._notify = pynotify.Notification(summary, body, self._icon)
self._handler_id = self._notify.connect('closed', closed_callback_wrapper)
self._notify.show()
def end(self):
if self._notify is not None:
if self._timeout_id is not None:
gobject.source_remove(self._timeout_id)
self._timeout_id = None
self._notify.disconnect(self._handler_id)
self._handler_id = None
try:
self._notify.close()
except gobject.GError:
# Throws a GError exception if the notification bubble has already been closed.
# Ignore the exception.
pass
self._notify = None
def _on_notification_closed(self, notification, get_reminder_message_func):
self._timeout_id = gobject.timeout_add(Notifier._NOTIFICATION_REDISPLAY_INTERVAL_SECONDS * 1000,
self._on_notification_redisplay_timeout,
get_reminder_message_func)
def _on_notification_redisplay_timeout(self, get_reminder_message_func):
message = get_reminder_message_func()
self._notify.props.body = message
self._notify.show()
self._timeout_id = None
return False
|