mirror of https://github.com/zulip/zulip.git
772 lines
26 KiB
JavaScript
772 lines
26 KiB
JavaScript
const render_compose_notification = require('../templates/compose_notification.hbs');
|
|
const render_notification = require('../templates/notification.hbs');
|
|
|
|
const notice_memory = new Map();
|
|
|
|
// When you start Zulip, window_has_focus should be true, but it might not be the
|
|
// case after a server-initiated reload.
|
|
let window_has_focus = document.hasFocus && document.hasFocus();
|
|
|
|
let supports_sound;
|
|
|
|
const unread_pms_favicon = '/static/images/favicon/favicon-pms.png';
|
|
let current_favicon;
|
|
let previous_favicon;
|
|
let flashing = false;
|
|
|
|
let notifications_api;
|
|
|
|
exports.set_notification_api = function (n) {
|
|
notifications_api = n;
|
|
};
|
|
|
|
if (window.webkitNotifications) {
|
|
notifications_api = window.webkitNotifications;
|
|
} else if (window.Notification) {
|
|
// Build a shim to the new notification API
|
|
notifications_api = {
|
|
checkPermission: function checkPermission() {
|
|
if (window.Notification.permission === 'granted') {
|
|
return 0;
|
|
}
|
|
return 2;
|
|
},
|
|
requestPermission: window.Notification.requestPermission,
|
|
createNotification: function createNotification(icon, title, content, tag) {
|
|
const notification_object = new window.Notification(title, {icon: icon,
|
|
body: content,
|
|
tag: tag});
|
|
notification_object.show = function () {};
|
|
notification_object.cancel = function () { notification_object.close(); };
|
|
return notification_object;
|
|
},
|
|
};
|
|
}
|
|
|
|
function cancel_notification_object(notification_object) {
|
|
// We must remove the .onclose so that it does not trigger on .cancel
|
|
notification_object.onclose = function () {};
|
|
notification_object.onclick = function () {};
|
|
notification_object.cancel();
|
|
}
|
|
|
|
exports.get_notifications = function () {
|
|
return notice_memory;
|
|
};
|
|
|
|
function get_audio_file_path(audio_element, audio_file_without_extension) {
|
|
if (audio_element.canPlayType('audio/ogg; codecs="vorbis"')) {
|
|
return audio_file_without_extension + ".ogg";
|
|
}
|
|
|
|
return audio_file_without_extension + ".mp3";
|
|
}
|
|
|
|
exports.initialize = function () {
|
|
$(window).focus(function () {
|
|
window_has_focus = true;
|
|
|
|
for (const notice_mem_entry of notice_memory.values()) {
|
|
cancel_notification_object(notice_mem_entry.obj);
|
|
}
|
|
notice_memory.clear();
|
|
|
|
// Update many places on the DOM to reflect unread
|
|
// counts.
|
|
unread_ops.process_visible();
|
|
|
|
}).blur(function () {
|
|
window_has_focus = false;
|
|
});
|
|
|
|
const audio = $("<audio>");
|
|
if (audio[0].canPlayType === undefined) {
|
|
supports_sound = false;
|
|
} else {
|
|
supports_sound = true;
|
|
|
|
$("#notifications-area").append(audio);
|
|
audio.append($("<source>").attr("loop", "yes"));
|
|
const source = $("#notifications-area audio source");
|
|
|
|
if (audio[0].canPlayType('audio/ogg; codecs="vorbis"')) {
|
|
source.attr("type", "audio/ogg");
|
|
} else {
|
|
source.attr("type", "audio/mpeg");
|
|
}
|
|
|
|
const audio_file_without_extension
|
|
= "/static/audio/notification_sounds/" + page_params.notification_sound;
|
|
source.attr("src", get_audio_file_path(audio[0], audio_file_without_extension));
|
|
}
|
|
};
|
|
|
|
function update_notification_sound_source() {
|
|
// Simplified version of the source creation in `exports.initialize`, for
|
|
// updating the source instead of creating it for the first time.
|
|
const audio = $("#notifications-area audio");
|
|
const source = $("#notifications-area audio source");
|
|
const audio_file_without_extension
|
|
= "/static/audio/notification_sounds/" + page_params.notification_sound;
|
|
source.attr("src", get_audio_file_path(audio[0], audio_file_without_extension));
|
|
|
|
// Load it so that it is ready to be played; without this the old sound
|
|
// is played.
|
|
$("#notifications-area").find("audio")[0].load();
|
|
}
|
|
|
|
exports.permission_state = function () {
|
|
if (window.Notification === undefined) {
|
|
// act like notifications are blocked if they do not have access to
|
|
// the notification API.
|
|
return "denied";
|
|
}
|
|
return window.Notification.permission;
|
|
};
|
|
|
|
let new_message_count = 0;
|
|
|
|
exports.update_title_count = function (count) {
|
|
new_message_count = count;
|
|
exports.redraw_title();
|
|
};
|
|
|
|
exports.redraw_title = function () {
|
|
// Update window title and favicon to reflect unread messages in current view
|
|
let n;
|
|
|
|
const new_title = (new_message_count ? "(" + new_message_count + ") " : "")
|
|
+ narrow.narrow_title + " - "
|
|
+ page_params.realm_name + " - "
|
|
+ "Zulip";
|
|
|
|
if (document.title === new_title) {
|
|
return;
|
|
}
|
|
|
|
document.title = new_title;
|
|
|
|
// IE doesn't support PNG favicons, *shrug*
|
|
if (!/msie/i.test(navigator.userAgent)) {
|
|
// Indicate the message count in the favicon
|
|
if (new_message_count) {
|
|
// Make sure we're working with a number, as a defensive programming
|
|
// measure. And we don't have images above 99, so display those as
|
|
// 'infinite'.
|
|
n = +new_message_count;
|
|
if (n > 99) {
|
|
n = 'infinite';
|
|
}
|
|
|
|
current_favicon = previous_favicon = '/static/images/favicon/favicon-' + n + '.png';
|
|
} else {
|
|
current_favicon = previous_favicon = '/static/favicon.ico?v=2';
|
|
}
|
|
favicon.set(current_favicon);
|
|
}
|
|
|
|
// Notify the current desktop app's UI about the new unread count.
|
|
if (window.electron_bridge !== undefined) {
|
|
window.electron_bridge.send_event('total_unread_count', new_message_count);
|
|
}
|
|
};
|
|
|
|
exports.show_history_limit_message = function () {
|
|
$(".top-messages-logo").hide();
|
|
$(".history-limited-box").show();
|
|
narrow.hide_empty_narrow_message();
|
|
};
|
|
|
|
exports.hide_history_limit_message = function () {
|
|
$(".top-messages-logo").show();
|
|
$(".history-limited-box").hide();
|
|
};
|
|
|
|
exports.hide_or_show_history_limit_message = function (msg_list) {
|
|
if (msg_list !== current_msg_list) {
|
|
return;
|
|
}
|
|
|
|
if (msg_list.fetch_status.history_limited()) {
|
|
exports.show_history_limit_message();
|
|
} else {
|
|
exports.hide_history_limit_message();
|
|
}
|
|
};
|
|
|
|
function flash_pms() {
|
|
// When you have unread PMs, toggle the favicon between the unread count and
|
|
// a special icon indicating that you have unread PMs.
|
|
if (unread.get_counts().private_message_count > 0) {
|
|
if (current_favicon === unread_pms_favicon) {
|
|
favicon.set(previous_favicon);
|
|
current_favicon = previous_favicon;
|
|
previous_favicon = unread_pms_favicon;
|
|
} else {
|
|
favicon.set(unread_pms_favicon);
|
|
previous_favicon = current_favicon;
|
|
current_favicon = unread_pms_favicon;
|
|
}
|
|
// Toggle every 2 seconds.
|
|
setTimeout(flash_pms, 2000);
|
|
} else {
|
|
flashing = false;
|
|
// You have no more unread PMs, so back to only showing the unread
|
|
// count.
|
|
favicon.set(current_favicon);
|
|
}
|
|
}
|
|
|
|
exports.update_pm_count = function () {
|
|
// TODO: Add a `window.electron_bridge.updatePMCount(new_pm_count);` call?
|
|
if (!flashing) {
|
|
flashing = true;
|
|
flash_pms();
|
|
}
|
|
};
|
|
|
|
exports.window_has_focus = function () {
|
|
return window_has_focus;
|
|
};
|
|
|
|
function in_browser_notify(message, title, content, raw_operators, opts) {
|
|
const notification_html = $(render_notification({
|
|
gravatar_url: people.small_avatar_url(message),
|
|
title: title,
|
|
content: content,
|
|
message_id: message.id,
|
|
}));
|
|
|
|
$(".top-right").notify({
|
|
message: {
|
|
html: notification_html,
|
|
},
|
|
fadeOut: {
|
|
enabled: true,
|
|
delay: 4000,
|
|
},
|
|
}).show();
|
|
|
|
$(".notification[data-message-id='" + message.id + "']").expectOne().data("narrow", {
|
|
raw_operators: raw_operators,
|
|
opts_notif: opts,
|
|
});
|
|
}
|
|
|
|
exports.notify_above_composebox = function (note, link_class, link_msg_id, link_text) {
|
|
const notification_html = $(render_compose_notification({
|
|
note: note,
|
|
link_class: link_class,
|
|
link_msg_id: link_msg_id,
|
|
link_text: link_text,
|
|
}));
|
|
exports.clear_compose_notifications();
|
|
$('#out-of-view-notification').append(notification_html);
|
|
$('#out-of-view-notification').show();
|
|
};
|
|
|
|
if (window.electron_bridge !== undefined) {
|
|
// The code below is for sending a message received from notification reply which
|
|
// is often refered to as inline reply feature. This is done so desktop app doesn't
|
|
// have to depend on channel.post for setting crsf_token and narrow.by_topic
|
|
// to narrow to the message being sent.
|
|
window.electron_bridge.send_notification_reply_message_supported = true;
|
|
window.electron_bridge.on_event('send_notification_reply_message', function (message_id, reply) {
|
|
const message = message_store.get(message_id);
|
|
const data = {
|
|
type: message.type,
|
|
content: reply,
|
|
to: message.type === 'private' ? message.reply_to : message.stream,
|
|
topic: message.topic,
|
|
};
|
|
|
|
function success() {
|
|
if (message.type === 'stream') {
|
|
narrow.by_topic(message_id, {trigger: 'desktop_notification_reply'});
|
|
} else {
|
|
narrow.by_recipient(message_id, {trigger: 'desktop_notification_reply'});
|
|
}
|
|
}
|
|
|
|
function error(error) {
|
|
window.electron_bridge.send_event('send_notification_reply_message_failed', {
|
|
data: data,
|
|
message_id: message_id,
|
|
error: error,
|
|
});
|
|
}
|
|
|
|
channel.post({
|
|
url: '/json/messages',
|
|
data: data,
|
|
success: success,
|
|
error: error,
|
|
});
|
|
});
|
|
}
|
|
|
|
function process_notification(notification) {
|
|
let i;
|
|
let notification_object;
|
|
let key;
|
|
let content;
|
|
let other_recipients;
|
|
const message = notification.message;
|
|
let title = message.sender_full_name;
|
|
let msg_count = 1;
|
|
let notification_source;
|
|
let raw_operators = [];
|
|
const opts = {trigger: "notification click"};
|
|
// Convert the content to plain text, replacing emoji with their alt text
|
|
content = $('<div/>').html(message.content);
|
|
ui.replace_emoji_with_text(content);
|
|
content = content.text();
|
|
|
|
const topic = message.topic;
|
|
|
|
if (message.is_me_message) {
|
|
content = message.sender_full_name + content.slice(3);
|
|
}
|
|
|
|
if (message.type === "private") {
|
|
if (page_params.pm_content_in_desktop_notifications !== undefined
|
|
&& !page_params.pm_content_in_desktop_notifications) {
|
|
content = "New private message from " + message.sender_full_name;
|
|
}
|
|
key = message.display_reply_to;
|
|
other_recipients = message.display_reply_to;
|
|
// 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 + ", ", "");
|
|
notification_source = 'pm';
|
|
} else {
|
|
key = message.sender_full_name + " to " +
|
|
message.stream + " > " + topic;
|
|
if (message.mentioned) {
|
|
notification_source = 'mention';
|
|
} else if (message.alerted) {
|
|
notification_source = 'alert';
|
|
} else {
|
|
notification_source = 'stream';
|
|
}
|
|
}
|
|
blueslip.debug("Desktop notification from source " + notification_source);
|
|
|
|
if (content.length > 150) {
|
|
// Truncate content at a word boundary
|
|
for (i = 150; i > 0; i -= 1) {
|
|
if (content[i] === ' ') {
|
|
break;
|
|
}
|
|
}
|
|
content = content.substring(0, i);
|
|
content += " [...]";
|
|
}
|
|
|
|
if (notice_memory.has(key)) {
|
|
msg_count = notice_memory.get(key).msg_count + 1;
|
|
title = msg_count + " messages from " + title;
|
|
notification_object = notice_memory.get(key).obj;
|
|
cancel_notification_object(notification_object);
|
|
}
|
|
|
|
if (message.type === "private") {
|
|
if (message.display_recipient.length > 2) {
|
|
// 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 + ")";
|
|
} else {
|
|
title += " (to you)";
|
|
}
|
|
|
|
raw_operators = [{operand: message.reply_to, operator: "pm-with"}];
|
|
}
|
|
|
|
if (message.type === "stream") {
|
|
title += " (to " + message.stream + " > " + topic + ")";
|
|
raw_operators = [
|
|
{operator: "stream", operand: message.stream},
|
|
{operator: "topic", operand: topic},
|
|
];
|
|
}
|
|
|
|
// Firefox on Ubuntu claims to do webkitNotifications but its notifications are terrible
|
|
if (notification.desktop_notify && /webkit/i.test(navigator.userAgent)) {
|
|
const icon_url = people.small_avatar_url(message);
|
|
notification_object =
|
|
notifications_api.createNotification(icon_url, title, content, message.id);
|
|
notice_memory.set(key, {
|
|
obj: notification_object,
|
|
msg_count: msg_count,
|
|
message_id: message.id,
|
|
});
|
|
notification_object.onclick = function () {
|
|
notification_object.cancel();
|
|
narrow.by_topic(message.id, {trigger: 'notification'});
|
|
window.focus();
|
|
};
|
|
notification_object.onclose = function () {
|
|
notice_memory.delete(key);
|
|
};
|
|
notification_object.show();
|
|
} else if (notification.desktop_notify && typeof Notification !== "undefined" && /mozilla/i.test(navigator.userAgent)) {
|
|
Notification.requestPermission(function (perm) {
|
|
if (perm === 'granted') {
|
|
notification_object = new Notification(title, {
|
|
body: content,
|
|
iconUrl: people.small_avatar_url(message),
|
|
tag: message.id,
|
|
});
|
|
notification_object.onclick = function () {
|
|
// We don't need to bring the browser window into focus explicitly
|
|
// by calling `window.focus()` as well as don't need to clear the
|
|
// notification since it is the default behavior in Firefox.
|
|
narrow.by_topic(message.id, {trigger: 'notification'});
|
|
};
|
|
} else {
|
|
in_browser_notify(message, title, content, raw_operators, opts);
|
|
}
|
|
});
|
|
} else {
|
|
in_browser_notify(message, title, content, raw_operators, opts);
|
|
}
|
|
}
|
|
|
|
exports.process_notification = process_notification;
|
|
|
|
exports.close_notification = function (message) {
|
|
for (const [key, notice_mem_entry] of notice_memory) {
|
|
if (notice_mem_entry.message_id === message.id) {
|
|
cancel_notification_object(notice_mem_entry.obj);
|
|
notice_memory.delete(key);
|
|
}
|
|
}
|
|
};
|
|
|
|
exports.message_is_notifiable = function (message) {
|
|
// Independent of the user's notification settings, are there
|
|
// properties of the message that unconditionally mean we
|
|
// shouldn't notify about it.
|
|
|
|
if (message.sent_by_me) {
|
|
return false;
|
|
}
|
|
|
|
// If a message is edited multiple times, we want to err on the side of
|
|
// not spamming notifications.
|
|
if (message.notification_sent) {
|
|
return false;
|
|
}
|
|
|
|
// @-<username> mentions take precedence over muted-ness. Note
|
|
// that @all mentions are still suppressed by muting.
|
|
if (message.mentioned_me_directly) {
|
|
return true;
|
|
}
|
|
|
|
// Messages to muted streams that don't mention us specifically
|
|
// are not notifiable.
|
|
if (message.type === "stream" &&
|
|
stream_data.is_muted(message.stream_id)) {
|
|
return false;
|
|
}
|
|
|
|
if (message.type === "stream" &&
|
|
muting.is_topic_muted(message.stream_id, message.topic)) {
|
|
return false;
|
|
}
|
|
|
|
// Everything else is on the table; next filter based on notification
|
|
// settings.
|
|
return true;
|
|
};
|
|
|
|
exports.should_send_desktop_notification = function (message) {
|
|
// For streams, send if desktop notifications are enabled for all
|
|
// message on this stream.
|
|
if (message.type === "stream" &&
|
|
stream_data.receives_notifications(message.stream, "desktop_notifications")) {
|
|
return true;
|
|
}
|
|
|
|
// enable_desktop_notifications determines whether we pop up a
|
|
// notification for PMs/mentions/alerts
|
|
if (!page_params.enable_desktop_notifications) {
|
|
return false;
|
|
}
|
|
|
|
// And then we need to check if the message is a PM, mention,
|
|
// wildcard mention with wildcard_mentions_notify, or alert.
|
|
if (message.type === "private") {
|
|
return true;
|
|
}
|
|
|
|
if (alert_words.notifies(message)) {
|
|
return true;
|
|
}
|
|
|
|
if (message.mentioned_me_directly) {
|
|
return true;
|
|
}
|
|
|
|
// wildcard mentions
|
|
if (message.mentioned &&
|
|
stream_data.receives_notifications(message.stream, "wildcard_mentions_notify")) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
exports.should_send_audible_notification = function (message) {
|
|
// For streams, ding if sounds are enabled for all messages on
|
|
// this stream.
|
|
if (message.type === "stream" &&
|
|
stream_data.receives_notifications(message.stream, "audible_notifications")) {
|
|
return true;
|
|
}
|
|
|
|
// enable_sounds determines whether we ding for PMs/mentions/alerts
|
|
if (!page_params.enable_sounds) {
|
|
return false;
|
|
}
|
|
|
|
// And then we need to check if the message is a PM, mention,
|
|
// wildcard mention with wildcard_mentions_notify, or alert.
|
|
if (message.type === "private") {
|
|
return true;
|
|
}
|
|
|
|
if (alert_words.notifies(message)) {
|
|
return true;
|
|
}
|
|
|
|
if (message.mentioned_me_directly) {
|
|
return true;
|
|
}
|
|
|
|
// wildcard mentions
|
|
if (message.mentioned &&
|
|
stream_data.receives_notifications(message.stream, "wildcard_mentions_notify")) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
exports.granted_desktop_notifications_permission = function () {
|
|
return notifications_api &&
|
|
// 0 is PERMISSION_ALLOWED
|
|
notifications_api.checkPermission() === 0;
|
|
};
|
|
|
|
|
|
exports.request_desktop_notifications_permission = function () {
|
|
if (notifications_api) {
|
|
return notifications_api.requestPermission();
|
|
}
|
|
};
|
|
|
|
exports.received_messages = function (messages) {
|
|
for (const message of messages) {
|
|
if (!exports.message_is_notifiable(message)) {
|
|
continue;
|
|
}
|
|
if (!unread.message_unread(message)) {
|
|
// The message is already read; Zulip is currently in focus.
|
|
continue;
|
|
}
|
|
|
|
message.notification_sent = true;
|
|
|
|
if (exports.should_send_desktop_notification(message)) {
|
|
process_notification({
|
|
message: message,
|
|
desktop_notify: exports.granted_desktop_notifications_permission(),
|
|
});
|
|
}
|
|
if (exports.should_send_audible_notification(message) && supports_sound) {
|
|
$("#notifications-area").find("audio")[0].play();
|
|
}
|
|
}
|
|
};
|
|
|
|
function get_message_header(message) {
|
|
if (message.type === "stream") {
|
|
return message.stream + " > " + message.topic;
|
|
}
|
|
if (message.display_recipient.length > 2) {
|
|
return i18n.t("group private messages with __recipient__",
|
|
{recipient: message.display_reply_to});
|
|
}
|
|
if (people.is_current_user(message.reply_to)) {
|
|
return i18n.t("private messages with yourself");
|
|
}
|
|
return i18n.t("private messages with __recipient__",
|
|
{recipient: message.display_reply_to});
|
|
}
|
|
|
|
exports.get_local_notify_mix_reason = function (message) {
|
|
const row = current_msg_list.get_row(message.id);
|
|
if (row.length > 0) {
|
|
// If our message is in the current message list, we do
|
|
// not have a mix, so we are happy.
|
|
return;
|
|
}
|
|
|
|
if (message.type === "stream" && muting.is_topic_muted(message.stream_id, message.topic)) {
|
|
return i18n.t("Sent! Your message was sent to a topic you have muted.");
|
|
}
|
|
|
|
if (message.type === "stream" && stream_data.is_muted(message.stream_id)) {
|
|
return i18n.t("Sent! Your message was sent to a stream you have muted.");
|
|
}
|
|
|
|
// offscreen because it is outside narrow
|
|
// we can only look for these on non-search (can_apply_locally) messages
|
|
// see also: exports.notify_messages_outside_current_search
|
|
return i18n.t("Sent! Your message is outside your current narrow.");
|
|
};
|
|
|
|
exports.notify_local_mixes = function (messages, need_user_to_scroll) {
|
|
/*
|
|
This code should only be called when we are displaying
|
|
messages sent by current client. It notifies users that
|
|
their messages aren't actually in the view that they
|
|
composed to.
|
|
|
|
This code is called after we insert messages into our
|
|
message list widgets. All of the conditions here are
|
|
checkable locally, so we may want to execute this code
|
|
earlier in the codepath at some point and possibly punt
|
|
on local rendering.
|
|
|
|
Possible cleanup: Arguably, we should call this function
|
|
unconditionally and just check if message.local_id is in
|
|
sent_messages.messages here.
|
|
*/
|
|
|
|
for (const message of messages) {
|
|
if (!people.is_my_user_id(message.sender_id)) {
|
|
// This can happen if the client is offline for a while
|
|
// around the time this client sends a message; see the
|
|
// caller of message_events.insert_new_messages.
|
|
blueslip.info('Slightly unexpected: A message not sent by us batches with those that were.');
|
|
continue;
|
|
}
|
|
|
|
let reason = exports.get_local_notify_mix_reason(message);
|
|
|
|
if (!reason) {
|
|
if (need_user_to_scroll) {
|
|
reason = i18n.t("Sent! Scroll down to view your message.");
|
|
exports.notify_above_composebox(reason, "", null, "");
|
|
setTimeout(function () {
|
|
$('#out-of-view-notification').hide();
|
|
}, 3000);
|
|
}
|
|
|
|
// This is the HAPPY PATH--for most messages we do nothing
|
|
// other than maybe sending the above message.
|
|
continue;
|
|
}
|
|
|
|
const link_msg_id = message.id;
|
|
const link_class = "compose_notification_narrow_by_topic";
|
|
const link_text = i18n.t("Narrow to __- message_recipient__",
|
|
{message_recipient: get_message_header(message)});
|
|
|
|
exports.notify_above_composebox(reason, link_class, link_msg_id, link_text);
|
|
}
|
|
};
|
|
|
|
// for callback when we have to check with the server if a message should be in
|
|
// the current_msg_list (!can_apply_locally; a.k.a. "a search").
|
|
exports.notify_messages_outside_current_search = function (messages) {
|
|
for (const message of messages) {
|
|
if (!people.is_current_user(message.sender_email)) {
|
|
continue;
|
|
}
|
|
const link_text = i18n.t("Narrow to __- message_recipient__",
|
|
{message_recipient: get_message_header(message)});
|
|
exports.notify_above_composebox(i18n.t("Sent! Your recent message is outside the current search."),
|
|
"compose_notification_narrow_by_topic",
|
|
message.id,
|
|
link_text);
|
|
}
|
|
};
|
|
|
|
exports.clear_compose_notifications = function () {
|
|
$('#out-of-view-notification').empty();
|
|
$('#out-of-view-notification').stop(true, true);
|
|
$('#out-of-view-notification').hide();
|
|
};
|
|
|
|
exports.reify_message_id = function (opts) {
|
|
const old_id = opts.old_id;
|
|
const new_id = opts.new_id;
|
|
|
|
// If a message ID that we're currently storing (as a link) has changed,
|
|
// update that link as well
|
|
for (const e of $('#out-of-view-notification a')) {
|
|
const elem = $(e);
|
|
const message_id = elem.data('message-id');
|
|
|
|
if (message_id === old_id) {
|
|
elem.data('message-id', new_id);
|
|
}
|
|
}
|
|
};
|
|
|
|
exports.register_click_handlers = function () {
|
|
$('#out-of-view-notification').on('click', '.compose_notification_narrow_by_topic', function (e) {
|
|
const message_id = $(e.currentTarget).data('message-id');
|
|
narrow.by_topic(message_id, {trigger: 'compose_notification'});
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
});
|
|
$('#out-of-view-notification').on('click', '.compose_notification_scroll_to_message', function (e) {
|
|
const message_id = $(e.currentTarget).data('message-id');
|
|
current_msg_list.select_id(message_id);
|
|
navigate.scroll_to_selected();
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
});
|
|
$('#out-of-view-notification').on('click', '.out-of-view-notification-close', function (e) {
|
|
exports.clear_compose_notifications();
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
});
|
|
};
|
|
|
|
exports.handle_global_notification_updates = function (notification_name, setting) {
|
|
// Update the global settings checked when determining if we should notify
|
|
// for a given message. These settings do not affect whether or not a
|
|
// particular stream should receive notifications.
|
|
if (settings_notifications.all_notification_settings_labels.includes(notification_name)) {
|
|
page_params[notification_name] = setting;
|
|
}
|
|
|
|
if (
|
|
settings_notifications.all_notifications.settings.stream_notification_settings.includes(
|
|
notification_name
|
|
)
|
|
) {
|
|
notification_name = notification_name.replace("enable_stream_", "");
|
|
stream_ui_updates.update_notification_setting_checkbox(notification_name);
|
|
}
|
|
|
|
if (notification_name === "notification_sound") {
|
|
// Change the sound source with the new page `notification_sound`.
|
|
update_notification_sound_source();
|
|
}
|
|
};
|
|
|
|
window.notifications = exports;
|