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:
parent
249f4296bf
commit
d1e4e7cd6f
|
@ -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>
|
||||||
|
|
|
@ -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],
|
||||||
|
|
|
@ -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();
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
Loading…
Reference in New Issue