2023-10-10 14:38:59 -04:00
|
|
|
import { click, fillIn, render, triggerKeyEvent } from "@ember/test-helpers";
|
2022-11-02 09:41:30 -04:00
|
|
|
import hbs from "htmlbars-inline-precompile";
|
|
|
|
import { module, test } from "qunit";
|
2023-10-10 14:38:59 -04:00
|
|
|
import { setupRenderingTest } from "discourse/tests/helpers/component-test";
|
2024-06-23 16:34:15 -04:00
|
|
|
import pretender, { response } from "discourse/tests/helpers/create-pretender";
|
2022-11-02 09:41:30 -04:00
|
|
|
|
|
|
|
function emojisResponse() {
|
|
|
|
return {
|
|
|
|
favorites: [
|
|
|
|
{
|
|
|
|
name: "grinning",
|
|
|
|
tonable: false,
|
|
|
|
url: "/images/emoji/twitter/grinning.png?v=12",
|
|
|
|
group: "smileys_\u0026_emotion",
|
|
|
|
search_aliases: ["smiley_cat", "star_struck"],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
"smileys_&_emotion": [
|
|
|
|
{
|
|
|
|
name: "grinning",
|
|
|
|
tonable: false,
|
|
|
|
url: "/images/emoji/twitter/grinning.png?v=12",
|
|
|
|
group: "smileys_\u0026_emotion",
|
|
|
|
search_aliases: ["smiley_cat", "star_struck"],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
"people_&_body": [
|
|
|
|
{
|
|
|
|
name: "raised_hands",
|
|
|
|
tonable: true,
|
|
|
|
url: "/images/emoji/twitter/raised_hands.png?v=12",
|
|
|
|
group: "people_&_body",
|
|
|
|
search_aliases: [],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "man_rowing_boat",
|
|
|
|
tonable: true,
|
|
|
|
url: "/images/emoji/twitter/man_rowing_boat.png?v=12",
|
|
|
|
group: "people_&_body",
|
|
|
|
search_aliases: [],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
objects: [
|
|
|
|
{
|
|
|
|
name: "womans_clothes",
|
|
|
|
tonable: false,
|
|
|
|
url: "/images/emoji/twitter/womans_clothes.png?v=12",
|
|
|
|
group: "objects",
|
|
|
|
search_aliases: [],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
module("Discourse Chat | Component | chat-emoji-picker", function (hooks) {
|
|
|
|
setupRenderingTest(hooks);
|
|
|
|
|
|
|
|
hooks.afterEach(function () {
|
|
|
|
this.emojiReactionStore.diversity = 1;
|
|
|
|
});
|
|
|
|
|
|
|
|
hooks.beforeEach(function () {
|
2024-06-23 16:34:15 -04:00
|
|
|
pretender.get("/chat/emojis.json", () => response(emojisResponse()));
|
2022-11-02 09:41:30 -04:00
|
|
|
|
|
|
|
this.chatEmojiPickerManager = this.container.lookup(
|
|
|
|
"service:chat-emoji-picker-manager"
|
|
|
|
);
|
DEV: Refactoring chat message actions for ChatMessage component usage in thread panel (#20756)
This commit is a major overhaul of how chat message actions work, to make it so they are reusable between the main chat channel and the chat thread panel, as well as many improvements and fixes for the thread panel.
There are now several new classes and concepts:
* ChatMessageInteractor - This is initialized from the ChatMessage, ChatMessageActionsDesktop, and ChatMessageActionsMobile components. This handles permissions about what actions can be done for each
message based on the context (thread or channel), handles the actions themselves (e.g. copyLink, delete, edit),
and interacts with the pane of the current context to modify the UI
* ChatChannelThreadPane and ChatChannelPane services - This represents the UI context which contains the
messages, and are mostly used for state management for things like message selection.
* ChatChannelThreadComposer and ChatChannelComposer - This handles interaction between the pane, the
message actions, and the composer, dealing with reply and edit message state.
* Scrolling logic for the messages has now been moved to a helper so it can be shared between the main channel pane and the thread pane
* Various improvements with the emoji picker on both mobile and desktop. The DOM node of each component is now located outside of the message which prevents a large range of issues.
The thread panel now also works in the chat drawer, and the thread messages have less
actions than the main panel, since some do not make sense there (e.g. moving messages to
a different channel). The thread panel title, excerpt, and message sender have also been removed
for now to save space.
This gives us a solid base to keep expanding on and fixing up threads. Subsequent PRs will
make the thread MessageBus subscriptions work and disable echo mode
for the initial release of threads.
Co-authored-by: Joffrey JAFFEUX <j.jaffeux@gmail.com>
2023-04-06 09:19:52 -04:00
|
|
|
this.chatEmojiPickerManager.open(() => {});
|
2022-11-02 09:41:30 -04:00
|
|
|
this.chatEmojiPickerManager.addVisibleSections([
|
|
|
|
"smileys_&_emotion",
|
|
|
|
"people_&_body",
|
|
|
|
"objects",
|
|
|
|
]);
|
|
|
|
|
|
|
|
this.emojiReactionStore = this.container.lookup(
|
|
|
|
"service:chat-emoji-reaction-store"
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("When displaying navigation", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(`.chat-emoji-picker__section-btn.active[data-section="favorites"]`)
|
|
|
|
.exists("it renders first section as active");
|
|
|
|
assert
|
|
|
|
.dom(`.chat-emoji-picker__section-btn[data-section="smileys_&_emotion"]`)
|
|
|
|
.exists();
|
|
|
|
assert
|
|
|
|
.dom(`.chat-emoji-picker__section-btn[data-section="people_&_body"]`)
|
|
|
|
.exists();
|
|
|
|
assert
|
|
|
|
.dom(`.chat-emoji-picker__section-btn[data-section="objects"]`)
|
|
|
|
.exists();
|
2022-11-02 09:41:30 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
test("When changing tone scale", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
await click(".chat-emoji-picker__fitzpatrick-modifier-btn.current.t1");
|
|
|
|
await click(".chat-emoji-picker__fitzpatrick-modifier-btn.t6");
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(`img[src="/images/emoji/twitter/raised_hands/6.png"]`)
|
|
|
|
.exists("it applies the tone to emojis");
|
|
|
|
assert
|
|
|
|
.dom(".chat-emoji-picker__fitzpatrick-modifier-btn.current.t6")
|
|
|
|
.exists("it changes the current scale to t6");
|
2022-11-02 09:41:30 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
test("When requesting section", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
|
|
|
|
assert.strictEqual(
|
|
|
|
document.querySelector("#ember-testing-container").scrollTop,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
|
|
|
|
await click(`.chat-emoji-picker__section-btn[data-section="objects"]`);
|
|
|
|
|
2022-12-22 08:35:18 -05:00
|
|
|
assert.true(
|
2023-04-13 14:35:13 -04:00
|
|
|
document.querySelector(".chat-emoji-picker__scrollable-content")
|
|
|
|
.scrollTop > 0,
|
2022-11-02 09:41:30 -04:00
|
|
|
"it scrolls to the section"
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
test("When filtering emojis", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
await fillIn(".dc-filter-input", "grinning");
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(".chat-emoji-picker__section.filtered > img")
|
|
|
|
.exists({ count: 1 }, "it filters the emojis list");
|
|
|
|
assert
|
|
|
|
.dom('.chat-emoji-picker__section.filtered > img[alt="grinning"]')
|
|
|
|
.exists("it filters the correct emoji");
|
2022-11-02 09:41:30 -04:00
|
|
|
|
|
|
|
await fillIn(".dc-filter-input", "Grinning");
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom('.chat-emoji-picker__section.filtered > img[alt="grinning"]')
|
|
|
|
.exists("it is case insensitive");
|
2022-11-02 09:41:30 -04:00
|
|
|
|
|
|
|
await fillIn(".dc-filter-input", "smiley_cat");
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom('.chat-emoji-picker__section.filtered > img[alt="grinning"]')
|
|
|
|
.exists("it filters the correct emoji using search alias");
|
2022-11-02 09:41:30 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
test("When selecting an emoji", async function (assert) {
|
2024-06-23 16:34:15 -04:00
|
|
|
this.didSelectEmoji = (emoji) => assert.step(emoji);
|
DEV: Refactoring chat message actions for ChatMessage component usage in thread panel (#20756)
This commit is a major overhaul of how chat message actions work, to make it so they are reusable between the main chat channel and the chat thread panel, as well as many improvements and fixes for the thread panel.
There are now several new classes and concepts:
* ChatMessageInteractor - This is initialized from the ChatMessage, ChatMessageActionsDesktop, and ChatMessageActionsMobile components. This handles permissions about what actions can be done for each
message based on the context (thread or channel), handles the actions themselves (e.g. copyLink, delete, edit),
and interacts with the pane of the current context to modify the UI
* ChatChannelThreadPane and ChatChannelPane services - This represents the UI context which contains the
messages, and are mostly used for state management for things like message selection.
* ChatChannelThreadComposer and ChatChannelComposer - This handles interaction between the pane, the
message actions, and the composer, dealing with reply and edit message state.
* Scrolling logic for the messages has now been moved to a helper so it can be shared between the main channel pane and the thread pane
* Various improvements with the emoji picker on both mobile and desktop. The DOM node of each component is now located outside of the message which prevents a large range of issues.
The thread panel now also works in the chat drawer, and the thread messages have less
actions than the main panel, since some do not make sense there (e.g. moving messages to
a different channel). The thread panel title, excerpt, and message sender have also been removed
for now to save space.
This gives us a solid base to keep expanding on and fixing up threads. Subsequent PRs will
make the thread MessageBus subscriptions work and disable echo mode
for the initial release of threads.
Co-authored-by: Joffrey JAFFEUX <j.jaffeux@gmail.com>
2023-04-06 09:19:52 -04:00
|
|
|
|
|
|
|
await render(
|
|
|
|
hbs`<ChatEmojiPicker @didSelectEmoji={{this.didSelectEmoji}} />`
|
|
|
|
);
|
2022-11-02 09:41:30 -04:00
|
|
|
await click('img.emoji[data-emoji="grinning"]');
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert.verifySteps(["grinning"]);
|
2022-11-02 09:41:30 -04:00
|
|
|
});
|
|
|
|
|
2023-02-03 09:36:45 -05:00
|
|
|
test("When navigating sections", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowDown");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"grinning",
|
|
|
|
"ArrowDown focuses on the first favorite emoji"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowDown");
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowDown");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"raised_hands",
|
|
|
|
"ArrowDown focuses on the first emoji form the third section"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowRight");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"man_rowing_boat",
|
|
|
|
"ArrowRight focuses on the emoji at the right"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowLeft");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"raised_hands",
|
|
|
|
"ArrowLeft focuses on the emoji at the left"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowUp");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"grinning",
|
|
|
|
"ArrowUp focuses on the first emoji form the second section"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
test("When navigating filtered emojis", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
await fillIn(".dc-filter-input", "man");
|
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowDown");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"man_rowing_boat",
|
|
|
|
"ArrowDown focuses on the first filtered emoji"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowRight");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"womans_clothes",
|
|
|
|
"ArrowRight focuses on the emoji at the right"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
|
|
|
|
await triggerKeyEvent(document.activeElement, "keydown", "ArrowLeft");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(document.activeElement)
|
|
|
|
.hasAttribute(
|
|
|
|
"data-emoji",
|
|
|
|
"man_rowing_boat",
|
|
|
|
"ArrowLeft focuses on the emoji at the left"
|
|
|
|
);
|
2023-02-03 09:36:45 -05:00
|
|
|
});
|
|
|
|
|
2022-11-02 09:41:30 -04:00
|
|
|
test("When selecting a toned an emoji", async function (assert) {
|
2024-06-23 16:34:15 -04:00
|
|
|
this.didSelectEmoji = (emoji) => assert.step(emoji);
|
DEV: Refactoring chat message actions for ChatMessage component usage in thread panel (#20756)
This commit is a major overhaul of how chat message actions work, to make it so they are reusable between the main chat channel and the chat thread panel, as well as many improvements and fixes for the thread panel.
There are now several new classes and concepts:
* ChatMessageInteractor - This is initialized from the ChatMessage, ChatMessageActionsDesktop, and ChatMessageActionsMobile components. This handles permissions about what actions can be done for each
message based on the context (thread or channel), handles the actions themselves (e.g. copyLink, delete, edit),
and interacts with the pane of the current context to modify the UI
* ChatChannelThreadPane and ChatChannelPane services - This represents the UI context which contains the
messages, and are mostly used for state management for things like message selection.
* ChatChannelThreadComposer and ChatChannelComposer - This handles interaction between the pane, the
message actions, and the composer, dealing with reply and edit message state.
* Scrolling logic for the messages has now been moved to a helper so it can be shared between the main channel pane and the thread pane
* Various improvements with the emoji picker on both mobile and desktop. The DOM node of each component is now located outside of the message which prevents a large range of issues.
The thread panel now also works in the chat drawer, and the thread messages have less
actions than the main panel, since some do not make sense there (e.g. moving messages to
a different channel). The thread panel title, excerpt, and message sender have also been removed
for now to save space.
This gives us a solid base to keep expanding on and fixing up threads. Subsequent PRs will
make the thread MessageBus subscriptions work and disable echo mode
for the initial release of threads.
Co-authored-by: Joffrey JAFFEUX <j.jaffeux@gmail.com>
2023-04-06 09:19:52 -04:00
|
|
|
|
|
|
|
await render(
|
|
|
|
hbs`<ChatEmojiPicker @didSelectEmoji={{this.didSelectEmoji}} />`
|
|
|
|
);
|
2022-11-02 09:41:30 -04:00
|
|
|
this.emojiReactionStore.diversity = 1;
|
|
|
|
await click('img.emoji[data-emoji="man_rowing_boat"]');
|
|
|
|
|
|
|
|
this.emojiReactionStore.diversity = 2;
|
|
|
|
await click('img.emoji[data-emoji="man_rowing_boat"]');
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert.verifySteps(["man_rowing_boat", "man_rowing_boat:t2"]);
|
2022-11-02 09:41:30 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
test("When opening the picker", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert.dom(document.activeElement).hasClass("dc-filter-input");
|
2022-11-02 09:41:30 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
test("When hovering an emoji", async function (assert) {
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(
|
2022-11-02 09:41:30 -04:00
|
|
|
'.chat-emoji-picker__section[data-section="people_&_body"] img.emoji:nth-child(1)'
|
2024-06-23 16:34:15 -04:00
|
|
|
)
|
|
|
|
.hasAttribute("title", ":raised_hands:", "first emoji has a title");
|
2022-11-02 09:41:30 -04:00
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom(
|
2022-11-02 09:41:30 -04:00
|
|
|
'.chat-emoji-picker__section[data-section="people_&_body"] img.emoji:nth-child(2)'
|
2024-06-23 16:34:15 -04:00
|
|
|
)
|
|
|
|
.hasAttribute("title", ":man_rowing_boat:", "second emoji has a title");
|
2022-11-02 09:41:30 -04:00
|
|
|
|
|
|
|
await fillIn(".dc-filter-input", "grinning");
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom('img.emoji[data-emoji="grinning"]')
|
|
|
|
.hasAttribute("title", ":grinning:", "filtered emoji have a title");
|
2022-11-02 09:41:30 -04:00
|
|
|
|
|
|
|
this.emojiReactionStore.diversity = 1;
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom('img.emoji[data-emoji="man_rowing_boat"]')
|
|
|
|
.hasAttribute(
|
|
|
|
"title",
|
|
|
|
":man_rowing_boat:",
|
|
|
|
"it has a title without the scale as diversity value is 1"
|
|
|
|
);
|
2022-11-02 09:41:30 -04:00
|
|
|
|
|
|
|
this.emojiReactionStore.diversity = 2;
|
|
|
|
await render(hbs`<ChatEmojiPicker />`);
|
|
|
|
|
2024-06-23 16:34:15 -04:00
|
|
|
assert
|
|
|
|
.dom('img.emoji[data-emoji="man_rowing_boat"]')
|
|
|
|
.hasAttribute(
|
|
|
|
"title",
|
|
|
|
":man_rowing_boat:t2:",
|
|
|
|
"it has a title with the scale"
|
|
|
|
);
|
2022-11-02 09:41:30 -04:00
|
|
|
});
|
|
|
|
});
|