upload: Convert module to TypeScript.

Signed-off-by: Anders Kaseorg <anders@zulip.com>
This commit is contained in:
Anders Kaseorg 2024-05-14 19:24:25 -07:00 committed by Tim Abbott
parent e74ed28f4f
commit fba81d1cd5
5 changed files with 120 additions and 64 deletions

View File

@ -261,7 +261,7 @@ EXEMPT_FILES = make_set(
"web/src/unread.ts",
"web/src/unread_ops.ts",
"web/src/unread_ui.ts",
"web/src/upload.js",
"web/src/upload.ts",
"web/src/upload_widget.ts",
"web/src/url-template.d.ts",
"web/src/user_card_popover.js",

View File

@ -74,6 +74,7 @@ export const realm_schema = z.object({
PRONOUNS: z.object({id: z.number(), name: z.string()}),
}),
max_avatar_file_size_mib: z.number(),
max_file_upload_size_mib: z.number(),
max_icon_file_size_mib: z.number(),
max_logo_file_size_mib: z.number(),
max_message_length: z.number(),

View File

@ -1,7 +1,9 @@
import type {UppyFile} from "@uppy/core";
import {Uppy} from "@uppy/core";
import XHRUpload from "@uppy/xhr-upload";
import $ from "jquery";
import assert from "minimalistic-assert";
import {z} from "zod";
import render_upload_banner from "../templates/compose_banner/upload_banner.hbs";
@ -18,27 +20,42 @@ import * as message_lists from "./message_lists";
import * as rows from "./rows";
import {realm} from "./state_data";
let drag_drop_img = null;
let compose_upload_object;
const upload_objects_by_message_edit_row = new Map();
let drag_drop_img: HTMLElement | null = null;
let compose_upload_object: Uppy;
const upload_objects_by_message_edit_row = new Map<number, Uppy>();
export function compose_upload_cancel() {
export function compose_upload_cancel(): void {
compose_upload_object.cancelAll();
}
export function feature_check() {
export function feature_check(): XMLHttpRequestUpload {
// Show the upload button only if the browser supports it.
return window.XMLHttpRequest && new window.XMLHttpRequest().upload;
}
export function get_translated_status(file) {
export function get_translated_status(file: File | UppyFile): string {
const status = $t({defaultMessage: "Uploading {filename}…"}, {filename: file.name});
return "[" + status + "]()";
}
export const compose_config = {
type Config = ({mode: "compose"} | {mode: "edit"; row: number}) & {
textarea: () => JQuery<HTMLTextAreaElement>;
send_button: () => JQuery;
banner_container: () => JQuery;
upload_banner_identifier: (file_id: string) => string;
upload_banner: (file_id: string) => JQuery;
upload_banner_cancel_button: (file_id: string) => JQuery;
upload_banner_hide_button: (file_id: string) => JQuery;
upload_banner_message: (file_id: string) => JQuery;
file_input_identifier: () => string;
source: () => string;
drag_drop_container: () => JQuery;
markdown_preview_hide_button: () => JQuery;
};
export const compose_config: Config = {
mode: "compose",
textarea: () => $("textarea#compose-textarea"),
textarea: () => $<HTMLTextAreaElement>("textarea#compose-textarea"),
send_button: () => $("#compose-send-button"),
banner_container: () => $("#compose_banners"),
upload_banner_identifier: (file_id) =>
@ -58,55 +75,58 @@ export const compose_config = {
),
upload_banner_message: (file_id) =>
$(`#compose_banners .upload_banner.file_${CSS.escape(file_id)} .upload_msg`),
file_input_identifier: () => "#compose .file_input",
file_input_identifier: () => "#compose input.file_input",
source: () => "compose-file-input",
drag_drop_container: () => $("#compose"),
markdown_preview_hide_button: () => $("#compose .undo_markdown_preview"),
};
export function edit_config(row) {
export function edit_config(row: number): Config {
return {
mode: "edit",
row,
textarea: () => $(`#edit_form_${CSS.escape(row)} .message_edit_content`),
send_button: () => $(`#edit_form_${CSS.escape(row)}`).find(".message_edit_save"),
banner_container: () => $(`#edit_form_${CSS.escape(row)} .edit_form_banners`),
textarea: () =>
$<HTMLTextAreaElement>(
`#edit_form_${CSS.escape(`${row}`)} textarea.message_edit_content`,
),
send_button: () => $(`#edit_form_${CSS.escape(`${row}`)}`).find(".message_edit_save"),
banner_container: () => $(`#edit_form_${CSS.escape(`${row}`)} .edit_form_banners`),
upload_banner_identifier: (file_id) =>
`#edit_form_${CSS.escape(row)} .upload_banner.file_${CSS.escape(file_id)}`,
`#edit_form_${CSS.escape(`${row}`)} .upload_banner.file_${CSS.escape(file_id)}`,
upload_banner: (file_id) =>
$(`#edit_form_${CSS.escape(row)} .upload_banner.file_${CSS.escape(file_id)}`),
$(`#edit_form_${CSS.escape(`${row}`)} .upload_banner.file_${CSS.escape(file_id)}`),
upload_banner_cancel_button: (file_id) =>
$(
`#edit_form_${CSS.escape(row)} .upload_banner.file_${CSS.escape(
`#edit_form_${CSS.escape(`${row}`)} .upload_banner.file_${CSS.escape(
file_id,
)} .upload_banner_cancel_button`,
),
upload_banner_hide_button: (file_id) =>
$(
`#edit_form_${CSS.escape(row)} .upload_banner.file_${CSS.escape(
`#edit_form_${CSS.escape(`${row}`)} .upload_banner.file_${CSS.escape(
file_id,
)} .main-view-banner-close-button`,
),
upload_banner_message: (file_id) =>
$(
`#edit_form_${CSS.escape(row)} .upload_banner.file_${CSS.escape(
`#edit_form_${CSS.escape(`${row}`)} .upload_banner.file_${CSS.escape(
file_id,
)} .upload_msg`,
),
file_input_identifier: () => `#edit_form_${CSS.escape(row)} .file_input`,
file_input_identifier: () => `#edit_form_${CSS.escape(`${row}`)} input.file_input`,
source: () => "message-edit-file-input",
drag_drop_container() {
assert(message_lists.current !== undefined);
return $(
`#message-row-${message_lists.current.id}-${CSS.escape(row)} .message_edit_form`,
`#message-row-${message_lists.current.id}-${CSS.escape(`${row}`)} .message_edit_form`,
);
},
markdown_preview_hide_button: () =>
$(`#edit_form_${CSS.escape(row)} .undo_markdown_preview`),
$(`#edit_form_${CSS.escape(`${row}`)} .undo_markdown_preview`),
};
}
export function hide_upload_banner(uppy, config, file_id) {
export function hide_upload_banner(uppy: Uppy, config: Config, file_id: string): void {
config.upload_banner(file_id).remove();
if (uppy.getFiles().length === 0) {
if (config.mode === "compose") {
@ -118,12 +138,12 @@ export function hide_upload_banner(uppy, config, file_id) {
}
function add_upload_banner(
config,
banner_type,
banner_text,
file_id,
config: Config,
banner_type: string,
banner_text: string,
file_id: string,
is_upload_process_tracker = false,
) {
): void {
const new_banner_html = render_upload_banner({
banner_type,
is_upload_process_tracker,
@ -137,10 +157,10 @@ function add_upload_banner(
}
export function show_error_message(
config,
config: Config,
message = $t({defaultMessage: "An unknown error occurred."}),
file_id = null,
) {
file_id: string | null = null,
): void {
if (file_id) {
$(`${config.upload_banner_identifier(file_id)} .moving_bar`).hide();
config.upload_banner(file_id).removeClass("info").addClass("error");
@ -153,7 +173,7 @@ export function show_error_message(
}
}
export async function upload_files(uppy, config, files) {
export function upload_files(uppy: Uppy, config: Config, files: File[] | FileList): void {
if (files.length === 0) {
return;
}
@ -179,6 +199,7 @@ export async function upload_files(uppy, config, files) {
}
for (const file of files) {
let file_id;
try {
compose_ui.insert_syntax_and_focus(
get_translated_status(file),
@ -187,7 +208,7 @@ export async function upload_files(uppy, config, files) {
1,
);
compose_ui.autosize_textarea(config.textarea());
file.id = uppy.addFile({
file_id = uppy.addFile({
source: config.source(),
name: file.name,
type: file.type,
@ -207,24 +228,24 @@ export async function upload_files(uppy, config, files) {
config,
"info",
$t({defaultMessage: "Uploading {filename}…"}, {filename: file.name}),
file.id,
file_id,
true,
);
config.upload_banner_cancel_button(file.id).one("click", () => {
config.upload_banner_cancel_button(file_id).one("click", () => {
compose_ui.replace_syntax(get_translated_status(file), "", config.textarea());
compose_ui.autosize_textarea(config.textarea());
config.textarea().trigger("focus");
uppy.removeFile(file.id);
hide_upload_banner(uppy, config, file.id);
uppy.removeFile(file_id);
hide_upload_banner(uppy, config, file_id);
});
config.upload_banner_hide_button(file.id).one("click", () => {
hide_upload_banner(uppy, config, file.id);
config.upload_banner_hide_button(file_id).one("click", () => {
hide_upload_banner(uppy, config, file_id);
});
}
}
export function setup_upload(config) {
export function setup_upload(config: Config): Uppy {
const uppy = new Uppy({
debug: false,
autoProceed: true,
@ -267,14 +288,16 @@ export function setup_upload(config) {
}
uppy.on("upload-progress", (file, progress) => {
assert(file !== undefined);
const percent_complete = (100 * progress.bytesUploaded) / progress.bytesTotal;
$(`${config.upload_banner_identifier(file.id)} .moving_bar`).css({
width: `${percent_complete}%`,
});
});
$(config.file_input_identifier()).on("change", (event) => {
$<HTMLInputElement>(config.file_input_identifier()).on("change", (event) => {
const files = event.target.files;
assert(files !== null);
upload_files(uppy, config, files);
config.textarea().trigger("focus");
event.target.value = "";
@ -291,12 +314,18 @@ export function setup_upload(config) {
);
const $drag_drop_container = config.drag_drop_container();
$drag_drop_container.on("dragover", (event) => event.preventDefault());
$drag_drop_container.on("dragenter", (event) => event.preventDefault());
$drag_drop_container.on("dragover", (event) => {
event.preventDefault();
});
$drag_drop_container.on("dragenter", (event) => {
event.preventDefault();
});
$drag_drop_container.on("drop", (event) => {
event.preventDefault();
event.stopPropagation();
assert(event.originalEvent !== undefined);
assert(event.originalEvent.dataTransfer !== null);
const files = event.originalEvent.dataTransfer.files;
if (config.mode === "compose" && !compose_state.composing()) {
compose_reply.respond_to_message({
@ -308,17 +337,18 @@ export function setup_upload(config) {
});
$drag_drop_container.on("paste", (event) => {
const clipboard_data = event.clipboardData || event.originalEvent.clipboardData;
assert(event.originalEvent instanceof ClipboardEvent);
const clipboard_data = event.originalEvent.clipboardData;
if (!clipboard_data) {
return;
}
const items = clipboard_data.items;
const files = [];
for (const item of items) {
if (item.kind !== "file") {
const file = item.getAsFile();
if (file === null) {
continue;
}
const file = item.getAsFile();
files.push(file);
}
if (files.length === 0) {
@ -338,7 +368,8 @@ export function setup_upload(config) {
});
uppy.on("upload-success", (file, response) => {
const url = response.body.uri;
assert(file !== undefined);
const {uri: url} = z.object({uri: z.string().optional()}).parse(response.body);
if (url === undefined) {
return;
}
@ -376,7 +407,9 @@ export function setup_upload(config) {
// TODO: Ideally, we'd be using the `.error()` hook or
// something, not parsing error message strings.
const infoList = uppy.getState().info;
assert(infoList !== undefined);
const info = infoList.at(-1);
assert(info !== undefined);
if (info.type === "error" && info.message === "No Internet connection") {
// server_events already handles the case of no internet.
return;
@ -397,11 +430,17 @@ export function setup_upload(config) {
});
uppy.on("upload-error", (file, _error, response) => {
assert(file !== undefined);
// The files with failed upload should be removed since uppy doesn't allow files in the store
// to be re-uploaded again.
uppy.removeFile(file.id);
const message = response ? response.body.msg : undefined;
let parsed;
const message =
response !== undefined &&
(parsed = z.object({msg: z.string()}).safeParse(response.body)).success
? parsed.data.msg
: undefined;
// Hide the upload status banner on error so only the error banner shows
hide_upload_banner(uppy, config, file.id);
show_error_message(config, message, file.id);
@ -410,6 +449,7 @@ export function setup_upload(config) {
});
uppy.on("restriction-failed", (file) => {
assert(file !== undefined);
compose_ui.replace_syntax(get_translated_status(file), "", config.textarea());
compose_ui.autosize_textarea(config.textarea());
});
@ -417,7 +457,7 @@ export function setup_upload(config) {
return uppy;
}
export function deactivate_upload(config) {
export function deactivate_upload(config: Config): void {
// Remove event listeners added for handling uploads.
$(config.file_input_identifier()).off("change");
config.banner_container().off("click");
@ -451,7 +491,7 @@ export function deactivate_upload(config) {
}
}
export function initialize() {
export function initialize(): void {
compose_upload_object = setup_upload(compose_config);
$(".app, #navbar-fixed-container").on("dragstart", (event) => {
@ -463,10 +503,14 @@ export function initialize() {
});
// Allow the app panel to receive drag/drop events.
$(".app, #navbar-fixed-container").on("dragover", (event) => event.preventDefault());
$(".app, #navbar-fixed-container").on("dragover", (event) => {
event.preventDefault();
});
// TODO: Do something visual to hint that drag/drop will work.
$(".app, #navbar-fixed-container").on("dragenter", (event) => event.preventDefault());
$(".app, #navbar-fixed-container").on("dragenter", (event) => {
event.preventDefault();
});
$(".app, #navbar-fixed-container").on("drop", (event) => {
event.preventDefault();
@ -477,6 +521,8 @@ export function initialize() {
}
const $drag_drop_edit_containers = $(".message_edit_form form");
assert(event.originalEvent !== undefined);
assert(event.originalEvent.dataTransfer !== null);
const files = event.originalEvent.dataTransfer.files;
const $last_drag_drop_edit_container = $drag_drop_edit_containers.last();
@ -496,6 +542,7 @@ export function initialize() {
return;
}
const edit_upload_object = upload_objects_by_message_edit_row.get(row_id);
assert(edit_upload_object !== undefined);
upload_files(edit_upload_object, edit_config(row_id), files);
} else if (message_lists.current?.selected_message()) {

View File

@ -10,8 +10,8 @@ const blueslip_stacktrace = zrequire("blueslip_stacktrace");
run_test("clean_path", () => {
// Local file
assert.strictEqual(
blueslip_stacktrace.clean_path("webpack:///web/src/upload.js"),
"/web/src/upload.js",
blueslip_stacktrace.clean_path("webpack:///web/src/upload.ts"),
"/web/src/upload.ts",
);
// Third party library (jQuery)
@ -36,9 +36,9 @@ run_test("clean_function_name", () => {
// Local file
assert.deepEqual(
blueslip_stacktrace.clean_function_name("Object../web/src/upload.js.exports.options"),
blueslip_stacktrace.clean_function_name("Object../web/src/upload.ts.exports.options"),
{
scope: "Object../web/src/upload.js.exports.",
scope: "Object../web/src/upload.ts.exports.",
name: "options",
},
);

View File

@ -7,6 +7,13 @@ const {run_test, noop} = require("./lib/test");
const $ = require("./lib/zjquery");
const {realm} = require("./lib/zpage_params");
class ClipboardEvent {
constructor({clipboardData}) {
this.clipboardData = clipboardData;
}
}
set_global("ClipboardEvent", ClipboardEvent);
set_global("navigator", {
userAgent: "Mozilla/5.0 (compatible; MSIE 9.0; Windows NT 6.1; Trident/5.0)",
});
@ -60,7 +67,7 @@ test("config", () => {
upload.compose_config.upload_banner_hide_button("id_2"),
$("#compose_banners .upload_banner.file_id_2 .main-view-banner-close-button"),
);
assert.equal(upload.compose_config.file_input_identifier(), "#compose .file_input");
assert.equal(upload.compose_config.file_input_identifier(), "#compose input.file_input");
assert.equal(upload.compose_config.source(), "compose-file-input");
assert.equal(upload.compose_config.drag_drop_container(), $("#compose"));
assert.equal(
@ -70,7 +77,7 @@ test("config", () => {
assert.equal(
upload.edit_config(1).textarea(),
$(`#edit_form_${CSS.escape(1)} .message_edit_content`),
$(`#edit_form_${CSS.escape(1)} textarea.message_edit_content`),
);
$(`#edit_form_${CSS.escape(2)}`).set_find_results(
@ -117,7 +124,7 @@ test("config", () => {
assert.equal(
upload.edit_config(123).file_input_identifier(),
`#edit_form_${CSS.escape(123)} .file_input`,
`#edit_form_${CSS.escape(123)} input.file_input`,
);
assert.equal(upload.edit_config(123).source(), "message-edit-file-input");
assert.equal(
@ -343,7 +350,7 @@ test("uppy_config", () => {
test("file_input", ({override_rewire}) => {
upload.setup_upload(upload.compose_config);
const change_handler = $("#compose .file_input").get_on_handler("change");
const change_handler = $("#compose input.file_input").get_on_handler("change");
const files = ["file1", "file2"];
const event = {
target: {
@ -417,7 +424,7 @@ test("copy_paste", ({override, override_rewire}) => {
const paste_handler = $("#compose").get_on_handler("paste");
let get_as_file_called = false;
let event = {
originalEvent: {
originalEvent: new ClipboardEvent({
clipboardData: {
items: [
{
@ -428,10 +435,11 @@ test("copy_paste", ({override, override_rewire}) => {
},
{
kind: "notfile",
getAsFile: () => null,
},
],
},
},
}),
preventDefault() {},
};
let upload_files_called = false;
@ -449,7 +457,7 @@ test("copy_paste", ({override, override_rewire}) => {
assert.ok(compose_actions_start_called);
upload_files_called = false;
event = {
originalEvent: {},
originalEvent: new ClipboardEvent({}),
};
paste_handler(event);
assert.equal(upload_files_called, false);
@ -585,7 +593,7 @@ test("uppy_events", ({override_rewire, mock_template}) => {
assert.ok(compose_ui_replace_syntax_called);
compose_ui_replace_syntax_called = false;
on_upload_error_callback(file, null, null);
on_upload_error_callback(file, null, undefined);
assert.ok(compose_ui_replace_syntax_called);
$("#compose_banners .upload_banner .upload_msg").text("");