2020-08-01 03:43:15 +02:00
|
|
|
"use strict";
|
|
|
|
|
2020-07-25 00:12:17 +02:00
|
|
|
const emoji = require("../shared/js/emoji");
|
|
|
|
|
2020-05-16 13:13:59 +02:00
|
|
|
const settings_config = require("./settings_config");
|
|
|
|
|
2017-05-31 08:53:09 +02:00
|
|
|
exports.dispatch_normal_event = function dispatch_normal_event(event) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const noop = function () {};
|
2017-05-31 08:53:09 +02:00
|
|
|
switch (event.type) {
|
2020-07-15 02:14:03 +02:00
|
|
|
case "alert_words":
|
|
|
|
alert_words.set_words(event.alert_words);
|
|
|
|
alert_words_ui.render_alert_words_ui();
|
|
|
|
break;
|
2020-06-11 12:12:12 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "attachment":
|
|
|
|
attachments_ui.update_attachments(event);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "custom_profile_fields":
|
|
|
|
page_params.custom_profile_fields = event.fields;
|
|
|
|
settings_profile_fields.populate_profile_fields(page_params.custom_profile_fields);
|
|
|
|
settings_account.add_custom_profile_fields_to_settings();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "default_streams":
|
|
|
|
stream_data.set_realm_default_streams(event.default_streams);
|
|
|
|
settings_streams.update_default_streams_table();
|
|
|
|
break;
|
2018-08-25 13:08:24 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "delete_message": {
|
|
|
|
const msg_ids = event.message_ids;
|
|
|
|
// message is passed to unread.get_unread_messages,
|
|
|
|
// which returns all the unread messages out of a given list.
|
|
|
|
// So double marking something as read would not occur
|
|
|
|
unread_ops.process_read_messages_event(msg_ids);
|
2020-08-04 11:12:42 +02:00
|
|
|
// This methods updates message_list too and since stream_topic_history relies on it
|
|
|
|
// this method should be called first.
|
|
|
|
ui.remove_messages(msg_ids);
|
2020-07-15 02:14:03 +02:00
|
|
|
|
|
|
|
if (event.message_type === "stream") {
|
|
|
|
stream_topic_history.remove_messages({
|
|
|
|
stream_id: event.stream_id,
|
|
|
|
topic_name: event.topic,
|
|
|
|
num_messages: msg_ids.length,
|
2020-08-04 11:12:42 +02:00
|
|
|
max_removed_msg_id: Math.max(...msg_ids),
|
2020-07-15 02:14:03 +02:00
|
|
|
});
|
|
|
|
stream_list.update_streams_sidebar();
|
2019-11-16 09:26:28 +01:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
|
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "has_zoom_token":
|
|
|
|
page_params.has_zoom_token = event.value;
|
|
|
|
if (event.value) {
|
|
|
|
for (const callback of compose.zoom_token_callbacks.values()) {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
compose.zoom_token_callbacks.clear();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "hotspots":
|
|
|
|
hotspots.load_new(event.hotspots);
|
2020-07-15 00:34:28 +02:00
|
|
|
page_params.hotspots = page_params.hotspots
|
|
|
|
? page_params.hotspots.concat(event.hotspots)
|
|
|
|
: event.hotspots;
|
2020-07-15 02:14:03 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case "invites_changed":
|
|
|
|
if ($("#admin-invites-list").length) {
|
|
|
|
settings_invites.set_up(false);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "muted_topics":
|
|
|
|
muting_ui.handle_updates(event.muted_topics);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "presence":
|
|
|
|
activity.update_presence_info(event.user_id, event.presence, event.server_timestamp);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "restart": {
|
|
|
|
const reload_options = {
|
|
|
|
save_pointer: true,
|
|
|
|
save_narrow: true,
|
|
|
|
save_compose: true,
|
|
|
|
message: "The application has been updated; reloading!",
|
|
|
|
};
|
|
|
|
if (event.immediate) {
|
|
|
|
reload_options.immediate = true;
|
|
|
|
}
|
|
|
|
reload.initiate(reload_options);
|
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
|
|
|
|
case "reaction":
|
|
|
|
if (event.op === "add") {
|
|
|
|
reactions.add_reaction(event);
|
|
|
|
} else if (event.op === "remove") {
|
|
|
|
reactions.remove_reaction(event);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "realm": {
|
|
|
|
const realm_settings = {
|
|
|
|
add_emoji_by_admins_only: settings_emoji.update_custom_emoji_ui,
|
|
|
|
allow_edit_history: noop,
|
|
|
|
allow_message_deleting: noop,
|
|
|
|
allow_message_editing: noop,
|
|
|
|
allow_community_topic_editing: noop,
|
|
|
|
user_group_edit_policy: noop,
|
|
|
|
avatar_changes_disabled: settings_account.update_avatar_change_display,
|
|
|
|
bot_creation_policy: settings_bots.update_bot_permissions_ui,
|
|
|
|
create_stream_policy: noop,
|
|
|
|
invite_to_stream_policy: noop,
|
|
|
|
default_code_block_language: noop,
|
|
|
|
default_language: noop,
|
|
|
|
default_twenty_four_hour_time: noop,
|
|
|
|
description: noop,
|
|
|
|
digest_emails_enabled: noop,
|
|
|
|
digest_weekday: noop,
|
|
|
|
email_address_visibility: noop,
|
|
|
|
email_changes_disabled: settings_account.update_email_change_display,
|
|
|
|
disallow_disposable_email_addresses: noop,
|
|
|
|
inline_image_preview: noop,
|
|
|
|
inline_url_embed_preview: noop,
|
|
|
|
invite_by_admins_only: noop,
|
|
|
|
invite_required: noop,
|
|
|
|
mandatory_topics: noop,
|
|
|
|
message_content_edit_limit_seconds: noop,
|
|
|
|
message_content_delete_limit_seconds: noop,
|
|
|
|
message_retention_days: noop,
|
|
|
|
name: notifications.redraw_title,
|
|
|
|
name_changes_disabled: settings_account.update_name_change_display,
|
|
|
|
notifications_stream_id: noop,
|
|
|
|
private_message_policy: noop,
|
|
|
|
send_welcome_emails: noop,
|
|
|
|
message_content_allowed_in_email_notifications: noop,
|
|
|
|
signup_notifications_stream_id: noop,
|
|
|
|
emails_restricted_to_domains: noop,
|
|
|
|
video_chat_provider: compose.update_video_chat_button_display,
|
|
|
|
waiting_period_threshold: noop,
|
|
|
|
};
|
2020-07-15 00:34:28 +02:00
|
|
|
if (
|
|
|
|
event.op === "update" &&
|
|
|
|
Object.prototype.hasOwnProperty.call(realm_settings, event.property)
|
|
|
|
) {
|
2020-07-15 02:14:03 +02:00
|
|
|
page_params["realm_" + event.property] = event.value;
|
|
|
|
realm_settings[event.property]();
|
|
|
|
settings_org.sync_realm_settings(event.property);
|
|
|
|
if (event.property === "create_stream_policy") {
|
2020-07-15 00:34:28 +02:00
|
|
|
// TODO: Add waiting_period_threshold logic here.
|
|
|
|
page_params.can_create_streams =
|
|
|
|
page_params.is_admin || page_params.realm_create_stream_policy === 1;
|
2020-07-15 02:14:03 +02:00
|
|
|
} else if (event.property === "invite_to_stream_policy") {
|
2020-07-15 00:34:28 +02:00
|
|
|
// TODO: Add waiting_period_threshold logic here.
|
|
|
|
page_params.can_invite_to_stream =
|
|
|
|
page_params.is_admin || page_params.realm_invite_to_stream_policy === 1;
|
2020-07-15 02:14:03 +02:00
|
|
|
}
|
2018-03-20 18:30:25 +01:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
if (event.property === "name" && window.electron_bridge !== undefined) {
|
|
|
|
window.electron_bridge.send_event("realm_name", event.value);
|
2017-08-20 03:08:53 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
} else if (event.op === "update_dict" && event.property === "default") {
|
|
|
|
for (const [key, value] of Object.entries(event.data)) {
|
|
|
|
page_params["realm_" + key] = value;
|
|
|
|
if (key === "allow_message_editing") {
|
|
|
|
message_edit.update_message_topic_editing_pencil();
|
|
|
|
}
|
|
|
|
if (Object.prototype.hasOwnProperty.call(realm_settings, key)) {
|
|
|
|
settings_org.sync_realm_settings(key);
|
|
|
|
}
|
2018-03-25 15:35:10 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
if (event.data.authentication_methods !== undefined) {
|
|
|
|
settings_org.populate_auth_methods(event.data.authentication_methods);
|
|
|
|
}
|
|
|
|
} else if (event.op === "update_dict" && event.property === "icon") {
|
|
|
|
page_params.realm_icon_url = event.data.icon_url;
|
|
|
|
page_params.realm_icon_source = event.data.icon_source;
|
|
|
|
realm_icon.rerender();
|
|
|
|
|
|
|
|
const electron_bridge = window.electron_bridge;
|
|
|
|
if (electron_bridge !== undefined) {
|
|
|
|
electron_bridge.send_event("realm_icon_url", event.data.icon_url);
|
|
|
|
}
|
|
|
|
} else if (event.op === "update_dict" && event.property === "logo") {
|
|
|
|
page_params.realm_logo_url = event.data.logo_url;
|
|
|
|
page_params.realm_logo_source = event.data.logo_source;
|
|
|
|
realm_logo.rerender();
|
|
|
|
} else if (event.op === "update_dict" && event.property === "night_logo") {
|
|
|
|
page_params.realm_night_logo_url = event.data.night_logo_url;
|
|
|
|
page_params.realm_night_logo_source = event.data.night_logo_source;
|
|
|
|
realm_logo.rerender();
|
|
|
|
} else if (event.op === "deactivated") {
|
|
|
|
window.location.href = "/accounts/deactivated/";
|
2020-02-06 04:23:20 +01:00
|
|
|
}
|
2017-05-31 08:53:09 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
if (page_params.is_admin) {
|
2020-07-15 00:34:28 +02:00
|
|
|
// Update the UI notice about the user's profile being
|
|
|
|
// incomplete, as we might have filled in the missing field(s).
|
2020-07-15 02:14:03 +02:00
|
|
|
panels.check_profile_incomplete();
|
|
|
|
}
|
|
|
|
break;
|
2020-03-26 08:15:26 +01:00
|
|
|
}
|
2017-05-31 08:53:09 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "realm_bot":
|
|
|
|
if (event.op === "add") {
|
|
|
|
bot_data.add(event.bot);
|
|
|
|
} else if (event.op === "remove") {
|
|
|
|
bot_data.deactivate(event.bot.user_id);
|
|
|
|
event.bot.is_active = false;
|
|
|
|
} else if (event.op === "delete") {
|
|
|
|
blueslip.info("ignoring bot deletion for live UI update");
|
|
|
|
break;
|
|
|
|
} else if (event.op === "update") {
|
|
|
|
bot_data.update(event.bot.user_id, event.bot);
|
|
|
|
}
|
|
|
|
settings_users.update_bot_data(event.bot.user_id);
|
2020-05-28 22:29:52 +02:00
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "realm_emoji":
|
2020-07-25 14:27:39 +02:00
|
|
|
// The authoritative data source is here.
|
2020-07-15 02:14:03 +02:00
|
|
|
emoji.update_emojis(event.realm_emoji);
|
2020-07-25 14:27:39 +02:00
|
|
|
|
|
|
|
// And then let other widgets know.
|
2020-07-24 00:42:44 +02:00
|
|
|
settings_emoji.populate_emoji();
|
2020-07-25 16:11:28 +02:00
|
|
|
emoji_picker.rebuild_catalog();
|
2020-07-15 02:14:03 +02:00
|
|
|
composebox_typeahead.update_emoji_data();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "realm_filters":
|
|
|
|
page_params.realm_filters = event.realm_filters;
|
|
|
|
markdown.update_realm_filter_rules(page_params.realm_filters);
|
|
|
|
settings_linkifiers.populate_filters(page_params.realm_filters);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "realm_domains": {
|
|
|
|
let i;
|
|
|
|
if (event.op === "add") {
|
|
|
|
page_params.realm_domains.push(event.realm_domain);
|
|
|
|
} else if (event.op === "change") {
|
|
|
|
for (i = 0; i < page_params.realm_domains.length; i += 1) {
|
|
|
|
if (page_params.realm_domains[i].domain === event.realm_domain.domain) {
|
|
|
|
page_params.realm_domains[i].allow_subdomains =
|
2020-07-15 00:34:28 +02:00
|
|
|
event.realm_domain.allow_subdomains;
|
2020-07-15 02:14:03 +02:00
|
|
|
break;
|
|
|
|
}
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
} else if (event.op === "remove") {
|
|
|
|
for (i = 0; i < page_params.realm_domains.length; i += 1) {
|
|
|
|
if (page_params.realm_domains[i].domain === event.domain) {
|
|
|
|
page_params.realm_domains.splice(i, 1);
|
|
|
|
break;
|
|
|
|
}
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
settings_org.populate_realm_domains(page_params.realm_domains);
|
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "realm_user":
|
|
|
|
if (event.op === "add") {
|
|
|
|
people.add_active_user(event.person);
|
|
|
|
} else if (event.op === "remove") {
|
|
|
|
people.deactivate(event.person);
|
|
|
|
stream_events.remove_deactivated_user_from_all_streams(event.person.user_id);
|
|
|
|
} else if (event.op === "update") {
|
|
|
|
user_events.update_person(event.person);
|
|
|
|
}
|
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "stream":
|
|
|
|
if (event.op === "update") {
|
2020-07-15 00:34:28 +02:00
|
|
|
// Legacy: Stream properties are still managed by subs.js on the client side.
|
|
|
|
stream_events.update_property(event.stream_id, event.property, event.value, {
|
|
|
|
rendered_description: event.rendered_description,
|
|
|
|
history_public_to_subscribers: event.history_public_to_subscribers,
|
|
|
|
});
|
2020-03-22 14:38:19 +01:00
|
|
|
settings_streams.update_default_streams_table();
|
2020-07-15 02:14:03 +02:00
|
|
|
} else if (event.op === "create") {
|
|
|
|
stream_data.create_streams(event.streams);
|
|
|
|
|
|
|
|
for (const stream of event.streams) {
|
|
|
|
const sub = stream_data.get_sub_by_id(stream.stream_id);
|
|
|
|
stream_data.update_calculated_fields(sub);
|
|
|
|
if (overlays.streams_open()) {
|
|
|
|
subs.add_sub_to_table(sub);
|
|
|
|
}
|
2017-06-10 07:03:53 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
} else if (event.op === "delete") {
|
|
|
|
for (const stream of event.streams) {
|
|
|
|
const was_subscribed = stream_data.get_sub_by_id(stream.stream_id).subscribed;
|
|
|
|
subs.remove_stream(stream.stream_id);
|
|
|
|
stream_data.delete_sub(stream.stream_id);
|
|
|
|
if (was_subscribed) {
|
|
|
|
stream_list.remove_sidebar_row(stream.stream_id);
|
|
|
|
}
|
|
|
|
settings_streams.update_default_streams_table();
|
|
|
|
stream_data.remove_default_stream(stream.stream_id);
|
|
|
|
if (page_params.realm_notifications_stream_id === stream.stream_id) {
|
|
|
|
page_params.realm_notifications_stream_id = -1;
|
|
|
|
settings_org.sync_realm_settings("notifications_stream_id");
|
|
|
|
}
|
|
|
|
if (page_params.realm_signup_notifications_stream_id === stream.stream_id) {
|
|
|
|
page_params.realm_signup_notifications_stream_id = -1;
|
|
|
|
settings_org.sync_realm_settings("signup_notifications_stream_id");
|
|
|
|
}
|
2017-10-20 16:55:04 +02:00
|
|
|
}
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "submessage": {
|
2020-07-15 00:34:28 +02:00
|
|
|
// The fields in the event don't quite exactly
|
|
|
|
// match the layout of a submessage, since there's
|
|
|
|
// an event id. We also want to be explicit here.
|
2020-07-15 02:14:03 +02:00
|
|
|
const submsg = {
|
|
|
|
id: event.submessage_id,
|
|
|
|
sender_id: event.sender_id,
|
|
|
|
msg_type: event.msg_type,
|
|
|
|
message_id: event.message_id,
|
|
|
|
content: event.content,
|
|
|
|
};
|
|
|
|
submessage.handle_event(submsg);
|
|
|
|
break;
|
|
|
|
}
|
2018-02-12 15:12:40 +01:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "subscription":
|
|
|
|
if (event.op === "add") {
|
|
|
|
for (const rec of event.subscriptions) {
|
|
|
|
const sub = stream_data.get_sub_by_id(rec.stream_id);
|
|
|
|
if (sub) {
|
|
|
|
stream_data.update_stream_email_address(sub, rec.email_address);
|
|
|
|
stream_events.mark_subscribed(sub, rec.subscribers, rec.color);
|
|
|
|
} else {
|
|
|
|
blueslip.error("Subscribing to unknown stream with ID " + rec.stream_id);
|
|
|
|
}
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
} else if (event.op === "peer_add") {
|
|
|
|
function add_peer(stream_id, user_id) {
|
|
|
|
const sub = stream_data.get_sub_by_id(stream_id);
|
|
|
|
|
|
|
|
if (!sub) {
|
|
|
|
blueslip.warn("Cannot find stream for peer_add: " + stream_id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!stream_data.add_subscriber(stream_id, user_id)) {
|
|
|
|
blueslip.warn("Cannot process peer_add event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
subs.update_subscribers_ui(sub);
|
|
|
|
compose_fade.update_faded_users();
|
2020-06-12 16:54:01 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
add_peer(event.stream_id, event.user_id);
|
|
|
|
} else if (event.op === "peer_remove") {
|
|
|
|
function remove_peer(stream_id, user_id) {
|
|
|
|
const sub = stream_data.get_sub_by_id(stream_id);
|
|
|
|
|
|
|
|
if (!sub) {
|
|
|
|
blueslip.warn("Cannot find stream for peer_remove: " + stream_id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!stream_data.remove_subscriber(sub.stream_id, user_id)) {
|
|
|
|
blueslip.warn("Cannot process peer_remove event.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
subs.update_subscribers_ui(sub);
|
|
|
|
compose_fade.update_faded_users();
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
remove_peer(event.stream_id, event.user_id);
|
|
|
|
} else if (event.op === "remove") {
|
|
|
|
for (const rec of event.subscriptions) {
|
|
|
|
const sub = stream_data.get_sub_by_id(rec.stream_id);
|
|
|
|
stream_events.mark_unsubscribed(sub);
|
2020-06-12 16:54:01 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
} else if (event.op === "update") {
|
2020-07-15 00:34:28 +02:00
|
|
|
stream_events.update_property(event.stream_id, event.property, event.value);
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "typing":
|
|
|
|
if (event.sender.user_id === page_params.user_id) {
|
2020-07-15 00:34:28 +02:00
|
|
|
// typing notifications are sent to the user who is typing
|
|
|
|
// as well as recipients; we ignore such self-generated events.
|
2020-07-15 02:14:03 +02:00
|
|
|
return;
|
|
|
|
}
|
2017-05-31 08:53:09 +02:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
if (event.op === "start") {
|
|
|
|
typing_events.display_notification(event);
|
|
|
|
} else if (event.op === "stop") {
|
|
|
|
typing_events.hide_notification(event);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "update_display_settings": {
|
|
|
|
const user_display_settings = [
|
|
|
|
"color_scheme",
|
|
|
|
"default_language",
|
|
|
|
"demote_inactive_streams",
|
|
|
|
"dense_mode",
|
|
|
|
"emojiset",
|
|
|
|
"fluid_layout_width",
|
|
|
|
"high_contrast_mode",
|
|
|
|
"left_side_userlist",
|
|
|
|
"timezone",
|
|
|
|
"twenty_four_hour_time",
|
|
|
|
"translate_emoticons",
|
|
|
|
"starred_message_counts",
|
|
|
|
];
|
|
|
|
if (user_display_settings.includes(event.setting_name)) {
|
|
|
|
page_params[event.setting_name] = event.setting;
|
|
|
|
}
|
|
|
|
if (event.setting_name === "default_language") {
|
2020-07-15 00:34:28 +02:00
|
|
|
// We additionally need to set the language name.
|
2020-07-15 02:14:03 +02:00
|
|
|
page_params.default_language_name = event.language_name;
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
if (event.setting_name === "twenty_four_hour_time") {
|
2020-07-15 00:34:28 +02:00
|
|
|
// Rerender the whole message list UI
|
2020-07-15 02:14:03 +02:00
|
|
|
home_msg_list.rerender();
|
|
|
|
if (current_msg_list === message_list.narrowed) {
|
|
|
|
message_list.narrowed.rerender();
|
2017-11-30 02:31:21 +01:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
}
|
|
|
|
if (event.setting_name === "high_contrast_mode") {
|
|
|
|
$("body").toggleClass("high-contrast");
|
|
|
|
}
|
|
|
|
if (event.setting_name === "demote_inactive_streams") {
|
|
|
|
stream_list.update_streams_sidebar();
|
|
|
|
stream_data.set_filter_out_inactives();
|
|
|
|
}
|
|
|
|
if (event.setting_name === "dense_mode") {
|
|
|
|
$("body").toggleClass("less_dense_mode");
|
|
|
|
$("body").toggleClass("more_dense_mode");
|
|
|
|
}
|
|
|
|
if (event.setting_name === "color_scheme") {
|
|
|
|
$("body").fadeOut(300);
|
|
|
|
setTimeout(() => {
|
|
|
|
if (event.setting === settings_config.color_scheme_values.night.code) {
|
|
|
|
night_mode.enable();
|
|
|
|
realm_logo.rerender();
|
|
|
|
} else if (event.setting === settings_config.color_scheme_values.day.code) {
|
|
|
|
night_mode.disable();
|
|
|
|
realm_logo.rerender();
|
|
|
|
} else {
|
|
|
|
night_mode.default_preference_checker();
|
|
|
|
realm_logo.rerender();
|
|
|
|
}
|
|
|
|
$("body").fadeIn(300);
|
|
|
|
}, 300);
|
|
|
|
}
|
|
|
|
if (event.setting_name === "starred_message_counts") {
|
|
|
|
starred_messages.rerender_ui();
|
|
|
|
}
|
|
|
|
if (event.setting_name === "fluid_layout_width") {
|
|
|
|
scroll_bar.set_layout_width();
|
|
|
|
}
|
|
|
|
if (event.setting_name === "left_side_userlist") {
|
2020-07-15 00:34:28 +02:00
|
|
|
// TODO: Make this change the view immediately rather
|
|
|
|
// than requiring a reload or page resize.
|
2020-07-15 02:14:03 +02:00
|
|
|
}
|
|
|
|
if (event.setting_name === "default_language") {
|
2020-07-15 00:34:28 +02:00
|
|
|
// TODO: Make this change the view immediately rather than
|
|
|
|
// requiring a reload. This is likely fairly difficult,
|
|
|
|
// because various i18n strings are rendered by the
|
|
|
|
// server; we may want to instead just trigger a page
|
|
|
|
// reload.
|
2020-07-15 02:14:03 +02:00
|
|
|
}
|
|
|
|
if (event.setting_name === "emojiset") {
|
|
|
|
settings_display.report_emojiset_change();
|
2018-01-02 21:11:23 +01:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
// Rerender the whole message list UI
|
|
|
|
home_msg_list.rerender();
|
|
|
|
if (current_msg_list === message_list.narrowed) {
|
|
|
|
message_list.narrowed.rerender();
|
|
|
|
}
|
2018-01-02 13:02:13 +01:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
settings_display.update_page();
|
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "update_global_notifications":
|
2020-07-15 00:34:28 +02:00
|
|
|
notifications.handle_global_notification_updates(
|
|
|
|
event.notification_name,
|
|
|
|
event.setting,
|
|
|
|
);
|
2020-07-15 02:14:03 +02:00
|
|
|
settings_notifications.update_page();
|
|
|
|
// TODO: This should also do a refresh of the stream_edit UI
|
|
|
|
// if it's currently displayed, possibly reusing some code
|
|
|
|
// from stream_events.js
|
|
|
|
// (E.g. update_stream_push_notifications).
|
|
|
|
break;
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "update_message_flags": {
|
|
|
|
const new_value = event.operation === "add";
|
|
|
|
switch (event.flag) {
|
|
|
|
case "starred":
|
|
|
|
for (const message_id of event.messages) {
|
|
|
|
message_flags.update_starred_flag(message_id, new_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event.operation === "add") {
|
|
|
|
starred_messages.add(event.messages);
|
|
|
|
} else {
|
|
|
|
starred_messages.remove(event.messages);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "read":
|
|
|
|
unread_ops.process_read_messages_event(event.messages);
|
|
|
|
break;
|
2018-08-17 03:46:32 +02:00
|
|
|
}
|
2017-05-31 08:53:09 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "user_group":
|
|
|
|
if (event.op === "add") {
|
|
|
|
user_groups.add(event.group);
|
|
|
|
} else if (event.op === "add_members") {
|
|
|
|
user_groups.add_members(event.group_id, event.user_ids);
|
|
|
|
} else if (event.op === "remove_members") {
|
|
|
|
user_groups.remove_members(event.group_id, event.user_ids);
|
|
|
|
} else if (event.op === "update") {
|
|
|
|
user_groups.update(event);
|
2019-01-23 19:25:22 +01:00
|
|
|
}
|
2020-07-15 02:14:03 +02:00
|
|
|
settings_user_groups.reload();
|
|
|
|
break;
|
2019-01-23 19:25:22 +01:00
|
|
|
|
2020-07-15 02:14:03 +02:00
|
|
|
case "user_status":
|
|
|
|
if (event.away !== undefined) {
|
|
|
|
if (event.away) {
|
|
|
|
activity.on_set_away(event.user_id);
|
|
|
|
} else {
|
|
|
|
activity.on_revoke_away(event.user_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event.status_text !== undefined) {
|
|
|
|
user_status.set_status_text({
|
|
|
|
user_id: event.user_id,
|
|
|
|
status_text: event.status_text,
|
|
|
|
});
|
|
|
|
activity.redraw_user(event.user_id);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "realm_export":
|
|
|
|
settings_exports.populate_exports_table(event.exports);
|
|
|
|
break;
|
2017-05-31 08:53:09 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-10-25 09:45:13 +02:00
|
|
|
window.server_events_dispatch = exports;
|