mirror of
https://github.com/discourse/discourse.git
synced 2025-01-06 11:53:49 +08:00
616f4a1118
Before this fix you could press send while upload was in progress and lose it as it was not yet uploaded.
364 lines
10 KiB
Ruby
364 lines
10 KiB
Ruby
# frozen_string_literal: true
|
||
|
||
RSpec.describe "Chat composer", type: :system, js: true do
|
||
fab!(:current_user) { Fabricate(:user) }
|
||
fab!(:channel_1) { Fabricate(:chat_channel) }
|
||
fab!(:message_1) { Fabricate(:chat_message, chat_channel: channel_1) }
|
||
|
||
let(:chat) { PageObjects::Pages::Chat.new }
|
||
let(:channel) { PageObjects::Pages::ChatChannel.new }
|
||
|
||
before { chat_system_bootstrap }
|
||
|
||
context "when loading a channel with a draft" do
|
||
fab!(:draft_1) do
|
||
Chat::Draft.create!(
|
||
chat_channel: channel_1,
|
||
user: current_user,
|
||
data: { message: "draft" }.to_json,
|
||
)
|
||
end
|
||
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "loads the draft" do
|
||
chat.visit_channel(channel_1)
|
||
|
||
expect(find(".chat-composer__input").value).to eq("draft")
|
||
end
|
||
|
||
context "with uploads" do
|
||
fab!(:upload_1) do
|
||
Fabricate(
|
||
:upload,
|
||
url: "/images/logo-dark.png",
|
||
original_filename: "logo_dark.png",
|
||
width: 400,
|
||
height: 300,
|
||
extension: "png",
|
||
)
|
||
end
|
||
|
||
fab!(:draft_1) do
|
||
Chat::Draft.create!(
|
||
chat_channel: channel_1,
|
||
user: current_user,
|
||
data: { message: "draft", uploads: [upload_1] }.to_json,
|
||
)
|
||
end
|
||
|
||
it "loads the draft with the upload" do
|
||
chat.visit_channel(channel_1)
|
||
|
||
expect(find(".chat-composer__input").value).to eq("draft")
|
||
expect(page).to have_selector(".chat-composer-upload--image", count: 1)
|
||
end
|
||
end
|
||
|
||
context "when replying" do
|
||
fab!(:draft_1) do
|
||
Chat::Draft.create!(
|
||
chat_channel: channel_1,
|
||
user: current_user,
|
||
data: {
|
||
message: "draft",
|
||
replyToMsg: {
|
||
id: message_1.id,
|
||
excerpt: message_1.excerpt,
|
||
user: {
|
||
id: message_1.user.id,
|
||
name: nil,
|
||
avatar_template: message_1.user.avatar_template,
|
||
username: message_1.user.username,
|
||
},
|
||
},
|
||
}.to_json,
|
||
)
|
||
end
|
||
|
||
it "loads the draft with replied to mesage" do
|
||
chat.visit_channel(channel_1)
|
||
|
||
expect(find(".chat-composer__input").value).to eq("draft")
|
||
expect(page).to have_selector(".chat-reply__username", text: message_1.user.username)
|
||
expect(page).to have_selector(".chat-reply__excerpt", text: message_1.excerpt)
|
||
end
|
||
end
|
||
end
|
||
|
||
context "when replying to a message" do
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "adds the reply indicator to the composer" do
|
||
chat.visit_channel(channel_1)
|
||
channel.reply_to(message_1)
|
||
|
||
expect(page).to have_selector(
|
||
".chat-composer-message-details .chat-reply__username",
|
||
text: message_1.user.username,
|
||
)
|
||
end
|
||
|
||
context "with HTML tags" do
|
||
before { message_1.update!(message: "<mark>not marked</mark>") }
|
||
|
||
it "renders text in the details" do
|
||
chat.visit_channel(channel_1)
|
||
channel.reply_to(message_1)
|
||
|
||
expect(
|
||
find(".chat-composer-message-details .chat-reply__excerpt")["innerHTML"].strip,
|
||
).to eq("not marked")
|
||
end
|
||
end
|
||
end
|
||
|
||
context "when editing a message" do
|
||
fab!(:message_2) { Fabricate(:chat_message, chat_channel: channel_1, user: current_user) }
|
||
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "adds the edit indicator" do
|
||
chat.visit_channel(channel_1)
|
||
channel.edit_message(message_2)
|
||
|
||
expect(page).to have_selector(
|
||
".chat-composer-message-details .chat-reply__username",
|
||
text: current_user.username,
|
||
)
|
||
expect(find(".chat-composer__input").value).to eq(message_2.message)
|
||
end
|
||
|
||
it "updates the message instantly" do
|
||
chat.visit_channel(channel_1)
|
||
page.driver.browser.network_conditions = { offline: true }
|
||
|
||
channel.edit_message(message_2)
|
||
find(".chat-composer__input").send_keys("instant")
|
||
channel.click_send_message
|
||
|
||
expect(channel).to have_message(text: message_2.message + "instant")
|
||
page.driver.browser.network_conditions = { offline: false }
|
||
end
|
||
|
||
context "when pressing escape" do
|
||
it "cancels editing" do
|
||
chat.visit_channel(channel_1)
|
||
channel.edit_message(message_2)
|
||
find(".chat-composer__input").send_keys(:escape)
|
||
|
||
expect(page).to have_no_selector(".chat-composer-message-details .chat-reply__username")
|
||
expect(find(".chat-composer__input").value).to eq("")
|
||
end
|
||
end
|
||
|
||
context "when closing edited message" do
|
||
it "cancels editing" do
|
||
chat.visit_channel(channel_1)
|
||
channel.edit_message(message_2)
|
||
find(".cancel-message-action").click
|
||
|
||
expect(page).to have_no_selector(".chat-composer-message-details .chat-reply__username")
|
||
expect(find(".chat-composer__input").value).to eq("")
|
||
end
|
||
end
|
||
end
|
||
|
||
context "when adding an emoji through the picker" do
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "adds the emoji to the composer" do
|
||
chat.visit_channel(channel_1)
|
||
channel.open_action_menu
|
||
channel.click_action_button("emoji")
|
||
find("[data-emoji='grimacing']").click(wait: 0.5)
|
||
|
||
expect(find(".chat-composer__input").value).to eq(":grimacing:")
|
||
end
|
||
|
||
it "removes denied emojis from insert emoji picker" do
|
||
SiteSetting.emoji_deny_list = "monkey|peach"
|
||
|
||
chat.visit_channel(channel_1)
|
||
channel.open_action_menu
|
||
channel.click_action_button("emoji")
|
||
|
||
expect(page).to have_no_selector("[data-emoji='monkey']")
|
||
expect(page).to have_no_selector("[data-emoji='peach']")
|
||
end
|
||
end
|
||
|
||
context "when adding an emoji through the autocomplete" do
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "adds the emoji to the composer" do
|
||
chat.visit_channel(channel_1)
|
||
find(".chat-composer__input").fill_in(with: ":gri")
|
||
find(".emoji-shortname", text: "grimacing").click
|
||
|
||
expect(find(".chat-composer__input").value).to eq(":grimacing: ")
|
||
end
|
||
|
||
it "doesn't suggest denied emojis and aliases" do
|
||
SiteSetting.emoji_deny_list = "peach|poop"
|
||
chat.visit_channel(channel_1)
|
||
|
||
find(".chat-composer__input").fill_in(with: ":peac")
|
||
expect(page).to have_no_selector(".emoji-shortname", text: "peach")
|
||
|
||
find(".chat-composer__input").fill_in(with: ":hank") # alias
|
||
expect(page).to have_no_selector(".emoji-shortname", text: "poop")
|
||
end
|
||
end
|
||
|
||
context "when opening emoji picker through more button of the autocomplete" do
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
xit "prefills the emoji picker filter input" do
|
||
chat.visit_channel(channel_1)
|
||
find(".chat-composer__input").fill_in(with: ":gri")
|
||
|
||
click_link(I18n.t("js.composer.more_emoji"))
|
||
|
||
expect(find(".chat-emoji-picker .dc-filter-input").value).to eq("gri")
|
||
end
|
||
|
||
xit "filters with the prefilled input" do
|
||
chat.visit_channel(channel_1)
|
||
find(".chat-composer__input").fill_in(with: ":fr")
|
||
|
||
click_link(I18n.t("js.composer.more_emoji"))
|
||
|
||
expect(page).to have_selector(".chat-emoji-picker [data-emoji='fr']")
|
||
expect(page).to have_no_selector(".chat-emoji-picker [data-emoji='grinning']")
|
||
end
|
||
end
|
||
|
||
context "when typing on keyboard" do
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "propagates keys to composer" do
|
||
chat.visit_channel(channel_1)
|
||
|
||
find("body").send_keys("b")
|
||
|
||
expect(find(".chat-composer__input").value).to eq("b")
|
||
|
||
find("body").send_keys("b")
|
||
|
||
expect(find(".chat-composer__input").value).to eq("bb")
|
||
|
||
find("body").send_keys(:enter) # special case
|
||
|
||
expect(find(".chat-composer__input").value).to eq("bb")
|
||
end
|
||
end
|
||
|
||
context "when pasting link over selected text" do
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "outputs a markdown link" do
|
||
modifier = /darwin/i =~ RbConfig::CONFIG["host_os"] ? :command : :control
|
||
select_text = <<-JS
|
||
const element = document.querySelector(arguments[0]);
|
||
element.focus();
|
||
element.setSelectionRange(0, element.value.length)
|
||
JS
|
||
|
||
chat.visit_channel(channel_1)
|
||
|
||
find("body").send_keys("https://www.discourse.org")
|
||
page.execute_script(select_text, ".chat-composer__input")
|
||
|
||
page.send_keys [modifier, "c"]
|
||
page.send_keys [:backspace]
|
||
|
||
find("body").send_keys("discourse")
|
||
page.execute_script(select_text, ".chat-composer__input")
|
||
|
||
page.send_keys [modifier, "v"]
|
||
|
||
expect(find(".chat-composer__input").value).to eq("[discourse](https://www.discourse.org)")
|
||
end
|
||
end
|
||
|
||
context "when posting a message with length equal to minimum length" do
|
||
before do
|
||
SiteSetting.chat_minimum_message_length = 1
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "works" do
|
||
chat.visit_channel(channel_1)
|
||
find("body").send_keys("1")
|
||
channel.click_send_message
|
||
|
||
expect(channel).to have_message(text: "1")
|
||
end
|
||
end
|
||
|
||
context "when posting a message with length superior to minimum length" do
|
||
before do
|
||
SiteSetting.chat_minimum_message_length = 2
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "doesn’t allow to send" do
|
||
chat.visit_channel(channel_1)
|
||
find("body").send_keys("1")
|
||
|
||
expect(page).to have_css(".chat-composer.is-send-disabled")
|
||
end
|
||
end
|
||
|
||
context "when upload is in progress" do
|
||
before do
|
||
channel_1.add(current_user)
|
||
sign_in(current_user)
|
||
end
|
||
|
||
it "doesn’t allow to send" do
|
||
chat.visit_channel(channel_1)
|
||
|
||
page.driver.browser.network_conditions = { latency: 20_000 }
|
||
|
||
file_path = file_from_fixtures("logo.png", "images").path
|
||
attach_file(file_path) do
|
||
channel.open_action_menu
|
||
channel.click_action_button("chat-upload-btn")
|
||
end
|
||
|
||
expect(page).to have_css(".chat-composer-upload--in-progress")
|
||
expect(page).to have_css(".chat-composer.is-send-disabled")
|
||
|
||
page.driver.browser.network_conditions = { latency: 0 }
|
||
end
|
||
end
|
||
end
|