mirror of
https://github.com/discourse/discourse.git
synced 2025-02-19 13:42:55 +08:00
data:image/s3,"s3://crabby-images/65e8e/65e8e483cf60c23366d3f4ea0c98b96828ae54a7" alt="Ghassan Maslamani"
This commit adds a delete button to the composer preview next to the image scale buttons. Reference: https://meta.discourse.org/t/image-remover-button-to-composer-preview/233005
379 lines
13 KiB
JavaScript
379 lines
13 KiB
JavaScript
import { click, fillIn, triggerKeyEvent, visit } from "@ember/test-helpers";
|
|
import {
|
|
acceptance,
|
|
count,
|
|
exists,
|
|
invisible,
|
|
query,
|
|
visible,
|
|
} from "discourse/tests/helpers/qunit-helpers";
|
|
import { test } from "qunit";
|
|
|
|
acceptance("Composer - Image Preview", function (needs) {
|
|
needs.user();
|
|
needs.settings({ enable_whispers: true });
|
|
needs.site({ can_tag_topics: true });
|
|
needs.pretender((server, helper) => {
|
|
server.post("/uploads/lookup-urls", () => {
|
|
return helper.response([]);
|
|
});
|
|
server.get("/posts/419", () => {
|
|
return helper.response({ id: 419 });
|
|
});
|
|
server.get("/u/is_local_username", () => {
|
|
return helper.response({
|
|
valid: [],
|
|
valid_groups: ["staff"],
|
|
mentionable_groups: [{ name: "staff", user_count: 30 }],
|
|
cannot_see: [],
|
|
max_users_notified_per_group_mention: 100,
|
|
});
|
|
});
|
|
});
|
|
|
|
const assertImageResized = (assert, uploads) => {
|
|
assert.strictEqual(
|
|
query(".d-editor-input").value,
|
|
uploads.join("\n"),
|
|
"it resizes uploaded image"
|
|
);
|
|
};
|
|
|
|
test("Image resizing buttons", async function (assert) {
|
|
await visit("/");
|
|
await click("#create-topic");
|
|
|
|
let uploads = [
|
|
// 0 Default markdown with dimensions- should work
|
|
"<a href='https://example.com'>data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|690x313"</a>",
|
|
// 1 Image with scaling percentage, should work
|
|
"data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|690x313,50%"",
|
|
// 2 image with scaling percentage and a proceeding whitespace, should work
|
|
"data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|690x313, 50%"",
|
|
// 3 No dimensions, should not work
|
|
"data:image/s3,"s3://crabby-images/ce960/ce96072c79223d92142ce107862650bf874dcbd1" alt="test"",
|
|
// 4 Wrapped in backticks should not work
|
|
"`data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|690x313"`",
|
|
// 5 html image - should not work
|
|
"<img src='/images/avatar.png' wight='20' height='20'>",
|
|
// 6 two images one the same line, but both are syntactically correct - both should work
|
|
"data:image/s3,"s3://crabby-images/61bf3/61bf3350a82c64c852c328a071f9e8299a615dd6" alt="onTheSameLine1|200x200" data:image/s3,"s3://crabby-images/66a45/66a45c6c2ec32498d0394b8039cbed2e9ee48929" alt="onTheSameLine2|250x250"",
|
|
// 7 & 8 Identical images - both should work
|
|
"data:image/s3,"s3://crabby-images/72bf0/72bf0e008384f64b7d44d2c0ffebf1c89b6a4db1" alt="identicalImage|300x300"",
|
|
"data:image/s3,"s3://crabby-images/72bf0/72bf0e008384f64b7d44d2c0ffebf1c89b6a4db1" alt="identicalImage|300x300"",
|
|
// 9 Image with whitespaces in alt - should work
|
|
"data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="image with spaces in alt|690x220"",
|
|
// 10 Image with markdown title - should work
|
|
`data:image/s3,"s3://crabby-images/cf414/cf414571584cea1a595defa001fd573149a1bfef" alt="image|690x220"`,
|
|
// 11 bbcode - should not work
|
|
"[img]/images/avatar.png[/img]",
|
|
// 12 Image with data attributes
|
|
"data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|foo=bar|690x313,50%|bar=baz"",
|
|
];
|
|
|
|
await fillIn(".d-editor-input", uploads.join("\n"));
|
|
|
|
assert.strictEqual(
|
|
count(".button-wrapper"),
|
|
10,
|
|
"it adds correct amount of scaling button groups"
|
|
);
|
|
|
|
// Default
|
|
uploads[0] =
|
|
"<a href='https://example.com'>data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|690x313, 50%"</a>";
|
|
await click(
|
|
".button-wrapper[data-image-index='0'] .scale-btn[data-scale='50']"
|
|
);
|
|
assertImageResized(assert, uploads);
|
|
|
|
// Targets the correct image if two on the same line
|
|
uploads[6] =
|
|
"data:image/s3,"s3://crabby-images/61bf3/61bf3350a82c64c852c328a071f9e8299a615dd6" alt="onTheSameLine1|200x200, 50%" data:image/s3,"s3://crabby-images/66a45/66a45c6c2ec32498d0394b8039cbed2e9ee48929" alt="onTheSameLine2|250x250"";
|
|
await click(
|
|
".button-wrapper[data-image-index='3'] .scale-btn[data-scale='50']"
|
|
);
|
|
assertImageResized(assert, uploads);
|
|
|
|
// Try the other image on the same line
|
|
uploads[6] =
|
|
"data:image/s3,"s3://crabby-images/61bf3/61bf3350a82c64c852c328a071f9e8299a615dd6" alt="onTheSameLine1|200x200, 50%" data:image/s3,"s3://crabby-images/66a45/66a45c6c2ec32498d0394b8039cbed2e9ee48929" alt="onTheSameLine2|250x250, 75%"";
|
|
await click(
|
|
".button-wrapper[data-image-index='4'] .scale-btn[data-scale='75']"
|
|
);
|
|
assertImageResized(assert, uploads);
|
|
|
|
// Make sure we target the correct image if there are duplicates
|
|
uploads[7] = "data:image/s3,"s3://crabby-images/72bf0/72bf0e008384f64b7d44d2c0ffebf1c89b6a4db1" alt="identicalImage|300x300, 50%"";
|
|
await click(
|
|
".button-wrapper[data-image-index='5'] .scale-btn[data-scale='50']"
|
|
);
|
|
assertImageResized(assert, uploads);
|
|
|
|
// Try the other dupe
|
|
uploads[8] = "data:image/s3,"s3://crabby-images/72bf0/72bf0e008384f64b7d44d2c0ffebf1c89b6a4db1" alt="identicalImage|300x300, 75%"";
|
|
await click(
|
|
".button-wrapper[data-image-index='6'] .scale-btn[data-scale='75']"
|
|
);
|
|
assertImageResized(assert, uploads);
|
|
|
|
// Don't mess with image titles
|
|
uploads[10] = `data:image/s3,"s3://crabby-images/cf414/cf414571584cea1a595defa001fd573149a1bfef" alt="image|690x220, 75%"`;
|
|
await click(
|
|
".button-wrapper[data-image-index='8'] .scale-btn[data-scale='75']"
|
|
);
|
|
assertImageResized(assert, uploads);
|
|
|
|
// Keep data attributes
|
|
uploads[12] = `data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|foo=bar|690x313, 75%|bar=baz"`;
|
|
await click(
|
|
".button-wrapper[data-image-index='9'] .scale-btn[data-scale='75']"
|
|
);
|
|
assertImageResized(assert, uploads);
|
|
|
|
await fillIn(
|
|
".d-editor-input",
|
|
`
|
|
data:image/s3,"s3://crabby-images/12720/12720f39685eb20d7e871189cea9ec129d01567f" alt="test|690x313"
|
|
|
|
\`<script>alert("xss")</script>\`
|
|
`
|
|
);
|
|
|
|
assert.ok(
|
|
!exists("script"),
|
|
"it does not unescape script tags in code blocks"
|
|
);
|
|
});
|
|
|
|
test("Editing alt text (with enter key) for single image in preview updates alt text in composer", async function (assert) {
|
|
const scaleButtonContainer = ".scale-btn-container";
|
|
|
|
const readonlyAltText = ".alt-text";
|
|
const editAltTextButton = ".alt-text-edit-btn";
|
|
|
|
const altTextInput = ".alt-text-input";
|
|
const altTextEditOk = ".alt-text-edit-ok";
|
|
const altTextEditCancel = ".alt-text-edit-cancel";
|
|
|
|
await visit("/");
|
|
|
|
await click("#create-topic");
|
|
await fillIn(".d-editor-input", `data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="zorro|200x200"`);
|
|
|
|
assert.equal(query(readonlyAltText).innerText, "zorro", "correct alt text");
|
|
assert.ok(visible(readonlyAltText), "alt text is visible");
|
|
assert.ok(visible(editAltTextButton), "alt text edit button is visible");
|
|
assert.ok(invisible(altTextInput), "alt text input is hidden");
|
|
assert.ok(invisible(altTextEditOk), "alt text edit ok button is hidden");
|
|
assert.ok(invisible(altTextEditCancel), "alt text edit cancel is hidden");
|
|
|
|
await click(editAltTextButton);
|
|
|
|
assert.ok(invisible(scaleButtonContainer), "scale buttons are hidden");
|
|
assert.ok(invisible(readonlyAltText), "alt text is hidden");
|
|
assert.ok(invisible(editAltTextButton), "alt text edit button is hidden");
|
|
assert.ok(visible(altTextInput), "alt text input is visible");
|
|
assert.ok(visible(altTextEditOk), "alt text edit ok button is visible");
|
|
assert.ok(visible(altTextEditCancel), "alt text edit cancel is hidden");
|
|
assert.equal(
|
|
query(altTextInput).value,
|
|
"zorro",
|
|
"correct alt text in input"
|
|
);
|
|
|
|
await triggerKeyEvent(altTextInput, "keypress", "[");
|
|
await triggerKeyEvent(altTextInput, "keypress", "]");
|
|
assert.equal(query(altTextInput).value, "zorro", "does not input [ ] keys");
|
|
|
|
await fillIn(altTextInput, "steak");
|
|
await triggerKeyEvent(altTextInput, "keypress", 13);
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value,
|
|
"data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="steak|200x200"",
|
|
"alt text updated"
|
|
);
|
|
assert.equal(
|
|
query(readonlyAltText).innerText,
|
|
"steak",
|
|
"shows the alt text"
|
|
);
|
|
assert.ok(visible(editAltTextButton), "alt text edit button is visible");
|
|
assert.ok(visible(scaleButtonContainer), "scale buttons are visible");
|
|
assert.ok(visible(editAltTextButton), "alt text edit button is visible");
|
|
assert.ok(invisible(altTextInput), "alt text input is hidden");
|
|
assert.ok(invisible(altTextEditOk), "alt text edit ok button is hidden");
|
|
assert.ok(invisible(altTextEditCancel), "alt text edit cancel is hidden");
|
|
});
|
|
|
|
test("Editing alt text (with check button) in preview updates alt text in composer", async function (assert) {
|
|
const scaleButtonContainer = ".scale-btn-container";
|
|
const readonlyAltText = ".alt-text";
|
|
const editAltTextButton = ".alt-text-edit-btn";
|
|
|
|
const altTextInput = ".alt-text-input";
|
|
const altTextEditOk = ".alt-text-edit-ok";
|
|
const altTextEditCancel = ".alt-text-edit-cancel";
|
|
|
|
await visit("/");
|
|
|
|
await click("#create-topic");
|
|
await fillIn(".d-editor-input", `data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="zorro|200x200"`);
|
|
|
|
await click(editAltTextButton);
|
|
|
|
await fillIn(altTextInput, "steak");
|
|
await click(altTextEditOk);
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value,
|
|
"data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="steak|200x200"",
|
|
"alt text updated"
|
|
);
|
|
assert.equal(
|
|
query(readonlyAltText).innerText,
|
|
"steak",
|
|
"shows the alt text"
|
|
);
|
|
|
|
assert.ok(visible(editAltTextButton), "alt text edit button is visible");
|
|
assert.ok(visible(scaleButtonContainer), "scale buttons are visible");
|
|
assert.ok(visible(editAltTextButton), "alt text edit button is visible");
|
|
assert.ok(invisible(altTextInput), "alt text input is hidden");
|
|
assert.ok(invisible(altTextEditOk), "alt text edit ok button is hidden");
|
|
assert.ok(invisible(altTextEditCancel), "alt text edit cancel is hidden");
|
|
});
|
|
|
|
test("Cancel alt text edit in preview does not update alt text in composer", async function (assert) {
|
|
const scaleButtonContainer = ".scale-btn-container";
|
|
|
|
const readonlyAltText = ".alt-text";
|
|
const editAltTextButton = ".alt-text-edit-btn";
|
|
|
|
const altTextInput = ".alt-text-input";
|
|
const altTextEditOk = ".alt-text-edit-ok";
|
|
const altTextEditCancel = ".alt-text-edit-cancel";
|
|
|
|
await visit("/");
|
|
|
|
await click("#create-topic");
|
|
await fillIn(".d-editor-input", `data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="zorro|200x200"`);
|
|
|
|
await click(editAltTextButton);
|
|
|
|
await fillIn(altTextInput, "steak");
|
|
await click(altTextEditCancel);
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value,
|
|
"data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="zorro|200x200"",
|
|
"alt text not updated"
|
|
);
|
|
assert.equal(
|
|
query(readonlyAltText).innerText,
|
|
"zorro",
|
|
"shows the unedited alt text"
|
|
);
|
|
|
|
assert.ok(visible(editAltTextButton), "alt text edit button is visible");
|
|
assert.ok(visible(scaleButtonContainer), "scale buttons are visible");
|
|
assert.ok(visible(editAltTextButton), "alt text edit button is visible");
|
|
assert.ok(invisible(altTextInput), "alt text input is hidden");
|
|
assert.ok(invisible(altTextEditOk), "alt text edit ok button is hidden");
|
|
assert.ok(invisible(altTextEditCancel), "alt text edit cancel is hidden");
|
|
});
|
|
|
|
test("Editing alt text for one of two images in preview updates correct alt text in composer", async function (assert) {
|
|
const editAltTextButton = ".alt-text-edit-btn";
|
|
const altTextInput = ".alt-text-input";
|
|
|
|
await visit("/");
|
|
await click("#create-topic");
|
|
|
|
await fillIn(
|
|
".d-editor-input",
|
|
`data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="zorro|200x200" data:image/s3,"s3://crabby-images/fa45f/fa45f9a5c0f95b823c7d4ab44a341c9036de3173" alt="not-zorro|200x200"`
|
|
);
|
|
await click(editAltTextButton);
|
|
|
|
await fillIn(altTextInput, "tomtom");
|
|
await triggerKeyEvent(altTextInput, "keypress", "Enter");
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value,
|
|
`data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="tomtom|200x200" data:image/s3,"s3://crabby-images/fa45f/fa45f9a5c0f95b823c7d4ab44a341c9036de3173" alt="not-zorro|200x200"`,
|
|
"the correct image's alt text updated"
|
|
);
|
|
});
|
|
|
|
test("Deleting alt text for image empties alt text in composer and allows further modification", async function (assert) {
|
|
const altText = ".alt-text";
|
|
const editAltTextButton = ".alt-text-edit-btn";
|
|
const altTextInput = ".alt-text-input";
|
|
|
|
await visit("/");
|
|
|
|
await click("#create-topic");
|
|
await fillIn(".d-editor-input", `data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="zorro|200x200"`);
|
|
|
|
await click(editAltTextButton);
|
|
|
|
await fillIn(altTextInput, "");
|
|
await triggerKeyEvent(altTextInput, "keypress", "Enter");
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value,
|
|
"data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="|200x200"",
|
|
"alt text updated"
|
|
);
|
|
assert.equal(query(altText).innerText, "", "shows the alt text");
|
|
|
|
await click(editAltTextButton);
|
|
|
|
await fillIn(altTextInput, "tomtom");
|
|
await triggerKeyEvent(altTextInput, "keypress", "Enter");
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value,
|
|
"data:image/s3,"s3://crabby-images/b103a/b103a5867276502eca90cf94efee184b80449097" alt="tomtom|200x200"",
|
|
"alt text updated"
|
|
);
|
|
});
|
|
|
|
test("Image delete button", async function (assert) {
|
|
await visit("/");
|
|
await click("#create-topic");
|
|
|
|
let uploads = [
|
|
"data:image/s3,"s3://crabby-images/9dd37/9dd3727a46a644ae42c378e3cb4b3ac01d68e13e" alt="image_example_0|666x500"",
|
|
"data:image/s3,"s3://crabby-images/482e5/482e54ef2daed6fd61344bc0061c06007a96038b" alt="image_example_1|481x480"",
|
|
];
|
|
|
|
await fillIn(".d-editor-input", uploads.join("\n"));
|
|
|
|
uploads[0] = ""; // delete the first image.
|
|
|
|
//click on the remove button of the first image
|
|
await click(".button-wrapper[data-image-index='0'] .delete-image-button");
|
|
|
|
assert.strictEqual(
|
|
query(".d-editor-input").value,
|
|
uploads.join("\n"),
|
|
"Image should be removed from the editor"
|
|
);
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value.includes("image_example_0"),
|
|
false,
|
|
"It shouldn't have the first image"
|
|
);
|
|
|
|
assert.equal(
|
|
query(".d-editor-input").value.includes("image_example_1"),
|
|
true,
|
|
"It should have the second image"
|
|
);
|
|
});
|
|
});
|