mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-04 05:53:43 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1534 lines
		
	
	
		
			51 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1534 lines
		
	
	
		
			51 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
var all_msg_list = new MessageList(
 | 
						|
    undefined, undefined,
 | 
						|
    {muting_enabled: false}
 | 
						|
);
 | 
						|
var home_msg_list = new MessageList('zhome',
 | 
						|
    new Filter([["in", "home"]]),
 | 
						|
    {
 | 
						|
        muting_enabled: true,
 | 
						|
        summarize_read: feature_flags.summarize_read_while_narrowed?'home':false
 | 
						|
    }
 | 
						|
);
 | 
						|
var narrowed_msg_list;
 | 
						|
var current_msg_list = home_msg_list;
 | 
						|
 | 
						|
// The following three Dicts point to the same objects
 | 
						|
// All people we've seen
 | 
						|
var people_dict = new Dict();
 | 
						|
var people_by_name_dict = new Dict();
 | 
						|
// People in this realm
 | 
						|
var realm_people_dict = new Dict();
 | 
						|
 | 
						|
var recent_subjects = new Dict({fold_case: true});
 | 
						|
 | 
						|
var queued_mark_as_read = [];
 | 
						|
var queued_flag_timer;
 | 
						|
 | 
						|
var get_updates_params = {
 | 
						|
    pointer: -1
 | 
						|
};
 | 
						|
var get_updates_failures = 0;
 | 
						|
 | 
						|
var load_more_enabled = true;
 | 
						|
// If the browser hasn't scrolled away from the top of the page
 | 
						|
// since the last time that we ran load_more_messages(), we do
 | 
						|
// not load_more_messages().
 | 
						|
var have_scrolled_away_from_top = true;
 | 
						|
 | 
						|
// Toggles re-centering the pointer in the window
 | 
						|
// when Home is next clicked by the user
 | 
						|
var recenter_pointer_on_display = false;
 | 
						|
var suppress_scroll_pointer_update = false;
 | 
						|
// Includes both scroll and arrow events. Negative means scroll up,
 | 
						|
// positive means scroll down.
 | 
						|
var last_viewport_movement_direction = 1;
 | 
						|
 | 
						|
var furthest_read = -1;
 | 
						|
var server_furthest_read = -1;
 | 
						|
var unread_messages_read_in_narrow = false;
 | 
						|
var pointer_update_in_flight = false;
 | 
						|
var suppress_unread_counts = true;
 | 
						|
 | 
						|
var events_stored_during_tutorial = [];
 | 
						|
 | 
						|
var waiting_on_browser_scroll = true;
 | 
						|
 | 
						|
var waiting_on_homeview_load = true;
 | 
						|
 | 
						|
function add_person(person, in_realm) {
 | 
						|
    page_params.people_list.push(person);
 | 
						|
    people_dict.set(person.email, person);
 | 
						|
    people_by_name_dict.set(person.full_name, person);
 | 
						|
    person.pm_recipient_count = 0;
 | 
						|
}
 | 
						|
 | 
						|
function add_person_in_realm(person) {
 | 
						|
    realm_people_dict.set(person.email, person);
 | 
						|
    add_person(person);
 | 
						|
}
 | 
						|
 | 
						|
function remove_person(person) {
 | 
						|
    var i;
 | 
						|
    for (i = 0; i < page_params.people_list.length; i++) {
 | 
						|
        if (page_params.people_list[i].email === person.email) {
 | 
						|
            page_params.people_list.splice(i, 1);
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    people_dict.del(person.email);
 | 
						|
    people_by_name_dict.del(person.full_name);
 | 
						|
    realm_people_dict.del(person.email);
 | 
						|
}
 | 
						|
 | 
						|
function reify_person(person) {
 | 
						|
    // If a locally sent message is a PM to
 | 
						|
    // an out-of-realm recipient, a people_dict
 | 
						|
    // entry is created with simply an email address
 | 
						|
    // Once we've received the full person object, replace
 | 
						|
    // it
 | 
						|
    if (! people_dict.has(person.email)) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    var old_person = people_dict.get(person.email);
 | 
						|
    var old_idx = page_params.people_list.indexOf(old_person);
 | 
						|
 | 
						|
    var new_person = _.extend({}, old_person, person);
 | 
						|
    people_dict.set(person.email, person);
 | 
						|
    people_by_name_dict.set(person.full_name, person);
 | 
						|
    page_params.people_list[old_idx] = new_person;
 | 
						|
 | 
						|
    if (people_by_name_dict.has(person.email)) {
 | 
						|
        people_by_name_dict.del(person.email);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function update_person(person) {
 | 
						|
    // Currently the only attribute that can change is full_name, so
 | 
						|
    // we just push out changes to that field.  As we add more things
 | 
						|
    // that can change, this will need to either get complicated or be
 | 
						|
    // replaced by MVC
 | 
						|
    if (! people_dict.has(person.email)) {
 | 
						|
        blueslip.error("Got update_person event for unexpected user",
 | 
						|
                       {email: person.email});
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    var person_obj = people_dict.get(person.email);
 | 
						|
 | 
						|
    if (_.has(person, 'full_name')) {
 | 
						|
        if (people_by_name_dict.has(person_obj.full_name)) {
 | 
						|
            people_by_name_dict.set(person.full_name, person_obj);
 | 
						|
            people_by_name_dict.del(person_obj.full_name);
 | 
						|
        }
 | 
						|
 | 
						|
        person_obj.full_name = person.full_name;
 | 
						|
 | 
						|
        if (person.email === page_params.email) {
 | 
						|
            page_params.fullname = person.full_name;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (_.has(person, 'is_admin')) {
 | 
						|
        person_obj.is_admin = person.is_admin;
 | 
						|
 | 
						|
        if (person.email === page_params.email) {
 | 
						|
            page_params.is_admin = person.is_admin;
 | 
						|
            admin.show_or_hide_menu_item();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    activity.set_user_statuses([]);
 | 
						|
 | 
						|
    // TODO: update sender names on messages
 | 
						|
}
 | 
						|
 | 
						|
function within_viewport(row_offset, row_height) {
 | 
						|
    // Returns true if a message is fully within the effectively visible
 | 
						|
    // part of the viewport.
 | 
						|
    var message_top = row_offset.top;
 | 
						|
    var message_bottom  = message_top + row_height;
 | 
						|
    var info = viewport.message_viewport_info();
 | 
						|
    var viewport_top = info.visible_top;
 | 
						|
    var viewport_bottom = viewport_top + info.visible_height;
 | 
						|
    return (message_top > viewport_top) && (message_bottom < viewport_bottom);
 | 
						|
}
 | 
						|
 | 
						|
function keep_pointer_in_view() {
 | 
						|
    // See recenter_view() for related logic to keep the pointer onscreen.
 | 
						|
    // This function mostly comes into place for mouse scrollers, and it
 | 
						|
    // keeps the pointer in view.  For people who purely scroll with the
 | 
						|
    // mouse, the pointer is kind of meaningless to them, but keyboard
 | 
						|
    // users will occasionally do big mouse scrolls, so this gives them
 | 
						|
    // a pointer reasonably close to the middle of the screen.
 | 
						|
    var candidate;
 | 
						|
    var next_row = current_msg_list.selected_row();
 | 
						|
 | 
						|
    if (next_row.length === 0) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    var info = viewport.message_viewport_info();
 | 
						|
    var top_threshold = info.visible_top + (1/10 * info.visible_height);
 | 
						|
    var bottom_threshold = info.visible_top + (9/10 * info.visible_height);
 | 
						|
 | 
						|
    function message_is_far_enough_down() {
 | 
						|
        if (viewport.at_top()) {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
 | 
						|
        var message_top = next_row.offset().top;
 | 
						|
 | 
						|
        // If the message starts after the very top of the screen, we just
 | 
						|
        // leave it alone.  This avoids bugs like #1608, where overzealousness
 | 
						|
        // about repositioning the pointer can cause users to miss messages.
 | 
						|
        if (message_top >= info.visible_top) {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
 | 
						|
 | 
						|
        // If at least part of the message is below top_threshold (10% from
 | 
						|
        // the top), then we also leave it alone.
 | 
						|
        var bottom_offset = message_top + next_row.outerHeight(true);
 | 
						|
        if (bottom_offset >= top_threshold) {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
 | 
						|
        // If we got this far, the message is not "in view."
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    function message_is_far_enough_up() {
 | 
						|
        return viewport.at_bottom() ||
 | 
						|
            (next_row.offset().top <= bottom_threshold);
 | 
						|
    }
 | 
						|
 | 
						|
    function adjust(in_view, get_next_row) {
 | 
						|
        // return true only if we make an actual adjustment, so
 | 
						|
        // that we know to short circuit the other direction
 | 
						|
        if (in_view(next_row)) {
 | 
						|
            return false;  // try other side
 | 
						|
        }
 | 
						|
        while (!in_view(next_row)) {
 | 
						|
            candidate = get_next_row(next_row);
 | 
						|
            if (candidate.length === 0) {
 | 
						|
                break;
 | 
						|
            }
 | 
						|
            next_row = candidate;
 | 
						|
        }
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!adjust(message_is_far_enough_down, rows.next_visible)) {
 | 
						|
        adjust(message_is_far_enough_up, rows.prev_visible);
 | 
						|
    }
 | 
						|
 | 
						|
    current_msg_list.select_id(rows.id(next_row), {from_scroll: true});
 | 
						|
}
 | 
						|
 | 
						|
function recenter_view(message, opts) {
 | 
						|
    opts = opts || {};
 | 
						|
 | 
						|
    // Barnowl-style recentering: if the pointer is too high, move it to
 | 
						|
    // the 1/2 marks. If the pointer is too low, move it to the 1/7 mark.
 | 
						|
    // See keep_pointer_in_view() for related logic to keep the pointer onscreen.
 | 
						|
 | 
						|
    var viewport_info = viewport.message_viewport_info();
 | 
						|
    var top_threshold = viewport_info.visible_top;
 | 
						|
 | 
						|
    var bottom_threshold = viewport_info.visible_top + viewport_info.visible_height;
 | 
						|
 | 
						|
    var message_top = message.offset().top;
 | 
						|
    var message_height = message.outerHeight(true);
 | 
						|
    var message_bottom = message_top + message_height;
 | 
						|
 | 
						|
    var is_above = message_top < top_threshold;
 | 
						|
    var is_below = message_bottom > bottom_threshold;
 | 
						|
 | 
						|
    if (opts.from_scroll) {
 | 
						|
        // If the message you're trying to center on is already in view AND
 | 
						|
        // you're already trying to move in the direction of that message,
 | 
						|
        // don't try to recenter. This avoids disorienting jumps when the
 | 
						|
        // pointer has gotten itself outside the threshold (e.g. by
 | 
						|
        // autoscrolling).
 | 
						|
        if (is_above && last_viewport_movement_direction >= 0) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        if (is_below && last_viewport_movement_direction <= 0) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (is_above || opts.force_center) {
 | 
						|
        viewport.set_message_position(message_top, message_height, viewport_info, 1/2);
 | 
						|
    } else if (is_below) {
 | 
						|
        viewport.set_message_position(message_top, message_height, viewport_info, 1/7);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function scroll_to_selected() {
 | 
						|
    var selected_row = current_msg_list.selected_row();
 | 
						|
    if (selected_row && (selected_row.length !== 0)) {
 | 
						|
        recenter_view(selected_row);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function maybe_scroll_to_selected() {
 | 
						|
    // If we have been previously instructed to re-center to the
 | 
						|
    // selected message, then do so
 | 
						|
    if (recenter_pointer_on_display) {
 | 
						|
        scroll_to_selected();
 | 
						|
        recenter_pointer_on_display = false;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function get_private_message_recipient(message, attr, fallback_attr) {
 | 
						|
    var recipient, i;
 | 
						|
    var other_recipients = _.filter(message.display_recipient,
 | 
						|
                                  function (element) {
 | 
						|
                                      return element.email !== page_params.email;
 | 
						|
                                  });
 | 
						|
    if (other_recipients.length === 0) {
 | 
						|
        // private message with oneself
 | 
						|
        return message.display_recipient[0][attr];
 | 
						|
    }
 | 
						|
 | 
						|
    recipient = other_recipients[0][attr];
 | 
						|
    if (recipient === undefined && fallback_attr !== undefined) {
 | 
						|
        recipient = other_recipients[0][fallback_attr];
 | 
						|
    }
 | 
						|
    for (i = 1; i < other_recipients.length; i++) {
 | 
						|
        var attr_value = other_recipients[i][attr];
 | 
						|
        if (attr_value === undefined && fallback_attr !== undefined) {
 | 
						|
            attr_value = other_recipients[i][fallback_attr];
 | 
						|
        }
 | 
						|
        recipient += ', ' + attr_value;
 | 
						|
    }
 | 
						|
    return recipient;
 | 
						|
}
 | 
						|
 | 
						|
// Returns messages from the given message list in the specified range, inclusive
 | 
						|
function message_range(msg_list, start, end) {
 | 
						|
    if (start === -1) {
 | 
						|
        blueslip.error("message_range given a start of -1");
 | 
						|
    }
 | 
						|
 | 
						|
    var all = msg_list.all();
 | 
						|
    var compare = function (a, b) { return a.id < b; };
 | 
						|
 | 
						|
    var start_idx = util.lower_bound(all, start, compare);
 | 
						|
    var end_idx   = util.lower_bound(all, end,   compare);
 | 
						|
    return all.slice(start_idx, end_idx + 1);
 | 
						|
}
 | 
						|
 | 
						|
function batched_flag_updater(flag, op) {
 | 
						|
    var queue = [];
 | 
						|
    var on_success;
 | 
						|
 | 
						|
    function server_request() {
 | 
						|
        // Wait for server IDs before sending flags
 | 
						|
        var real_msgs = _.filter(queue, function (msg) {
 | 
						|
            return msg.local_id === undefined;
 | 
						|
        });
 | 
						|
 | 
						|
        channel.post({
 | 
						|
            url:      '/json/update_message_flags',
 | 
						|
            idempotent: true,
 | 
						|
            data:     {messages: JSON.stringify(real_msgs),
 | 
						|
                       op:       op,
 | 
						|
                       flag:     flag},
 | 
						|
            success:  on_success
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    var start = _.debounce(server_request, 1000);
 | 
						|
 | 
						|
    on_success = function on_success(data, status, jqXHR) {
 | 
						|
        if (data ===  undefined || data.messages === undefined) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        queue = _.filter(queue, function (message) {
 | 
						|
            return data.messages.indexOf(message) === -1;
 | 
						|
        });
 | 
						|
 | 
						|
        if (queue.length > 0) {
 | 
						|
            start();
 | 
						|
        }
 | 
						|
    };
 | 
						|
 | 
						|
    function add(message) {
 | 
						|
        if (message.flags === undefined) {
 | 
						|
            message.flags = [];
 | 
						|
        }
 | 
						|
        if (op === 'add')  {
 | 
						|
            message.flags.push(flag);
 | 
						|
        } else {
 | 
						|
            message.flags = _.without(message.flags, flag);
 | 
						|
        }
 | 
						|
        queue.push(message.id);
 | 
						|
        start();
 | 
						|
    }
 | 
						|
 | 
						|
    return add;
 | 
						|
}
 | 
						|
 | 
						|
var send_read = batched_flag_updater('read', 'add');
 | 
						|
var send_summarize_in_stream = batched_flag_updater('summarize_in_stream', 'add');
 | 
						|
var send_summarize_in_home = batched_flag_updater('summarize_in_home', 'add');
 | 
						|
 | 
						|
function update_unread_counts() {
 | 
						|
    if (suppress_unread_counts) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Pure computation:
 | 
						|
    var res = unread.get_counts();
 | 
						|
 | 
						|
    // Side effects from here down:
 | 
						|
    // This updates some DOM elements directly, so try to
 | 
						|
    // avoid excessive calls to this.
 | 
						|
    stream_list.update_dom_with_unread_counts(res);
 | 
						|
    notifications.update_title_count(res.home_unread_messages);
 | 
						|
    notifications.update_pm_count(res.private_message_count);
 | 
						|
    notifications_bar.update(res.home_unread_messages);
 | 
						|
}
 | 
						|
 | 
						|
function enable_unread_counts() {
 | 
						|
    suppress_unread_counts = false;
 | 
						|
    update_unread_counts();
 | 
						|
}
 | 
						|
 | 
						|
function mark_all_as_read(cont) {
 | 
						|
    _.each(all_msg_list.all(), function (msg) {
 | 
						|
        msg.flags = msg.flags || [];
 | 
						|
        msg.flags.push('read');
 | 
						|
    });
 | 
						|
    unread.declare_bankruptcy();
 | 
						|
    update_unread_counts();
 | 
						|
 | 
						|
    channel.post({
 | 
						|
        url:      '/json/update_message_flags',
 | 
						|
        idempotent: true,
 | 
						|
        data:     {messages: JSON.stringify([]),
 | 
						|
                   all:      true,
 | 
						|
                   op:       'add',
 | 
						|
                   flag:     'read'},
 | 
						|
        success:  cont});
 | 
						|
}
 | 
						|
 | 
						|
function process_loaded_for_unread(messages) {
 | 
						|
    activity.process_loaded_messages(messages);
 | 
						|
    activity.update_huddles();
 | 
						|
    unread.process_loaded_messages(messages);
 | 
						|
    update_unread_counts();
 | 
						|
    ui.resize_page_components();
 | 
						|
}
 | 
						|
 | 
						|
// Takes a list of messages and marks them as read
 | 
						|
function mark_messages_as_read(messages, options) {
 | 
						|
    options = options || {};
 | 
						|
    var processed = false;
 | 
						|
 | 
						|
    _.each(messages, function (message) {
 | 
						|
        if (!unread.message_unread(message)) {
 | 
						|
            // Don't do anything if the message is already read.
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        if (current_msg_list === narrowed_msg_list) {
 | 
						|
            unread_messages_read_in_narrow = true;
 | 
						|
        }
 | 
						|
 | 
						|
        if (options.from !== "server") {
 | 
						|
            send_read(message);
 | 
						|
        }
 | 
						|
        summary.maybe_mark_summarized(message);
 | 
						|
 | 
						|
        message.unread = false;
 | 
						|
        unread.process_read_message(message, options);
 | 
						|
        home_msg_list.show_message_as_read(message, options);
 | 
						|
        all_msg_list.show_message_as_read(message, options);
 | 
						|
        if (narrowed_msg_list) {
 | 
						|
            narrowed_msg_list.show_message_as_read(message, options);
 | 
						|
        }
 | 
						|
        notifications.close_notification(message);
 | 
						|
        processed = true;
 | 
						|
    });
 | 
						|
 | 
						|
    if (processed) {
 | 
						|
        update_unread_counts();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function mark_message_as_read(message, options) {
 | 
						|
    mark_messages_as_read([message], options);
 | 
						|
}
 | 
						|
 | 
						|
// If we ever materially change the algorithm for this function, we
 | 
						|
// may need to update notifications.received_messages as well.
 | 
						|
function process_visible_unread_messages(update_cursor) {
 | 
						|
    if (! notifications.window_has_focus()) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (feature_flags.mark_read_at_bottom) {
 | 
						|
        if (viewport.bottom_message_visible()) {
 | 
						|
            mark_current_list_as_read();
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        mark_messages_as_read(viewport.visible_messages(true));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function mark_current_list_as_read(options) {
 | 
						|
    mark_messages_as_read(current_msg_list.all(), options);
 | 
						|
}
 | 
						|
 | 
						|
function respond_to_message(opts) {
 | 
						|
    var message, msg_type;
 | 
						|
    // Before initiating a reply to a message, if there's an
 | 
						|
    // in-progress composition, snapshot it.
 | 
						|
    compose.snapshot_message();
 | 
						|
 | 
						|
    message = current_msg_list.selected_message();
 | 
						|
 | 
						|
    if (message === undefined) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    mark_message_as_read(message);
 | 
						|
 | 
						|
    var stream = '';
 | 
						|
    var subject = '';
 | 
						|
    if (message.type === "stream") {
 | 
						|
        stream = message.stream;
 | 
						|
        subject = message.subject;
 | 
						|
    }
 | 
						|
 | 
						|
    var pm_recipient = message.reply_to;
 | 
						|
    if (opts.reply_type === "personal" && message.type === "private") {
 | 
						|
        // reply_to for private messages is everyone involved, so for
 | 
						|
        // personals replies we need to set the the private message
 | 
						|
        // recipient to just the sender
 | 
						|
        pm_recipient = message.sender_email;
 | 
						|
    }
 | 
						|
    if (opts.reply_type === 'personal' || message.type === 'private') {
 | 
						|
        msg_type = 'private';
 | 
						|
    } else {
 | 
						|
        msg_type = message.type;
 | 
						|
    }
 | 
						|
    compose.start(msg_type, {'stream': stream, 'subject': subject,
 | 
						|
                             'private_message_recipient': pm_recipient,
 | 
						|
                             'replying_to_message': message,
 | 
						|
                             'trigger': opts.trigger});
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
function update_pointer() {
 | 
						|
    if (!pointer_update_in_flight) {
 | 
						|
        pointer_update_in_flight = true;
 | 
						|
        return channel.post({
 | 
						|
            url:      '/json/update_pointer',
 | 
						|
            idempotent: true,
 | 
						|
            data:     {pointer: furthest_read},
 | 
						|
            success: function () {
 | 
						|
                server_furthest_read = furthest_read;
 | 
						|
                pointer_update_in_flight = false;
 | 
						|
            },
 | 
						|
            error: function () {
 | 
						|
                pointer_update_in_flight = false;
 | 
						|
            }
 | 
						|
        });
 | 
						|
    } else {
 | 
						|
        // Return an empty, resolved Deferred.
 | 
						|
        return $.when();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function send_pointer_update() {
 | 
						|
    // Only bother if you've read new messages.
 | 
						|
    if (furthest_read > server_furthest_read) {
 | 
						|
        update_pointer();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function unconditionally_send_pointer_update() {
 | 
						|
    if (pointer_update_in_flight) {
 | 
						|
        // Keep trying.
 | 
						|
        var deferred = $.Deferred();
 | 
						|
 | 
						|
        setTimeout(function () {
 | 
						|
            deferred.resolve(unconditionally_send_pointer_update());
 | 
						|
        }, 100);
 | 
						|
        return deferred;
 | 
						|
    } else {
 | 
						|
        return update_pointer();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function process_message_for_recent_subjects(message, remove_message) {
 | 
						|
    var current_timestamp = 0;
 | 
						|
    var count = 0;
 | 
						|
    var stream = message.stream;
 | 
						|
    var canon_subject = stream_data.canonicalized_name(message.subject);
 | 
						|
 | 
						|
    if (! recent_subjects.has(stream)) {
 | 
						|
        recent_subjects.set(stream, []);
 | 
						|
    } else {
 | 
						|
        recent_subjects.set(stream,
 | 
						|
                            _.filter(recent_subjects.get(stream), function (item) {
 | 
						|
                                var is_duplicate = (item.canon_subject.toLowerCase() === canon_subject.toLowerCase());
 | 
						|
                                if (is_duplicate) {
 | 
						|
                                    current_timestamp = item.timestamp;
 | 
						|
                                    count = item.count;
 | 
						|
                                }
 | 
						|
 | 
						|
                                return !is_duplicate;
 | 
						|
                            }));
 | 
						|
    }
 | 
						|
 | 
						|
    var recents = recent_subjects.get(stream);
 | 
						|
 | 
						|
    if (remove_message !== undefined) {
 | 
						|
        count = count - 1;
 | 
						|
    } else {
 | 
						|
        count = count + 1;
 | 
						|
    }
 | 
						|
 | 
						|
    if (count !== 0) {
 | 
						|
        recents.push({subject: message.subject,
 | 
						|
                      canon_subject: canon_subject,
 | 
						|
                      count: count,
 | 
						|
                      timestamp: Math.max(message.timestamp, current_timestamp)});
 | 
						|
    }
 | 
						|
 | 
						|
    recents.sort(function (a, b) {
 | 
						|
        return b.timestamp - a.timestamp;
 | 
						|
    });
 | 
						|
 | 
						|
    recent_subjects.set(stream, recents);
 | 
						|
}
 | 
						|
 | 
						|
function set_topic_edit_properties(message) {
 | 
						|
    message.always_visible_topic_edit = false;
 | 
						|
    message.on_hover_topic_edit = false;
 | 
						|
    if (feature_flags.disable_message_editing) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Messages with no topics should always have an edit icon visible
 | 
						|
    // to encourage updating them. Admins can also edit any topic.
 | 
						|
    if (message.subject === compose.empty_subject_placeholder()) {
 | 
						|
        message.always_visible_topic_edit = true;
 | 
						|
    } else if (page_params.is_admin) {
 | 
						|
        message.on_hover_topic_edit = true;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
var msg_metadata_cache = {};
 | 
						|
function add_message_metadata(message) {
 | 
						|
    var cached_msg = msg_metadata_cache[message.id];
 | 
						|
    if (cached_msg !== undefined) {
 | 
						|
        // Copy the match subject and content over if they exist on
 | 
						|
        // the new message
 | 
						|
        if (message.match_subject !== undefined) {
 | 
						|
            cached_msg.match_subject = message.match_subject;
 | 
						|
            cached_msg.match_content = message.match_content;
 | 
						|
        }
 | 
						|
        return cached_msg;
 | 
						|
    }
 | 
						|
    get_updates_params.last = Math.max(get_updates_params.last || 0, message.id);
 | 
						|
 | 
						|
    var involved_people;
 | 
						|
 | 
						|
    message.sent_by_me = (message.sender_email === page_params.email);
 | 
						|
 | 
						|
    message.flags = message.flags || [];
 | 
						|
    message.historical = (message.flags !== undefined &&
 | 
						|
                          message.flags.indexOf('historical') !== -1);
 | 
						|
    message.starred = message.flags.indexOf("starred") !== -1;
 | 
						|
    message.mentioned = message.flags.indexOf("mentioned") !== -1 ||
 | 
						|
                        message.flags.indexOf("wildcard_mentioned") !== -1;
 | 
						|
    message.collapsed = message.flags.indexOf("collapsed") !== -1;
 | 
						|
    message.alerted = message.flags.indexOf("has_alert_word") !== -1;
 | 
						|
 | 
						|
    switch (message.type) {
 | 
						|
    case 'stream':
 | 
						|
        message.is_stream = true;
 | 
						|
        message.stream = message.display_recipient;
 | 
						|
        composebox_typeahead.add_topic(message.stream, message.subject);
 | 
						|
        message.reply_to = message.sender_email;
 | 
						|
 | 
						|
        process_message_for_recent_subjects(message);
 | 
						|
 | 
						|
        involved_people = [{'full_name': message.sender_full_name,
 | 
						|
                            'email': message.sender_email}];
 | 
						|
        set_topic_edit_properties(message);
 | 
						|
        break;
 | 
						|
 | 
						|
    case 'private':
 | 
						|
        message.is_private = true;
 | 
						|
        message.reply_to = util.normalize_recipients(
 | 
						|
                get_private_message_recipient(message, 'email'));
 | 
						|
        message.display_reply_to = get_private_message_recipient(message, 'full_name', 'email');
 | 
						|
 | 
						|
        involved_people = message.display_recipient;
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    // Add new people involved in this message to the people list
 | 
						|
    _.each(involved_people, function (person) {
 | 
						|
        // Do the hasOwnProperty() call via the prototype to avoid problems
 | 
						|
        // with keys like "hasOwnProperty"
 | 
						|
        if (! people_dict.has(person.email)) {
 | 
						|
            add_person(person);
 | 
						|
        }
 | 
						|
 | 
						|
        if (people_dict.get(person.email).full_name !== person.full_name) {
 | 
						|
            reify_person(person);
 | 
						|
        }
 | 
						|
 | 
						|
        if (message.type === 'private' && message.sent_by_me) {
 | 
						|
            // Track the number of PMs we've sent to this person to improve autocomplete
 | 
						|
            people_dict.get(person.email).pm_recipient_count += 1;
 | 
						|
        }
 | 
						|
    });
 | 
						|
 | 
						|
    alert_words.process_message(message);
 | 
						|
    msg_metadata_cache[message.id] = message;
 | 
						|
    return message;
 | 
						|
}
 | 
						|
 | 
						|
function add_messages(messages, msg_list, opts) {
 | 
						|
    if (!messages) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    opts = _.extend({messages_are_new: false, delay_render: false}, opts);
 | 
						|
 | 
						|
    util.destroy_loading_indicator($('#page_loading_indicator'));
 | 
						|
    util.destroy_first_run_message();
 | 
						|
 | 
						|
    msg_list.add_messages(messages, opts);
 | 
						|
 | 
						|
    if (msg_list === home_msg_list && opts.messages_are_new) {
 | 
						|
        _.each(messages, function (message) {
 | 
						|
            if (message.sent_by_me) {
 | 
						|
                compose.mark_end_to_end_receive_time(message.id);
 | 
						|
                setTimeout(function () {
 | 
						|
                    compose.mark_end_to_end_display_time(message.id);
 | 
						|
                }, 0);
 | 
						|
            }
 | 
						|
        });
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function maybe_add_narrowed_messages(messages, msg_list, messages_are_new) {
 | 
						|
    var ids = [];
 | 
						|
    _.each(messages, function (elem) {
 | 
						|
        ids.push(elem.id);
 | 
						|
    });
 | 
						|
 | 
						|
    channel.post({
 | 
						|
        url:      '/json/messages_in_narrow',
 | 
						|
        idempotent: true,
 | 
						|
        data:     {msg_ids: JSON.stringify(ids),
 | 
						|
                   narrow:  JSON.stringify(narrow.public_operators())},
 | 
						|
        timeout:  5000,
 | 
						|
        success: function (data) {
 | 
						|
            if (msg_list !== current_msg_list) {
 | 
						|
                // We unnarrowed in the mean time
 | 
						|
                return;
 | 
						|
            }
 | 
						|
 | 
						|
            var new_messages = [];
 | 
						|
            var elsewhere_messages = [];
 | 
						|
            _.each(messages, function (elem) {
 | 
						|
                if (data.messages.hasOwnProperty(elem.id)) {
 | 
						|
                    elem.match_subject = data.messages[elem.id].match_subject;
 | 
						|
                    elem.match_content = data.messages[elem.id].match_content;
 | 
						|
                    new_messages.push(elem);
 | 
						|
                } else {
 | 
						|
                    elsewhere_messages.push(elem);
 | 
						|
                }
 | 
						|
            });
 | 
						|
 | 
						|
            new_messages = _.map(new_messages, add_message_metadata);
 | 
						|
            add_messages(new_messages, msg_list, {messages_are_new: messages_are_new});
 | 
						|
            process_visible_unread_messages();
 | 
						|
            notifications.possibly_notify_new_messages_outside_viewport(new_messages);
 | 
						|
            notifications.notify_messages_outside_current_search(elsewhere_messages);
 | 
						|
        },
 | 
						|
        error: function (xhr) {
 | 
						|
            // We might want to be more clever here
 | 
						|
            setTimeout(function () {
 | 
						|
                if (msg_list === current_msg_list) {
 | 
						|
                    // Don't actually try again if we unnarrowed
 | 
						|
                    // while waiting
 | 
						|
                    maybe_add_narrowed_messages(messages, msg_list, messages_are_new);
 | 
						|
                }
 | 
						|
            }, 5000);
 | 
						|
        }});
 | 
						|
}
 | 
						|
 | 
						|
function update_messages(events) {
 | 
						|
    _.each(events, function (event) {
 | 
						|
        var msg = all_msg_list.get(event.message_id);
 | 
						|
        if (msg === undefined) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        msg.alerted = event.flags.indexOf("has_alert_word") !== -1;
 | 
						|
        msg.mentioned = event.flags.indexOf("mentioned") !== -1 ||
 | 
						|
                        event.flags.indexOf("wildcard_mentioned") !== -1;
 | 
						|
 | 
						|
        ui.un_cache_message_content_height(msg.id);
 | 
						|
 | 
						|
        if (event.rendered_content !== undefined) {
 | 
						|
            msg.content = event.rendered_content;
 | 
						|
        }
 | 
						|
 | 
						|
        if (event.subject !== undefined) {
 | 
						|
            // A topic edit may affect multiple messages, listed in
 | 
						|
            // event.message_ids. event.message_id is still the first message
 | 
						|
            // where the user initiated the edit.
 | 
						|
            _.each(event.message_ids, function (id) {
 | 
						|
                var msg = all_msg_list.get(id);
 | 
						|
                if (msg === undefined) {
 | 
						|
                    return;
 | 
						|
                }
 | 
						|
 | 
						|
                // Remove the recent subjects entry for the old subject;
 | 
						|
                // must be called before we update msg.subject
 | 
						|
                process_message_for_recent_subjects(msg, true);
 | 
						|
                // Update the unread counts; again, this must be called
 | 
						|
                // before we update msg.subject
 | 
						|
                unread.update_unread_subjects(msg, event);
 | 
						|
 | 
						|
                msg.subject = event.subject;
 | 
						|
                msg.subject_links = event.subject_links;
 | 
						|
                set_topic_edit_properties(msg);
 | 
						|
                // Add the recent subjects entry for the new subject; must
 | 
						|
                // be called after we update msg.subject
 | 
						|
                process_message_for_recent_subjects(msg);
 | 
						|
            });
 | 
						|
        }
 | 
						|
 | 
						|
        var row = current_msg_list.get_row(event.message_id);
 | 
						|
        if (row.length > 0) {
 | 
						|
            message_edit.end(row);
 | 
						|
        }
 | 
						|
 | 
						|
        msg.last_edit_timestamp = event.edit_timestamp;
 | 
						|
        delete msg.last_edit_timestr;
 | 
						|
 | 
						|
        notifications.received_messages([msg]);
 | 
						|
        alert_words.process_message(msg);
 | 
						|
    });
 | 
						|
 | 
						|
    home_msg_list.rerender();
 | 
						|
    if (current_msg_list === narrowed_msg_list) {
 | 
						|
        narrowed_msg_list.rerender();
 | 
						|
    }
 | 
						|
    update_unread_counts();
 | 
						|
    stream_list.update_streams_sidebar();
 | 
						|
}
 | 
						|
 | 
						|
function insert_new_messages(messages) {
 | 
						|
    messages = _.map(messages, add_message_metadata);
 | 
						|
 | 
						|
    if (feature_flags.summarize_read_while_narrowed) {
 | 
						|
        _.each(messages, function (message) {
 | 
						|
            if (message.sent_by_me) {
 | 
						|
                summary.maybe_mark_summarized(message);
 | 
						|
            }
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    // You must add add messages to home_msg_list BEFORE
 | 
						|
    // calling process_loaded_for_unread.
 | 
						|
    add_messages(messages, home_msg_list, {messages_are_new: true});
 | 
						|
    add_messages(messages, all_msg_list, {messages_are_new: true});
 | 
						|
 | 
						|
    if (narrow.active()) {
 | 
						|
        if (narrow.filter().can_apply_locally()) {
 | 
						|
            add_messages(messages, narrowed_msg_list, {messages_are_new: true});
 | 
						|
            notifications.possibly_notify_new_messages_outside_viewport(messages);
 | 
						|
        } else {
 | 
						|
            // if we cannot apply locally, we have to wait for this callback to happen to notify
 | 
						|
            maybe_add_narrowed_messages(messages, narrowed_msg_list, true);
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        notifications.possibly_notify_new_messages_outside_viewport(messages);
 | 
						|
    }
 | 
						|
 | 
						|
    process_loaded_for_unread(messages);
 | 
						|
 | 
						|
    if (narrow.narrowed_by_reply()) {
 | 
						|
        // If you send a message when narrowed to a recipient, move the
 | 
						|
        // pointer to it.
 | 
						|
 | 
						|
        var i;
 | 
						|
        var selected_id = current_msg_list.selected_id();
 | 
						|
 | 
						|
        // Iterate backwards to find the last message sent_by_me, stopping at
 | 
						|
        // the pointer position.
 | 
						|
        for (i = messages.length-1; i>=0; i--){
 | 
						|
            var id = messages[i].id;
 | 
						|
            if (id <= selected_id) {
 | 
						|
                break;
 | 
						|
            }
 | 
						|
            if (messages[i].sent_by_me && current_msg_list.get(id) !== undefined) {
 | 
						|
                // If this is a reply we just sent, advance the pointer to it.
 | 
						|
                current_msg_list.select_id(messages[i].id, {then_scroll: true,
 | 
						|
                                                            from_scroll: true});
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    process_visible_unread_messages();
 | 
						|
    notifications.received_messages(messages);
 | 
						|
    stream_list.update_streams_sidebar();
 | 
						|
}
 | 
						|
 | 
						|
function get_updates_success(data) {
 | 
						|
    var messages = [];
 | 
						|
    var messages_to_update = [];
 | 
						|
    var new_pointer;
 | 
						|
 | 
						|
    _.each(data.events, function (event) {
 | 
						|
        get_updates_params.last_event_id = Math.max(get_updates_params.last_event_id,
 | 
						|
                                                    event.id);
 | 
						|
    });
 | 
						|
 | 
						|
    if (tutorial.is_running()) {
 | 
						|
        events_stored_during_tutorial = events_stored_during_tutorial.concat(data.events);
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (events_stored_during_tutorial.length > 0) {
 | 
						|
        data.events = events_stored_during_tutorial.concat(data.events);
 | 
						|
        events_stored_during_tutorial = [];
 | 
						|
    }
 | 
						|
 | 
						|
    _.each(data.events, function (event) {
 | 
						|
        switch (event.type) {
 | 
						|
        case 'message':
 | 
						|
            var msg = event.message;
 | 
						|
            msg.flags = event.flags;
 | 
						|
            if (event.local_message_id !== undefined) {
 | 
						|
                msg.local_id = event.local_message_id;
 | 
						|
            }
 | 
						|
            messages.push(msg);
 | 
						|
            break;
 | 
						|
        case 'pointer':
 | 
						|
            new_pointer = event.pointer;
 | 
						|
            break;
 | 
						|
        case 'restart':
 | 
						|
            reload.initiate({message: "The application has been updated; reloading!"});
 | 
						|
            break;
 | 
						|
        case 'update_message':
 | 
						|
            messages_to_update.push(event);
 | 
						|
            break;
 | 
						|
        case 'realm_user':
 | 
						|
            if (event.op === 'add') {
 | 
						|
                add_person_in_realm(event.person);
 | 
						|
            } else if (event.op === 'remove') {
 | 
						|
                remove_person(event.person);
 | 
						|
            } else if (event.op === 'update') {
 | 
						|
                update_person(event.person);
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case 'stream':
 | 
						|
            if (event.op === 'update') {
 | 
						|
                // Legacy: Stream properties are still managed by subs.js on the client side.
 | 
						|
                subs.update_subscription_properties(event.name, event.property, event.value);
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case 'subscriptions':
 | 
						|
            if (event.op === 'add') {
 | 
						|
                _.each(event.subscriptions, function (subscription) {
 | 
						|
                    $(document).trigger($.Event('subscription_add.zulip',
 | 
						|
                                                {subscription: subscription}));
 | 
						|
                });
 | 
						|
            } else if (event.op === 'remove') {
 | 
						|
                _.each(event.subscriptions, function (subscription) {
 | 
						|
                    $(document).trigger($.Event('subscription_remove.zulip',
 | 
						|
                                                {subscription: subscription}));
 | 
						|
                });
 | 
						|
            } else if (event.op === 'update') {
 | 
						|
                subs.update_subscription_properties(event.name, event.property, event.value);
 | 
						|
            } else if (event.op === 'peer_add' || event.op === 'peer_remove') {
 | 
						|
                _.each(event.subscriptions, function (sub) {
 | 
						|
                    var js_event_type;
 | 
						|
                    if (event.op === 'peer_add') {
 | 
						|
                        js_event_type = 'peer_subscribe.zulip';
 | 
						|
 | 
						|
                        stream_data.add_subscriber(sub, event.user_email);
 | 
						|
                    } else if (event.op === 'peer_remove') {
 | 
						|
                        js_event_type = 'peer_unsubscribe.zulip';
 | 
						|
 | 
						|
                        stream_data.remove_subscriber(sub, event.user_email);
 | 
						|
                    }
 | 
						|
 | 
						|
                    $(document).trigger(js_event_type, {stream_name: sub,
 | 
						|
                                                        user_email: event.user_email});
 | 
						|
                });
 | 
						|
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case 'presence':
 | 
						|
            var users = {};
 | 
						|
            users[event.email] = event.presence;
 | 
						|
            activity.set_user_statuses(users, event.server_timestamp);
 | 
						|
            break;
 | 
						|
        case 'update_message_flags':
 | 
						|
            var new_value = event.operation === "add";
 | 
						|
            switch(event.flag) {
 | 
						|
            case 'starred':
 | 
						|
                _.each(event.messages, function (message_id) {
 | 
						|
                    ui.update_starred(message_id, new_value);
 | 
						|
                });
 | 
						|
                break;
 | 
						|
            case 'read':
 | 
						|
                var msgs_to_update = _.map(event.messages, function (message_id) {
 | 
						|
                    return msg_metadata_cache[message_id];
 | 
						|
                });
 | 
						|
                mark_messages_as_read(msgs_to_update, {from: "server"});
 | 
						|
                break;
 | 
						|
            }
 | 
						|
            break;
 | 
						|
        case 'referral':
 | 
						|
            referral.update_state(event.referrals.granted, event.referrals.used);
 | 
						|
            break;
 | 
						|
        case 'realm_emoji':
 | 
						|
            emoji.update_emojis(event.realm_emoji);
 | 
						|
            break;
 | 
						|
        case 'alert_words':
 | 
						|
            alert_words.words = event.alert_words;
 | 
						|
            break;
 | 
						|
        case 'muted_topics':
 | 
						|
            muting_ui.handle_updates(event.muted_topics);
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    });
 | 
						|
 | 
						|
    if (messages.length !== 0) {
 | 
						|
        messages = echo.process_from_server(messages);
 | 
						|
        insert_new_messages(messages);
 | 
						|
    }
 | 
						|
 | 
						|
    if (new_pointer !== undefined
 | 
						|
        && new_pointer > furthest_read)
 | 
						|
    {
 | 
						|
        furthest_read = new_pointer;
 | 
						|
        server_furthest_read = new_pointer;
 | 
						|
        home_msg_list.select_id(new_pointer, {then_scroll: true, use_closest: true});
 | 
						|
    }
 | 
						|
 | 
						|
    if ((home_msg_list.selected_id() === -1) && !home_msg_list.empty()) {
 | 
						|
        home_msg_list.select_id(home_msg_list.first().id, {then_scroll: false});
 | 
						|
    }
 | 
						|
 | 
						|
    if (messages_to_update.length !== 0) {
 | 
						|
        update_messages(messages_to_update);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
var get_updates_xhr;
 | 
						|
var get_updates_timeout;
 | 
						|
function get_updates(options) {
 | 
						|
    options = _.extend({dont_block: false}, options);
 | 
						|
 | 
						|
    get_updates_params.pointer = furthest_read;
 | 
						|
    get_updates_params.dont_block = options.dont_block || get_updates_failures > 0;
 | 
						|
    if (get_updates_params.queue_id === undefined) {
 | 
						|
        get_updates_params.queue_id = page_params.event_queue_id;
 | 
						|
        get_updates_params.last_event_id = page_params.last_event_id;
 | 
						|
    }
 | 
						|
 | 
						|
    if (get_updates_xhr !== undefined) {
 | 
						|
        get_updates_xhr.abort();
 | 
						|
    }
 | 
						|
    if (get_updates_timeout !== undefined) {
 | 
						|
        clearTimeout(get_updates_timeout);
 | 
						|
    }
 | 
						|
    get_updates_timeout = undefined;
 | 
						|
    get_updates_xhr = channel.post({
 | 
						|
        url:      '/json/get_events',
 | 
						|
        data:     get_updates_params,
 | 
						|
        idempotent: true,
 | 
						|
        timeout:  page_params.poll_timeout,
 | 
						|
        success: function (data) {
 | 
						|
            get_updates_xhr = undefined;
 | 
						|
            get_updates_failures = 0;
 | 
						|
            $('#connection-error').hide();
 | 
						|
 | 
						|
            get_updates_success(data);
 | 
						|
            get_updates_timeout = setTimeout(get_updates, 0);
 | 
						|
        },
 | 
						|
        error: function (xhr, error_type, exn) {
 | 
						|
            get_updates_xhr = undefined;
 | 
						|
            // If we are old enough to have messages outside of the
 | 
						|
            // Tornado cache or if we're old enough that our message
 | 
						|
            // queue has been garbage collected, immediately reload.
 | 
						|
            if ((xhr.status === 400) &&
 | 
						|
                ($.parseJSON(xhr.responseText).msg.indexOf("too old") !== -1 ||
 | 
						|
                 $.parseJSON(xhr.responseText).msg.indexOf("Bad event queue id") !== -1)) {
 | 
						|
                page_params.event_queue_expired = true;
 | 
						|
                reload.initiate({immediate: true});
 | 
						|
            }
 | 
						|
 | 
						|
            if (error_type === 'abort') {
 | 
						|
                // Don't restart if we explicitly aborted
 | 
						|
                return;
 | 
						|
            } else if (error_type === 'timeout') {
 | 
						|
                // Retry indefinitely on timeout.
 | 
						|
                get_updates_failures = 0;
 | 
						|
                $('#connection-error').hide();
 | 
						|
            } else {
 | 
						|
                get_updates_failures += 1;
 | 
						|
            }
 | 
						|
 | 
						|
            if (get_updates_failures >= 5) {
 | 
						|
                $('#connection-error').show();
 | 
						|
            } else {
 | 
						|
                $('#connection-error').hide();
 | 
						|
            }
 | 
						|
 | 
						|
            var retry_sec = Math.min(90, Math.exp(get_updates_failures/2));
 | 
						|
            get_updates_timeout = setTimeout(get_updates, retry_sec*1000);
 | 
						|
        }
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
function force_get_updates() {
 | 
						|
    get_updates_timeout = setTimeout(get_updates, 0);
 | 
						|
}
 | 
						|
 | 
						|
function home_view_loaded() {
 | 
						|
    if (!waiting_on_homeview_load) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    waiting_on_homeview_load = false;
 | 
						|
 | 
						|
    $(document).trigger("home_view_loaded.zulip");
 | 
						|
}
 | 
						|
 | 
						|
function cleanup_event_queue() {
 | 
						|
    // Submit a request to the server to cleanup our event queue
 | 
						|
    if (page_params.event_queue_expired === true) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    channel.del({
 | 
						|
        url:      '/json/events',
 | 
						|
        data:     {queue_id: page_params.event_queue_id}
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
window.addEventListener("beforeunload", function (event) {
 | 
						|
    cleanup_event_queue();
 | 
						|
});
 | 
						|
 | 
						|
function process_result(messages, opts) {
 | 
						|
    $('#get_old_messages_error').hide();
 | 
						|
 | 
						|
    if ((messages.length === 0) && (current_msg_list === narrowed_msg_list) &&
 | 
						|
        narrowed_msg_list.empty()) {
 | 
						|
        // Even after trying to load more messages, we have no
 | 
						|
        // messages to display in this narrow.
 | 
						|
        narrow.show_empty_narrow_message();
 | 
						|
    }
 | 
						|
 | 
						|
    messages = _.map(messages, add_message_metadata);
 | 
						|
 | 
						|
    // If we're loading more messages into the home view, save them to
 | 
						|
    // the all_msg_list as well, as the home_msg_list is reconstructed
 | 
						|
    // from all_msg_list.
 | 
						|
    if (opts.msg_list === home_msg_list) {
 | 
						|
        process_loaded_for_unread(messages);
 | 
						|
        add_messages(messages, all_msg_list, {messages_are_new: false});
 | 
						|
    }
 | 
						|
 | 
						|
    if (messages.length !== 0 && !opts.cont_will_add_messages) {
 | 
						|
        add_messages(messages, opts.msg_list, {messages_are_new: false});
 | 
						|
    }
 | 
						|
 | 
						|
    stream_list.update_streams_sidebar();
 | 
						|
 | 
						|
    if (opts.cont !== undefined) {
 | 
						|
        opts.cont(messages);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function get_old_messages_success(data, opts) {
 | 
						|
    if (waiting_on_browser_scroll) {
 | 
						|
        setTimeout(function () {
 | 
						|
            get_old_messages_success(data, opts);
 | 
						|
        }, 25);
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    if (tutorial.is_running()) {
 | 
						|
        // Don't actually process the messages until the tutorial is
 | 
						|
        // finished, but do disable the loading indicator so it isn't
 | 
						|
        // distracting in the background
 | 
						|
        util.destroy_loading_indicator($('#page_loading_indicator'));
 | 
						|
        tutorial.defer(function () { get_old_messages_success(data, opts); });
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (opts.msg_list.narrowed && opts.msg_list !== current_msg_list) {
 | 
						|
        // We unnarrowed before receiving new messages so
 | 
						|
        // don't bother processing the newly arrived messages.
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    if (! data) {
 | 
						|
        // The server occationally returns no data during a
 | 
						|
        // restart.  Ignore those responses and try again
 | 
						|
        setTimeout(function () {
 | 
						|
            load_old_messages(opts);
 | 
						|
        }, 0);
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    process_result(data.messages, opts);
 | 
						|
    ui.resize_bottom_whitespace();
 | 
						|
 | 
						|
    if (waiting_on_homeview_load && opts.msg_list === home_msg_list) {
 | 
						|
        home_view_loaded();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function load_old_messages(opts) {
 | 
						|
    opts = _.extend({cont_will_add_messages: false}, opts);
 | 
						|
 | 
						|
    var data = {anchor: opts.anchor,
 | 
						|
                num_before: opts.num_before,
 | 
						|
                num_after: opts.num_after};
 | 
						|
 | 
						|
    if (opts.msg_list.narrowed && narrow.active()) {
 | 
						|
        var operators = narrow.public_operators();
 | 
						|
        if (page_params.narrow !== undefined) {
 | 
						|
            operators = operators.concat(page_params.narrow);
 | 
						|
        }
 | 
						|
        data.narrow = JSON.stringify(operators);
 | 
						|
    }
 | 
						|
    if (opts.msg_list === home_msg_list && page_params.narrow_stream !== undefined) {
 | 
						|
        data.narrow = JSON.stringify(page_params.narrow);
 | 
						|
    }
 | 
						|
 | 
						|
    channel.post({
 | 
						|
        url:      '/json/get_old_messages',
 | 
						|
        data:     data,
 | 
						|
        idempotent: true,
 | 
						|
        success: function (data) {
 | 
						|
            get_old_messages_success(data, opts);
 | 
						|
        },
 | 
						|
        error: function (xhr, error_type, exn) {
 | 
						|
            if (opts.msg_list.narrowed && opts.msg_list !== current_msg_list) {
 | 
						|
                // We unnarrowed before getting an error so don't
 | 
						|
                // bother trying again or doing further processing.
 | 
						|
                return;
 | 
						|
            }
 | 
						|
            if (xhr.status === 400) {
 | 
						|
                // Bad request: We probably specified a narrow operator
 | 
						|
                // for a nonexistent stream or something.  We shouldn't
 | 
						|
                // retry or display a connection error.
 | 
						|
                //
 | 
						|
                // FIXME: Warn the user when this has happened?
 | 
						|
                process_result([], opts);
 | 
						|
                return;
 | 
						|
            }
 | 
						|
 | 
						|
            // We might want to be more clever here
 | 
						|
            $('#get_old_messages_error').show();
 | 
						|
            setTimeout(function () {
 | 
						|
                load_old_messages(opts);
 | 
						|
            }, 5000);
 | 
						|
        }
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
function restart_get_updates(options) {
 | 
						|
    get_updates(options);
 | 
						|
}
 | 
						|
 | 
						|
function reset_load_more_status() {
 | 
						|
    load_more_enabled = true;
 | 
						|
    have_scrolled_away_from_top = true;
 | 
						|
    ui.hide_loading_more_messages_indicator();
 | 
						|
}
 | 
						|
 | 
						|
function load_more_messages(msg_list) {
 | 
						|
    var batch_size = 100;
 | 
						|
    var oldest_message_id;
 | 
						|
    if (!load_more_enabled) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    ui.show_loading_more_messages_indicator();
 | 
						|
    load_more_enabled = false;
 | 
						|
    if (msg_list.first() === undefined) {
 | 
						|
        oldest_message_id = page_params.initial_pointer;
 | 
						|
    } else {
 | 
						|
        oldest_message_id = msg_list.first().id;
 | 
						|
    }
 | 
						|
    load_old_messages({
 | 
						|
        anchor: oldest_message_id,
 | 
						|
        num_before: batch_size,
 | 
						|
        num_after: 0,
 | 
						|
        msg_list: msg_list,
 | 
						|
        cont: function (messages) {
 | 
						|
            ui.hide_loading_more_messages_indicator();
 | 
						|
            if (messages.length >= batch_size) {
 | 
						|
                load_more_enabled = true;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
var watchdog_time = $.now();
 | 
						|
setInterval(function () {
 | 
						|
    var new_time = $.now();
 | 
						|
    if ((new_time - watchdog_time) > 20000) { // 20 seconds.
 | 
						|
        // Defensively reset watchdog_time here in case there's an
 | 
						|
        // exception in one of the event handlers
 | 
						|
        watchdog_time = new_time;
 | 
						|
        // Our app's JS wasn't running, which probably means the machine was
 | 
						|
        // asleep.
 | 
						|
        $(document).trigger($.Event('unsuspend'));
 | 
						|
    }
 | 
						|
    watchdog_time = new_time;
 | 
						|
}, 5000);
 | 
						|
 | 
						|
$(function () {
 | 
						|
    $(document).on('unsuspend', function () {
 | 
						|
        // Immediately poll for new updates on unsuspend
 | 
						|
        blueslip.log("Restarting get_updates due to unsuspend");
 | 
						|
        get_updates_failures = 0;
 | 
						|
        restart_get_updates({dont_block: true});
 | 
						|
    });
 | 
						|
});
 | 
						|
 | 
						|
function fast_forward_pointer() {
 | 
						|
    channel.post({
 | 
						|
        url: '/json/get_profile',
 | 
						|
        idempotent: true,
 | 
						|
        data: {email: page_params.email},
 | 
						|
        success: function (data) {
 | 
						|
            mark_all_as_read(function () {
 | 
						|
                furthest_read = data.max_message_id;
 | 
						|
                unconditionally_send_pointer_update().then(function () {
 | 
						|
                    ui.change_tab_to('#home');
 | 
						|
                    reload.initiate({immediate: true, save_state: false});
 | 
						|
                });
 | 
						|
            });
 | 
						|
        }
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
function consider_bankruptcy() {
 | 
						|
    // Until we've handled possibly declaring bankruptcy, don't show
 | 
						|
    // unread counts since they only consider messages that are loaded
 | 
						|
    // client side and may be different from the numbers reported by
 | 
						|
    // the server.
 | 
						|
 | 
						|
    if (!page_params.furthest_read_time) {
 | 
						|
        // We've never read a message.
 | 
						|
        enable_unread_counts();
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    var now = new XDate(true).getTime() / 1000;
 | 
						|
    if ((page_params.unread_count > 500) &&
 | 
						|
        (now - page_params.furthest_read_time > 60 * 60 * 24 * 2)) { // 2 days.
 | 
						|
        var unread_info = templates.render('bankruptcy_modal',
 | 
						|
                                           {"unread_count": page_params.unread_count});
 | 
						|
        $('#bankruptcy-unread-count').html(unread_info);
 | 
						|
        $('#bankruptcy').modal('show');
 | 
						|
    } else {
 | 
						|
        enable_unread_counts();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
_.each(page_params.people_list, function (person) {
 | 
						|
    people_dict.set(person.email, person);
 | 
						|
    people_by_name_dict.set(person.full_name, person);
 | 
						|
    realm_people_dict.set(person.email, person);
 | 
						|
    person.pm_recipient_count = 0;
 | 
						|
});
 | 
						|
 | 
						|
// The special account feedback@zulip.com is used for in-app
 | 
						|
// feedback and should always show up as an autocomplete option.
 | 
						|
if (! people_dict.has('feedback@zulip.com')){
 | 
						|
    add_person({"email": "feedback@zulip.com",
 | 
						|
                "full_name": "Zulip Feedback Bot"});
 | 
						|
}
 | 
						|
 | 
						|
function main() {
 | 
						|
    activity.set_user_statuses(page_params.initial_presences,
 | 
						|
                               page_params.initial_servertime);
 | 
						|
 | 
						|
    server_furthest_read = page_params.initial_pointer;
 | 
						|
    if (page_params.orig_initial_pointer !== undefined &&
 | 
						|
        page_params.orig_initial_pointer > server_furthest_read) {
 | 
						|
        server_furthest_read = page_params.orig_initial_pointer;
 | 
						|
    }
 | 
						|
    furthest_read = server_furthest_read;
 | 
						|
 | 
						|
    // Before trying to load messages: is this user way behind?
 | 
						|
    consider_bankruptcy();
 | 
						|
 | 
						|
    // We only send pointer updates when the user has been idle for a
 | 
						|
    // short while to avoid hammering the server
 | 
						|
    $(document).idle({idle: 1000,
 | 
						|
                      onIdle: send_pointer_update,
 | 
						|
                      keepTracking: true});
 | 
						|
 | 
						|
    $(document).on('message_selected.zulip', function (event) {
 | 
						|
        // Only advance the pointer when not narrowed
 | 
						|
        if (event.id === -1) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        // Additionally, don't advance the pointer server-side
 | 
						|
        // if the selected message is local-only
 | 
						|
        if (event.msg_list === home_msg_list && page_params.narrow_stream === undefined) {
 | 
						|
            if (event.id > furthest_read &&
 | 
						|
                home_msg_list.get(event.id).local_id === undefined) {
 | 
						|
                furthest_read = event.id;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if (event.mark_read && event.previously_selected !== -1) {
 | 
						|
            // Mark messages between old pointer and new pointer as read
 | 
						|
            var messages;
 | 
						|
            if (event.id < event.previously_selected) {
 | 
						|
                messages = message_range(event.msg_list, event.id, event.previously_selected);
 | 
						|
            } else {
 | 
						|
                messages = message_range(event.msg_list, event.previously_selected, event.id);
 | 
						|
            }
 | 
						|
            mark_messages_as_read(messages, {from: 'pointer'});
 | 
						|
        }
 | 
						|
    });
 | 
						|
 | 
						|
    // get the initial message list
 | 
						|
    function load_more(messages) {
 | 
						|
 | 
						|
        // If we received the initially selected message, select it on the client side,
 | 
						|
        // but not if the user has already selected another one during load.
 | 
						|
        //
 | 
						|
        // We fall back to the closest selected id, as the user may have removed
 | 
						|
        // a stream from the home before already
 | 
						|
        if (home_msg_list.selected_id() === -1 && !home_msg_list.empty()) {
 | 
						|
            home_msg_list.select_id(page_params.initial_pointer,
 | 
						|
                {then_scroll: true, use_closest: true});
 | 
						|
        }
 | 
						|
 | 
						|
        // catch the user up
 | 
						|
        if (messages.length !== 0) {
 | 
						|
            var latest_id = messages[messages.length-1].id;
 | 
						|
            if (latest_id < page_params.max_message_id) {
 | 
						|
                load_old_messages({
 | 
						|
                    anchor: latest_id,
 | 
						|
                    num_before: 0,
 | 
						|
                    num_after: 400,
 | 
						|
                    msg_list: home_msg_list,
 | 
						|
                    cont: load_more
 | 
						|
                });
 | 
						|
                return;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        // now start subscribing to updates
 | 
						|
        get_updates();
 | 
						|
 | 
						|
        // backfill more messages after the user is idle
 | 
						|
        var backfill_batch_size = 1000;
 | 
						|
        $(document).idle({'idle': 1000*10,
 | 
						|
                          'onIdle': function () {
 | 
						|
                              var first_id = all_msg_list.first().id;
 | 
						|
                              load_old_messages({
 | 
						|
                                  anchor: first_id,
 | 
						|
                                  num_before: backfill_batch_size,
 | 
						|
                                  num_after: 0,
 | 
						|
                                  msg_list: home_msg_list
 | 
						|
                              });
 | 
						|
                          }});
 | 
						|
    }
 | 
						|
 | 
						|
    if (page_params.have_initial_messages) {
 | 
						|
        load_old_messages({
 | 
						|
            anchor: page_params.initial_pointer,
 | 
						|
            num_before: 200,
 | 
						|
            num_after: 200,
 | 
						|
            msg_list: home_msg_list,
 | 
						|
            cont: load_more
 | 
						|
        });
 | 
						|
    } else {
 | 
						|
        home_view_loaded();
 | 
						|
        get_updates();
 | 
						|
    }
 | 
						|
 | 
						|
    $(document).on('message_id_changed', function (event) {
 | 
						|
        var old_id = event.old_id, new_id = event.new_id;
 | 
						|
        if (furthest_read === old_id) {
 | 
						|
            furthest_read = new_id;
 | 
						|
        }
 | 
						|
        if (get_updates_params.pointer === old_id) {
 | 
						|
            get_updates_params.pointer = new_id;
 | 
						|
        }
 | 
						|
        if (msg_metadata_cache[old_id]) {
 | 
						|
            msg_metadata_cache[new_id] = msg_metadata_cache[old_id];
 | 
						|
            delete msg_metadata_cache[old_id];
 | 
						|
        }
 | 
						|
 | 
						|
        // This handler cannot be in the MessageList constructor, which is the logical place
 | 
						|
        // If it's there, the event handler creates a closure with a reference to the message
 | 
						|
        // list itself. When narrowing, the old narrow message list is discarded and a new one
 | 
						|
        // created, but due to the closure, the old list is not garbage collected. This also leads
 | 
						|
        // to the old list receiving the change id events, and throwing errors as it does not
 | 
						|
        // have the messages that you would expect in its internal data structures.
 | 
						|
        _.each([all_msg_list, home_msg_list, narrowed_msg_list], function (msg_list) {
 | 
						|
            if (msg_list !== undefined) {
 | 
						|
                msg_list.change_message_id(old_id, new_id);
 | 
						|
 | 
						|
                if (msg_list.view !== undefined) {
 | 
						|
                    msg_list.view.change_message_id(old_id, new_id);
 | 
						|
                }
 | 
						|
            }
 | 
						|
        });
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
function install_main_scroll_handler() {
 | 
						|
    waiting_on_browser_scroll = false;
 | 
						|
    // Unregister this special un-throttled scroll handler.
 | 
						|
    $(window).unbind("scroll");
 | 
						|
    // Register the normal throttled scroll handler.
 | 
						|
    ui.register_scroll_handler();
 | 
						|
}
 | 
						|
 | 
						|
$(function () {
 | 
						|
    // On a page load or reload, the browser will, at an idle time, scroll to
 | 
						|
    // the top of the page. We can't intercept this browser-induced scroll, so
 | 
						|
    // to make sure it doesn't in interfere with our scrolling to the correct
 | 
						|
    // place in your message feed, we let the browser scroll happen first.
 | 
						|
 | 
						|
    // After some time, if we haven't seen the browser scroll, give up
 | 
						|
    // and call main.
 | 
						|
    var browser_scroll_timer = setTimeout(function () {
 | 
						|
        install_main_scroll_handler();
 | 
						|
    }, 500);
 | 
						|
 | 
						|
    $(window).scroll(function () {
 | 
						|
        if (viewport.scrollTop() < viewport.height()) {
 | 
						|
            // This is the browser-induced scroll to the top of the
 | 
						|
            // page that we were waiting for. It only happens once,
 | 
						|
            // so stop waiting and call main.
 | 
						|
            install_main_scroll_handler();
 | 
						|
            clearTimeout(browser_scroll_timer);
 | 
						|
        }
 | 
						|
    });
 | 
						|
 | 
						|
    main();
 | 
						|
});
 |