matrix-react-sdk
Version:
SDK for matrix.org using React
859 lines (824 loc) • 136 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
exports.shouldFormContinuation = shouldFormContinuation;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireWildcard(require("react"));
var _reactDom = _interopRequireDefault(require("react-dom"));
var _classnames = _interopRequireDefault(require("classnames"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _logger = require("matrix-js-sdk/src/logger");
var _utils = require("matrix-js-sdk/src/utils");
var _shouldHideEvent = _interopRequireDefault(require("../../shouldHideEvent"));
var _DateUtils = require("../../DateUtils");
var _MatrixClientPeg = require("../../MatrixClientPeg");
var _SettingsStore = _interopRequireDefault(require("../../settings/SettingsStore"));
var _RoomContext = _interopRequireWildcard(require("../../contexts/RoomContext"));
var _Layout = require("../../settings/enums/Layout");
var _EventTile = _interopRequireWildcard(require("../views/rooms/EventTile"));
var _IRCTimelineProfileResizer = _interopRequireDefault(require("../views/elements/IRCTimelineProfileResizer"));
var _dispatcher = _interopRequireDefault(require("../../dispatcher/dispatcher"));
var _WhoIsTypingTile = _interopRequireDefault(require("../views/rooms/WhoIsTypingTile"));
var _ScrollPanel = _interopRequireDefault(require("./ScrollPanel"));
var _DateSeparator = _interopRequireDefault(require("../views/messages/DateSeparator"));
var _TimelineSeparator = _interopRequireWildcard(require("../views/messages/TimelineSeparator"));
var _ErrorBoundary = _interopRequireDefault(require("../views/elements/ErrorBoundary"));
var _Spinner = _interopRequireDefault(require("../views/elements/Spinner"));
var _actions = require("../../dispatcher/actions");
var _EventRenderingUtils = require("../../utils/EventRenderingUtils");
var _EventTileFactory = require("../../events/EventTileFactory");
var _Editing = require("../../Editing");
var _EventUtils = require("../../utils/EventUtils");
var _MainGrouper = require("./grouper/MainGrouper");
var _CreationGrouper = require("./grouper/CreationGrouper");
var _languageHandler = require("../../languageHandler");
var _LateEventGrouper = require("./grouper/LateEventGrouper");
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
/*
Copyright 2024 New Vector Ltd.
Copyright 2016-2023 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const CONTINUATION_MAX_INTERVAL = 5 * 60 * 1000; // 5 minutes
const continuedTypes = [_matrix.EventType.Sticker, _matrix.EventType.RoomMessage];
// check if there is a previous event and it has the same sender as this event
// and the types are the same/is in continuedTypes and the time between them is <= CONTINUATION_MAX_INTERVAL
function shouldFormContinuation(prevEvent, mxEvent, matrixClient, showHiddenEvents, timelineRenderingType) {
if (timelineRenderingType === _RoomContext.TimelineRenderingType.ThreadsList) return false;
// sanity check inputs
if (!prevEvent?.sender || !mxEvent.sender) return false;
// check if within the max continuation period
if (mxEvent.getTs() - prevEvent.getTs() > CONTINUATION_MAX_INTERVAL) return false;
// As we summarise redactions, do not continue a redacted event onto a non-redacted one and vice-versa
if (mxEvent.isRedacted() !== prevEvent.isRedacted()) return false;
// Some events should appear as continuations from previous events of different types.
if (mxEvent.getType() !== prevEvent.getType() && (!continuedTypes.includes(mxEvent.getType()) || !continuedTypes.includes(prevEvent.getType()))) return false;
// Check if the sender is the same and hasn't changed their displayname/avatar between these events
if (mxEvent.sender.userId !== prevEvent.sender.userId || mxEvent.sender.name !== prevEvent.sender.name || mxEvent.sender.getMxcAvatarUrl() !== prevEvent.sender.getMxcAvatarUrl()) return false;
// Thread summaries in the main timeline should break up a continuation on both sides
if (((0, _EventUtils.hasThreadSummary)(mxEvent) || (0, _EventUtils.hasThreadSummary)(prevEvent)) && timelineRenderingType !== _RoomContext.TimelineRenderingType.Thread) {
return false;
}
// if we don't have tile for previous event then it was shown by showHiddenEvents and has no SenderProfile
if (!(0, _EventTileFactory.haveRendererForEvent)(prevEvent, matrixClient, showHiddenEvents)) return false;
return true;
}
/* (almost) stateless UI component which builds the event tiles in the room timeline.
*/
class MessagePanel extends _react.default.Component {
constructor(props, context) {
super(props, context);
// opaque readreceipt info for each userId; used by ReadReceiptMarker
// to manage its animations
(0, _defineProperty2.default)(this, "readReceiptMap", {});
// Track read receipts by event ID. For each _shown_ event ID, we store
// the list of read receipts to display:
// [
// {
// userId: string,
// member: RoomMember,
// ts: number,
// },
// ]
// This is recomputed on each render. It's only stored on the component
// for ease of passing the data around since it's computed in one pass
// over all events.
(0, _defineProperty2.default)(this, "readReceiptsByEvent", new Map());
// Track read receipts by user ID. For each user ID we've ever shown a
// a read receipt for, we store an object:
// {
// lastShownEventId: string,
// receipt: {
// userId: string,
// member: RoomMember,
// ts: number,
// },
// }
// so that we can always keep receipts displayed by reverting back to
// the last shown event for that user ID when needed. This may feel like
// it duplicates the receipt storage in the room, but at this layer, we
// are tracking _shown_ event IDs, which the JS SDK knows nothing about.
// This is recomputed on each render, using the data from the previous
// render as our fallback for any user IDs we can't match a receipt to a
// displayed event in the current render cycle.
(0, _defineProperty2.default)(this, "readReceiptsByUserId", new Map());
(0, _defineProperty2.default)(this, "_showHiddenEvents", void 0);
(0, _defineProperty2.default)(this, "isMounted", false);
(0, _defineProperty2.default)(this, "readMarkerNode", /*#__PURE__*/(0, _react.createRef)());
(0, _defineProperty2.default)(this, "whoIsTyping", /*#__PURE__*/(0, _react.createRef)());
(0, _defineProperty2.default)(this, "scrollPanel", /*#__PURE__*/(0, _react.createRef)());
(0, _defineProperty2.default)(this, "showTypingNotificationsWatcherRef", void 0);
(0, _defineProperty2.default)(this, "eventTiles", {});
// A map to allow groupers to maintain consistent keys even if their first event is uprooted due to back-pagination.
(0, _defineProperty2.default)(this, "grouperKeyMap", new WeakMap());
(0, _defineProperty2.default)(this, "calculateRoomMembersCount", () => {
this.setState({
hideSender: this.shouldHideSender()
});
});
(0, _defineProperty2.default)(this, "onShowTypingNotificationsChange", () => {
this.setState({
showTypingNotifications: _SettingsStore.default.getValue("showTypingNotifications")
});
});
(0, _defineProperty2.default)(this, "isUnmounting", () => {
return !this.isMounted;
});
(0, _defineProperty2.default)(this, "collectGhostReadMarker", node => {
if (node) {
// now the element has appeared, change the style which will trigger the CSS transition
requestAnimationFrame(() => {
node.style.width = "10%";
node.style.opacity = "0";
});
}
});
(0, _defineProperty2.default)(this, "onGhostTransitionEnd", ev => {
// we can now clean up the ghost element
const finishedEventId = ev.target.dataset.eventid;
this.setState({
ghostReadMarkers: this.state.ghostReadMarkers.filter(eid => eid !== finishedEventId)
});
});
(0, _defineProperty2.default)(this, "collectEventTile", (eventId, node) => {
this.eventTiles[eventId] = node;
});
// once dynamic content in the events load, make the scrollPanel check the
// scroll offsets.
(0, _defineProperty2.default)(this, "onHeightChanged", () => {
const scrollPanel = this.scrollPanel.current;
if (scrollPanel) {
scrollPanel.checkScroll();
}
});
(0, _defineProperty2.default)(this, "onTypingShown", () => {
const scrollPanel = this.scrollPanel.current;
// this will make the timeline grow, so checkScroll
scrollPanel?.checkScroll();
if (scrollPanel && scrollPanel.getScrollState().stuckAtBottom) {
scrollPanel.preventShrinking();
}
});
(0, _defineProperty2.default)(this, "onTypingHidden", () => {
const scrollPanel = this.scrollPanel.current;
if (scrollPanel) {
// as hiding the typing notifications doesn't
// update the scrollPanel, we tell it to apply
// the shrinking prevention once the typing notifs are hidden
scrollPanel.updatePreventShrinking();
// order is important here as checkScroll will scroll down to
// reveal added padding to balance the notifs disappearing.
scrollPanel.checkScroll();
}
});
this.state = {
// previous positions the read marker has been in, so we can
// display 'ghost' read markers that are animating away
ghostReadMarkers: [],
showTypingNotifications: _SettingsStore.default.getValue("showTypingNotifications"),
hideSender: this.shouldHideSender()
};
// Cache these settings on mount since Settings is expensive to query,
// and we check this in a hot code path. This is also cached in our
// RoomContext, however we still need a fallback for roomless MessagePanels.
this._showHiddenEvents = _SettingsStore.default.getValue("showHiddenEventsInTimeline");
this.showTypingNotificationsWatcherRef = _SettingsStore.default.watchSetting("showTypingNotifications", null, this.onShowTypingNotificationsChange);
}
componentDidMount() {
this.calculateRoomMembersCount();
this.props.room?.currentState.on(_matrix.RoomStateEvent.Update, this.calculateRoomMembersCount);
this.isMounted = true;
}
componentWillUnmount() {
this.isMounted = false;
this.props.room?.currentState.off(_matrix.RoomStateEvent.Update, this.calculateRoomMembersCount);
_SettingsStore.default.unwatchSetting(this.showTypingNotificationsWatcherRef);
this.readReceiptMap = {};
}
componentDidUpdate(prevProps, prevState) {
if (prevProps.layout !== this.props.layout) {
this.calculateRoomMembersCount();
}
if (prevProps.readMarkerVisible && prevProps.readMarkerEventId && this.props.readMarkerEventId !== prevProps.readMarkerEventId) {
const ghostReadMarkers = this.state.ghostReadMarkers;
ghostReadMarkers.push(prevProps.readMarkerEventId);
this.setState({
ghostReadMarkers
});
}
const pendingEditItem = this.pendingEditItem;
if (!this.props.editState && this.props.room && pendingEditItem) {
const event = this.props.room.findEventById(pendingEditItem);
_dispatcher.default.dispatch({
action: _actions.Action.EditEvent,
event: !event?.isRedacted() ? event : null,
timelineRenderingType: this.context.timelineRenderingType
});
}
}
shouldHideSender() {
return !!this.props.room && this.props.room.getInvitedAndJoinedMemberCount() <= 2 && this.props.layout === _Layout.Layout.Bubble;
}
/* get the DOM node representing the given event */
getNodeForEventId(eventId) {
if (!this.eventTiles) {
return undefined;
}
return this.eventTiles[eventId]?.ref?.current ?? undefined;
}
getTileForEventId(eventId) {
if (!this.eventTiles || !eventId) {
return undefined;
}
return this.eventTiles[eventId];
}
/* return true if the content is fully scrolled down right now; else false.
*/
isAtBottom() {
return this.scrollPanel.current?.isAtBottom();
}
/* get the current scroll state. See ScrollPanel.getScrollState for
* details.
*
* returns null if we are not mounted.
*/
getScrollState() {
return this.scrollPanel.current?.getScrollState() ?? null;
}
// returns one of:
//
// null: there is no read marker
// -1: read marker is above the window
// 0: read marker is within the window
// +1: read marker is below the window
getReadMarkerPosition() {
const readMarker = this.readMarkerNode.current;
const messageWrapper = this.scrollPanel.current;
if (!readMarker || !messageWrapper) {
return null;
}
const wrapperRect = _reactDom.default.findDOMNode(messageWrapper).getBoundingClientRect();
const readMarkerRect = readMarker.getBoundingClientRect();
// the read-marker pretends to have zero height when it is actually
// two pixels high; +2 here to account for that.
if (readMarkerRect.bottom + 2 < wrapperRect.top) {
return -1;
} else if (readMarkerRect.top < wrapperRect.bottom) {
return 0;
} else {
return 1;
}
}
/* jump to the top of the content.
*/
scrollToTop() {
this.scrollPanel.current?.scrollToTop();
}
/* jump to the bottom of the content.
*/
scrollToBottom() {
this.scrollPanel.current?.scrollToBottom();
}
/**
* Scroll up/down in response to a scroll key
*
* @param {KeyboardEvent} ev: the keyboard event to handle
*/
handleScrollKey(ev) {
this.scrollPanel.current?.handleScrollKey(ev);
}
/* jump to the given event id.
*
* offsetBase gives the reference point for the pixelOffset. 0 means the
* top of the container, 1 means the bottom, and fractional values mean
* somewhere in the middle. If omitted, it defaults to 0.
*
* pixelOffset gives the number of pixels *above* the offsetBase that the
* node (specifically, the bottom of it) will be positioned. If omitted, it
* defaults to 0.
*/
scrollToEvent(eventId, pixelOffset, offsetBase) {
this.scrollPanel.current?.scrollToToken(eventId, pixelOffset, offsetBase);
}
scrollToEventIfNeeded(eventId) {
const node = this.getNodeForEventId(eventId);
if (node) {
node.scrollIntoView({
block: "nearest",
behavior: "instant"
});
}
}
get showHiddenEvents() {
return this.context?.showHiddenEvents ?? this._showHiddenEvents;
}
// TODO: Implement granular (per-room) hide options
shouldShowEvent(mxEv, forceHideEvents = false) {
if (this.props.hideThreadedMessages && this.props.room) {
const {
shouldLiveInRoom
} = this.props.room.eventShouldLiveIn(mxEv, this.props.events);
if (!shouldLiveInRoom) {
return false;
}
}
if (_MatrixClientPeg.MatrixClientPeg.safeGet().isUserIgnored(mxEv.getSender())) {
return false; // ignored = no show (only happens if the ignore happens after an event was received)
}
if (this.showHiddenEvents && !forceHideEvents) {
return true;
}
if (!(0, _EventTileFactory.haveRendererForEvent)(mxEv, _MatrixClientPeg.MatrixClientPeg.safeGet(), this.showHiddenEvents)) {
return false; // no tile = no show
}
// Always show highlighted event
if (this.props.highlightedEventId === mxEv.getId()) return true;
return !(0, _shouldHideEvent.default)(mxEv, this.context);
}
readMarkerForEvent(eventId, isLastEvent) {
if (this.context.timelineRenderingType === _RoomContext.TimelineRenderingType.File) return null;
const visible = !isLastEvent && this.props.readMarkerVisible;
if (this.props.readMarkerEventId === eventId) {
let hr;
// if the read marker comes at the end of the timeline (except
// for local echoes, which are excluded from RMs, because they
// don't have useful event ids), we don't want to show it, but
// we still want to create the <li/> for it so that the
// algorithms which depend on its position on the screen aren't
// confused.
if (visible) {
hr = /*#__PURE__*/_react.default.createElement("hr", {
style: {
opacity: 1,
width: "99%"
}
});
}
return /*#__PURE__*/_react.default.createElement("li", {
key: "readMarker_" + eventId,
ref: this.readMarkerNode,
className: "mx_MessagePanel_myReadMarker",
"data-scroll-tokens": eventId
}, hr);
} else if (this.state.ghostReadMarkers.includes(eventId)) {
// We render 'ghost' read markers in the DOM while they
// transition away. This allows the actual read marker
// to be in the right place straight away without having
// to wait for the transition to finish.
// There are probably much simpler ways to do this transition,
// possibly using react-transition-group which handles keeping
// elements in the DOM whilst they transition out, although our
// case is a little more complex because only some of the items
// transition (ie. the read markers do but the event tiles do not)
// and TransitionGroup requires that all its children are Transitions.
const hr = /*#__PURE__*/_react.default.createElement("hr", {
ref: this.collectGhostReadMarker,
onTransitionEnd: this.onGhostTransitionEnd,
"data-eventid": eventId
});
// give it a key which depends on the event id. That will ensure that
// we get a new DOM node (restarting the animation) when the ghost
// moves to a different event.
return /*#__PURE__*/_react.default.createElement("li", {
key: "_readuptoghost_" + eventId,
className: "mx_MessagePanel_myReadMarker"
}, hr);
}
return null;
}
/**
* Find the next event in the list, and the next visible event in the list.
*
* @param events - the list of events to look in and whether they are shown
* @param i - where in the list we are now
*
* @returns { nextEvent, nextTile }
*
* nextEvent is the event after i in the supplied array.
*
* nextTile is the first event in the array after i that we will show a tile
* for. It is used to to determine the 'last successful' flag when rendering
* the tile.
*/
getNextEventInfo(events, i) {
// WARNING: this method is on a hot path.
const nextEventAndShouldShow = i < events.length - 1 ? events[i + 1] : null;
const nextTile = findFirstShownAfter(i, events);
return {
nextEventAndShouldShow,
nextTile
};
}
get pendingEditItem() {
if (!this.props.room) {
return null;
}
try {
return localStorage.getItem((0, _Editing.editorRoomKey)(this.props.room.roomId, this.context.timelineRenderingType));
} catch (err) {
_logger.logger.error(err);
return null;
}
}
isSentState(ev) {
const status = ev.getAssociatedStatus();
// A falsey state applies to events which have come down sync, including remote echoes
return !status || status === _matrix.EventStatus.SENT;
}
getEventTiles() {
// first figure out which is the last event in the list which we're
// actually going to show; this allows us to behave slightly
// differently for the last event in the list. (eg show timestamp)
//
// we also need to figure out which is the last event we show which isn't
// a local echo, to manage the read-marker.
let lastShownEvent;
const events = this.props.events.map(event => {
return {
event,
shouldShow: this.shouldShowEvent(event)
};
});
const userId = _MatrixClientPeg.MatrixClientPeg.safeGet().getSafeUserId();
let foundLastSuccessfulEvent = false;
let lastShownNonLocalEchoIndex = -1;
// Find the indices of the last successful event we sent and the last non-local-echo events shown
for (let i = events.length - 1; i >= 0; i--) {
const {
event,
shouldShow
} = events[i];
if (!shouldShow) {
continue;
}
if (lastShownEvent === undefined) {
lastShownEvent = event;
}
if (!foundLastSuccessfulEvent && this.isSentState(event) && (0, _EventTile.isEligibleForSpecialReceipt)(event)) {
foundLastSuccessfulEvent = true;
// If we are not sender of this last successful event eligible for special receipt then we stop here
// As we do not want to render our sent receipt if there are more receipts below it and events sent
// by other users get a synthetic read receipt for their sent events.
if (event.getSender() === userId) {
events[i].lastSuccessfulWeSent = true;
}
}
if (lastShownNonLocalEchoIndex < 0 && !event.status) {
lastShownNonLocalEchoIndex = i;
}
if (lastShownNonLocalEchoIndex >= 0 && foundLastSuccessfulEvent) {
break;
}
}
const ret = [];
let prevEvent = null; // the last event we showed
// Note: the EventTile might still render a "sent/sending receipt" independent of
// this information. When not providing read receipt information, the tile is likely
// to assume that sent receipts are to be shown more often.
this.readReceiptsByEvent = new Map();
if (this.props.showReadReceipts) {
this.readReceiptsByEvent = this.getReadReceiptsByShownEvent(events);
}
let grouper = null;
for (let i = 0; i < events.length; i++) {
const wrappedEvent = events[i];
const {
event,
shouldShow
} = wrappedEvent;
const eventId = event.getId();
const last = event === lastShownEvent;
const {
nextEventAndShouldShow,
nextTile
} = this.getNextEventInfo(events, i);
if (grouper) {
if (grouper.shouldGroup(wrappedEvent)) {
grouper.add(wrappedEvent);
continue;
} else {
// not part of group, so get the group tiles, close the
// group, and continue like a normal event
ret.push(...grouper.getTiles());
prevEvent = grouper.getNewPrevEvent();
grouper = null;
}
}
for (const Grouper of groupers) {
if (Grouper.canStartGroup(this, wrappedEvent) && !this.props.disableGrouping) {
grouper = new Grouper(this, wrappedEvent, prevEvent, lastShownEvent, nextEventAndShouldShow, nextTile);
break; // break on first grouper
}
}
if (!grouper) {
if (shouldShow) {
// make sure we unpack the array returned by getTilesForEvent,
// otherwise React will auto-generate keys, and we will end up
// replacing all the DOM elements every time we paginate.
ret.push(...this.getTilesForEvent(prevEvent, wrappedEvent, last, false, nextEventAndShouldShow, nextTile));
prevEvent = event;
}
const readMarker = this.readMarkerForEvent(eventId, i >= lastShownNonLocalEchoIndex);
if (readMarker) ret.push(readMarker);
}
}
if (grouper) {
ret.push(...grouper.getTiles());
}
return ret;
}
getTilesForEvent(prevEvent, wrappedEvent, last = false, isGrouped = false, nextEvent = null, nextEventWithTile = null) {
const mxEv = wrappedEvent.event;
const ret = [];
const isEditing = this.props.editState?.getEvent().getId() === mxEv.getId();
// local echoes have a fake date, which could even be yesterday. Treat them as 'today' for the date separators.
const ts1 = mxEv.getTs() ?? Date.now();
// do we need a separator since the last event?
const wantsSeparator = this.wantsSeparator(prevEvent, mxEv);
if (!isGrouped && this.props.room) {
if (wantsSeparator === _TimelineSeparator.SeparatorKind.Date) {
ret.push( /*#__PURE__*/_react.default.createElement("li", {
key: ts1
}, /*#__PURE__*/_react.default.createElement(_DateSeparator.default, {
key: ts1,
roomId: this.props.room.roomId,
ts: ts1
})));
} else if (wantsSeparator === _TimelineSeparator.SeparatorKind.LateEvent) {
const text = (0, _languageHandler._t)("timeline|late_event_separator", {
dateTime: (0, _DateUtils.formatDate)(mxEv.getDate() ?? new Date())
});
ret.push( /*#__PURE__*/_react.default.createElement("li", {
key: ts1
}, /*#__PURE__*/_react.default.createElement(_TimelineSeparator.default, {
key: ts1,
label: text
}, text)));
}
}
const cli = _MatrixClientPeg.MatrixClientPeg.safeGet();
let lastInSection = true;
if (nextEventWithTile) {
const nextEv = nextEventWithTile;
const willWantSeparator = this.wantsSeparator(mxEv, nextEv);
lastInSection = willWantSeparator === _TimelineSeparator.SeparatorKind.Date || mxEv.getSender() !== nextEv.getSender() || (0, _EventRenderingUtils.getEventDisplayInfo)(cli, nextEv, this.showHiddenEvents).isInfoMessage || !shouldFormContinuation(mxEv, nextEv, cli, this.showHiddenEvents, this.context.timelineRenderingType);
}
// is this a continuation of the previous message?
const continuation = wantsSeparator === _TimelineSeparator.SeparatorKind.None && shouldFormContinuation(prevEvent, mxEv, cli, this.showHiddenEvents, this.context.timelineRenderingType);
const eventId = mxEv.getId();
const highlight = eventId === this.props.highlightedEventId;
const readReceipts = this.readReceiptsByEvent.get(eventId);
const callEventGrouper = this.props.callEventGroupers.get(mxEv.getContent().call_id);
// use txnId as key if available so that we don't remount during sending
ret.push( /*#__PURE__*/_react.default.createElement(_EventTile.default, {
key: mxEv.getTxnId() || eventId,
as: "li",
ref: this.collectEventTile.bind(this, eventId),
alwaysShowTimestamps: this.props.alwaysShowTimestamps,
mxEvent: mxEv,
continuation: continuation,
isRedacted: mxEv.isRedacted(),
replacingEventId: mxEv.replacingEventId(),
editState: isEditing ? this.props.editState : undefined,
onHeightChanged: this.onHeightChanged,
readReceipts: readReceipts,
readReceiptMap: this.readReceiptMap,
showUrlPreview: this.props.showUrlPreview,
checkUnmounting: this.isUnmounting,
eventSendStatus: mxEv.getAssociatedStatus() ?? undefined,
isTwelveHour: this.props.isTwelveHour,
permalinkCreator: this.props.permalinkCreator,
last: last,
lastInSection: lastInSection,
lastSuccessful: wrappedEvent.lastSuccessfulWeSent,
isSelectedEvent: highlight,
getRelationsForEvent: this.props.getRelationsForEvent,
showReactions: this.props.showReactions,
layout: this.props.layout,
showReadReceipts: this.props.showReadReceipts,
callEventGrouper: callEventGrouper,
hideSender: this.state.hideSender
}));
return ret;
}
wantsSeparator(prevEvent, mxEvent) {
if (this.context.timelineRenderingType === _RoomContext.TimelineRenderingType.ThreadsList) {
return _TimelineSeparator.SeparatorKind.None;
}
if (prevEvent !== null) {
// If the previous event was late but current is not then show a date separator for orientation
// Otherwise if the current event is of a different late group than the previous show a late event separator
const lateEventInfo = (0, _LateEventGrouper.getLateEventInfo)(mxEvent);
if (lateEventInfo?.group_id !== (0, _LateEventGrouper.getLateEventInfo)(prevEvent)?.group_id) {
return lateEventInfo !== undefined ? _TimelineSeparator.SeparatorKind.LateEvent : _TimelineSeparator.SeparatorKind.Date;
}
}
// first event in the panel: depends on if we could back-paginate from here.
if (prevEvent === null && !this.props.canBackPaginate) {
return _TimelineSeparator.SeparatorKind.Date;
}
const nextEventDate = mxEvent.getDate() ?? new Date();
if (prevEvent !== null && (0, _DateUtils.wantsDateSeparator)(prevEvent.getDate() || undefined, nextEventDate)) {
return _TimelineSeparator.SeparatorKind.Date;
}
return _TimelineSeparator.SeparatorKind.None;
}
// Get a list of read receipts that should be shown next to this event
// Receipts are objects which have a 'userId', 'roomMember' and 'ts'.
getReadReceiptsForEvent(event) {
const myUserId = _MatrixClientPeg.MatrixClientPeg.safeGet().credentials.userId;
// get list of read receipts, sorted most recent first
const {
room
} = this.props;
if (!room) {
return null;
}
const receiptDestination = this.context.threadId ? room.getThread(this.context.threadId) : room;
const receipts = [];
if (!receiptDestination) {
_logger.logger.debug("Discarding request, could not find the receiptDestination for event: " + this.context.threadId);
return receipts;
}
receiptDestination.getReceiptsForEvent(event).forEach(r => {
if (!r.userId || !(0, _utils.isSupportedReceiptType)(r.type) || r.userId === myUserId) {
return; // ignore non-read receipts and receipts from self.
}
if (_MatrixClientPeg.MatrixClientPeg.safeGet().isUserIgnored(r.userId)) {
return; // ignore ignored users
}
const member = room.getMember(r.userId);
receipts.push({
userId: r.userId,
roomMember: member,
ts: r.data ? r.data.ts : 0
});
});
return receipts;
}
// Get an object that maps from event ID to a list of read receipts that
// should be shown next to that event. If a hidden event has read receipts,
// they are folded into the receipts of the last shown event.
getReadReceiptsByShownEvent(events) {
const receiptsByEvent = new Map();
const receiptsByUserId = new Map();
let lastShownEventId;
for (const event of this.props.events) {
if (this.shouldShowEvent(event)) {
lastShownEventId = event.getId();
}
if (!lastShownEventId) {
continue;
}
const existingReceipts = receiptsByEvent.get(lastShownEventId) || [];
const newReceipts = this.getReadReceiptsForEvent(event);
if (!newReceipts) continue;
receiptsByEvent.set(lastShownEventId, existingReceipts.concat(newReceipts));
// Record these receipts along with their last shown event ID for
// each associated user ID.
for (const receipt of newReceipts) {
receiptsByUserId.set(receipt.userId, {
lastShownEventId,
receipt
});
}
}
// It's possible in some cases (for example, when a read receipt
// advances before we have paginated in the new event that it's marking
// received) that we can temporarily not have a matching event for
// someone which had one in the last. By looking through our previous
// mapping of receipts by user ID, we can cover recover any receipts
// that would have been lost by using the same event ID from last time.
for (const userId of this.readReceiptsByUserId.keys()) {
if (receiptsByUserId.get(userId)) {
continue;
}
const {
lastShownEventId,
receipt
} = this.readReceiptsByUserId.get(userId);
const existingReceipts = receiptsByEvent.get(lastShownEventId) || [];
receiptsByEvent.set(lastShownEventId, existingReceipts.concat(receipt));
receiptsByUserId.set(userId, {
lastShownEventId,
receipt
});
}
this.readReceiptsByUserId = receiptsByUserId;
// After grouping receipts by shown events, do another pass to sort each
// receipt list.
for (const receipts of receiptsByEvent.values()) {
receipts.sort((r1, r2) => {
return r2.ts - r1.ts;
});
}
return receiptsByEvent;
}
updateTimelineMinHeight() {
const scrollPanel = this.scrollPanel.current;
if (scrollPanel) {
const isAtBottom = scrollPanel.isAtBottom();
const whoIsTyping = this.whoIsTyping.current;
const isTypingVisible = whoIsTyping && whoIsTyping.isVisible();
// when messages get added to the timeline,
// but somebody else is still typing,
// update the min-height, so once the last
// person stops typing, no jumping occurs
if (isAtBottom && isTypingVisible) {
scrollPanel.preventShrinking();
}
}
}
onTimelineReset() {
const scrollPanel = this.scrollPanel.current;
if (scrollPanel) {
scrollPanel.clearPreventShrinking();
}
}
render() {
let topSpinner;
let bottomSpinner;
if (this.props.backPaginating) {
topSpinner = /*#__PURE__*/_react.default.createElement("li", {
key: "_topSpinner"
}, /*#__PURE__*/_react.default.createElement(_Spinner.default, null));
}
if (this.props.forwardPaginating) {
bottomSpinner = /*#__PURE__*/_react.default.createElement("li", {
key: "_bottomSpinner"
}, /*#__PURE__*/_react.default.createElement(_Spinner.default, null));
}
const style = this.props.hidden ? {
display: "none"
} : {};
let whoIsTyping;
if (this.props.room && this.state.showTypingNotifications && this.context.timelineRenderingType === _RoomContext.TimelineRenderingType.Room) {
whoIsTyping = /*#__PURE__*/_react.default.createElement(_WhoIsTypingTile.default, {
room: this.props.room,
onShown: this.onTypingShown,
onHidden: this.onTypingHidden,
ref: this.whoIsTyping
});
}
let ircResizer;
if (this.props.layout == _Layout.Layout.IRC) {
ircResizer = /*#__PURE__*/_react.default.createElement(_IRCTimelineProfileResizer.default, {
minWidth: 20,
maxWidth: 600,
roomId: this.props.room?.roomId ?? null
});
}
const classes = (0, _classnames.default)(this.props.className, {
mx_MessagePanel_narrow: this.context.narrow
});
return /*#__PURE__*/_react.default.createElement(_ErrorBoundary.default, null, /*#__PURE__*/_react.default.createElement(_ScrollPanel.default, {
ref: this.scrollPanel,
className: classes,
onScroll: this.props.onScroll,
onFillRequest: this.props.onFillRequest,
onUnfillRequest: this.props.onUnfillRequest,
style: style,
stickyBottom: this.props.stickyBottom,
resizeNotifier: this.props.resizeNotifier,
fixedChildren: ircResizer
}, topSpinner, this.getEventTiles(), whoIsTyping, bottomSpinner));
}
}
/**
* Holds on to an event, caching the information about it in the context of the current messages list.
* Avoids calling shouldShowEvent more times than we need to.
* Simplifies threading of event context like whether it's the last successful event we sent which cannot be determined
* by a consumer from the event alone, so has to be done by the event list processing code earlier.
*/
exports.default = MessagePanel;
(0, _defineProperty2.default)(MessagePanel, "contextType", _RoomContext.default);
(0, _defineProperty2.default)(MessagePanel, "defaultProps", {
disableGrouping: false
});
// all the grouper classes that we use, ordered by priority
const groupers = [_CreationGrouper.CreationGrouper, _MainGrouper.MainGrouper];
/**
* Look through the supplied list of WrappedEvent, and return the first
* event that is >start items through the list, and is shown.
*/
function findFirstShownAfter(start, events) {
// Note: this could be done with something like:
// events.slice(i + 1).find((e) => e.shouldShow)?.event ?? null;
// but it is ~10% slower, and this is on the critical path.
for (let n = start + 1; n < events.length; n++) {
const {
event,
shouldShow
} = events[n];
if (shouldShow) {
return event;
}
}
return null;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_reactDom","_interopRequireDefault","_classnames","_matrix","_logger","_utils","_shouldHideEvent","_DateUtils","_MatrixClientPeg","_SettingsStore","_RoomContext","_Layout","_EventTile","_IRCTimelineProfileResizer","_dispatcher","_WhoIsTypingTile","_ScrollPanel","_DateSeparator","_TimelineSeparator","_ErrorBoundary","_Spinner","_actions","_EventRenderingUtils","_EventTileFactory","_Editing","_EventUtils","_MainGrouper","_CreationGrouper","_languageHandler","_LateEventGrouper","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","CONTINUATION_MAX_INTERVAL","continuedTypes","EventType","Sticker","RoomMessage","shouldFormContinuation","prevEvent","mxEvent","matrixClient","showHiddenEvents","timelineRenderingType","TimelineRenderingType","ThreadsList","sender","getTs","isRedacted","getType","includes","userId","name","getMxcAvatarUrl","hasThreadSummary","Thread","haveRendererForEvent","MessagePanel","React","Component","constructor","props","context","_defineProperty2","Map","createRef","setState","hideSender","shouldHideSender","showTypingNotifications","SettingsStore","getValue","isMounted","node","requestAnimationFrame","style","width","opacity","ev","finishedEventId","target","dataset","eventid","ghostReadMarkers","state","filter","eid","eventId","eventTiles","scrollPanel","current","checkScroll","getScrollState","stuckAtBottom","preventShrinking","updatePreventShrinking","_showHiddenEvents","showTypingNotificationsWatcherRef","watchSetting","onShowTypingNotificationsChange","componentDidMount","calculateRoomMembersCount","room","currentState","on","RoomStateEvent","Update","componentWillUnmount","off","unwatchSetting","readReceiptMap","componentDidUpdate","prevProps","prevState","layout","readMarkerVisible","readMarkerEventId","push","pendingEditItem","editState","event","findEventById","defaultDispatcher","dispatch","action","Action","EditEvent","getInvitedAndJoinedMemberCount","Layout","Bubble","getNodeForEventId","undefined","ref","getTileForEventId","isAtBottom","getReadMarkerPosition","readMarker","readMarkerNode","messageWrapper","wrapperRect","ReactDOM","findDOMNode","getBoundingClientRect","readMarkerRect","bottom","top","scrollToTop","scrollToBottom","handleScrollKey","scrollToEvent","pixelOffset","offsetBase","scrollToToken","scrollToEventIfNeeded","scrollIntoView","block","behavior","shouldShowEvent","mxEv","forceHideEvents","hideThreadedMessages","shouldLiveInRoom","eventShouldLiveIn","events","MatrixClientPeg","safeGet","isUserIgnored","getSender","highlightedEventId","getId","shouldHideEvent","readMarkerForEvent","isLastEvent","File","visible","hr","createElement","key","className","collectGhostReadMarker","onTransitionEnd","onGhostTransitionEnd","getNextEventInfo","nextEventAndShouldShow","length","nextTile","findFirstShownAfter","localStorage","getItem","editorRoomKey","roomId","err","logger","error","isSentState","status","getAssociatedStatus","EventStatus","SENT","getEventTiles","lastShownEvent","map","shouldShow","getSafeUserId","foundLastSuccessfulEvent","lastShownNonLocalEchoIndex","isEligibleForSpecialReceipt","lastSuccessfulWeSent","ret","readReceiptsByEvent","showReadReceipts","getReadReceiptsByShownEvent","grouper","wrappedEvent","last","shouldGroup","add","getTiles","getNewPrevEvent","Grouper","groupers","canStartGroup","disableGrouping","getTilesForEvent","isGrouped","nextEvent","nextEventWithTile","isEditing","getEvent","ts1","Date","now","wantsSeparator","SeparatorKind","ts","LateEvent","text","_t","dateTime","formatDate","getDate","label","cli","lastInSection","nextEv","willWantSeparator","getEventDisplayInfo","isInfoMessage","continuation","None","highlight","readReceipts","callEventGrouper","callEventGroupers","getContent","call_id","getTxnId","as","collectEventTile","bind","alwaysShowTimestamps","replacingEventId","onHeightChanged","showUrlPreview","checkUnmounting","isUnmounting","eventSendStatus","isTwelveHour","permalinkCreator","lastSuccessful","isSelectedEvent","getRelationsForEvent","showReactions","lateEventInfo","getLateEventInfo","group_id","canBackPaginate","nextEventDate","wantsDateSeparator","getReadReceiptsForEvent","myUserId","credentials","receiptDestination","threadId","getThread","receipts","debug","getReceiptsForEvent","forEach","isSupportedReceiptType","type","member","getMember","roomMember","data","receiptsByEvent","receiptsByUserId","lastShownEventId","existingReceipts","newReceipts","concat","receipt","readReceiptsByUserId","keys","values","sort","r1","r2","updateTimelineMinHeight","whoIsTyping","isTypingVisible","isVisible","onTimelineReset","clearPreventShrinking","render","topSpinner","bottomSpinner","backPaginating","forwardPaginating","hidden","display","Room","onShown","onTypingShown","onHidden","onTypingHidden","ircResizer","IRC","minWidth","maxWidth","classes","classNames","mx_MessagePanel_narrow","narrow","onScroll","onFillRequest","onUnfillRequest","stickyBottom","resizeNotifier","fixedChildren","exports","RoomContext","CreationGrouper","MainGrouper","start"],"sources":["../../../src/components/structures/MessagePanel.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2016-2023 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React, { createRef, ReactNode, TransitionEvent } from \"react\";\nimport ReactDOM from \"react-dom\";\nimport classNames from \"classnames\";\nimport { Room, MatrixClient, RoomStateEvent, EventStatus, MatrixEvent, EventType } from \"matrix-js-sdk/src/matrix\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport { isSupportedReceiptType } from \"matrix-js-sdk/src/utils\";\n\nimport shouldHideEvent from \"../../shouldHideEvent\";\nimport { formatDate, wantsDateSeparator } from \"../../DateUtils\";\nimport { MatrixClientPeg } from \"../../MatrixClientPeg\";\nimport SettingsStore from \"../../settings/SettingsStore\";\nimport RoomContext, { TimelineRenderingType } from \"../../contexts/RoomContext\";\nimport { Layout } from \"../../settings/enums/Layout\";\nimport EventTile, {\n    GetRelationsForEvent,\n    IReadReceiptProps,\n    isEligibleForSpecialReceipt,\n    UnwrappedEventTile,\n} from \"../views/rooms/EventTile\";\nimport IRCTimelineProfileResizer from \"../views/elements/IRCTimelineProfileResizer\";\nimport defaultDispatcher from \"../../dispatcher/dispatcher\";\nimport LegacyCallEventGrouper from \"./LegacyCallEventGrouper\";\nimport WhoIsTypingTile from \"../views/rooms/WhoIsTypingTile\";\nimport ScrollPanel, { IScrollState } from \"./ScrollPanel\";\nimport DateSeparator from \"../views/messages/DateSeparator\";\nimport TimelineSeparator, { SeparatorKind } from \"../views/messages/TimelineSeparator\";\nimport ErrorBoundary from \"../views/elements/ErrorBoundary\";\nimport ResizeNotifier from \"../../utils/ResizeNotifier\";\nimport Spinner from \"../views/elements/Spinner\";\nimport { RoomPermalinkCreator } from \"../../utils/permalinks/Permalinks\";\nimport EditorStateTransfer from \"../../utils/EditorStateTransfer\";\nimport { Action } from \"../../dispatcher/actions\";\nimport { getEventDisplayInfo } from \"../../utils/EventRenderingUtils\";\nimport { IReadReceiptPosition } from \"../views/rooms/ReadReceiptMarker\";\nimport { haveRendererForEvent } from \"../../events/EventTileFactory\";\nimport { editorRoomKey } from \"../../Editing\";\nimport { hasThreadSummary } from \"../../utils/EventUtils\";\nimport { BaseGrouper } from \"./grouper/BaseGrouper\";\nimport { MainGrouper } from \"./grouper/MainGrouper\";\nimport { CreationGrouper } from \"./grouper/CreationGrouper\";\nimport { _t } from \"../../languageHandler\";\nimport { getLateEventInfo } from \"./grouper/LateEventGrouper\";\n\nconst CONTINUATION_MAX_INTERVAL = 5 * 60 * 1000; // 5 minutes\nconst continuedTypes = [EventType.Sticker, EventType.RoomMessage];\n\n// check if there is a previous event and it has the same sender as this event\n// and the types are the same/is in continuedTypes and the time between them is <= CONTINUATION_MAX_INTERVAL\nexport function shouldFormContinuation(\n    prevEvent: MatrixEvent | null,\n    mxEvent: MatrixEvent,\n    matrixClient: MatrixClient,\n    showHiddenEvents: boolean,\n    timelineRenderingType?: TimelineRenderingType,\n): boolean {\n    if (timelineRenderingType === TimelineRenderingType.ThreadsList) return false;\n    // sanity check inputs\n    if (!prevEvent?.sender || !mxEvent.sender) return false;\n    // check if within the max continuation period\n    if (mxEvent.getTs() - prevEvent.getTs() > CONTINUATION_MAX_INTERVAL) return false;\n\n    // As we summarise redactions, do not continue a redacted event onto a non-redacted one and vice-versa\n    if (mxEvent.isRedacted() !== prevEvent.isRedacted()) return false;\n\n    // Some events should appear as continuations from previous events of different types.\n    if (\n        mxEvent.getType() !== prevEvent.getType() &&\n        (!continuedTypes.includes(mxEvent.getType() as EventType) ||\n            !continuedTypes.includes(prevEvent.getType() as EventType))\n    )\n        return false;\n\n    // Check if the sender is the same and hasn't changed their displayname/avatar between these events\n    if (\n        mxEvent.sender.userId !== prevEvent.sender.userId ||\n        mxEvent.sender.name !== prevEvent.sender.name ||\n        mxEvent.sender.getMxcAvatarUrl() !== prevEvent.sender.getMxcAvatarUrl()\n    )\n        return false;\n\n    // Thread summaries in the main timeline should break up a cont