mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-04 05:53:43 +00:00 
			
		
		
		
	We are trying to phase out the trigger-event way
of telling modules to do something.
In this case we not only remove the indirection
of the event handler, but we also get to remove
`compose_fade` from the `ui_init` startup sequence.
This also has us update `compose_fade` outside
the loop, although that's only a theoretical
improvement, since I don't think `peer_add` events
every actually include multiple streams.
To make the dispatch tests a little flatter, I
added a one-line change to zjsunit to add
`make_stub` to `global`.
To manually test:
    * have Aaron reply to Denmark (keep compose box open)
    * have Iago add Hamlet to Denmark
    * have Hamlet unsubscribe
		
	
		
			
				
	
	
		
			528 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			528 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
exports.dispatch_normal_event = function dispatch_normal_event(event) {
 | 
						|
    const noop = function () {};
 | 
						|
    switch (event.type) {
 | 
						|
    case 'alert_words':
 | 
						|
        alert_words.set_words(event.alert_words);
 | 
						|
        alert_words_ui.render_alert_words_ui();
 | 
						|
        break;
 | 
						|
 | 
						|
    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;
 | 
						|
 | 
						|
    case 'delete_message': {
 | 
						|
        const msg_id = event.message_id;
 | 
						|
        // 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_id]);
 | 
						|
        if (event.message_type === 'stream') {
 | 
						|
            stream_topic_history.remove_message({
 | 
						|
                stream_id: event.stream_id,
 | 
						|
                topic_name: event.topic,
 | 
						|
            });
 | 
						|
            stream_list.update_streams_sidebar();
 | 
						|
        }
 | 
						|
        ui.remove_message(msg_id);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    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);
 | 
						|
        page_params.hotspots = page_params.hotspots ?
 | 
						|
            page_params.hotspots.concat(event.hotspots) :
 | 
						|
            event.hotspots;
 | 
						|
        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;
 | 
						|
    }
 | 
						|
 | 
						|
    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,
 | 
						|
            google_hangouts_domain: 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,
 | 
						|
        };
 | 
						|
        if (event.op === 'update' && Object.prototype.hasOwnProperty.call(realm_settings, event.property)) {
 | 
						|
            page_params['realm_' + event.property] = event.value;
 | 
						|
            realm_settings[event.property]();
 | 
						|
            settings_org.sync_realm_settings(event.property);
 | 
						|
            if (event.property === 'create_stream_policy') {
 | 
						|
                // TODO: Add waiting_period_threshold logic here.
 | 
						|
                page_params.can_create_streams = page_params.is_admin ||
 | 
						|
                    page_params.realm_create_stream_policy === 1;
 | 
						|
            } else if (event.property === 'invite_to_stream_policy') {
 | 
						|
                // 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;
 | 
						|
            }
 | 
						|
 | 
						|
            if (event.property === 'name' && window.electron_bridge !== undefined) {
 | 
						|
                window.electron_bridge.send_event('realm_name', event.value);
 | 
						|
            }
 | 
						|
        } 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);
 | 
						|
                }
 | 
						|
            }
 | 
						|
            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/";
 | 
						|
        }
 | 
						|
 | 
						|
        if (page_params.is_admin) {
 | 
						|
            // Update the UI notice about the user's profile being
 | 
						|
            // incomplete, as we might have filled in the missing field(s).
 | 
						|
            panels.check_profile_incomplete();
 | 
						|
        }
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    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);
 | 
						|
        break;
 | 
						|
 | 
						|
    case 'realm_emoji':
 | 
						|
        // Update `page_params.realm_emoji` so that settings page
 | 
						|
        // can display it properly when reopened without refresh.
 | 
						|
        page_params.realm_emoji = event.realm_emoji;
 | 
						|
        emoji.update_emojis(event.realm_emoji);
 | 
						|
        settings_emoji.populate_emoji(event.realm_emoji);
 | 
						|
        emoji_picker.generate_emoji_picker_data(emoji.active_realm_emojis);
 | 
						|
        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 =
 | 
						|
                        event.realm_domain.allow_subdomains;
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        } 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;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
        settings_org.populate_realm_domains(page_params.realm_domains);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    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;
 | 
						|
 | 
						|
    case 'stream':
 | 
						|
        if (event.op === 'update') {
 | 
						|
            // 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,
 | 
						|
                });
 | 
						|
            settings_streams.update_default_streams_table();
 | 
						|
        } 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);
 | 
						|
                }
 | 
						|
            }
 | 
						|
        } 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');
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
        break;
 | 
						|
 | 
						|
    case 'submessage': {
 | 
						|
        // 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.
 | 
						|
        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;
 | 
						|
    }
 | 
						|
 | 
						|
    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);
 | 
						|
                }
 | 
						|
            }
 | 
						|
        } else if (event.op === 'peer_add') {
 | 
						|
            for (const sub of event.subscriptions) {
 | 
						|
                if (stream_data.add_subscriber(sub, event.user_id)) {
 | 
						|
                    $(document).trigger('peer_subscribe.zulip', {stream_name: sub});
 | 
						|
                } else {
 | 
						|
                    blueslip.warn('Cannot process peer_add event');
 | 
						|
                }
 | 
						|
            }
 | 
						|
            compose_fade.update_faded_users();
 | 
						|
        } else if (event.op === 'peer_remove') {
 | 
						|
            for (const sub of event.subscriptions) {
 | 
						|
                if (stream_data.remove_subscriber(sub, event.user_id)) {
 | 
						|
                    $(document).trigger('peer_unsubscribe.zulip', {stream_name: sub});
 | 
						|
                } else {
 | 
						|
                    blueslip.warn('Cannot process peer_remove event.');
 | 
						|
                }
 | 
						|
            }
 | 
						|
            compose_fade.update_faded_users();
 | 
						|
        } 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);
 | 
						|
            }
 | 
						|
        } else if (event.op === 'update') {
 | 
						|
            stream_events.update_property(
 | 
						|
                event.stream_id,
 | 
						|
                event.property,
 | 
						|
                event.value
 | 
						|
            );
 | 
						|
        }
 | 
						|
        break;
 | 
						|
 | 
						|
    case 'typing':
 | 
						|
        if (event.sender.user_id === page_params.user_id) {
 | 
						|
            // typing notifications are sent to the user who is typing
 | 
						|
            // as well as recipients; we ignore such self-generated events.
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        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 = [
 | 
						|
            'default_language',
 | 
						|
            'demote_inactive_streams',
 | 
						|
            'dense_mode',
 | 
						|
            'emojiset',
 | 
						|
            'fluid_layout_width',
 | 
						|
            'high_contrast_mode',
 | 
						|
            'night_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') {
 | 
						|
            // We additionally need to set the language name.
 | 
						|
            page_params.default_language_name = event.language_name;
 | 
						|
        }
 | 
						|
        if (event.setting_name === 'twenty_four_hour_time') {
 | 
						|
            // Rerender the whole message list UI
 | 
						|
            home_msg_list.rerender();
 | 
						|
            if (current_msg_list === message_list.narrowed) {
 | 
						|
                message_list.narrowed.rerender();
 | 
						|
            }
 | 
						|
        }
 | 
						|
        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 === 'night_mode') {
 | 
						|
            $("body").fadeOut(300);
 | 
						|
            setTimeout(function () {
 | 
						|
                if (event.setting === true) {
 | 
						|
                    night_mode.enable();
 | 
						|
                    realm_logo.rerender();
 | 
						|
                } else {
 | 
						|
                    night_mode.disable();
 | 
						|
                    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') {
 | 
						|
            // TODO: Make this change the view immediately rather
 | 
						|
            // than requiring a reload or page resize.
 | 
						|
        }
 | 
						|
        if (event.setting_name === 'default_language') {
 | 
						|
            // 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.
 | 
						|
        }
 | 
						|
        if (event.setting_name === 'emojiset') {
 | 
						|
            settings_display.report_emojiset_change();
 | 
						|
 | 
						|
            // Rerender the whole message list UI
 | 
						|
            home_msg_list.rerender();
 | 
						|
            if (current_msg_list === message_list.narrowed) {
 | 
						|
                message_list.narrowed.rerender();
 | 
						|
            }
 | 
						|
        }
 | 
						|
        settings_display.update_page();
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    case 'update_global_notifications':
 | 
						|
        notifications.handle_global_notification_updates(event.notification_name,
 | 
						|
                                                         event.setting);
 | 
						|
        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;
 | 
						|
 | 
						|
    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;
 | 
						|
        }
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    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);
 | 
						|
        }
 | 
						|
        settings_user_groups.reload();
 | 
						|
        break;
 | 
						|
 | 
						|
    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;
 | 
						|
    }
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
window.server_events_dispatch = exports;
 |