matrix-react-sdk
Version:
SDK for matrix.org using React
161 lines (152 loc) • 25.7 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _lodash = require("lodash");
var _matrix = require("matrix-js-sdk/src/matrix");
var _types = require("matrix-js-sdk/src/types");
var _MatrixClientPeg = require("../MatrixClientPeg");
var _QueryMatcher = _interopRequireDefault(require("./QueryMatcher"));
var _Components = require("./Components");
var _AutocompleteProvider = _interopRequireDefault(require("./AutocompleteProvider"));
var _languageHandler = require("../languageHandler");
var _Permalinks = require("../utils/permalinks/Permalinks");
var _MemberAvatar = _interopRequireDefault(require("../components/views/avatars/MemberAvatar"));
var _UserIdentifier = _interopRequireDefault(require("../customisations/UserIdentifier"));
/*
Copyright 2024 New Vector Ltd.
Copyright 2018 Michael Telatynski <7t3chguy@gmail.com>
Copyright 2017, 2018 New Vector Ltd
Copyright 2017 Vector Creations Ltd
Copyright 2016 Aviral Dasgupta
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const USER_REGEX = /\B@\S*/g;
// used when you hit 'tab' - we allow some separator chars at the beginning
// to allow you to tab-complete /mat into /(matthew)
const FORCED_USER_REGEX = /[^/,:; \t\n]\S*/g;
class UserProvider extends _AutocompleteProvider.default {
constructor(_room, renderingType) {
super({
commandRegex: USER_REGEX,
forcedCommandRegex: FORCED_USER_REGEX,
renderingType
});
(0, _defineProperty2.default)(this, "matcher", void 0);
(0, _defineProperty2.default)(this, "users", void 0);
(0, _defineProperty2.default)(this, "room", void 0);
(0, _defineProperty2.default)(this, "onRoomTimeline", (ev, room, toStartOfTimeline, removed, data) => {
if (!room) return; // notification timeline, we'll get this event again with a room specific timeline
if (removed) return;
if (room.roomId !== this.room.roomId) return;
// ignore events from filtered timelines
if (data.timeline.getTimelineSet() !== room.getUnfilteredTimelineSet()) return;
// ignore anything but real-time updates at the end of the room:
// updates from pagination will happen when the paginate completes.
if (toStartOfTimeline || !data || !data.liveEvent) return;
// TODO: lazyload if we have no ev.sender room member?
this.onUserSpoke(ev.sender);
});
(0, _defineProperty2.default)(this, "onRoomStateUpdate", state => {
// ignore updates in other rooms
if (state.roomId !== this.room.roomId) return;
// blow away the users cache
this.users = undefined;
});
this.room = _room;
this.matcher = new _QueryMatcher.default([], {
keys: ["name"],
funcs: [obj => obj.userId.slice(1)],
// index by user id minus the leading '@'
shouldMatchWordsOnly: false
});
_MatrixClientPeg.MatrixClientPeg.safeGet().on(_matrix.RoomEvent.Timeline, this.onRoomTimeline);
_MatrixClientPeg.MatrixClientPeg.safeGet().on(_matrix.RoomStateEvent.Update, this.onRoomStateUpdate);
}
destroy() {
_MatrixClientPeg.MatrixClientPeg.get()?.removeListener(_matrix.RoomEvent.Timeline, this.onRoomTimeline);
_MatrixClientPeg.MatrixClientPeg.get()?.removeListener(_matrix.RoomStateEvent.Update, this.onRoomStateUpdate);
}
async getCompletions(rawQuery, selection, force = false, limit = -1) {
// lazy-load user list into matcher
if (!this.users) this.makeUsers();
const {
command,
range
} = this.getCurrentCommand(rawQuery, selection, force);
const fullMatch = command?.[0];
// Don't search if the query is a single "@"
if (fullMatch && fullMatch !== "@") {
// Don't include the '@' in our search query - it's only used as a way to trigger completion
const query = fullMatch.startsWith("@") ? fullMatch.substring(1) : fullMatch;
return this.matcher.match(query, limit).map(user => {
const description = _UserIdentifier.default.getDisplayUserIdentifier?.(user.userId, {
roomId: this.room.roomId,
withDisplayName: true
});
const displayName = user.name || user.userId || "";
return {
// Length of completion should equal length of text in decorator. draft-js
// relies on the length of the entity === length of the text in the decoration.
completion: user.rawDisplayName,
completionId: user.userId,
type: "user",
suffix: selection.beginning && range.start === 0 ? ": " : " ",
href: (0, _Permalinks.makeUserPermalink)(user.userId),
component: /*#__PURE__*/_react.default.createElement(_Components.PillCompletion, {
title: displayName,
description: description
}, /*#__PURE__*/_react.default.createElement(_MemberAvatar.default, {
member: user,
size: "24px"
})),
range: range
};
});
}
return [];
}
getName() {
return (0, _languageHandler._t)("composer|autocomplete|user_description");
}
makeUsers() {
const events = this.room.getLiveTimeline().getEvents();
const lastSpoken = {};
for (const event of events) {
lastSpoken[event.getSender()] = event.getTs();
}
const currentUserId = _MatrixClientPeg.MatrixClientPeg.safeGet().credentials.userId;
this.users = this.room.getJoinedMembers().filter(({
userId
}) => userId !== currentUserId);
this.users = this.users.concat(this.room.getMembersWithMembership(_types.KnownMembership.Invite));
this.users = (0, _lodash.sortBy)(this.users, member => 1e20 - lastSpoken[member.userId] || 1e20);
this.matcher.setObjects(this.users);
}
onUserSpoke(user) {
if (!this.users) return;
if (!user) return;
if (user.userId === _MatrixClientPeg.MatrixClientPeg.safeGet().getSafeUserId()) return;
// Move the user that spoke to the front of the array
this.users.splice(this.users.findIndex(user2 => user2.userId === user.userId), 1);
this.users = [user, ...this.users];
this.matcher.setObjects(this.users);
}
renderCompletions(completions) {
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Autocomplete_Completion_container_pill",
role: "presentation",
"aria-label": (0, _languageHandler._t)("composer|autocomplete|user_a11y")
}, completions);
}
shouldForceComplete() {
return true;
}
}
exports.default = UserProvider;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_lodash","_matrix","_types","_MatrixClientPeg","_QueryMatcher","_Components","_AutocompleteProvider","_languageHandler","_Permalinks","_MemberAvatar","_UserIdentifier","USER_REGEX","FORCED_USER_REGEX","UserProvider","AutocompleteProvider","constructor","room","renderingType","commandRegex","forcedCommandRegex","_defineProperty2","default","ev","toStartOfTimeline","removed","data","roomId","timeline","getTimelineSet","getUnfilteredTimelineSet","liveEvent","onUserSpoke","sender","state","users","undefined","matcher","QueryMatcher","keys","funcs","obj","userId","slice","shouldMatchWordsOnly","MatrixClientPeg","safeGet","on","RoomEvent","Timeline","onRoomTimeline","RoomStateEvent","Update","onRoomStateUpdate","destroy","get","removeListener","getCompletions","rawQuery","selection","force","limit","makeUsers","command","range","getCurrentCommand","fullMatch","query","startsWith","substring","match","map","user","description","UserIdentifierCustomisations","getDisplayUserIdentifier","withDisplayName","displayName","name","completion","rawDisplayName","completionId","type","suffix","beginning","start","href","makeUserPermalink","component","createElement","PillCompletion","title","member","size","getName","_t","events","getLiveTimeline","getEvents","lastSpoken","event","getSender","getTs","currentUserId","credentials","getJoinedMembers","filter","concat","getMembersWithMembership","KnownMembership","Invite","sortBy","setObjects","getSafeUserId","splice","findIndex","user2","renderCompletions","completions","className","role","shouldForceComplete","exports"],"sources":["../../src/autocomplete/UserProvider.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2018 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2017, 2018 New Vector Ltd\nCopyright 2017 Vector Creations Ltd\nCopyright 2016 Aviral Dasgupta\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 from \"react\";\nimport { sortBy } from \"lodash\";\nimport {\n    MatrixEvent,\n    Room,\n    RoomEvent,\n    RoomMember,\n    RoomState,\n    RoomStateEvent,\n    IRoomTimelineData,\n} from \"matrix-js-sdk/src/matrix\";\nimport { KnownMembership } from \"matrix-js-sdk/src/types\";\n\nimport { MatrixClientPeg } from \"../MatrixClientPeg\";\nimport QueryMatcher from \"./QueryMatcher\";\nimport { PillCompletion } from \"./Components\";\nimport AutocompleteProvider from \"./AutocompleteProvider\";\nimport { _t } from \"../languageHandler\";\nimport { makeUserPermalink } from \"../utils/permalinks/Permalinks\";\nimport { ICompletion, ISelectionRange } from \"./Autocompleter\";\nimport MemberAvatar from \"../components/views/avatars/MemberAvatar\";\nimport { TimelineRenderingType } from \"../contexts/RoomContext\";\nimport UserIdentifierCustomisations from \"../customisations/UserIdentifier\";\n\nconst USER_REGEX = /\\B@\\S*/g;\n\n// used when you hit 'tab' - we allow some separator chars at the beginning\n// to allow you to tab-complete /mat into /(matthew)\nconst FORCED_USER_REGEX = /[^/,:; \\t\\n]\\S*/g;\n\nexport default class UserProvider extends AutocompleteProvider {\n    public matcher: QueryMatcher<RoomMember>;\n    public users?: RoomMember[];\n    public room: Room;\n\n    public constructor(room: Room, renderingType?: TimelineRenderingType) {\n        super({\n            commandRegex: USER_REGEX,\n            forcedCommandRegex: FORCED_USER_REGEX,\n            renderingType,\n        });\n        this.room = room;\n        this.matcher = new QueryMatcher<RoomMember>([], {\n            keys: [\"name\"],\n            funcs: [(obj) => obj.userId.slice(1)], // index by user id minus the leading '@'\n            shouldMatchWordsOnly: false,\n        });\n\n        MatrixClientPeg.safeGet().on(RoomEvent.Timeline, this.onRoomTimeline);\n        MatrixClientPeg.safeGet().on(RoomStateEvent.Update, this.onRoomStateUpdate);\n    }\n\n    public destroy(): void {\n        MatrixClientPeg.get()?.removeListener(RoomEvent.Timeline, this.onRoomTimeline);\n        MatrixClientPeg.get()?.removeListener(RoomStateEvent.Update, this.onRoomStateUpdate);\n    }\n\n    private onRoomTimeline = (\n        ev: MatrixEvent,\n        room: Room | undefined,\n        toStartOfTimeline: boolean | undefined,\n        removed: boolean,\n        data: IRoomTimelineData,\n    ): void => {\n        if (!room) return; // notification timeline, we'll get this event again with a room specific timeline\n        if (removed) return;\n        if (room.roomId !== this.room.roomId) return;\n\n        // ignore events from filtered timelines\n        if (data.timeline.getTimelineSet() !== room.getUnfilteredTimelineSet()) return;\n\n        // ignore anything but real-time updates at the end of the room:\n        // updates from pagination will happen when the paginate completes.\n        if (toStartOfTimeline || !data || !data.liveEvent) return;\n\n        // TODO: lazyload if we have no ev.sender room member?\n        this.onUserSpoke(ev.sender);\n    };\n\n    private onRoomStateUpdate = (state: RoomState): void => {\n        // ignore updates in other rooms\n        if (state.roomId !== this.room.roomId) return;\n\n        // blow away the users cache\n        this.users = undefined;\n    };\n\n    public async getCompletions(\n        rawQuery: string,\n        selection: ISelectionRange,\n        force = false,\n        limit = -1,\n    ): Promise<ICompletion[]> {\n        // lazy-load user list into matcher\n        if (!this.users) this.makeUsers();\n\n        const { command, range } = this.getCurrentCommand(rawQuery, selection, force);\n\n        const fullMatch = command?.[0];\n        // Don't search if the query is a single \"@\"\n        if (fullMatch && fullMatch !== \"@\") {\n            // Don't include the '@' in our search query - it's only used as a way to trigger completion\n            const query = fullMatch.startsWith(\"@\") ? fullMatch.substring(1) : fullMatch;\n            return this.matcher.match(query, limit).map((user) => {\n                const description = UserIdentifierCustomisations.getDisplayUserIdentifier?.(user.userId, {\n                    roomId: this.room.roomId,\n                    withDisplayName: true,\n                });\n                const displayName = user.name || user.userId || \"\";\n                return {\n                    // Length of completion should equal length of text in decorator. draft-js\n                    // relies on the length of the entity === length of the text in the decoration.\n                    completion: user.rawDisplayName,\n                    completionId: user.userId,\n                    type: \"user\",\n                    suffix: selection.beginning && range!.start === 0 ? \": \" : \" \",\n                    href: makeUserPermalink(user.userId),\n                    component: (\n                        <PillCompletion title={displayName} description={description}>\n                            <MemberAvatar member={user} size=\"24px\" />\n                        </PillCompletion>\n                    ),\n                    range: range!,\n                };\n            });\n        }\n        return [];\n    }\n\n    public getName(): string {\n        return _t(\"composer|autocomplete|user_description\");\n    }\n\n    private makeUsers(): void {\n        const events = this.room.getLiveTimeline().getEvents();\n        const lastSpoken: Record<string, number> = {};\n\n        for (const event of events) {\n            lastSpoken[event.getSender()!] = event.getTs();\n        }\n\n        const currentUserId = MatrixClientPeg.safeGet().credentials.userId;\n        this.users = this.room.getJoinedMembers().filter(({ userId }) => userId !== currentUserId);\n        this.users = this.users.concat(this.room.getMembersWithMembership(KnownMembership.Invite));\n\n        this.users = sortBy(this.users, (member) => 1e20 - lastSpoken[member.userId] || 1e20);\n\n        this.matcher.setObjects(this.users);\n    }\n\n    public onUserSpoke(user: RoomMember | null): void {\n        if (!this.users) return;\n        if (!user) return;\n        if (user.userId === MatrixClientPeg.safeGet().getSafeUserId()) return;\n\n        // Move the user that spoke to the front of the array\n        this.users.splice(\n            this.users.findIndex((user2) => user2.userId === user.userId),\n            1,\n        );\n        this.users = [user, ...this.users];\n\n        this.matcher.setObjects(this.users);\n    }\n\n    public renderCompletions(completions: React.ReactNode[]): React.ReactNode {\n        return (\n            <div\n                className=\"mx_Autocomplete_Completion_container_pill\"\n                role=\"presentation\"\n                aria-label={_t(\"composer|autocomplete|user_a11y\")}\n            >\n                {completions}\n            </div>\n        );\n    }\n\n    public shouldForceComplete(): boolean {\n        return true;\n    }\n}\n"],"mappings":"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","ignoreList":[]}