FIX: play audio sound on message in non group DMs (#27112)

Prior to this change, only mentions would get a notification and a sound. This change will not create a notification for this case, but will play a sound. This is still respecting notification settings, not playing the sound when you are viewing the channel or not following it.

---------

Co-authored-by: Régis Hanol <regis@hanol.fr>
This commit is contained in:
Joffrey JAFFEUX 2024-05-21 21:51:22 +02:00 committed by GitHub
parent 13a2ca8b09
commit bc0ef9f7ee
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 177 additions and 2 deletions

View File

@ -37,11 +37,13 @@ export default class ChatAudioManager extends Service {
}
async play(soundName) {
if (isTesting()) {
return;
}
const audio =
this._audioCache[soundName] || this._audioCache[DEFAULT_SOUND_NAME];
audio.muted = isTesting();
if (!audio.paused) {
audio.pause();
if (typeof audio.fastSeek === "function") {

View File

@ -0,0 +1,51 @@
import Service, { service } from "@ember/service";
export default class ChatChannelNotificationSound extends Service {
@service chat;
@service chatAudioManager;
@service currentUser;
@service site;
async play(channel) {
if (channel.isCategoryChannel) {
return false;
}
if (channel.chatable.group) {
return false;
}
if (this.currentUser.isInDoNotDisturb()) {
return false;
}
if (!this.currentUser.chat_sound) {
return false;
}
if (this.site.mobileView) {
return false;
}
const membership = channel.currentUserMembership;
if (!membership.following) {
return false;
}
if (membership.desktopNotificationLevel !== "always") {
return false;
}
if (membership.muted) {
return false;
}
if (this.chat.activeChannel === channel) {
return false;
}
await this.chatAudioManager.play(this.currentUser.chat_sound);
return true;
}
}

View File

@ -7,6 +7,7 @@ import ChatChannelArchive from "../models/chat-channel-archive";
export default class ChatSubscriptionsManager extends Service {
@service store;
@service chatChannelsManager;
@service chatChannelNotificationSound;
@service chatTrackingStateManager;
@service currentUser;
@service appEvents;
@ -205,6 +206,8 @@ export default class ChatSubscriptionsManager extends Service {
channel.tracking.unreadCount++;
}
this.chatChannelNotificationSound.play(channel);
// Thread should be considered unread if not already.
if (busData.thread_id && channel.threadingEnabled) {
channel.threadsManager

View File

@ -0,0 +1,119 @@
import { getOwner } from "@ember/application";
import { test } from "qunit";
import {
acceptance,
updateCurrentUser,
} from "discourse/tests/helpers/qunit-helpers";
import ChatFabricators from "discourse/plugins/chat/discourse/lib/fabricators";
function buildDirectMessageChannel(owner) {
const channel = new ChatFabricators(owner).directMessageChannel();
buildMembership(channel);
return channel;
}
function buildCategoryMessageChannel(owner) {
const channel = new ChatFabricators(owner).channel();
buildMembership(channel);
return channel;
}
function buildMembership(channel) {
channel.currentUserMembership = {
following: true,
desktop_notification_level: "always",
muted: false,
};
return channel;
}
acceptance(
"Discourse Chat | Unit | Service | chat-channel-notification-sound",
function (needs) {
needs.hooks.beforeEach(function () {
Object.defineProperty(this, "subject", {
get: () =>
this.container.lookup("service:chat-channel-notification-sound"),
});
Object.defineProperty(this, "site", {
get: () => this.container.lookup("service:site"),
});
Object.defineProperty(this, "chat", {
get: () => this.container.lookup("service:chat"),
});
updateCurrentUser({ chat_sound: "ding" });
});
needs.user();
test("in do not disturb", async function (assert) {
updateCurrentUser({ do_not_disturb_until: moment().add(1, "hour") });
const channel = buildDirectMessageChannel(getOwner(this));
assert.deepEqual(await this.subject.play(channel), false);
});
test("no chat sound", async function (assert) {
updateCurrentUser({ chat_sound: null });
const channel = buildDirectMessageChannel(getOwner(this));
assert.deepEqual(await this.subject.play(channel), false);
});
test("mobile", async function (assert) {
const channel = buildDirectMessageChannel(getOwner(this));
this.site.mobileView = true;
assert.deepEqual(await this.subject.play(channel), false);
});
test("plays sound", async function (assert) {
const channel = buildDirectMessageChannel(getOwner(this));
assert.deepEqual(await this.subject.play(channel), true);
});
test("muted", async function (assert) {
const channel = buildDirectMessageChannel(getOwner(this));
channel.currentUserMembership.muted = true;
assert.deepEqual(await this.subject.play(channel), false);
});
test("not following", async function (assert) {
const channel = buildDirectMessageChannel(getOwner(this));
channel.currentUserMembership.following = false;
assert.deepEqual(await this.subject.play(channel), false);
});
test("no notification", async function (assert) {
const channel = buildDirectMessageChannel(getOwner(this));
channel.currentUserMembership.desktopNotificationLevel = "never";
assert.deepEqual(await this.subject.play(channel), false);
});
test("currently active channel", async function (assert) {
const channel = buildDirectMessageChannel(getOwner(this));
this.chat.activeChannel = channel;
assert.deepEqual(await this.subject.play(channel), false);
});
test("category channel", async function (assert) {
const channel = buildCategoryMessageChannel(getOwner(this));
assert.deepEqual(await this.subject.play(channel), false);
});
test("group", async function (assert) {
const channel = buildDirectMessageChannel(getOwner(this));
channel.chatable.group = true;
assert.deepEqual(await this.subject.play(channel), false);
});
}
);