message-edit: Use last_moved_timestamp for edited/moved indicators.

Use both the last_moved_timestamp and last_edit_timestamp to show
edited and moved indicators/tooltips in the message list view of
the web app, instead of parsing the message edit history array.

We still maintain and build the message edit history array as it's
used for calculating the narrow terms when there is a near operator
and a message has been moved to a different channel or topic.

Updates the tooltip for message edit indicators to include both
the moved and edited time if a message has been both moved and
edited.
This commit is contained in:
Lauryn Menard
2025-03-10 22:27:42 +01:00
committed by Tim Abbott
parent 7c653165af
commit 0f5246400b
9 changed files with 138 additions and 228 deletions

View File

@@ -332,8 +332,11 @@ export function initialize(): void {
} }
if ( if (
realm.realm_message_edit_history_visibility_policy !== realm.realm_message_edit_history_visibility_policy ===
message_edit_history_visibility_policy_values.never.code message_edit_history_visibility_policy_values.always.code ||
(realm.realm_message_edit_history_visibility_policy ===
message_edit_history_visibility_policy_values.moves_only.code &&
message.last_moved_timestamp !== undefined)
) { ) {
fetch_and_render_message_history(message); fetch_and_render_message_history(message);
$("#message-history-overlay .exit-sign").trigger("focus"); $("#message-history-overlay .exit-sign").trigger("focus");

View File

@@ -3,6 +3,8 @@ import _ from "lodash";
import assert from "minimalistic-assert"; import assert from "minimalistic-assert";
import {z} from "zod"; import {z} from "zod";
import * as resolved_topic from "../shared/src/resolved_topic.ts";
import * as activity from "./activity.ts"; import * as activity from "./activity.ts";
import * as alert_words from "./alert_words.ts"; import * as alert_words from "./alert_words.ts";
import * as channel from "./channel.ts"; import * as channel from "./channel.ts";
@@ -375,6 +377,16 @@ export function insert_new_messages(
return messages; return messages;
} }
function topic_resolve_toggled(new_topic: string, original_topic: string): boolean {
if (resolved_topic.is_resolved(new_topic) && new_topic.slice(2) === original_topic) {
return true;
}
if (resolved_topic.is_resolved(original_topic) && original_topic.slice(2) === new_topic) {
return true;
}
return false;
}
export function update_messages(events: UpdateMessageEvent[]): void { export function update_messages(events: UpdateMessageEvent[]): void {
const messages_to_rerender: Message[] = []; const messages_to_rerender: Message[] = [];
let changed_narrow = false; let changed_narrow = false;
@@ -582,7 +594,15 @@ export function update_messages(events: UpdateMessageEvent[]): void {
...moved_message.edit_history, ...moved_message.edit_history,
]; ];
} }
moved_message.last_edit_timestamp = event.edit_timestamp;
if (stream_changed) {
moved_message.last_moved_timestamp = event.edit_timestamp;
} else if (topic_edited) {
assert(new_topic !== undefined);
if (!topic_resolve_toggled(new_topic, orig_topic)) {
moved_message.last_moved_timestamp = event.edit_timestamp;
}
}
// Update the unread counts; again, this must be called // Update the unread counts; again, this must be called
// before we modify the topic field on the message. // before we modify the topic field on the message.
@@ -753,7 +773,7 @@ export function update_messages(events: UpdateMessageEvent[]): void {
// flag is used to indicated update_message events that are // flag is used to indicated update_message events that are
// triggered by server latency optimizations, not user // triggered by server latency optimizations, not user
// interactions; these should not generate edit history updates. // interactions; these should not generate edit history updates.
if (!event.rendering_only) { if (!event.rendering_only && any_message_content_edited) {
anchor_message.last_edit_timestamp = event.edit_timestamp; anchor_message.last_edit_timestamp = event.edit_timestamp;
} }

View File

@@ -384,20 +384,36 @@ export function initialize(): void {
if (!message_container.modified) { if (!message_container.modified) {
return false; return false;
} }
// We know the message has been modified, so we either have a timestamp let edited_time_string = "";
let moved_time_string = "";
if (message_container.edited) {
// We know the message has been edited, so we either have a timestamp
// from the server or from a local edit. // from the server or from a local edit.
assert(message_container.last_edit_timestamp !== undefined); assert(message_container.last_edit_timestamp !== undefined);
const last_modified_time_string = get_time_string( edited_time_string = get_time_string(message_container.last_edit_timestamp);
message_container.last_edit_timestamp, }
); if (message_container.moved) {
// We know the message has been moved, so we have a timestamp from
// the server.
assert(message_container.last_moved_timestamp !== undefined);
moved_time_string = get_time_string(message_container.last_moved_timestamp);
}
const edit_history_access =
realm.realm_message_edit_history_visibility_policy ===
message_edit_history_visibility_policy_values.always.code;
const message_moved_and_move_history_access =
realm.realm_message_edit_history_visibility_policy ===
message_edit_history_visibility_policy_values.moves_only.code &&
message_container.moved;
instance.setContent( instance.setContent(
parse_html( parse_html(
render_message_edit_notice_tooltip({ render_message_edit_notice_tooltip({
moved: message_container.moved, moved: message_container.moved,
last_edit_timestr: last_modified_time_string, edited: message_container.edited,
realm_message_edit_history_is_visible: edited_time_string,
realm.realm_message_edit_history_visibility_policy !== moved_time_string,
message_edit_history_visibility_policy_values.never.code, edit_history_access,
message_moved_and_move_history_access,
}), }),
), ),
); );

View File

@@ -55,11 +55,13 @@ export type MessageContainer = {
include_sender: boolean; include_sender: boolean;
is_hidden: boolean; is_hidden: boolean;
last_edit_timestamp: number | undefined; last_edit_timestamp: number | undefined;
last_moved_timestamp: number | undefined;
mention_classname: string | undefined; mention_classname: string | undefined;
message_edit_notices_in_left_col: boolean; message_edit_notices_in_left_col: boolean;
message_edit_notices_alongside_sender: boolean; message_edit_notices_alongside_sender: boolean;
message_edit_notices_for_status_message: boolean; message_edit_notices_for_status_message: boolean;
modified: boolean; modified: boolean;
edited: boolean;
moved: boolean; moved: boolean;
msg: Message; msg: Message;
sender_is_bot: boolean; sender_is_bot: boolean;
@@ -164,69 +166,6 @@ function same_recipient(a: MessageContainer | undefined, b: MessageContainer | u
return util.same_recipient(a.msg, b.msg); return util.same_recipient(a.msg, b.msg);
} }
function analyze_edit_history(
message: Message,
last_edit_timestamp: number | undefined,
): {
edited: boolean;
moved: boolean;
resolve_toggled: boolean;
} {
// Returns a dict of booleans that describe the message's history:
// * edited: if the message has had its content edited
// * moved: if the message has had its stream/topic edited
// * resolve_toggled: if the message has had a topic resolve/unresolve edit
let edited = false;
let moved = false;
let resolve_toggled = false;
if (message.edit_history !== undefined) {
for (const edit_history_event of message.edit_history) {
if (edit_history_event.prev_content) {
edited = true;
}
if (edit_history_event.prev_stream) {
moved = true;
}
if (edit_history_event.prev_topic !== undefined) {
// TODO: Possibly this assert could be removed if we tightened the type
// on edit history elements such that a `prev_topic` being present means a
// `topic` element is.
assert(edit_history_event.topic !== undefined);
// We know it has a topic edit. Now we need to determine if
// it was a true move or a resolve/unresolve.
if (
resolved_topic.is_resolved(edit_history_event.topic) &&
edit_history_event.topic.slice(2) === edit_history_event.prev_topic
) {
// Resolved.
resolve_toggled = true;
continue;
}
if (
resolved_topic.is_resolved(edit_history_event.prev_topic) &&
edit_history_event.prev_topic.slice(2) === edit_history_event.topic
) {
// Unresolved.
resolve_toggled = true;
continue;
}
// Otherwise, it is a real topic rename/move.
moved = true;
}
}
} else if (last_edit_timestamp !== undefined) {
// When the edit_history is disabled for the organization, we do not receive the edit_history
// variable in the message object. In this case, we will check if the last_edit_timestamp is
// available or not. Since we don't have the edit_history, we can't determine if the message
// was moved or edited. Therefore, we simply mark the messages as edited.
edited = true;
}
return {edited, moved, resolve_toggled};
}
function get_group_display_date(message: Message, display_year: boolean): string { function get_group_display_date(message: Message, display_year: boolean): string {
const time = new Date(message.timestamp * 1000); const time = new Date(message.timestamp * 1000);
const date_element = timerender.render_date(time, display_year); const date_element = timerender.render_date(time, display_year);
@@ -592,8 +531,10 @@ export class MessageListView {
); );
} }
_get_message_edited_vars(message: Message): { _get_message_edited_and_moved_vars(message: Message): {
last_edit_timestamp: number | undefined; last_edit_timestamp: number | undefined;
last_moved_timestamp: number | undefined;
edited: boolean;
moved: boolean; moved: boolean;
modified: boolean; modified: boolean;
} { } {
@@ -603,25 +544,14 @@ export class MessageListView {
} else { } else {
last_edit_timestamp = message.last_edit_timestamp; last_edit_timestamp = message.last_edit_timestamp;
} }
const edit_history_details = analyze_edit_history(message, last_edit_timestamp); const last_moved_timestamp = message.last_moved_timestamp;
if (!last_edit_timestamp || !(edit_history_details.moved || edit_history_details.edited)) {
// For messages whose edit history at most includes
// resolving topics, we don't display an EDITED/MOVED
// notice at all. (The message actions popover will still
// display an edit history option, so you can see when it
// was marked as resolved if you need to).
return {
last_edit_timestamp: undefined,
moved: false,
modified: false,
};
}
return { return {
last_edit_timestamp, last_edit_timestamp,
moved: edit_history_details.moved && !edit_history_details.edited, last_moved_timestamp,
modified: true, edited: last_edit_timestamp !== undefined,
moved: last_moved_timestamp !== undefined,
modified: last_edit_timestamp !== undefined || last_moved_timestamp !== undefined,
}; };
} }
@@ -646,6 +576,8 @@ export class MessageListView {
include_sender: boolean; include_sender: boolean;
status_message: string | false; status_message: string | false;
last_edit_timestamp: number | undefined; last_edit_timestamp: number | undefined;
last_moved_timestamp: number | undefined;
edited: boolean;
moved: boolean; moved: boolean;
modified: boolean; modified: boolean;
} { } {
@@ -745,7 +677,7 @@ export class MessageListView {
mention_classname, mention_classname,
include_sender, include_sender,
...this._maybe_get_me_message(is_hidden, message), ...this._maybe_get_me_message(is_hidden, message),
...this._get_message_edited_vars(message), ...this._get_message_edited_and_moved_vars(message),
}; };
} }

View File

@@ -74,6 +74,7 @@ export const raw_message_schema = z.intersection(
id: z.number(), id: z.number(),
is_me_message: z.boolean(), is_me_message: z.boolean(),
last_edit_timestamp: z.optional(z.number()), last_edit_timestamp: z.optional(z.number()),
last_moved_timestamp: z.optional(z.number()),
reactions: z.array(message_reaction_schema), reactions: z.array(message_reaction_schema),
recipient_id: z.number(), recipient_id: z.number(),
sender_email: z.string(), sender_email: z.string(),

View File

@@ -49,6 +49,7 @@ export const server_message_schema = z
id: z.number(), id: z.number(),
is_me_message: z.boolean(), is_me_message: z.boolean(),
last_edit_timestamp: z.number().optional(), last_edit_timestamp: z.number().optional(),
last_moved_timestamp: z.number().optional(),
reactions: message_reaction_schema, reactions: message_reaction_schema,
recipient_id: z.number(), recipient_id: z.number(),
sender_email: z.string(), sender_email: z.string(),

View File

@@ -4,13 +4,13 @@
<div class="message_edit_notice"> <div class="message_edit_notice">
{{t "SAVING"}} {{t "SAVING"}}
</div> </div>
{{else if moved}} {{else if edited}}
<div class="message_edit_notice">
{{t "MOVED"}}
</div>
{{else}}
<div class="message_edit_notice"> <div class="message_edit_notice">
{{t "EDITED"}} {{t "EDITED"}}
</div> </div>
{{else}}
<div class="message_edit_notice">
{{t "MOVED"}}
</div>
{{/if}} {{/if}}
{{/if}} {{/if}}

View File

@@ -1,15 +1,30 @@
<div> <div>
{{#if realm_message_edit_history_is_visible}} {{#if edit_history_access}}
{{#if edited}}
{{#if moved}}
<div>{{t "View edit and move history"}}</div>
{{else}}
<div>{{t "View edit history"}}</div> <div>{{t "View edit history"}}</div>
{{/if}} {{/if}}
<div class="tooltip-inner-content italic"> {{else if moved}}
{{#if moved}} <div>{{t "View move history"}}</div>
{{t 'Last moved {last_edit_timestr}.'}}
{{else}}
{{t 'Last edited {last_edit_timestr}.'}}
{{/if}} {{/if}}
{{else if message_moved_and_move_history_access}}
<div>{{t "View move history"}}</div>
{{/if}}
{{#if edited}}
<div class="tooltip-inner-content italic">
{{t 'Last edited {edited_time_string}.'}}
</div> </div>
{{/if}}
{{#if moved}}
<div class="tooltip-inner-content italic">
{{t 'Last moved {moved_time_string}.'}}
</div>
{{/if}}
</div> </div>
{{#if realm_message_edit_history_is_visible}} {{#if edit_history_access}}
{{tooltip_hotkey_hints "Shift" "H"}}
{{else if message_moved_and_move_history_access}}
{{tooltip_hotkey_hints "Shift" "H"}} {{tooltip_hotkey_hints "Shift" "H"}}
{{/if}} {{/if}}

View File

@@ -45,27 +45,9 @@ function test(label, f) {
}); });
} }
test("msg_moved_var", () => { test("msg_edited_and_moved_vars", () => {
// This is a test to verify that when the stream or topic is changed // This is a test to verify that when the stream or topic is changed
// (and the content is not), the message says "MOVED" rather than "EDITED." // (and the content is not), the message says "MOVED" rather than "EDITED."
// See the end of the test for the list of cases verified.
function build_message_context(message = {}, message_context = {}) {
message_context = {
...message_context,
};
if ("edit_history" in message) {
message_context.msg = {
last_edit_timestamp: (next_timestamp += 1),
...message,
};
} else {
message_context.msg = {
...message,
};
}
return message_context;
}
function build_message_group(messages) { function build_message_group(messages) {
return {message_containers: messages}; return {message_containers: messages};
@@ -83,85 +65,29 @@ test("msg_moved_var", () => {
return list; return list;
} }
function assert_moved_true(message_container) {
assert.equal(message_container.moved, true);
}
function assert_moved_false(message_container) {
assert.equal(message_container.moved, false);
}
(function test_msg_moved_var() { (function test_msg_moved_var() {
const messages = [ const messages = [
// no edit history: NO LABEL // no edit or moved timestamps
build_message_context({}), {msg: {}},
// stream changed: MOVED // edit timestamp: EDITED
build_message_context({
edit_history: [{prev_stream: 1, timestamp: 1000, user_id: 1}],
}),
// topic changed (not resolved/unresolved): MOVED
build_message_context({
edit_history: [
{prev_topic: "test_topic", topic: "new_topic", timestamp: 1000, user_id: 1},
],
}),
// content edited: EDITED
build_message_context({
edit_history: [{prev_content: "test_content", timestamp: 1000, user_id: 1}],
}),
// stream and topic edited: MOVED
build_message_context({
edit_history: [
{ {
prev_stream: 1, msg: {
prev_topic: "test_topic", last_edit_timestamp: (next_timestamp += 1),
topic: "new_topic",
timestamp: 1000,
user_id: 1,
}, },
], },
}), // moved timestamp: MOVED
// topic and content changed: EDITED
build_message_context({
edit_history: [
{ {
prev_topic: "test_topic", msg: {
topic: "new_topic", last_moved_timestamp: (next_timestamp += 1),
prev_content: "test_content", },
timestamp: 1000, },
user_id: 1, // both edit and moved timestamp: EDITED
{
msg: {
last_edit_timestamp: (next_timestamp += 1),
last_moved_timestamp: (next_timestamp += 1),
},
}, },
],
}),
// only topic resolved: NO LABEL
build_message_context({
edit_history: [
{prev_topic: "test_topic", topic: "✔ test_topic", timestamp: 1000, user_id: 1},
],
}),
// only topic unresolved: NO LABEL
build_message_context({
edit_history: [
{prev_topic: "✔ test_topic", topic: "test_topic", timestamp: 1000, user_id: 1},
],
}),
// multiple edit history logs, with at least one content edit: EDITED
build_message_context({
edit_history: [
{prev_stream: 1, timestamp: 1000, user_id: 1},
{prev_topic: "old_topic", topic: "test_topic", timestamp: 1001, user_id: 1},
{prev_content: "test_content", timestamp: 1002, user_id: 1},
{prev_topic: "test_topic", topic: "✔ test_topic", timestamp: 1003, user_id: 1},
],
}),
// multiple edit history logs with no content edit: MOVED
build_message_context({
edit_history: [
{prev_stream: 1, timestamp: 1000, user_id: 1},
{prev_topic: "old_topic", topic: "test_topic", timestamp: 1001, user_id: 1},
{prev_topic: "test_topic", topic: "✔ test_topic", timestamp: 1002, user_id: 1},
{prev_topic: "✔ test_topic", topic: "test_topic", timestamp: 1003, user_id: 1},
],
}),
]; ];
const message_group = build_message_group(messages); const message_group = build_message_group(messages);
@@ -171,32 +97,28 @@ test("msg_moved_var", () => {
Object.assign( Object.assign(
message_container, message_container,
list._maybe_get_me_message(message_container.is_hidden, message_container.msg), list._maybe_get_me_message(message_container.is_hidden, message_container.msg),
list._get_message_edited_vars(message_container.msg), list._get_message_edited_and_moved_vars(message_container.msg),
); );
} }
const result = list._message_groups[0].message_containers; const result = list._message_groups[0].message_containers;
// no edit history: false // no edit or moved timestamps
assert_moved_false(result[0]); assert.equal(result[0].edited, false);
// stream changed: true assert.equal(result[0].moved, false);
assert_moved_true(result[1]); assert.equal(result[0].modified, false);
// topic changed: true // edit timestamp: EDITED
assert_moved_true(result[2]); assert.equal(result[1].edited, true);
// content edited: false assert.equal(result[1].moved, false);
assert_moved_false(result[3]); assert.equal(result[1].modified, true);
// stream and topic edited: true // moved timestamp: MOVED
assert_moved_true(result[4]); assert.equal(result[2].edited, false);
// topic and content changed: false assert.equal(result[2].moved, true);
assert_moved_false(result[5]); assert.equal(result[2].modified, true);
// only topic resolved: false // both edit and moved timestamp: EDITED
assert_moved_false(result[6]); assert.equal(result[3].edited, true);
// only topic unresolved: false assert.equal(result[3].moved, true);
assert_moved_false(result[7]); assert.equal(result[3].modified, true);
// multiple edits with content edit: false
assert_moved_false(result[8]);
// multiple edits without content edit: true
assert_moved_true(result[9]);
})(); })();
}); });
@@ -283,7 +205,7 @@ test("message_edited_vars", () => {
Object.assign( Object.assign(
message_container, message_container,
list._maybe_get_me_message(message_container.is_hidden, message_container.msg), list._maybe_get_me_message(message_container.is_hidden, message_container.msg),
list._get_message_edited_vars(message_container.msg), list._get_message_edited_and_moved_vars(message_container.msg),
); );
} }
@@ -368,7 +290,7 @@ test("muted_message_vars", () => {
]; ];
const message_group = build_message_group(messages); const message_group = build_message_group(messages);
const list = build_list([message_group]); const list = build_list([message_group]);
list._get_message_edited_vars = noop; list._get_message_edited_and_moved_vars = noop;
// Sender is not muted. // Sender is not muted.
let result = calculate_variables(list, messages); let result = calculate_variables(list, messages);