mirror of
https://github.com/zulip/zulip.git
synced 2025-11-04 22:13:26 +00:00
Now that we no longer use tables for our message list, we can more logically group messages together. (imported from commit 9923a092f91a45fe3ef06f2f00e23e4e3fb62a37)
258 lines
8.4 KiB
JavaScript
258 lines
8.4 KiB
JavaScript
var compose_fade = (function () {
|
|
|
|
var exports = {};
|
|
|
|
var focused_recipient;
|
|
var normal_display = false;
|
|
|
|
var fade_heuristic = feature_flags.fade_at_stream_granularity ?
|
|
util.same_major_recipient : util.same_recipient;
|
|
|
|
exports.set_focused_recipient = function (msg_type) {
|
|
if (msg_type === undefined) {
|
|
focused_recipient = undefined;
|
|
}
|
|
|
|
// Construct focused_recipient as a mocked up element which has all the
|
|
// fields of a message used by util.same_recipient()
|
|
focused_recipient = {
|
|
type: msg_type
|
|
};
|
|
|
|
if (focused_recipient.type === "stream") {
|
|
focused_recipient.stream = $('#stream').val();
|
|
focused_recipient.subject = $('#subject').val();
|
|
} else {
|
|
// Normalize the recipient list so it matches the one used when
|
|
// adding the message (see message_store.add_message_metadata()).
|
|
focused_recipient.reply_to = util.normalize_recipients(
|
|
$('#private_message_recipient').val());
|
|
}
|
|
};
|
|
|
|
function _display_messages_normally() {
|
|
rows.get_table(current_msg_list.table_name).find(".recipient_row, .message_row")
|
|
.removeClass("faded").removeClass("unfaded");
|
|
|
|
normal_display = true;
|
|
ui.update_floating_recipient_bar();
|
|
}
|
|
|
|
function _display_users_normally() {
|
|
if (!feature_flags.fade_users_when_composing) {
|
|
return;
|
|
}
|
|
$('.user_sidebar_entry').removeClass('faded').removeClass('unfaded');
|
|
}
|
|
|
|
function change_fade_state(elt, should_fade_message) {
|
|
if (should_fade_message) {
|
|
elt.removeClass("unfaded").addClass("faded");
|
|
} else {
|
|
elt.removeClass("faded").addClass("unfaded");
|
|
}
|
|
}
|
|
|
|
function _fade_messages() {
|
|
var i;
|
|
var should_fade_message = false;
|
|
var visible_messages = viewport.visible_messages(false);
|
|
|
|
normal_display = false;
|
|
|
|
// Update the visible messages first, before the compose box opens
|
|
for (i = 0; i < visible_messages.length; i++) {
|
|
should_fade_message = !fade_heuristic(focused_recipient, visible_messages[i]);
|
|
var elt = current_msg_list.get_row(visible_messages[i].id);
|
|
var recipient_row = rows.get_message_recipient_row(elt);
|
|
|
|
change_fade_state(elt, should_fade_message);
|
|
change_fade_state(recipient_row, should_fade_message);
|
|
}
|
|
|
|
ui.update_floating_recipient_bar();
|
|
|
|
// Defer updating all messages so that the compose box can open sooner
|
|
setTimeout(function (expected_msg_list, expected_recipient) {
|
|
var all_groups = rows.get_table(current_msg_list.table_name).find(".recipient_row");
|
|
|
|
if (current_msg_list !== expected_msg_list ||
|
|
!compose.composing() ||
|
|
compose.recipient() !== expected_recipient) {
|
|
return;
|
|
}
|
|
|
|
should_fade_message = false;
|
|
|
|
// Note: The below algorithm relies on the fact that all_elts is
|
|
// sorted as it would be displayed in the message view
|
|
for (i = 0; i < all_groups.length; i++) {
|
|
var group_elt = $(all_groups[i]);
|
|
should_fade_message = !fade_heuristic(focused_recipient, rows.recipient_from_group(group_elt));
|
|
change_fade_state(group_elt, should_fade_message);
|
|
}
|
|
}, 0, current_msg_list, compose.recipient());
|
|
}
|
|
|
|
exports.would_receive_message = function (email) {
|
|
// Given the current focused_recipient, this function returns true if
|
|
// the user in question would definitely receive this message, false if
|
|
// they would definitely not receive this message, and undefined if we
|
|
// don't know (e.g. the recipient is a stream we're not subscribed to).
|
|
//
|
|
// Yes it's slightly weird to have three return values, but this will be
|
|
// helpful if we want to emphasize the '.unfaded' class later (applied
|
|
// to users who will definitely receive the message).
|
|
|
|
if (email === page_params.email) {
|
|
// We never want to fade you yourself, so pretend it's true even if
|
|
// it's not.
|
|
return true;
|
|
}
|
|
|
|
if (focused_recipient.type === 'stream') {
|
|
var user = people.realm_get(email);
|
|
var sub = stream_data.get_sub(focused_recipient.stream);
|
|
if (user && sub && user.is_bot && !sub.invite_only) {
|
|
// Bots may receive messages on public streams even if they are
|
|
// not subscribed.
|
|
return undefined;
|
|
}
|
|
return stream_data.user_is_subscribed(focused_recipient.stream, email);
|
|
}
|
|
|
|
// PM, so check if the given email is in the recipients list.
|
|
var recipients = focused_recipient.reply_to.split(',');
|
|
return recipients.indexOf(email) !== -1;
|
|
};
|
|
|
|
function _fade_users() {
|
|
if (!feature_flags.fade_users_when_composing) {
|
|
return;
|
|
}
|
|
_.forEach($('.user_sidebar_entry'), function (elt) {
|
|
elt = $(elt);
|
|
var would_receive = exports.would_receive_message(elt.attr('data-email'));
|
|
if (would_receive === true) {
|
|
elt.addClass('unfaded').removeClass('faded');
|
|
} else if (would_receive === false) {
|
|
elt.addClass('faded').removeClass('unfaded');
|
|
} else {
|
|
elt.removeClass('faded').removeClass('unfaded');
|
|
}
|
|
});
|
|
}
|
|
|
|
function _want_normal_display() {
|
|
// If we're not composing show a normal display.
|
|
if (focused_recipient === undefined) {
|
|
return true;
|
|
}
|
|
|
|
// If the user really hasn't specified anything let, then we want a normal display
|
|
if (focused_recipient.type === "stream") {
|
|
// If a stream doesn't exist, there is no real chance of a mix, so fading
|
|
// is just noise to the user.
|
|
if (!stream_data.get_sub(focused_recipient.stream)) {
|
|
return true;
|
|
}
|
|
|
|
// This is kind of debatable. If the topic is empty, it could be that
|
|
// the user simply hasn't started typing it yet, but disabling fading here
|
|
// means the feature doesn't help realms where topics aren't mandatory
|
|
// (which is most realms as of this writing).
|
|
if (focused_recipient.subject === "") {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return focused_recipient.type === "private" && focused_recipient.reply_to === "";
|
|
}
|
|
|
|
function _update_faded_messages() {
|
|
// See also update_faded_messages(), which just wraps this with a debounce.
|
|
// FIXME: This fades users too now, as well as messages, so should have
|
|
// a better name.
|
|
if (_want_normal_display()) {
|
|
if (!normal_display) {
|
|
_display_messages_normally();
|
|
_display_users_normally();
|
|
}
|
|
}
|
|
else {
|
|
_fade_messages();
|
|
_fade_users();
|
|
}
|
|
}
|
|
|
|
// This one only updates the users, not both, like update_faded_messages.
|
|
// This is for when new presence information comes in, redrawing the presence
|
|
// list.
|
|
exports.update_faded_users = function () {
|
|
if (_want_normal_display()) {
|
|
_display_users_normally();
|
|
} else {
|
|
_fade_users();
|
|
}
|
|
};
|
|
|
|
// See trac #1633. For fast typists, calls to _update_faded_messages can
|
|
// cause typing sluggishness.
|
|
exports.update_faded_messages = _.debounce(_update_faded_messages, 50);
|
|
|
|
exports.start_compose = function (msg_type) {
|
|
exports.set_focused_recipient(msg_type);
|
|
_update_faded_messages();
|
|
};
|
|
|
|
exports.clear_compose = function () {
|
|
focused_recipient = undefined;
|
|
_display_messages_normally();
|
|
_display_users_normally();
|
|
};
|
|
|
|
exports.update_message_list = function () {
|
|
if (_want_normal_display()) {
|
|
_display_messages_normally();
|
|
}
|
|
else {
|
|
_fade_messages();
|
|
}
|
|
};
|
|
|
|
exports.update_rendered_messages = function (messages, get_elements) {
|
|
if (_want_normal_display()) {
|
|
return;
|
|
}
|
|
|
|
// This loop is superficially similar to some code in _fade_messages, but an
|
|
// important difference here is that we look at each message individually, whereas
|
|
// the other code takes advantage of blocks beneath recipient bars.
|
|
//
|
|
// get_elements() is plural, because we can get up to two elements:
|
|
// the message (always)
|
|
// the recipient bar (sometimes)
|
|
_.each(messages, function (message) {
|
|
var elts = get_elements(message);
|
|
var should_fade_message = !fade_heuristic(focused_recipient, message);
|
|
|
|
_.each(elts, function (elt) {
|
|
change_fade_state(elt, should_fade_message);
|
|
});
|
|
});
|
|
};
|
|
|
|
$(function () {
|
|
$(document).on('peer_subscribe.zulip', function (e) {
|
|
exports.update_faded_users();
|
|
});
|
|
$(document).on('peer_unsubscribe.zulip', function (e) {
|
|
exports.update_faded_users();
|
|
});
|
|
});
|
|
|
|
|
|
return exports;
|
|
|
|
}());
|