import { acceptance, exists, publishToMessageBus, query, } from "discourse/tests/helpers/qunit-helpers"; import { click, fillIn, settled, triggerKeyEvent, visit, } from "@ember/test-helpers"; import { skip, test } from "qunit"; import { baseChatPretenders, chatChannelPretender, } from "../helpers/chat-pretenders"; acceptance("Discourse Chat - Composer", function (needs) { needs.user({ has_chat_enabled: true }); needs.settings({ chat_enabled: true, enable_rich_text_paste: true }); needs.pretender((server, helper) => { baseChatPretenders(server, helper); chatChannelPretender(server, helper); server.get("/chat/:id/messages.json", () => helper.response({ chat_messages: [], meta: {} }) ); server.get("/chat/emojis.json", () => helper.response({ favorites: [{ name: "grinning" }] }) ); server.post("/chat/drafts", () => { return helper.response([]); }); }); needs.hooks.beforeEach(function () { Object.defineProperty(this, "chatService", { get: () => this.container.lookup("service:chat"), }); }); test("when pasting html in composer", async function (assert) { await visit("/chat/channel/11/another-category"); const clipboardEvent = new Event("paste", { bubbles: true }); clipboardEvent.clipboardData = { types: ["text/html"], getData: (type) => { if (type === "text/html") { return "Foo"; } }, }; document .querySelector(".chat-composer-input") .dispatchEvent(clipboardEvent); await settled(); assert.equal(document.querySelector(".chat-composer-input").value, "Foo"); }); test("when selecting an emoji from the picker", async function (assert) { const emojiReactionStore = this.container.lookup( "service:chat-emoji-reaction-store" ); assert.deepEqual( emojiReactionStore.favorites, this.siteSettings.default_emoji_reactions.split("|") ); await visit("/chat/channel/11/-"); await click(".chat-composer-dropdown__trigger-btn"); await click(".chat-composer-dropdown__action-btn.emoji"); await click(`[data-emoji="grinning"]`); assert.deepEqual( emojiReactionStore.favorites, ["grinning"].concat(this.siteSettings.default_emoji_reactions.split("|")), "it tracks the emoji" ); }); skip("when selecting an emoji from the autocomplete", async function (assert) { const emojiReactionStore = this.container.lookup( "service:chat-emoji-reaction-store" ); assert.deepEqual( emojiReactionStore.favorites, this.siteSettings.default_emoji_reactions.split("|") ); await visit("/chat/channel/11/-"); await fillIn(".chat-composer-input", "test :grinni"); await triggerKeyEvent(".chat-composer-input", "keyup", "ArrowDown"); // necessary to show the menu await click(".autocomplete.ac-emoji ul li:first-child a"); assert.deepEqual( emojiReactionStore.favorites, ["grinning"].concat(this.siteSettings.default_emoji_reactions.split("|")), "it tracks the emoji" ); }); }); let sendAttempt = 0; acceptance("Discourse Chat - Composer - unreliable network", function (needs) { needs.user({ id: 1, has_chat_enabled: true }); needs.settings({ chat_enabled: true }); needs.pretender((server, helper) => { chatChannelPretender(server, helper); server.get("/chat/:id/messages.json", () => helper.response({ chat_messages: [], meta: {} }) ); server.post("/chat/drafts", () => helper.response(500, {})); server.post("/chat/:id.json", () => { sendAttempt += 1; return sendAttempt === 1 ? helper.response(500, {}) : helper.response({ success: true }); }); }); needs.hooks.beforeEach(function () { Object.defineProperty(this, "chatService", { get: () => this.container.lookup("service:chat"), }); }); needs.hooks.afterEach(function () { sendAttempt = 0; }); test("Sending a message with unreliable network", async function (assert) { await visit("/chat/channel/11/-"); await fillIn(".chat-composer-input", "network-error-message"); await click(".send-btn"); assert.ok( exists(".chat-message-container[data-id='1'] .retry-staged-message-btn"), "it adds a retry button" ); await fillIn(".chat-composer-input", "network-error-message"); await click(".send-btn"); await publishToMessageBus(`/chat/11`, { type: "sent", stagedId: 1, chat_message: { cooked: "network-error-message", id: 175, user: { id: 1 }, }, }); assert.notOk( exists(".chat-message-container[data-id='1'] .retry-staged-message-btn"), "it removes the staged message" ); assert.ok( exists(".chat-message-container[data-id='175']"), "it sends the message" ); assert.strictEqual( query(".chat-composer-input").value, "", "it clears the input" ); }); test("Draft with unreliable network", async function (assert) { await visit("/chat/channel/11/-"); this.chatService.set("isNetworkUnreliable", true); await settled(); assert.ok( exists(".chat-composer__unreliable-network"), "it displays a network error icon" ); }); });