Add webkit desktop notifications

Partially addresses Trac #31

(imported from commit f1f04b5c50bf9f33b69c90926fd67015cd2dc219)
This commit is contained in:
Jeff Arnold 2012-11-23 17:53:38 -05:00
parent a6ec875268
commit 4cb0c6225e
6 changed files with 126 additions and 0 deletions

View File

@ -48,6 +48,7 @@
<script type="text/javascript" src="{{ static_hidden }}js/search.js"></script> <script type="text/javascript" src="{{ static_hidden }}js/search.js"></script>
<script type="text/javascript" src="{{ static_hidden }}js/composebox_typeahead.js"></script> <script type="text/javascript" src="{{ static_hidden }}js/composebox_typeahead.js"></script>
<script type="text/javascript" src="{{ static_hidden }}js/hotkey.js"></script> <script type="text/javascript" src="{{ static_hidden }}js/hotkey.js"></script>
<script type="text/javascript" src="{{ static_hidden }}js/notifications.js"></script>
<script type="text/javascript" src="{{ static_hidden }}js/zephyr.js"></script> <script type="text/javascript" src="{{ static_hidden }}js/zephyr.js"></script>
{% if debug %} {% if debug %}
@ -62,6 +63,7 @@ var initial_pointer = {{ user_profile.pointer }};
var email = "{{ user_profile.user.email|escapejs }}"; var email = "{{ user_profile.user.email|escapejs }}";
var have_initial_messages = {{ have_initial_messages|escapejs }}; var have_initial_messages = {{ have_initial_messages|escapejs }};
var desktop_notifications_enabled = "{{ user_profile.enable_desktop_notifications|escapejs }}" === "True";
var stream_list = [ var stream_list = [
{% for stream in streams %} {% for stream in streams %}

View File

@ -7,6 +7,7 @@ var globals =
// index.html // index.html
+ ' initial_pointer email stream_list people_list have_initial_messages' + ' initial_pointer email stream_list people_list have_initial_messages'
+ ' desktop_notifications_enabled'
// common.js // common.js
+ ' status_classes' + ' status_classes'
@ -41,6 +42,9 @@ var globals =
// typeahead_helper.js // typeahead_helper.js
+ ' typeahead_helper' + ' typeahead_helper'
// notifications.js
+ ' notifications'
// ui.js // ui.js
+ ' ui' + ' ui'

View File

@ -0,0 +1 @@
../../../static/js/notifications.js

View File

@ -0,0 +1,113 @@
var notifications = (function () {
var exports = {};
var window_has_focus = true;
exports.initialize = function () {
if (!window.webkitNotifications) {
return;
}
$(window).focus(function () {
window_has_focus = true;
}).blur(function () {
window_has_focus = false;
});
$(document).click(function () {
if (!desktop_notifications_enabled) {
return;
}
if (window.webkitNotifications.checkPermission() !== 0) { // 0 is PERMISSION_ALLOWED
window.webkitNotifications.requestPermission();
}
});
};
function gravatar_url(message) {
return "https://secure.gravatar.com/avatar/" + message.gravatar_hash +
"?d=identicon&s=30?stamp=" + ui.get_gravatar_stamp();
}
var notice_memory = {};
function process_message(message) {
var i, notification_object;
var key = message.display_reply_to;
var title = message.sender_full_name;
var content = $('<div/>').html(message.content).text();
var other_recipients = message.display_reply_to;
var msg_count = 1;
// Remove the sender from the list of other recipients
other_recipients = other_recipients.replace(", " + message.sender_full_name, "");
other_recipients = other_recipients.replace(message.sender_full_name + ", ", "");
if (content.length > 150) {
// Truncate content at a word boundary
for (i = 150; i > 0; i--) {
if (content[i] === ' ') {
break;
}
}
content = content.substring(0, i);
content += " [...]";
}
if (notice_memory[key] !== undefined) {
msg_count = notice_memory[key].msg_count + 1;
title = msg_count + " messages from " + title;
notification_object = notice_memory[key].obj;
// We must remove the .onclose so that it does not trigger on .cancel
notification_object.onclose = function () {};
notification_object.onclick = function () {};
notification_object.cancel();
}
if (message.type === "huddle") {
// If the message has too many recipients to list them all...
if (content.length + title.length + other_recipients.length > 230) {
// Then count how many people are in the conversation and summarize
// by saying the conversation is with "you and [number] other people"
other_recipients = other_recipients.replace(/[^,]/g, "").length +
" other people";
}
title += " (to you and " + other_recipients + ")";
}
notice_memory[key] = {
obj: window.webkitNotifications.createNotification(
gravatar_url(message), title, content),
msg_count: msg_count
};
notification_object = notice_memory[key].obj;
notification_object.onclick = function () {
notification_object.cancel();
delete notice_memory[key];
};
notification_object.onclose = function () {
delete notice_memory[key];
};
notification_object.show();
}
exports.received_messages = function (messages) {
var i;
if (!window.webkitNotifications ||
!desktop_notifications_enabled ||
window_has_focus) {
return;
}
$.each(messages, function (index, message) {
if (message.sender_email !== email &&
(message.type === "personal" || message.type === "huddle")) {
process_message(message);
}
});
};
return exports;
}());

View File

@ -412,6 +412,10 @@ $(function () {
} }
update_gravatars(); update_gravatars();
if (result.enable_desktop_notifications !== undefined) {
desktop_notifications_enabled = result.enable_desktop_notifications;
}
settings_status.removeClass(status_classes) settings_status.removeClass(status_classes)
.addClass('alert-success') .addClass('alert-success')
.text(message).stop(true).fadeTo(0,1); .text(message).stop(true).fadeTo(0,1);
@ -473,6 +477,7 @@ $(function () {
composebox_typeahead.initialize(); composebox_typeahead.initialize();
search.initialize(); search.initialize();
notifications.initialize();
$("body").bind('click', function() { $("body").bind('click', function() {
ui.hide_userinfo_popover(); ui.hide_userinfo_popover();

View File

@ -594,6 +594,7 @@ function get_updates(options) {
if (data.messages.length !== 0) { if (data.messages.length !== 0) {
add_messages(data.messages, true); add_messages(data.messages, true);
notifications.received_messages(data.messages);
} }
if (data.zephyr_mirror_active === false) { if (data.zephyr_mirror_active === false) {