DEV: use service worker for chat sound (#29388)

This change makes use of service workers to determine if we should play chat sounds in the current browser tab. Since users can have multiple tabs open, we currently attempt to play sound across all active tabs.

With this change we iterate over all clients and check if client.focused is true (ie. the current tab/window we have open), if so we allow playing the audio in the current tab and for all other hidden tabs/windows we return false.

---------

Co-authored-by: Bianca Nenciu <nbianca@users.noreply.github.com>
This commit is contained in:
David Battersby 2024-10-29 13:15:53 +04:00 committed by GitHub
parent f902e0fdd7
commit 7bcd46b87d
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 84 additions and 25 deletions

View File

@ -769,3 +769,19 @@ export function cleanNullQueryParams(params) {
export function getElement(node) { export function getElement(node) {
return node.nodeType === Node.TEXT_NODE ? node.parentElement : node; return node.nodeType === Node.TEXT_NODE ? node.parentElement : node;
} }
export function isPrimaryTab() {
return new Promise((resolve) => {
if (capabilities.supportsServiceWorker) {
navigator.serviceWorker.addEventListener("message", (event) => {
resolve(event.data.primaryTab);
});
navigator.serviceWorker.ready.then((registration) => {
registration.active.postMessage({ action: "primaryTab" });
});
} else {
resolve(true);
}
});
}

View File

@ -43,6 +43,16 @@ class Capabilities {
!("userActivation" in navigator) || navigator.userActivation.hasBeenActive !("userActivation" in navigator) || navigator.userActivation.hasBeenActive
); );
} }
get supportsServiceWorker() {
return (
"serviceWorker" in navigator &&
typeof ServiceWorkerRegistration !== "undefined" &&
!this.isAppWebview &&
navigator.serviceWorker.controller &&
navigator.serviceWorker.controller.state === "activated"
);
}
} }
export const capabilities = new Capabilities(); export const capabilities = new Capabilities();

View File

@ -99,7 +99,6 @@ self.addEventListener('notificationclick', function(event) {
} }
}); });
self.addEventListener('pushsubscriptionchange', function(event) { self.addEventListener('pushsubscriptionchange', function(event) {
event.waitUntil( event.waitUntil(
Promise.all( Promise.all(
@ -126,6 +125,22 @@ self.addEventListener('pushsubscriptionchange', function(event) {
); );
}); });
self.addEventListener('message', function(event) {
if (event.data?.action !== "primaryTab") {
return;
}
event.waitUntil(
self.clients.matchAll().then(function(clients) {
clients.forEach(function(client) {
client.postMessage({
primaryTab: client.focused
});
});
})
);
});
<% DiscoursePluginRegistry.service_workers.each do |js| %> <% DiscoursePluginRegistry.service_workers.each do |js| %>
<%=raw "#{File.read(js)}" %> <%=raw "#{File.read(js)}" %>
<% end %> <% end %>

View File

@ -1,4 +1,5 @@
import { withPluginApi } from "discourse/lib/plugin-api"; import { withPluginApi } from "discourse/lib/plugin-api";
import { isPrimaryTab } from "discourse/lib/utilities";
import { INDICATOR_PREFERENCES } from "discourse/plugins/chat/discourse/lib/chat-constants"; import { INDICATOR_PREFERENCES } from "discourse/plugins/chat/discourse/lib/chat-constants";
const MENTION = 29; const MENTION = 29;
@ -15,6 +16,10 @@ export default {
return; return;
} }
this.canPlaySound = async () => {
return await isPrimaryTab();
};
withPluginApi("0.12.1", (api) => { withPluginApi("0.12.1", (api) => {
api.registerDesktopNotificationHandler((data, siteSettings, user) => { api.registerDesktopNotificationHandler((data, siteSettings, user) => {
const indicatorType = user.user_option.chat_header_indicator_preference; const indicatorType = user.user_option.chat_header_indicator_preference;
@ -24,10 +29,7 @@ export default {
return; return;
} }
if ( if (!user.chat_sound || indicatorType === INDICATOR_PREFERENCES.never) {
!user.user_option.chat_sound ||
indicatorType === INDICATOR_PREFERENCES.never
) {
return; return;
} }
@ -47,10 +49,14 @@ export default {
} }
if (CHAT_NOTIFICATION_TYPES.includes(data.notification_type)) { if (CHAT_NOTIFICATION_TYPES.includes(data.notification_type)) {
const chatAudioManager = container.lookup( this.canPlaySound().then((success) => {
"service:chat-audio-manager" if (success) {
); const chatAudioManager = container.lookup(
chatAudioManager.play(user.chat_sound); "service:chat-audio-manager"
);
chatAudioManager.play(user.chat_sound);
}
});
} }
}); });
}); });

View File

@ -19,14 +19,19 @@ module("Discourse Chat | Unit | chat-audio", function (hooks) {
this.siteSettings = getOwner(this).lookup("service:site-settings"); this.siteSettings = getOwner(this).lookup("service:site-settings");
this.siteSettings.chat_enabled = true; this.siteSettings.chat_enabled = true;
this.currentUser.chat_sound = "ding";
this.currentUser.user_option.has_chat_enabled = true; this.currentUser.user_option.has_chat_enabled = true;
this.currentUser.user_option.chat_sound = "ding";
this.currentUser.user_option.chat_header_indicator_preference = "all_new"; this.currentUser.user_option.chat_header_indicator_preference = "all_new";
withPluginApi("0.12.1", async (api) => { withPluginApi("0.12.1", async (api) => {
this.stub = sinon.spy(api, "registerDesktopNotificationHandler"); this.stub = sinon.spy(api, "registerDesktopNotificationHandler");
chatAudioInitializer.initialize(getOwner(this)); chatAudioInitializer.initialize(getOwner(this));
// stub the service worker response
sinon
.stub(chatAudioInitializer, "canPlaySound")
.returns(Promise.resolve(true));
this.notificationHandler = this.stub.getCall(0).callback; this.notificationHandler = this.stub.getCall(0).callback;
this.playStub = sinon.stub(chatAudioManager, "play"); this.playStub = sinon.stub(chatAudioManager, "play");
@ -43,58 +48,65 @@ module("Discourse Chat | Unit | chat-audio", function (hooks) {
assert.ok(this.stub.calledOnce); assert.ok(this.stub.calledOnce);
}); });
test("it plays chat sound", function (assert) { test("it plays chat sound", async function (assert) {
this.handleNotification(); await this.handleNotification();
assert.ok(this.playStub.calledOnce); assert.ok(this.playStub.calledOnce);
}); });
test("it skips chat sound for user in DND mode", function (assert) { test("it skips chat sound for user in DND mode", async function (assert) {
this.currentUser.isInDoNotDisturb = () => true; this.currentUser.isInDoNotDisturb = () => true;
this.handleNotification(); await this.handleNotification();
assert.ok(this.playStub.notCalled); assert.ok(this.playStub.notCalled);
}); });
test("it skips chat sound for user with no chat sound set", function (assert) { test("it skips chat sound for user with no chat sound set", async function (assert) {
this.currentUser.user_option.chat_sound = null; this.currentUser.chat_sound = null;
this.handleNotification(); await this.handleNotification();
assert.ok(this.playStub.notCalled); assert.ok(this.playStub.notCalled);
}); });
test("it plays a chat sound for mentions", function (assert) { test("it plays a chat sound for mentions", async function (assert) {
this.currentUser.user_option.chat_header_indicator_preference = this.currentUser.user_option.chat_header_indicator_preference =
"only_mentions"; "only_mentions";
this.handleNotification({ notification_type: 29 }); await this.handleNotification({ notification_type: 29 });
assert.ok(this.playStub.calledOnce); assert.ok(this.playStub.calledOnce);
}); });
test("it skips chat sound for non-mentions", function (assert) { test("it skips chat sound for non-mentions", async function (assert) {
this.currentUser.user_option.chat_header_indicator_preference = this.currentUser.user_option.chat_header_indicator_preference =
"only_mentions"; "only_mentions";
this.handleNotification(); await this.handleNotification();
assert.ok(this.playStub.notCalled); assert.ok(this.playStub.notCalled);
}); });
test("it plays a chat sound for DMs", function (assert) { test("it plays a chat sound for DMs", async function (assert) {
this.currentUser.user_option.chat_header_indicator_preference = this.currentUser.user_option.chat_header_indicator_preference =
"dm_and_mentions"; "dm_and_mentions";
this.handleNotification({ is_direct_message_channel: true }); await this.handleNotification({ is_direct_message_channel: true });
assert.ok(this.playStub.calledOnce); assert.ok(this.playStub.calledOnce);
}); });
test("it skips chat sound for non-DM messages", function (assert) { test("it skips chat sound for non-DM messages", async function (assert) {
this.currentUser.user_option.chat_header_indicator_preference = this.currentUser.user_option.chat_header_indicator_preference =
"dm_and_mentions"; "dm_and_mentions";
this.handleNotification({ is_direct_message_channel: false }); await this.handleNotification({ is_direct_message_channel: false });
assert.ok(this.playStub.notCalled);
});
test("it skips chat sound when service worker returns false", async function (assert) {
chatAudioInitializer.canPlaySound.returns(Promise.resolve(false));
await this.handleNotification();
assert.ok(this.playStub.notCalled); assert.ok(this.playStub.notCalled);
}); });