zulip/static/js/notifications.js

748 lines
24 KiB
JavaScript

import $ from "jquery";
import render_compose_notification from "../templates/compose_notification.hbs";
import * as alert_words from "./alert_words";
import * as blueslip from "./blueslip";
import * as channel from "./channel";
import * as favicon from "./favicon";
import * as hash_util from "./hash_util";
import {$t} from "./i18n";
import * as message_lists from "./message_lists";
import * as message_store from "./message_store";
import * as muted_topics from "./muted_topics";
import * as narrow from "./narrow";
import * as narrow_state from "./narrow_state";
import * as navigate from "./navigate";
import {page_params} from "./page_params";
import * as people from "./people";
import {realm_user_settings_defaults} from "./realm_user_settings_defaults";
import * as settings_config from "./settings_config";
import * as spoilers from "./spoilers";
import * as stream_data from "./stream_data";
import * as stream_ui_updates from "./stream_ui_updates";
import * as ui from "./ui";
import * as unread from "./unread";
import * as unread_ops from "./unread_ops";
import {user_settings} from "./user_settings";
const notice_memory = new Map();
// When you start Zulip, window_focused should be true, but it might not be the
// case after a server-initiated reload.
let window_focused = document.hasFocus && document.hasFocus();
let NotificationAPI;
export function set_notification_api(n) {
NotificationAPI = n;
}
if (window.electron_bridge && window.electron_bridge.new_notification) {
class ElectronBridgeNotification extends EventTarget {
constructor(title, options) {
super();
Object.assign(
this,
window.electron_bridge.new_notification(title, options, (type, eventInit) =>
this.dispatchEvent(new Event(type, eventInit)),
),
);
}
static get permission() {
return Notification.permission;
}
static async requestPermission(callback) {
if (callback) {
callback(await Promise.resolve(Notification.permission));
}
return Notification.permission;
}
}
NotificationAPI = ElectronBridgeNotification;
} else if (window.Notification) {
NotificationAPI = window.Notification;
}
export function get_notifications() {
return notice_memory;
}
export function initialize() {
$(window)
.on("focus", () => {
window_focused = true;
for (const notice_mem_entry of notice_memory.values()) {
notice_mem_entry.obj.close();
}
notice_memory.clear();
// Update many places on the DOM to reflect unread
// counts.
unread_ops.process_visible();
})
.on("blur", () => {
window_focused = false;
});
update_notification_sound_source($("#user-notification-sound-audio"), user_settings);
update_notification_sound_source(
$("#realm-default-notification-sound-audio"),
realm_user_settings_defaults,
);
}
export function update_notification_sound_source(container_elem, settings_object) {
const notification_sound = settings_object.notification_sound;
const audio_file_without_extension = "/static/audio/notification_sounds/" + notification_sound;
container_elem
.find(".notification-sound-source-ogg")
.attr("src", `${audio_file_without_extension}.ogg`);
container_elem
.find(".notification-sound-source-mp3")
.attr("src", `${audio_file_without_extension}.mp3`);
if (notification_sound !== "none") {
// Load it so that it is ready to be played; without this the old sound
// is played.
container_elem[0].load();
}
}
export function permission_state() {
if (NotificationAPI === undefined) {
// act like notifications are blocked if they do not have access to
// the notification API.
return "denied";
}
return NotificationAPI.permission;
}
let unread_count = 0;
let pm_count = 0;
export function redraw_title() {
// Update window title to reflect unread messages in current view
const new_title =
(unread_count ? "(" + unread_count + ") " : "") +
narrow.narrow_title +
" - " +
page_params.realm_name +
" - " +
"Zulip";
document.title = new_title;
}
export function update_unread_counts(new_unread_count, new_pm_count) {
if (new_unread_count === unread_count && new_pm_count === pm_count) {
return;
}
unread_count = new_unread_count;
pm_count = new_pm_count;
// Indicate the message count in the favicon
favicon.update_favicon(unread_count, pm_count);
// 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", unread_count);
}
// TODO: Add a `window.electron_bridge.updatePMCount(new_pm_count);` call?
redraw_title();
}
export function is_window_focused() {
return window_focused;
}
export function notify_above_composebox(
note,
link_class,
above_composebox_narrow_url,
link_msg_id,
link_text,
) {
const notification_html = $(
render_compose_notification({
note,
link_class,
above_composebox_narrow_url,
link_msg_id,
link_text,
}),
);
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 referred 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.
if (window.electron_bridge.set_send_notification_reply_message_supported !== undefined) {
window.electron_bridge.set_send_notification_reply_message_supported(true);
}
window.electron_bridge.on_event("send_notification_reply_message", (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,
message_id,
error,
});
}
channel.post({
url: "/json/messages",
data,
success,
error,
});
});
}
export 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;
// Convert the content to plain text, replacing emoji with their alt text
content = $("<div/>").html(message.content);
ui.replace_emoji_with_text(content);
spoilers.hide_spoilers_in_notification(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" || message.type === "test-notification") {
if (
user_settings.pm_content_in_desktop_notifications !== undefined &&
!user_settings.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.slice(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;
notification_object.close();
}
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)";
}
}
if (message.type === "stream") {
title += " (to " + message.stream + " > " + topic + ")";
}
if (notification.desktop_notify) {
const icon_url = people.small_avatar_url(message);
notification_object = new NotificationAPI(title, {
icon: icon_url,
body: content,
tag: message.id,
});
notice_memory.set(key, {
obj: notification_object,
msg_count,
message_id: message.id,
});
if (typeof notification_object.addEventListener === "function") {
// Sadly, some third-party Electron apps like Franz/Ferdi
// misimplement the Notification API not inheriting from
// EventTarget. This results in addEventListener being
// unavailable for them.
notification_object.addEventListener("click", () => {
notification_object.close();
if (message.type !== "test-notification") {
narrow.by_topic(message.id, {trigger: "notification"});
}
window.focus();
});
notification_object.addEventListener("close", () => {
notice_memory.delete(key);
});
}
}
}
export function close_notification(message) {
for (const [key, notice_mem_entry] of notice_memory) {
if (notice_mem_entry.message_id === message.id) {
notice_mem_entry.obj.close();
notice_memory.delete(key);
}
}
}
export function message_is_notifiable(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" &&
muted_topics.is_topic_muted(message.stream_id, message.topic)
) {
return false;
}
// Everything else is on the table; next filter based on notification
// settings.
return true;
}
export function should_send_desktop_notification(message) {
// Always notify for testing notifications.
if (message.type === "test-notification") {
return true;
}
// For streams, send if desktop notifications are enabled for all
// message on this stream.
if (
message.type === "stream" &&
stream_data.receives_notifications(message.stream_id, "desktop_notifications")
) {
return true;
}
// enable_desktop_notifications determines whether we pop up a
// notification for PMs/mentions/alerts
if (!user_settings.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_id, "wildcard_mentions_notify")
) {
return true;
}
return false;
}
export function should_send_audible_notification(message) {
// If `None` is selected as the notification sound, never send
// audible notifications regardless of other configuration.
if (user_settings.notification_sound === "none") {
return false;
}
// For streams, ding if sounds are enabled for all messages on
// this stream.
if (
message.type === "stream" &&
stream_data.receives_notifications(message.stream_id, "audible_notifications")
) {
return true;
}
// enable_sounds determines whether we ding for PMs/mentions/alerts
if (!user_settings.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" || message.type === "test-notification") {
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_id, "wildcard_mentions_notify")
) {
return true;
}
return false;
}
export function granted_desktop_notifications_permission() {
return NotificationAPI && NotificationAPI.permission === "granted";
}
export function request_desktop_notifications_permission() {
if (NotificationAPI) {
NotificationAPI.requestPermission();
}
}
export function received_messages(messages) {
for (const message of messages) {
if (!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 (should_send_desktop_notification(message)) {
process_notification({
message,
desktop_notify: granted_desktop_notifications_permission(),
});
}
if (should_send_audible_notification(message)) {
$("#user-notification-sound-audio")[0].play();
}
}
}
export function send_test_notification(content) {
received_messages([
{
id: Math.random(),
type: "test-notification",
sender_email: "notification-bot@zulip.com",
sender_full_name: "Notification Bot",
display_reply_to: "Notification Bot",
content,
unread: true,
},
]);
}
// Note that this returns values that are not HTML-escaped, for use in
// handlebars templates that will do further escaping.
function get_message_header(message) {
if (message.type === "stream") {
return message.stream + " > " + message.topic;
}
if (message.display_recipient.length > 2) {
return $t(
{defaultMessage: "group private messages with {recipient}"},
{recipient: message.display_reply_to},
);
}
if (people.is_current_user(message.reply_to)) {
return $t({defaultMessage: "private messages with yourself"});
}
return $t(
{defaultMessage: "private messages with {recipient}"},
{recipient: message.display_reply_to},
);
}
export function get_local_notify_mix_reason(message) {
const row = message_lists.current.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 undefined;
}
if (
message.type === "stream" &&
muted_topics.is_topic_muted(message.stream_id, message.topic)
) {
return $t({defaultMessage: "Sent! Your message was sent to a topic you have muted."});
}
if (message.type === "stream" && stream_data.is_muted(message.stream_id)) {
return $t({defaultMessage: "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
const current_filter = narrow_state.filter();
if (
current_filter &&
current_filter.can_apply_locally() &&
!current_filter.predicate()(message)
) {
return $t({defaultMessage: "Sent! Your message is outside your current narrow."});
}
return undefined;
}
export function notify_local_mixes(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 = get_local_notify_mix_reason(message);
const above_composebox_narrow_url = get_above_composebox_narrow_url(message);
if (!reason) {
if (need_user_to_scroll) {
reason = $t({defaultMessage: "Sent! Scroll down to view your message."});
notify_above_composebox(reason, "", above_composebox_narrow_url, null, "");
setTimeout(() => {
$("#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 = $t(
{defaultMessage: "Narrow to {message_recipient}"},
{message_recipient: get_message_header(message)},
);
notify_above_composebox(
reason,
link_class,
above_composebox_narrow_url,
link_msg_id,
link_text,
);
}
}
function get_above_composebox_narrow_url(message) {
let above_composebox_narrow_url;
if (message.type === "stream") {
above_composebox_narrow_url = hash_util.by_stream_topic_uri(
message.stream_id,
message.topic,
);
} else {
above_composebox_narrow_url = message.pm_with_url;
}
return above_composebox_narrow_url;
}
// for callback when we have to check with the server if a message should be in
// the message_lists.current (!can_apply_locally; a.k.a. "a search").
export function notify_messages_outside_current_search(messages) {
for (const message of messages) {
if (!people.is_current_user(message.sender_email)) {
continue;
}
const above_composebox_narrow_url = get_above_composebox_narrow_url(message);
const link_text = $t(
{defaultMessage: "Narrow to {message_recipient}"},
{message_recipient: get_message_header(message)},
);
notify_above_composebox(
$t({defaultMessage: "Sent! Your recent message is outside the current search."}),
"compose_notification_narrow_by_topic",
above_composebox_narrow_url,
message.id,
link_text,
);
}
}
export function clear_compose_notifications() {
$("#out-of-view-notification").empty();
$("#out-of-view-notification").stop(true, true);
$("#out-of-view-notification").hide();
}
export function reify_message_id(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);
}
}
}
export function register_click_handlers() {
$("#out-of-view-notification").on("click", ".compose_notification_narrow_by_topic", (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", (e) => {
const message_id = $(e.currentTarget).data("message-id");
message_lists.current.select_id(message_id);
navigate.scroll_to_selected();
e.stopPropagation();
e.preventDefault();
});
$("#out-of-view-notification").on("click", ".out-of-view-notification-close", (e) => {
clear_compose_notifications();
e.stopPropagation();
e.preventDefault();
});
}
export function handle_global_notification_updates(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_config.all_notification_settings.includes(notification_name)) {
user_settings[notification_name] = setting;
}
if (settings_config.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($("#user-notification-sound-audio"), user_settings);
}
}