"use strict";
const ClipboardJS = require("clipboard");
const XDate = require("xdate");
const render_message_edit_form = require("../templates/message_edit_form.hbs");
const render_topic_edit_form = require("../templates/topic_edit_form.hbs");
const currently_editing_messages = new Map();
let currently_deleting_messages = [];
let currently_topic_editing_messages = [];
const currently_echoing_messages = new Map();
// These variables are designed to preserve the user's most recent
// choices when editing a group of messages, to make it convenient to
// move several topics in a row with the same settings.
exports.notify_old_thread_default = true;
exports.notify_new_thread_default = true;
const editability_types = {
NO: 1,
NO_LONGER: 2,
// Note: TOPIC_ONLY does not include stream messages with no topic sent
// by someone else. You can edit the topic of such a message by editing
// the topic of the whole recipient_row it appears in, but you can't
// directly edit the topic of such a message.
// Similar story for messages whose topic you can change only because
// you are an admin.
TOPIC_ONLY: 3,
FULL: 4,
};
exports.editability_types = editability_types;
function is_topic_editable(message, edit_limit_seconds_buffer) {
const now = new XDate();
edit_limit_seconds_buffer = edit_limit_seconds_buffer || 0;
if (!page_params.realm_allow_message_editing) {
// If message editing is disabled, so is topic editing.
return false;
}
// Organization admins and message senders can edit message topics indefinitely.
if (page_params.is_admin) {
return true;
}
if (message.sent_by_me) {
return true;
}
if (!page_params.realm_allow_community_topic_editing) {
// If you're another non-admin user, you need community topic editing enabled.
return false;
}
// If you're using community topic editing, there's a deadline.
return (
page_params.realm_community_topic_editing_limit_seconds +
edit_limit_seconds_buffer +
now.diffSeconds(message.timestamp * 1000) >
0
);
}
function get_editability(message, edit_limit_seconds_buffer) {
edit_limit_seconds_buffer = edit_limit_seconds_buffer || 0;
if (!message) {
return editability_types.NO;
}
if (!is_topic_editable(message, edit_limit_seconds_buffer)) {
return editability_types.NO;
}
if (message.failed_request) {
// TODO: For completely failed requests, we should be able
// to "edit" the message, but it won't really be like
// other message updates. This commit changed the result
// from FULL to NO, since the prior implementation was
// buggy.
return editability_types.NO;
}
// Locally echoed messages are not editable, since the message hasn't
// finished being sent yet.
if (message.locally_echoed) {
return editability_types.NO;
}
if (!page_params.realm_allow_message_editing) {
return editability_types.NO;
}
if (page_params.realm_message_content_edit_limit_seconds === 0 && message.sent_by_me) {
return editability_types.FULL;
}
if (currently_echoing_messages.has(message.id)) {
return editability_types.NO;
}
const now = new XDate();
if (
page_params.realm_message_content_edit_limit_seconds +
edit_limit_seconds_buffer +
now.diffSeconds(message.timestamp * 1000) >
0 &&
message.sent_by_me
) {
return editability_types.FULL;
}
// time's up!
if (message.type === "stream") {
return editability_types.TOPIC_ONLY;
}
return editability_types.NO_LONGER;
}
exports.get_editability = get_editability;
exports.is_topic_editable = is_topic_editable;
exports.get_deletability = function (message) {
if (page_params.is_admin) {
return true;
}
if (!message.sent_by_me) {
return false;
}
if (message.locally_echoed) {
return false;
}
if (!page_params.realm_allow_message_deleting) {
return false;
}
if (page_params.realm_message_content_delete_limit_seconds === 0) {
// This means no time limit for message deletion.
return true;
}
if (page_params.realm_allow_message_deleting) {
const now = new XDate();
if (
page_params.realm_message_content_delete_limit_seconds +
now.diffSeconds(message.timestamp * 1000) >
0
) {
return true;
}
}
return false;
};
exports.update_message_topic_editing_pencil = function () {
if (page_params.realm_allow_message_editing) {
$(".on_hover_topic_edit, .always_visible_topic_edit").show();
} else {
$(".on_hover_topic_edit, .always_visible_topic_edit").hide();
}
};
exports.hide_message_edit_spinner = function (row) {
const spinner = row.find(".message_edit_spinner");
loading.destroy_indicator(spinner);
$("#message_edit_form .message_edit_save").show();
$("#message_edit_form .message_edit_cancel").show();
};
exports.show_message_edit_spinner = function (row) {
const spinner = row.find(".message_edit_spinner");
loading.make_indicator(spinner);
$("#message_edit_form .message_edit_save").hide();
$("#message_edit_form .message_edit_cancel").hide();
};
exports.show_topic_edit_spinner = function (row) {
const spinner = row.find(".topic_edit_spinner");
loading.make_indicator(spinner);
spinner.css({height: ""});
$(".topic_edit_save").hide();
$(".topic_edit_cancel").hide();
};
exports.hide_topic_move_spinner = function () {
const spinner = $("#move_topic_modal .topic_move_spinner");
loading.destroy_indicator(spinner);
$("#move_topic_modal .modal-footer").show();
};
exports.show_topic_move_spinner = function () {
const spinner = $("#move_topic_modal .topic_move_spinner");
loading.make_indicator(spinner);
$("#move_topic_modal .modal-footer").hide();
};
exports.end_if_focused_on_inline_topic_edit = function () {
const focused_elem = $(".topic_edit_form").find(":focus");
if (focused_elem.length === 1) {
focused_elem.trigger("blur");
const recipient_row = focused_elem.closest(".recipient_row");
exports.end_inline_topic_edit(recipient_row);
}
};
exports.end_if_focused_on_message_row_edit = function () {
const focused_elem = $(".message_edit").find(":focus");
if (focused_elem.length === 1) {
focused_elem.trigger("blur");
const row = focused_elem.closest(".message_row");
exports.end_message_row_edit(row);
}
};
function handle_message_row_edit_keydown(e) {
const code = e.keyCode || e.which;
switch (code) {
case 13:
if ($(e.target).hasClass("message_edit_content")) {
// Pressing Enter to save edits is coupled with Enter to send
if (composebox_typeahead.should_enter_send(e)) {
const row = $(".message_edit_content").filter(":focus").closest(".message_row");
const message_edit_save_button = row.find(".message_edit_save");
if (message_edit_save_button.prop("disabled")) {
// In cases when the save button is disabled
// we need to disable save on pressing Enter
// Prevent default to avoid new-line on pressing
// Enter inside the textarea in this case
e.preventDefault();
return;
}
exports.save_message_row_edit(row);
e.stopPropagation();
e.preventDefault();
} else {
composebox_typeahead.handle_enter($(e.target), e);
return;
}
} else if (
$(e.target).hasClass("message_edit_topic") ||
$(e.target).hasClass("message_edit_topic_propagate")
) {
const row = $(e.target).closest(".message_row");
exports.save_message_row_edit(row);
e.stopPropagation();
e.preventDefault();
}
return;
case 27: // Handle escape keys in the message_edit form.
exports.end_if_focused_on_message_row_edit();
e.stopPropagation();
e.preventDefault();
return;
default:
return;
}
}
function handle_inline_topic_edit_keydown(e) {
let row;
const code = e.keyCode || e.which;
switch (code) {
case 13: // Handle Enter key in the recipient bar/inline topic edit form
row = $(e.target).closest(".recipient_row");
exports.save_inline_topic_edit(row);
e.stopPropagation();
e.preventDefault();
return;
case 27: // handle Esc
exports.end_if_focused_on_inline_topic_edit();
e.stopPropagation();
e.preventDefault();
return;
default:
return;
}
}
function timer_text(seconds_left) {
const minutes = Math.floor(seconds_left / 60);
const seconds = seconds_left % 60;
if (minutes >= 1) {
return i18n.t("__minutes__ min to edit", {minutes: minutes.toString()});
} else if (seconds_left >= 10) {
return i18n.t("__seconds__ sec to edit", {seconds: (seconds - (seconds % 5)).toString()});
}
return i18n.t("__seconds__ sec to edit", {seconds: seconds.toString()});
}
function edit_message(row, raw_content) {
row.find(".message_reactions").hide();
condense.hide_message_expander(row);
const content_top = row.find(".message_top_line")[0].getBoundingClientRect().top;
const message = current_msg_list.get(rows.id(row));
// We potentially got to this function by clicking a button that implied the
// user would be able to edit their message. Give a little bit of buffer in
// case the button has been around for a bit, e.g. we show the
// edit_content_button (hovering pencil icon) as long as the user would have
// been able to click it at the time the mouse entered the message_row. Also
// a buffer in case their computer is slow, or stalled for a second, etc
// If you change this number also change edit_limit_buffer in
// zerver.views.message_edit.update_message_backend
const seconds_left_buffer = 5;
const editability = get_editability(message, seconds_left_buffer);
const is_editable =
editability === exports.editability_types.TOPIC_ONLY ||
editability === exports.editability_types.FULL;
const max_file_upload_size = page_params.max_file_upload_size_mib;
let file_upload_enabled = false;
if (max_file_upload_size > 0) {
file_upload_enabled = true;
}
const show_video_chat_button = compose.compute_show_video_chat_button();
const show_edit_stream = message.is_stream && page_params.is_admin;
// current message's stream has been already been added and selected in handlebar
const available_streams = show_edit_stream
? stream_data.subscribed_subs().filter((s) => s.stream_id !== message.stream_id)
: null;
const form = $(
render_message_edit_form({
is_stream: message.type === "stream",
message_id: message.id,
is_editable,
is_content_editable: editability === exports.editability_types.FULL,
has_been_editable: editability !== editability_types.NO,
topic: message.topic,
content: raw_content,
file_upload_enabled,
show_video_chat_button,
minutes_to_edit: Math.floor(page_params.realm_message_content_edit_limit_seconds / 60),
show_edit_stream,
available_streams,
stream_id: message.stream_id,
stream_name: message.stream,
notify_new_thread: exports.notify_new_thread_default,
notify_old_thread: exports.notify_old_thread_default,
}),
);
const edit_obj = {form, raw_content};
currently_editing_messages.set(message.id, edit_obj);
current_msg_list.show_edit_message(row, edit_obj);
form.on("keydown", handle_message_row_edit_keydown);
upload.feature_check($("#attach_files_" + rows.id(row)));
const message_edit_stream = row.find("#select_stream_id_" + message.id);
const stream_header_colorblock = row.find(".stream_header_colorblock");
const message_edit_content = row.find("textarea.message_edit_content");
const message_edit_topic = row.find("input.message_edit_topic");
const message_edit_topic_propagate = row.find("select.message_edit_topic_propagate");
const message_edit_breadcrumb_messages = row.find("div.message_edit_breadcrumb_messages");
const message_edit_countdown_timer = row.find(".message_edit_countdown_timer");
const copy_message = row.find(".copy_message");
ui_util.decorate_stream_bar(message.stream, stream_header_colorblock, false);
message_edit_stream.on("change", function () {
const stream_name = stream_data.maybe_get_stream_name(parseInt(this.value, 10));
ui_util.decorate_stream_bar(stream_name, stream_header_colorblock, false);
});
if (editability === editability_types.NO) {
message_edit_content.attr("readonly", "readonly");
message_edit_topic.attr("readonly", "readonly");
new ClipboardJS(copy_message[0]);
} else if (editability === editability_types.NO_LONGER) {
// You can currently only reach this state in non-streams. If that
// changes (e.g. if we stop allowing topics to be modified forever
// in streams), then we'll need to disable
// row.find('input.message_edit_topic') as well.
message_edit_content.attr("readonly", "readonly");
message_edit_countdown_timer.text(i18n.t("View source"));
new ClipboardJS(copy_message[0]);
} else if (editability === editability_types.TOPIC_ONLY) {
message_edit_content.attr("readonly", "readonly");
// Hint why you can edit the topic but not the message content
message_edit_countdown_timer.text(i18n.t("Topic editing only"));
new ClipboardJS(copy_message[0]);
} else if (editability === editability_types.FULL) {
copy_message.remove();
const edit_id = "#message_edit_content_" + rows.id(row);
const listeners = resize.watch_manual_resize(edit_id);
if (listeners) {
currently_editing_messages.get(rows.id(row)).listeners = listeners;
}
composebox_typeahead.initialize_compose_typeahead(edit_id);
compose.handle_keyup(null, $(edit_id).expectOne());
$(edit_id).on("keydown", function (event) {
compose.handle_keydown(event, $(this).expectOne());
});
$(edit_id).on("keyup", function (event) {
compose.handle_keyup(event, $(this).expectOne());
});
}
// Add tooltip
if (
editability !== editability_types.NO &&
page_params.realm_message_content_edit_limit_seconds > 0
) {
row.find(".message-edit-timer-control-group").show();
row.find("#message_edit_tooltip").tooltip({
animation: false,
placement: "left",
template:
'
',
});
}
// add timer
if (
editability === editability_types.FULL &&
page_params.realm_message_content_edit_limit_seconds > 0
) {
// Give them at least 10 seconds.
// If you change this number also change edit_limit_buffer in
// zerver.views.message_edit.update_message_backend
const min_seconds_to_edit = 10;
const now = new XDate();
let seconds_left =
page_params.realm_message_content_edit_limit_seconds +
now.diffSeconds(message.timestamp * 1000);
seconds_left = Math.floor(Math.max(seconds_left, min_seconds_to_edit));
// I believe this needs to be defined outside the countdown_timer, since
// row just refers to something like the currently selected message, and
// can change out from under us
const message_edit_save = row.find("button.message_edit_save");
// Do this right away, rather than waiting for the timer to do its first update,
// since otherwise there is a noticeable lag
message_edit_countdown_timer.text(timer_text(seconds_left));
const countdown_timer = setInterval(() => {
seconds_left -= 1;
if (seconds_left <= 0) {
clearInterval(countdown_timer);
message_edit_content.prop("readonly", "readonly");
if (message.type === "stream") {
message_edit_topic.prop("readonly", "readonly");
message_edit_topic_propagate.hide();
message_edit_breadcrumb_messages.hide();
}
// We don't go directly to a "TOPIC_ONLY" type state (with an active Save button),
// since it isn't clear what to do with the half-finished edit. It's nice to keep
// the half-finished edit around so that they can copy-paste it, but we don't want
// people to think "Save" will save the half-finished edit.
message_edit_save.addClass("disabled");
message_edit_countdown_timer.text(i18n.t("Time's up!"));
} else {
message_edit_countdown_timer.text(timer_text(seconds_left));
}
}, 1000);
}
if (!is_editable) {
row.find(".message_edit_close").trigger("focus");
} else if (message.type === "stream" && message.topic === compose.empty_topic_placeholder()) {
message_edit_topic.val("");
message_edit_topic.trigger("focus");
} else if (editability === editability_types.TOPIC_ONLY) {
row.find(".message_edit_topic").trigger("focus");
} else {
message_edit_content.trigger("focus");
// Put cursor at end of input.
const contents = message_edit_content.val();
message_edit_content.val("");
message_edit_content.val(contents);
}
// Scroll to keep the top of the message content text in the same
// place visually, adjusting for border and padding.
const edit_top = message_edit_content[0].getBoundingClientRect().top;
const scroll_by = edit_top - content_top + 5 - 14;
edit_obj.scrolled_by = scroll_by;
message_viewport.scrollTop(message_viewport.scrollTop() + scroll_by);
const original_stream_id = message.stream_id;
const original_topic = message.topic;
function set_propagate_selector_display() {
const new_topic = message_edit_topic.val();
const new_stream_id = parseInt(message_edit_stream.val(), 10);
const is_topic_edited = new_topic !== original_topic && new_topic !== "";
const is_stream_edited = new_stream_id !== original_stream_id;
message_edit_topic_propagate.toggle(is_topic_edited || is_stream_edited);
message_edit_breadcrumb_messages.toggle(is_stream_edited);
}
if (!message.locally_echoed) {
message_edit_topic.on("keyup", () => {
set_propagate_selector_display();
});
message_edit_stream.on("change", () => {
set_propagate_selector_display();
});
}
}
function start_edit_maintaining_scroll(row, content) {
edit_message(row, content);
const row_bottom = row.height() + row.offset().top;
const composebox_top = $("#compose").offset().top;
if (row_bottom > composebox_top) {
message_viewport.scrollTop(message_viewport.scrollTop() + row_bottom - composebox_top);
}
}
function start_edit_with_content(row, content, edit_box_open_callback) {
start_edit_maintaining_scroll(row, content);
if (edit_box_open_callback) {
edit_box_open_callback();
}
upload.setup_upload({
mode: "edit",
row: rows.id(row),
});
}
exports.start = function (row, edit_box_open_callback) {
const message = current_msg_list.get(rows.id(row));
if (message === undefined) {
blueslip.error("Couldn't find message ID for edit " + rows.id(row));
return;
}
if (message.raw_content) {
start_edit_with_content(row, message.raw_content, edit_box_open_callback);
return;
}
const msg_list = current_msg_list;
channel.get({
url: "/json/messages/" + message.id,
idempotent: true,
success(data) {
if (current_msg_list === msg_list) {
message.raw_content = data.raw_content;
start_edit_with_content(row, message.raw_content, edit_box_open_callback);
}
},
});
};
exports.start_topic_edit = function (recipient_row) {
const form = $(render_topic_edit_form());
current_msg_list.show_edit_topic_on_recipient_row(recipient_row, form);
form.on("keydown", handle_inline_topic_edit_keydown);
const msg_id = rows.id_for_recipient_row(recipient_row);
const message = current_msg_list.get(msg_id);
let topic = message.topic;
if (topic === compose.empty_topic_placeholder()) {
topic = "";
}
form.find(".inline_topic_edit").val(topic).trigger("select").trigger("focus");
};
exports.is_editing = function (id) {
return currently_editing_messages.has(id);
};
exports.end_inline_topic_edit = function (row) {
current_msg_list.hide_edit_topic_on_recipient_row(row);
};
exports.end_message_row_edit = function (row) {
const message = current_msg_list.get(rows.id(row));
if (message !== undefined && currently_editing_messages.has(message.id)) {
const scroll_by = currently_editing_messages.get(message.id).scrolled_by;
message_viewport.scrollTop(message_viewport.scrollTop() - scroll_by);
// Clean up resize event listeners
const listeners = currently_editing_messages.get(message.id).listeners;
const edit_box = document.querySelector("#message_edit_content_" + message.id);
if (listeners !== undefined) {
// Event listeners to cleanup are only set in some edit types
edit_box.removeEventListener("mousedown", listeners[0]);
document.body.removeEventListener("mouseup", listeners[1]);
}
currently_editing_messages.delete(message.id);
current_msg_list.hide_edit_message(row);
compose.abort_zoom(message.id);
}
condense.show_message_expander(row);
row.find(".message_reactions").show();
// We have to blur out text fields, or else hotkeys.js
// thinks we are still editing.
row.find(".message_edit").trigger("blur");
};
exports.save_inline_topic_edit = function (row) {
const msg_list = current_msg_list;
let message_id = rows.id_for_recipient_row(row);
const message = current_msg_list.get(message_id);
const old_topic = message.topic;
const new_topic = row.find(".inline_topic_edit").val();
const topic_changed = new_topic !== old_topic && new_topic.trim() !== "";
if (!topic_changed) {
// this means the inline_topic_edit was opened and submitted without
// changing anything, therefore, we should just close the inline topic edit.
exports.end_inline_topic_edit(row);
return;
}
exports.show_topic_edit_spinner(row);
if (message.locally_echoed) {
if (topic_changed) {
echo.edit_locally(message, {new_topic});
row = current_msg_list.get_row(message_id);
}
exports.end_inline_topic_edit(row);
return;
}
const request = {
message_id: message.id,
topic: new_topic,
propagate_mode: "change_later",
};
channel.patch({
url: "/json/messages/" + message.id,
data: request,
success() {
const spinner = row.find(".topic_edit_spinner");
loading.destroy_indicator(spinner);
},
error(xhr) {
const spinner = row.find(".topic_edit_spinner");
loading.destroy_indicator(spinner);
if (msg_list === current_msg_list) {
message_id = rows.id_for_recipient_row(row);
const message = channel.xhr_error_message(i18n.t("Error saving edit"), xhr);
row.find(".edit_error").text(message).css("display", "inline-block");
}
},
});
};
exports.save_message_row_edit = function (row) {
const msg_list = current_msg_list;
let message_id = rows.id(row);
const message = current_msg_list.get(message_id);
let changed = false;
let edit_locally_echoed = false;
const new_content = row.find(".message_edit_content").val();
let topic_changed = false;
let new_topic;
const old_topic = message.topic;
let stream_changed = false;
let new_stream_id;
const old_stream_id = message.stream_id;
exports.show_message_edit_spinner(row);
if (message.type === "stream") {
new_topic = row.find(".message_edit_topic").val();
topic_changed = new_topic !== old_topic && new_topic.trim() !== "";
new_stream_id = parseInt($("#select_stream_id_" + message_id).val(), 10);
stream_changed = new_stream_id !== old_stream_id;
}
// Editing a not-yet-acked message (because the original send attempt failed)
// just results in the in-memory message being changed
if (message.locally_echoed) {
if (new_content !== message.raw_content || topic_changed || stream_changed) {
// `edit_locally` handles the case where `new_topic/new_stream_id` is undefined
echo.edit_locally(message, {
raw_content: new_content,
new_topic,
new_stream_id,
});
row = current_msg_list.get_row(message_id);
}
exports.end_message_row_edit(row);
return;
}
const request = {message_id: message.id};
if (topic_changed || stream_changed) {
const selected_topic_propagation =
row.find("select.message_edit_topic_propagate").val() || "change_later";
const send_notification_to_old_thread = row
.find(".send_notification_to_old_thread")
.is(":checked");
const send_notification_to_new_thread = row
.find(".send_notification_to_new_thread")
.is(":checked");
request.propagate_mode = selected_topic_propagation;
request.send_notification_to_old_thread = send_notification_to_old_thread;
request.send_notification_to_new_thread = send_notification_to_new_thread;
exports.notify_old_thread_default = send_notification_to_old_thread;
exports.notify_new_thread_default = send_notification_to_new_thread;
changed = true;
}
if (topic_changed) {
request.topic = new_topic;
}
if (stream_changed) {
request.stream_id = new_stream_id;
}
if (new_content !== message.raw_content) {
request.content = new_content;
changed = true;
}
if (!changed) {
// If they didn't change anything, just cancel it.
exports.end_message_row_edit(row);
return;
}
if (
changed &&
!topic_changed &&
!stream_changed &&
!markdown.contains_backend_only_syntax(new_content)
) {
// If the topic isn't changed, and the new message content
// could have been locally echoed, than we can locally echo
// the edit.
currently_echoing_messages.set(message_id, {
raw_content: new_content,
orig_content: message.content,
orig_raw_content: message.raw_content,
// Store flags that are about user interaction with the
// message so that echo.edit_locally() can restore these
// flags.
starred: message.starred,
historical: message.historical,
collapsed: message.collapsed,
// These flags are rendering artifacts we'll want if the
// edit fails and we need to revert to the original
// rendering of the message.
alerted: message.alerted,
mentioned: message.mentioned,
mentioned_me_directly: message.mentioned,
});
edit_locally_echoed = true;
// Settings these attributes causes a "SAVING" notice to
// briefly appear where "EDITED" would normally appear until
// the message is acknowledged by the server.
message.local_edit_timestamp = Math.round(new Date().getTime() / 1000);
echo.edit_locally(message, currently_echoing_messages.get(message_id));
row = current_msg_list.get_row(message_id);
exports.end_message_row_edit(row);
}
channel.patch({
url: "/json/messages/" + message.id,
data: request,
success() {
if (edit_locally_echoed) {
delete message.local_edit_timestamp;
currently_echoing_messages.delete(message_id);
}
exports.hide_message_edit_spinner(row);
},
error(xhr) {
if (msg_list === current_msg_list) {
message_id = rows.id(row);
if (edit_locally_echoed) {
const echoed_message = message_store.get(message_id);
const echo_data = currently_echoing_messages.get(message_id);
delete echoed_message.local_edit_timestamp;
currently_echoing_messages.delete(message_id);
// Restore the original content.
echo.edit_locally(echoed_message, {
content: echo_data.orig_content,
raw_content: echo_data.orig_raw_content,
mentioned: echo_data.mentioned,
mentioned_me_directly: echo_data.mentioned_me_directly,
alerted: echo_data.alerted,
});
row = current_msg_list.get_row(message_id);
if (!exports.is_editing(message_id)) {
// Return to the message editing open UI state.
start_edit_maintaining_scroll(row, echo_data.orig_raw_content);
}
}
exports.hide_message_edit_spinner(row);
const message = channel.xhr_error_message(i18n.t("Error saving edit"), xhr);
row.find(".edit_error").text(message).show();
}
},
});
// The message will automatically get replaced via message_list.update_message.
};
exports.maybe_show_edit = function (row, id) {
if (currently_editing_messages.has(id)) {
current_msg_list.show_edit_message(row, currently_editing_messages.get(id));
}
};
exports.edit_last_sent_message = function () {
const msg = current_msg_list.get_last_message_sent_by_me();
if (!msg) {
return;
}
if (!msg.id) {
blueslip.error("Message has invalid id in edit_last_sent_message.");
return;
}
const msg_editability_type = exports.get_editability(msg, 5);
if (msg_editability_type !== editability_types.FULL) {
return;
}
const msg_row = current_msg_list.get_row(msg.id);
if (!msg_row) {
// This should never happen, since we got the message above
// from current_msg_list.
blueslip.error("Could not find row for id " + msg.id);
return;
}
current_msg_list.select_id(msg.id, {then_scroll: true, from_scroll: true});
// Finally do the real work!
compose_actions.cancel();
exports.start(msg_row, () => {
$("#message_edit_content").trigger("focus");
});
};
function hide_delete_btn_show_spinner(deleting) {
if (deleting) {
$("do_delete_message_button").prop("disabled", true);
$("#delete_message_modal > div.modal-footer > button").hide();
const delete_spinner = $("#do_delete_message_spinner");
loading.make_indicator(delete_spinner, {abs_positioned: true});
} else {
loading.destroy_indicator($("#do_delete_message_spinner"));
$("#do_delete_message_button").prop("disabled", false);
$("#delete_message_modal > div.modal-footer > button").show();
}
}
exports.delete_message = function (msg_id) {
$("#delete-message-error").html("");
$("#delete_message_modal").modal("show");
if (currently_deleting_messages.includes(msg_id)) {
hide_delete_btn_show_spinner(true);
} else {
hide_delete_btn_show_spinner(false);
}
$("#do_delete_message_button")
.off()
.on("click", (e) => {
e.stopPropagation();
e.preventDefault();
currently_deleting_messages.push(msg_id);
hide_delete_btn_show_spinner(true);
channel.del({
url: "/json/messages/" + msg_id,
success() {
$("#delete_message_modal").modal("hide");
currently_deleting_messages = currently_deleting_messages.filter(
(id) => id !== msg_id,
);
hide_delete_btn_show_spinner(false);
},
error(xhr) {
currently_deleting_messages = currently_deleting_messages.filter(
(id) => id !== msg_id,
);
hide_delete_btn_show_spinner(false);
ui_report.error(
i18n.t("Error deleting message"),
xhr,
$("#delete-message-error"),
);
},
});
});
};
exports.delete_topic = function (stream_id, topic_name) {
channel.post({
url: "/json/streams/" + stream_id + "/delete_topic",
data: {
topic_name,
},
success() {
$("#delete_topic_modal").modal("hide");
},
});
};
exports.handle_narrow_deactivated = function () {
for (const [idx, elem] of currently_editing_messages) {
if (current_msg_list.get(idx) !== undefined) {
const row = current_msg_list.get_row(idx);
current_msg_list.show_edit_message(row, elem);
}
}
};
exports.move_topic_containing_message_to_stream = function (
message_id,
new_stream_id,
new_topic_name,
send_notification_to_new_thread,
send_notification_to_old_thread,
) {
function reset_modal_ui() {
currently_topic_editing_messages = currently_topic_editing_messages.filter(
(id) => id !== message_id,
);
exports.hide_topic_move_spinner();
$("#move_topic_modal").modal("hide");
}
if (currently_topic_editing_messages.includes(message_id)) {
exports.hide_topic_move_spinner();
$("#topic_stream_edit_form_error .error-msg").text(
i18n.t("A Topic Move already in progress."),
);
$("#topic_stream_edit_form_error").show();
return;
}
currently_topic_editing_messages.push(message_id);
const request = {
stream_id: new_stream_id,
propagate_mode: "change_all",
topic: new_topic_name,
send_notification_to_old_thread,
send_notification_to_new_thread,
};
exports.notify_old_thread_default = send_notification_to_old_thread;
exports.notify_new_thread_default = send_notification_to_new_thread;
channel.patch({
url: "/json/messages/" + message_id,
data: request,
success() {
// The main UI will update via receiving the event
// from server_events.js.
reset_modal_ui();
},
error(xhr) {
reset_modal_ui();
ui_report.error(i18n.t("Error moving the topic"), xhr, $("#home-error"), 4000);
},
});
};
window.message_edit = exports;