FIX: more precise chat-replying-indicator (#21451)

- correctly subscribe/unsubscribe channel
- instantly changes users list
- adds a test for testing channel change
- rewrites tests to be less verbose
- ensures users is always an array
This commit is contained in:
Joffrey JAFFEUX 2023-05-09 13:25:33 +02:00 committed by GitHub
parent 249f4296bf
commit d1e4e7cd6f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 61 additions and 120 deletions

View File

@ -5,9 +5,8 @@
(if this.presenceChannel.subscribed "is-subscribed") (if this.presenceChannel.subscribed "is-subscribed")
}} }}
{{did-insert this.subscribe}} {{did-insert this.subscribe}}
{{did-update this.handleDraft @channel.isDraft}} {{did-update this.updateSubscription @channel.id}}
{{did-update this.subscribe this.channelName}} {{will-destroy this.unsubscribe}}
{{will-destroy this.teardown}}
> >
{{#if this.shouldRender}} {{#if this.shouldRender}}
<span class="chat-replying-indicator__text">{{this.text}}</span> <span class="chat-replying-indicator__text">{{this.text}}</span>

View File

@ -1,4 +1,4 @@
import { isBlank, isPresent } from "@ember/utils"; import { isPresent } from "@ember/utils";
import Component from "@glimmer/component"; import Component from "@glimmer/component";
import { inject as service } from "@ember/service"; import { inject as service } from "@ember/service";
import I18n from "I18n"; import I18n from "I18n";
@ -13,16 +13,27 @@ export default class ChatReplyingIndicator extends Component {
@tracked users = []; @tracked users = [];
@action @action
async subscribe() { async updateSubscription() {
this.presenceChannel = this.presence.getChannel(this.channelName); await this.unsubscribe();
this.presenceChannel.on("change", this.handlePresenceChange); await this.subscribe();
await this.presenceChannel.subscribe();
} }
@action @action
async resubscribe() { async subscribe() {
await this.teardown(); this.presenceChannel = this.presence.getChannel(this.channelName);
await this.subscribe(); await this.presenceChannel.subscribe();
this.users = this.presenceChannel.users;
this.presenceChannel.on("change", this.handlePresenceChange);
}
@action
async unsubscribe() {
this.users = [];
if (this.presenceChannel.subscribed) {
this.presenceChannel.off("change", this.handlePresenceChange);
await this.presenceChannel.unsubscribe();
}
} }
@action @action
@ -30,22 +41,6 @@ export default class ChatReplyingIndicator extends Component {
this.users = presenceChannel.users || []; this.users = presenceChannel.users || [];
} }
@action
async handleDraft() {
if (this.args.channel.isDraft) {
await this.teardown();
} else {
await this.resubscribe();
}
}
@action
async teardown() {
if (this.presenceChannel) {
await this.presenceChannel.unsubscribe();
}
}
get usernames() { get usernames() {
return this.users return this.users
.filter((u) => u.id !== this.currentUser.id) .filter((u) => u.id !== this.currentUser.id)
@ -53,10 +48,6 @@ export default class ChatReplyingIndicator extends Component {
} }
get text() { get text() {
if (isBlank(this.usernames)) {
return;
}
if (this.usernames.length === 1) { if (this.usernames.length === 1) {
return I18n.t("chat.replying_indicator.single_user", { return I18n.t("chat.replying_indicator.single_user", {
username: this.usernames[0], username: this.usernames[0],

View File

@ -1,11 +1,19 @@
import { setupRenderingTest } from "discourse/tests/helpers/component-test"; import { setupRenderingTest } from "discourse/tests/helpers/component-test";
import { exists, query } from "discourse/tests/helpers/qunit-helpers"; import { query } from "discourse/tests/helpers/qunit-helpers";
import hbs from "htmlbars-inline-precompile"; import hbs from "htmlbars-inline-precompile";
import fabricators from "../helpers/fabricators"; import fabricators from "../helpers/fabricators";
import { module, test } from "qunit"; import { module, test } from "qunit";
import { render, settled } from "@ember/test-helpers"; import { render } from "@ember/test-helpers";
import { joinChannel } from "discourse/tests/helpers/presence-pretender"; import { joinChannel } from "discourse/tests/helpers/presence-pretender";
async function addUserToChannel(channelId, id, username) {
await joinChannel(`/chat-reply/${channelId}`, {
id,
avatar_template: "/images/avatar.png",
username,
});
}
module( module(
"Discourse Chat | Component | chat-replying-indicator", "Discourse Chat | Component | chat-replying-indicator",
function (hooks) { function (hooks) {
@ -16,7 +24,7 @@ module(
await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`); await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`);
assert.false(exists(".chat-replying-indicator__text")); assert.dom(".chat-replying-indicator__text").doesNotExist();
}); });
test("displays indicator when user is replying", async function (assert) { test("displays indicator when user is replying", async function (assert) {
@ -24,11 +32,7 @@ module(
await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`); await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`);
await joinChannel("/chat-reply/1", { await addUserToChannel(1, 1, "sam");
id: 1,
avatar_template: "/images/avatar.png",
username: "sam",
});
assert.strictEqual( assert.strictEqual(
query(".chat-replying-indicator__text").innerText, query(".chat-replying-indicator__text").innerText,
@ -41,22 +45,12 @@ module(
await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`); await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`);
await joinChannel("/chat-reply/1", { await addUserToChannel(1, 1, "sam");
id: 1, await addUserToChannel(1, 2, "mark");
avatar_template: "/images/avatar.png",
username: "sam",
});
await joinChannel("/chat-reply/1", { assert
id: 2, .dom(".chat-replying-indicator__text")
avatar_template: "/images/avatar.png", .hasText("sam and mark are typing");
username: "mark",
});
assert.strictEqual(
query(".chat-replying-indicator__text").innerText,
`sam and mark are typing`
);
}); });
test("displays indicator when 3 users are replying", async function (assert) { test("displays indicator when 3 users are replying", async function (assert) {
@ -64,28 +58,13 @@ module(
await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`); await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`);
await joinChannel("/chat-reply/1", { await addUserToChannel(1, 1, "sam");
id: 1, await addUserToChannel(1, 2, "mark");
avatar_template: "/images/avatar.png", await addUserToChannel(1, 3, "joffrey");
username: "sam",
});
await joinChannel("/chat-reply/1", { assert
id: 2, .dom(".chat-replying-indicator__text")
avatar_template: "/images/avatar.png", .hasText("sam, mark and joffrey are typing");
username: "mark",
});
await joinChannel("/chat-reply/1", {
id: 3,
avatar_template: "/images/avatar.png",
username: "joffrey",
});
assert.strictEqual(
query(".chat-replying-indicator__text").innerText,
`sam, mark and joffrey are typing`
);
}); });
test("displays indicator when more than 3 users are replying", async function (assert) { test("displays indicator when more than 3 users are replying", async function (assert) {
@ -93,34 +72,14 @@ module(
await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`); await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`);
await joinChannel("/chat-reply/1", { await addUserToChannel(1, 1, "sam");
id: 1, await addUserToChannel(1, 2, "mark");
avatar_template: "/images/avatar.png", await addUserToChannel(1, 3, "joffrey");
username: "sam", await addUserToChannel(1, 4, "taylor");
});
await joinChannel("/chat-reply/1", { assert
id: 2, .dom(".chat-replying-indicator__text")
avatar_template: "/images/avatar.png", .hasText("sam, mark and 2 others are typing");
username: "mark",
});
await joinChannel("/chat-reply/1", {
id: 3,
avatar_template: "/images/avatar.png",
username: "joffrey",
});
await joinChannel("/chat-reply/1", {
id: 4,
avatar_template: "/images/avatar.png",
username: "taylor",
});
assert.strictEqual(
query(".chat-replying-indicator__text").innerText,
`sam, mark and 2 others are typing`
);
}); });
test("filters current user from list of repliers", async function (assert) { test("filters current user from list of repliers", async function (assert) {
@ -128,32 +87,24 @@ module(
await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`); await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`);
await joinChannel("/chat-reply/1", { await addUserToChannel(1, 1, "sam");
id: 1, await addUserToChannel(1, this.currentUser.id, this.currentUser.username);
avatar_template: "/images/avatar.png",
username: "sam", assert.dom(".chat-replying-indicator__text").hasText("sam is typing");
}); });
await joinChannel("/chat-reply/1", this.currentUser); test("resets presence when channel changes", async function (assert) {
this.set("channel", fabricators.chatChannel());
assert.strictEqual( await addUserToChannel(1, 1, "sam");
query(".chat-replying-indicator__text").innerText,
`sam is typing`
);
});
test("resets presence when channel is draft", async function (assert) {
this.channel = fabricators.chatChannel();
await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`); await render(hbs`<ChatReplyingIndicator @channel={{this.channel}} />`);
assert.dom(".chat-replying-indicator.is-subscribed").exists(); assert.dom(".chat-replying-indicator__text").hasText("sam is typing");
this.channel.isDraft = true; this.set("channel", fabricators.chatChannel({ id: 2 }));
await settled(); assert.dom(".chat-replying-indicator__text").doesNotExist();
assert.dom(".chat-replying-indicator.is-subscribed").doesNotExist();
}); });
} }
); );