Odoo18-Base/addons/mail/static/tests/thread/attachment_list.test.js
2025-01-06 10:57:38 +07:00

433 lines
14 KiB
JavaScript

import {
assertSteps,
click,
contains,
defineMailModels,
onRpcBefore,
openDiscuss,
start,
startServer,
step,
} from "@mail/../tests/mail_test_helpers";
import { describe, expect, test } from "@odoo/hoot";
import { getOrigin } from "@web/core/utils/urls";
describe.current.tags("desktop");
defineMailModels();
test("simplest layout", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.txt",
mimetype: "text/plain",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-Message .o-mail-AttachmentList");
expect(".o-mail-AttachmentCard:first").toHaveAttribute("title", "test.txt");
await contains(".o-mail-AttachmentCard-image");
expect(".o-mail-AttachmentCard-image:first").toHaveClass("o_image"); // required for mimetype.scss style
expect(".o-mail-AttachmentCard-image:first").toHaveAttribute("data-mimetype", "text/plain"); // required for mimetype.scss style
await contains(".o-mail-AttachmentCard-aside button", { count: 2 });
await contains(".o-mail-AttachmentCard-unlink");
await contains(".o-mail-AttachmentCard-aside button[title='Download']");
});
test("layout with card details and filename and extension", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.txt",
mimetype: "text/plain",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentCard", { text: "test.txt" });
await contains(".o-mail-AttachmentCard small", { text: "txt" });
});
test("link-type attachment should have open button instead of download button", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachment_ids = pyEnv["ir.attachment"].create([
{
name: "url.example",
mimetype: "text/plain",
type: "url",
url: "https://www.odoo.com",
},
{
name: "test.txt",
mimetype: "text/plain",
},
]);
pyEnv["mail.message"].create({
attachment_ids,
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentCard", { count: 2 });
await contains(".o-mail-AttachmentCard:eq(0)", { text: "url.example" });
await contains(".o-mail-AttachmentCard:eq(1)", { text: "test.txt" });
await contains(
".o-mail-AttachmentCard:eq(0) .o-mail-AttachmentCard-aside a[title='Open Link']"
);
await contains(
".o-mail-AttachmentCard:eq(0) .o-mail-AttachmentCard-aside button[title='Download']",
{ count: 0 }
);
await contains(
".o-mail-AttachmentCard:eq(1) .o-mail-AttachmentCard-aside button[title='Download']"
);
await contains(`.o-mail-AttachmentCard-aside a[title='Open Link'][target='_blank']`);
});
test("clicking on the delete attachment button multiple times should do the rpc only once", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.txt",
mimetype: "text/plain",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
onRpcBefore("/mail/attachment/delete", () => step("attachment_unlink"));
await start();
await openDiscuss(channelId);
await click(".o-mail-AttachmentCard-unlink");
await click(".modal-footer .btn-primary");
await click(".modal-footer .btn-primary");
await click(".modal-footer .btn-primary");
await contains(".o-mail-AttachmentCard-unlink", { count: 0 });
await assertSteps(["attachment_unlink"]); // The unlink method must be called once
});
test("view attachment", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.png",
mimetype: "image/png",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentImage img");
await click(".o-mail-AttachmentImage");
await contains(".o-FileViewer");
});
test("can view pdf url", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "url.pdf.example",
mimetype: "application/pdf",
type: "url",
url: "https://pdfobject.com/pdf/sample.pdf",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await click(".o-mail-AttachmentCard", { text: "url.pdf.example" });
await contains(".o-FileViewer");
await contains(
`iframe.o-FileViewer-view[data-src="/web/static/lib/pdfjs/web/viewer.html?file=${encodeURIComponent(
`${getOrigin()}/web/content/${attachmentId}?filename=url.pdf.example`
)}#pagemode=none"]`
);
});
test("close attachment viewer", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.png",
mimetype: "image/png",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentImage img");
await click(".o-mail-AttachmentImage");
await contains(".o-FileViewer");
await click(".o-FileViewer div[aria-label='Close']");
await contains(".o-FileViewer", { count: 0 });
});
test("[technical] does not crash when the viewer is closed before image load", async () => {
/**
* When images are displayed using "src" attribute for the 1st time, it fetches the resource.
* In this case, images are actually displayed (fully fetched and rendered on screen) when
* "<image>" intercepts "load" event.
*
* Current code needs to be aware of load state of image, to display spinner when loading
* and actual image when loaded. This test asserts no crash from mishandling image becoming
* loaded from being viewed for 1st time, but viewer being closed while image is loading.
*/
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.png",
mimetype: "image/png",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await click(".o-mail-AttachmentImage");
await contains(".o-FileViewer-viewImage");
await click(".o-FileViewer div[aria-label='Close']");
// Simulate image becoming loaded.
expect(() => {
document
.querySelector(".o-FileViewer-viewImage")
.dispatchEvent(new Event("load", { bubbles: true }));
}).not.toThrow();
});
test("plain text file is viewable", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.txt",
mimetype: "text/plain",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentCard.o-viewable");
});
test("HTML file is viewable", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.html",
mimetype: "text/html",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentCard.o-viewable");
});
test("ODT file is not viewable", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.odt",
mimetype: "application/vnd.oasis.opendocument.text",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentCard:not(.o-viewable)");
});
test("DOCX file is not viewable", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.docx",
mimetype: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentCard:not(.o-viewable)");
});
test("should not view attachment from click on non-viewable attachment in list containing a viewable attachment", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const [attachmentId_1, attachmentId_2] = pyEnv["ir.attachment"].create([
{
name: "test.png",
mimetype: "image/png",
},
{
name: "test.odt",
mimetype: "application/vnd.oasis.opendocument.text",
},
]);
pyEnv["mail.message"].create({
attachment_ids: [attachmentId_1, attachmentId_2],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(".o-mail-AttachmentImage[title='test.png'].o-viewable");
await contains(".o-mail-AttachmentCard:not(.o-viewable)", { text: "test.odt" });
await click(".o-mail-AttachmentCard", { text: "test.odt" });
// weak test, no guarantee that we waited long enough for the potential file viewer to show
await contains(".o-FileViewer", { count: 0 });
await click(".o-mail-AttachmentImage[title='test.png']");
await contains(".o-FileViewer");
});
test("img file has proper src in discuss.channel", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.png",
mimetype: "image/png",
res_id: channelId,
res_model: "discuss.channel",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(
`.o-mail-AttachmentImage[title='test.png'] img[data-src*='${getOrigin()}/discuss/channel/${channelId}/image/${attachmentId}?filename=test.png']`
);
});
test("download url of non-viewable binary file", async () => {
const pyEnv = await startServer();
const channelId = pyEnv["discuss.channel"].create({
channel_type: "channel",
name: "channel1",
});
const attachmentId = pyEnv["ir.attachment"].create({
name: "test.o",
mimetype: "application/octet-stream",
type: "binary",
});
pyEnv["mail.message"].create({
attachment_ids: [attachmentId],
body: "<p>Test</p>",
model: "discuss.channel",
res_id: channelId,
message_type: "comment",
});
await start();
await openDiscuss(channelId);
await contains(
`button[data-download-url="${getOrigin()}/web/content/${attachmentId}?filename=test.o&download=true"]`
);
});