mirror of
https://github.com/zulip/zulip.git
synced 2025-11-06 06:53:25 +00:00
compose: Extract "compose_validate.js".
This commit first moves the compose.validate() function out with the functions that are needed by it. Then one by one checked for which function is now not needed in compose.js. This moves all validation related functions out of "compose.js" to "compose_validate.js". Splitting compose announce variables out of compose.js. This commit moves the "user_acknowledged_all_everyone" and "user_acknowledged_announce" out of compose.js to reduce cyclic dependency of compose_validate on compose.js. Moving wildcard mentions to compose_validate. The wildcard mention settings are mostly used while validating. Also to reduce the cyclic dependence of compose in compose_validate, the related wildcard mentions are moved out to compose_vaidate.js. This also converts reset_acknowledged functions to set values by passing values.
This commit is contained in:
@@ -2,10 +2,7 @@ import $ from "jquery";
|
||||
import _ from "lodash";
|
||||
|
||||
import render_compose from "../templates/compose.hbs";
|
||||
import render_compose_all_everyone from "../templates/compose_all_everyone.hbs";
|
||||
import render_compose_announce from "../templates/compose_announce.hbs";
|
||||
import render_compose_invite_users from "../templates/compose_invite_users.hbs";
|
||||
import render_compose_not_subscribed from "../templates/compose_not_subscribed.hbs";
|
||||
import render_compose_private_stream_alert from "../templates/compose_private_stream_alert.hbs";
|
||||
|
||||
import * as blueslip from "./blueslip";
|
||||
@@ -14,9 +11,9 @@ import * as common from "./common";
|
||||
import * as compose_actions from "./compose_actions";
|
||||
import * as compose_error from "./compose_error";
|
||||
import * as compose_fade from "./compose_fade";
|
||||
import * as compose_pm_pill from "./compose_pm_pill";
|
||||
import * as compose_state from "./compose_state";
|
||||
import * as compose_ui from "./compose_ui";
|
||||
import * as compose_validate from "./compose_validate";
|
||||
import * as drafts from "./drafts";
|
||||
import * as echo from "./echo";
|
||||
import * as giphy from "./giphy";
|
||||
@@ -34,7 +31,6 @@ import * as rows from "./rows";
|
||||
import * as rtl from "./rtl";
|
||||
import * as sent_messages from "./sent_messages";
|
||||
import * as server_events from "./server_events";
|
||||
import * as settings_config from "./settings_config";
|
||||
import * as settings_data from "./settings_data";
|
||||
import * as stream_data from "./stream_data";
|
||||
import * as stream_edit from "./stream_edit";
|
||||
@@ -56,13 +52,8 @@ import * as zcommand from "./zcommand";
|
||||
false: user typed @all/@everyone;
|
||||
true: user clicked YES */
|
||||
|
||||
let user_acknowledged_all_everyone;
|
||||
let user_acknowledged_announce;
|
||||
let wildcard_mention;
|
||||
let uppy;
|
||||
|
||||
export let wildcard_mention_large_stream_threshold = 15;
|
||||
export const announce_warn_threshold = 60;
|
||||
export const uploads_domain = document.location.protocol + "//" + document.location.host;
|
||||
export const uploads_path = "/user_uploads";
|
||||
|
||||
@@ -76,24 +67,6 @@ function make_uploads_relative(content) {
|
||||
return content.replace(uploads_re, "]($1)");
|
||||
}
|
||||
|
||||
function show_all_everyone_warnings(stream_id) {
|
||||
const stream_count = peer_data.get_subscriber_count(stream_id) || 0;
|
||||
|
||||
const all_everyone_template = render_compose_all_everyone({
|
||||
count: stream_count,
|
||||
mention: wildcard_mention,
|
||||
});
|
||||
const error_area_all_everyone = $("#compose-all-everyone");
|
||||
|
||||
// only show one error for any number of @all or @everyone mentions
|
||||
if (!error_area_all_everyone.is(":visible")) {
|
||||
error_area_all_everyone.append(all_everyone_template);
|
||||
}
|
||||
|
||||
error_area_all_everyone.show();
|
||||
user_acknowledged_all_everyone = false;
|
||||
}
|
||||
|
||||
export function compute_show_video_chat_button() {
|
||||
const available_providers = page_params.realm_available_video_chat_providers;
|
||||
if (page_params.realm_video_chat_provider === available_providers.disabled.id) {
|
||||
@@ -116,37 +89,6 @@ export function update_video_chat_button_display() {
|
||||
$(".message-edit-feature-group .video_link").toggle(show_video_chat_button);
|
||||
}
|
||||
|
||||
export function clear_all_everyone_warnings() {
|
||||
$("#compose-all-everyone").hide();
|
||||
$("#compose-all-everyone").empty();
|
||||
$("#compose-send-status").hide();
|
||||
}
|
||||
|
||||
function show_sending_indicator(whats_happening) {
|
||||
$("#sending-indicator").text(whats_happening);
|
||||
$("#sending-indicator").show();
|
||||
}
|
||||
|
||||
function show_announce_warnings(stream_id) {
|
||||
const stream_count = peer_data.get_subscriber_count(stream_id) || 0;
|
||||
|
||||
const announce_template = render_compose_announce({count: stream_count});
|
||||
const error_area_announce = $("#compose-announce");
|
||||
|
||||
if (!error_area_announce.is(":visible")) {
|
||||
error_area_announce.append(announce_template);
|
||||
}
|
||||
|
||||
error_area_announce.show();
|
||||
user_acknowledged_announce = false;
|
||||
}
|
||||
|
||||
export function clear_announce_warnings() {
|
||||
$("#compose-announce").hide();
|
||||
$("#compose-announce").empty();
|
||||
$("#compose-send-status").hide();
|
||||
}
|
||||
|
||||
export function clear_invites() {
|
||||
$("#compose_invite_users").hide();
|
||||
$("#compose_invite_users").empty();
|
||||
@@ -157,14 +99,6 @@ export function clear_private_stream_alert() {
|
||||
$("#compose_private_stream_alert").empty();
|
||||
}
|
||||
|
||||
export function reset_user_acknowledged_all_everyone_flag() {
|
||||
user_acknowledged_all_everyone = undefined;
|
||||
}
|
||||
|
||||
export function reset_user_acknowledged_announce_flag() {
|
||||
user_acknowledged_announce = undefined;
|
||||
}
|
||||
|
||||
export function clear_preview_area() {
|
||||
$("#compose-textarea").show();
|
||||
$("#compose .undo_markdown_preview").hide();
|
||||
@@ -363,7 +297,7 @@ export function finish() {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
if (!validate()) {
|
||||
if (!compose_validate.validate()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@@ -395,422 +329,6 @@ export function update_email(user_id, new_email) {
|
||||
compose_state.private_message_recipient(reply_to);
|
||||
}
|
||||
|
||||
export function get_invalid_recipient_emails() {
|
||||
const private_recipients = util.extract_pm_recipients(
|
||||
compose_state.private_message_recipient(),
|
||||
);
|
||||
const invalid_recipients = private_recipients.filter(
|
||||
(email) => !people.is_valid_email_for_compose(email),
|
||||
);
|
||||
|
||||
return invalid_recipients;
|
||||
}
|
||||
|
||||
function check_unsubscribed_stream_for_send(stream_name, autosubscribe) {
|
||||
let result;
|
||||
if (!autosubscribe) {
|
||||
return "not-subscribed";
|
||||
}
|
||||
|
||||
// In the rare circumstance of the autosubscribe option, we
|
||||
// *Synchronously* try to subscribe to the stream before sending
|
||||
// the message. This is deprecated and we hope to remove it; see
|
||||
// #4650.
|
||||
channel.post({
|
||||
url: "/json/subscriptions/exists",
|
||||
data: {stream: stream_name, autosubscribe: true},
|
||||
async: false,
|
||||
success(data) {
|
||||
if (data.subscribed) {
|
||||
result = "subscribed";
|
||||
} else {
|
||||
result = "not-subscribed";
|
||||
}
|
||||
},
|
||||
error(xhr) {
|
||||
if (xhr.status === 404) {
|
||||
result = "does-not-exist";
|
||||
} else {
|
||||
result = "error";
|
||||
}
|
||||
},
|
||||
});
|
||||
return result;
|
||||
}
|
||||
|
||||
export function wildcard_mention_allowed() {
|
||||
if (
|
||||
page_params.realm_wildcard_mention_policy ===
|
||||
settings_config.wildcard_mention_policy_values.by_everyone.code
|
||||
) {
|
||||
return true;
|
||||
}
|
||||
if (
|
||||
page_params.realm_wildcard_mention_policy ===
|
||||
settings_config.wildcard_mention_policy_values.nobody.code
|
||||
) {
|
||||
return false;
|
||||
}
|
||||
if (
|
||||
page_params.realm_wildcard_mention_policy ===
|
||||
settings_config.wildcard_mention_policy_values.by_stream_admins_only.code
|
||||
) {
|
||||
// TODO: Check the user's stream-level role once stream-level admins exist.
|
||||
return page_params.is_admin;
|
||||
}
|
||||
|
||||
if (
|
||||
page_params.realm_wildcard_mention_policy ===
|
||||
settings_config.wildcard_mention_policy_values.by_moderators_only.code
|
||||
) {
|
||||
return page_params.is_admin || page_params.is_moderator;
|
||||
}
|
||||
// TODO: Uncomment when we add support for stream-level administrators.
|
||||
// if (
|
||||
// page_params.realm_wildcard_mention_policy ===
|
||||
// settings_config.wildcard_mention_policy_values.by_admins_only.code
|
||||
// ) {
|
||||
// return page_params.is_admin;
|
||||
// }
|
||||
if (
|
||||
page_params.realm_wildcard_mention_policy ===
|
||||
settings_config.wildcard_mention_policy_values.by_full_members.code
|
||||
) {
|
||||
if (page_params.is_admin) {
|
||||
return true;
|
||||
}
|
||||
const person = people.get_by_user_id(page_params.user_id);
|
||||
const current_datetime = new Date(Date.now());
|
||||
const person_date_joined = new Date(person.date_joined);
|
||||
const days = (current_datetime - person_date_joined) / 1000 / 86400;
|
||||
|
||||
return days >= page_params.realm_waiting_period_threshold && !page_params.is_guest;
|
||||
}
|
||||
return !page_params.is_guest;
|
||||
}
|
||||
|
||||
export function set_wildcard_mention_large_stream_threshold(value) {
|
||||
wildcard_mention_large_stream_threshold = value;
|
||||
}
|
||||
|
||||
function validate_stream_message_mentions(stream_id) {
|
||||
const stream_count = peer_data.get_subscriber_count(stream_id) || 0;
|
||||
|
||||
// If the user is attempting to do a wildcard mention in a large
|
||||
// stream, check if they permission to do so.
|
||||
if (wildcard_mention !== null && stream_count > wildcard_mention_large_stream_threshold) {
|
||||
if (!wildcard_mention_allowed()) {
|
||||
compose_error.show(
|
||||
$t_html({
|
||||
defaultMessage:
|
||||
"You do not have permission to use wildcard mentions in this stream.",
|
||||
}),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (
|
||||
user_acknowledged_all_everyone === undefined ||
|
||||
user_acknowledged_all_everyone === false
|
||||
) {
|
||||
// user has not seen a warning message yet if undefined
|
||||
show_all_everyone_warnings(stream_id);
|
||||
|
||||
$("#compose-send-button").prop("disabled", false);
|
||||
$("#sending-indicator").hide();
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
// the message no longer contains @all or @everyone
|
||||
clear_all_everyone_warnings();
|
||||
}
|
||||
// at this point, the user has either acknowledged the warning or removed @all / @everyone
|
||||
user_acknowledged_all_everyone = undefined;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
function validate_stream_message_announce(sub) {
|
||||
const stream_count = peer_data.get_subscriber_count(sub.stream_id) || 0;
|
||||
|
||||
if (sub.name === "announce" && stream_count > announce_warn_threshold) {
|
||||
if (user_acknowledged_announce === undefined || user_acknowledged_announce === false) {
|
||||
// user has not seen a warning message yet if undefined
|
||||
show_announce_warnings(sub.stream_id);
|
||||
|
||||
$("#compose-send-button").prop("disabled", false);
|
||||
$("#sending-indicator").hide();
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
clear_announce_warnings();
|
||||
}
|
||||
// at this point, the user has acknowledged the warning
|
||||
user_acknowledged_announce = undefined;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
function validate_stream_message_post_policy(sub) {
|
||||
if (page_params.is_admin) {
|
||||
return true;
|
||||
}
|
||||
|
||||
const stream_post_permission_type = stream_data.stream_post_policy_values;
|
||||
const stream_post_policy = sub.stream_post_policy;
|
||||
|
||||
if (stream_post_policy === stream_post_permission_type.admins.code) {
|
||||
compose_error.show(
|
||||
$t_html({
|
||||
defaultMessage: "Only organization admins are allowed to post to this stream.",
|
||||
}),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (page_params.is_moderator) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (stream_post_policy === stream_post_permission_type.moderators.code) {
|
||||
compose_error.show(
|
||||
$t_html({
|
||||
defaultMessage:
|
||||
"Only organization admins and moderators are allowed to post to this stream.",
|
||||
}),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (page_params.is_guest && stream_post_policy !== stream_post_permission_type.everyone.code) {
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "Guests are not allowed to post to this stream."}),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
const person = people.get_by_user_id(page_params.user_id);
|
||||
const current_datetime = new Date(Date.now());
|
||||
const person_date_joined = new Date(person.date_joined);
|
||||
const days = (current_datetime - person_date_joined) / 1000 / 86400;
|
||||
let error_html;
|
||||
if (
|
||||
stream_post_policy === stream_post_permission_type.non_new_members.code &&
|
||||
days < page_params.realm_waiting_period_threshold
|
||||
) {
|
||||
error_html = $t_html(
|
||||
{
|
||||
defaultMessage:
|
||||
"New members are not allowed to post to this stream.<br />Permission will be granted in {days} days.",
|
||||
},
|
||||
{days},
|
||||
);
|
||||
compose_error.show(error_html);
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
export function validation_error(error_type, stream_name) {
|
||||
let response;
|
||||
|
||||
switch (error_type) {
|
||||
case "does-not-exist":
|
||||
response = $t_html(
|
||||
{
|
||||
defaultMessage:
|
||||
"<p>The stream <b>{stream_name}</b> does not exist.</p><p>Manage your subscriptions <z-link>on your Streams page</z-link>.</p>",
|
||||
},
|
||||
{
|
||||
stream_name,
|
||||
"z-link": (content_html) => `<a href='#streams/all'>${content_html}</a>`,
|
||||
},
|
||||
);
|
||||
compose_error.show(response, $("#stream_message_recipient_stream"));
|
||||
return false;
|
||||
case "error":
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "Error checking subscription"}),
|
||||
$("#stream_message_recipient_stream"),
|
||||
);
|
||||
return false;
|
||||
case "not-subscribed": {
|
||||
const sub = stream_data.get_sub(stream_name);
|
||||
const new_row = render_compose_not_subscribed({
|
||||
should_display_sub_button: stream_data.can_toggle_subscription(sub),
|
||||
});
|
||||
compose_error.show_not_subscribed(new_row, $("#stream_message_recipient_stream"));
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
export function validate_stream_message_address_info(stream_name) {
|
||||
if (stream_data.is_subscribed(stream_name)) {
|
||||
return true;
|
||||
}
|
||||
const autosubscribe = page_params.narrow_stream !== undefined;
|
||||
const error_type = check_unsubscribed_stream_for_send(stream_name, autosubscribe);
|
||||
return validation_error(error_type, stream_name);
|
||||
}
|
||||
|
||||
function validate_stream_message() {
|
||||
const stream_name = compose_state.stream_name();
|
||||
if (stream_name === "") {
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "Please specify a stream"}),
|
||||
$("#stream_message_recipient_stream"),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (page_params.realm_mandatory_topics) {
|
||||
const topic = compose_state.topic();
|
||||
if (topic === "") {
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "Please specify a topic"}),
|
||||
$("#stream_message_recipient_topic"),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
const sub = stream_data.get_sub(stream_name);
|
||||
if (!sub) {
|
||||
return validation_error("does-not-exist", stream_name);
|
||||
}
|
||||
|
||||
if (!validate_stream_message_post_policy(sub)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/* Note: This is a global and thus accessible in the functions
|
||||
below; it's important that we update this state here before
|
||||
proceeding with further validation. */
|
||||
wildcard_mention = util.find_wildcard_mentions(compose_state.message_content());
|
||||
|
||||
// If both `@all` is mentioned and it's in `#announce`, just validate
|
||||
// for `@all`. Users shouldn't have to hit "yes" more than once.
|
||||
if (wildcard_mention !== null && stream_name === "announce") {
|
||||
if (
|
||||
!validate_stream_message_address_info(stream_name) ||
|
||||
!validate_stream_message_mentions(sub.stream_id)
|
||||
) {
|
||||
return false;
|
||||
}
|
||||
// If either criteria isn't met, just do the normal validation.
|
||||
} else {
|
||||
if (
|
||||
!validate_stream_message_address_info(stream_name) ||
|
||||
!validate_stream_message_mentions(sub.stream_id) ||
|
||||
!validate_stream_message_announce(sub)
|
||||
) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
// The function checks whether the recipients are users of the realm or cross realm users (bots
|
||||
// for now)
|
||||
function validate_private_message() {
|
||||
const user_ids = compose_pm_pill.get_user_ids();
|
||||
|
||||
if (
|
||||
page_params.realm_private_message_policy === 2 && // Frontend check for for PRIVATE_MESSAGE_POLICY_DISABLED
|
||||
(user_ids.length !== 1 || !people.get_by_user_id(user_ids[0]).is_bot)
|
||||
) {
|
||||
// Unless we're composing to a bot
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "Private messages are disabled in this organization."}),
|
||||
$("#private_message_recipient"),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (compose_state.private_message_recipient().length === 0) {
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "Please specify at least one valid recipient"}),
|
||||
$("#private_message_recipient"),
|
||||
);
|
||||
return false;
|
||||
} else if (page_params.realm_is_zephyr_mirror_realm) {
|
||||
// For Zephyr mirroring realms, the frontend doesn't know which users exist
|
||||
return true;
|
||||
}
|
||||
|
||||
const invalid_recipients = get_invalid_recipient_emails();
|
||||
|
||||
let context = {};
|
||||
if (invalid_recipients.length === 1) {
|
||||
context = {recipient: invalid_recipients.join(",")};
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "The recipient {recipient} is not valid"}, context),
|
||||
$("#private_message_recipient"),
|
||||
);
|
||||
return false;
|
||||
} else if (invalid_recipients.length > 1) {
|
||||
context = {recipients: invalid_recipients.join(",")};
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "The recipients {recipients} are not valid"}, context),
|
||||
$("#private_message_recipient"),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
for (const user_id of user_ids) {
|
||||
if (!people.is_person_active(user_id)) {
|
||||
context = {full_name: people.get_by_user_id(user_id).full_name};
|
||||
compose_error.show(
|
||||
$t_html(
|
||||
{defaultMessage: "You cannot send messages to deactivated users."},
|
||||
context,
|
||||
),
|
||||
$("#private_message_recipient"),
|
||||
);
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
export function validate() {
|
||||
$("#compose-send-button").prop("disabled", true).trigger("blur");
|
||||
const message_content = compose_state.message_content();
|
||||
if (reminder.is_deferred_delivery(message_content)) {
|
||||
show_sending_indicator($t({defaultMessage: "Scheduling..."}));
|
||||
} else {
|
||||
show_sending_indicator($t({defaultMessage: "Sending..."}));
|
||||
}
|
||||
|
||||
if (/^\s*$/.test(message_content)) {
|
||||
compose_error.show(
|
||||
$t_html({defaultMessage: "You have nothing to send!"}),
|
||||
$("#compose-textarea"),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($("#zephyr-mirror-error").is(":visible")) {
|
||||
compose_error.show(
|
||||
$t_html({
|
||||
defaultMessage:
|
||||
"You need to be running Zephyr mirroring in order to send messages!",
|
||||
}),
|
||||
);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (compose_state.get_message_type() === "private") {
|
||||
return validate_private_message();
|
||||
}
|
||||
return validate_stream_message();
|
||||
}
|
||||
|
||||
export function handle_keydown(event, textarea) {
|
||||
// The event.key property will have uppercase letter if
|
||||
// the "Shift + <key>" combo was used or the Caps Lock
|
||||
@@ -1121,8 +639,8 @@ export function initialize() {
|
||||
event.preventDefault();
|
||||
|
||||
$(event.target).parents(".compose-all-everyone").remove();
|
||||
user_acknowledged_all_everyone = true;
|
||||
clear_all_everyone_warnings();
|
||||
compose_validate.set_user_acknowledged_all_everyone_flag(true);
|
||||
compose_validate.clear_all_everyone_warnings();
|
||||
finish();
|
||||
});
|
||||
|
||||
@@ -1130,8 +648,8 @@ export function initialize() {
|
||||
event.preventDefault();
|
||||
|
||||
$(event.target).parents(".compose-announce").remove();
|
||||
user_acknowledged_announce = true;
|
||||
clear_announce_warnings();
|
||||
compose_validate.set_user_acknowledged_announce_flag(true);
|
||||
compose_validate.clear_announce_warnings();
|
||||
finish();
|
||||
});
|
||||
|
||||
|
||||
Reference in New Issue
Block a user