2017-06-01 14:18:06 +00:00
|
|
|
/*
|
2017-06-01 16:29:40 +00:00
|
|
|
Copyright 2016 Aviral Dasgupta
|
2018-06-19 10:52:48 +00:00
|
|
|
Copyright 2017, 2018 New Vector Ltd
|
2017-06-01 14:18:06 +00:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2021-06-18 15:13:55 +00:00
|
|
|
import { ReactElement } from "react";
|
2023-08-04 07:36:16 +00:00
|
|
|
import { Room } from "matrix-js-sdk/src/matrix";
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2016-06-01 11:24:21 +00:00
|
|
|
import CommandProvider from "./CommandProvider";
|
2016-06-12 11:32:46 +00:00
|
|
|
import RoomProvider from "./RoomProvider";
|
|
|
|
import UserProvider from "./UserProvider";
|
2016-06-16 23:28:09 +00:00
|
|
|
import EmojiProvider from "./EmojiProvider";
|
2017-11-02 18:54:25 +00:00
|
|
|
import NotifProvider from "./NotifProvider";
|
2021-06-18 15:13:55 +00:00
|
|
|
import { timeout } from "../utils/promise";
|
2021-06-29 12:11:58 +00:00
|
|
|
import AutocompleteProvider, { ICommand } from "./AutocompleteProvider";
|
2021-05-18 08:39:01 +00:00
|
|
|
import SpaceProvider from "./SpaceProvider";
|
2021-11-18 12:47:11 +00:00
|
|
|
import { TimelineRenderingType } from "../contexts/RoomContext";
|
2023-02-24 15:28:40 +00:00
|
|
|
import { filterBoolean } from "../utils/arrays";
|
2016-09-13 10:11:52 +00:00
|
|
|
|
2020-04-20 18:00:54 +00:00
|
|
|
export interface ISelectionRange {
|
|
|
|
beginning?: boolean; // whether the selection is in the first block of the editor or not
|
|
|
|
start: number; // byte offset relative to the start anchor of the current editor selection.
|
|
|
|
end: number; // byte offset relative to the end anchor of the current editor selection.
|
|
|
|
}
|
2016-09-13 10:11:52 +00:00
|
|
|
|
2020-04-20 18:00:54 +00:00
|
|
|
export interface ICompletion {
|
2022-11-30 11:32:56 +00:00
|
|
|
type?: "at-room" | "command" | "community" | "room" | "user";
|
2020-06-18 13:32:43 +00:00
|
|
|
completion: string;
|
2020-04-20 18:00:54 +00:00
|
|
|
completionId?: string;
|
2023-04-17 07:31:58 +00:00
|
|
|
component: ReactElement;
|
2020-06-18 13:32:43 +00:00
|
|
|
range: ISelectionRange;
|
|
|
|
command?: string;
|
2020-04-20 18:00:54 +00:00
|
|
|
suffix?: string;
|
2017-07-20 15:49:23 +00:00
|
|
|
// If provided, apply a LINK entity to the completion with the
|
|
|
|
// data = { url: href }.
|
2020-06-18 13:32:43 +00:00
|
|
|
href?: string;
|
2020-04-20 18:00:54 +00:00
|
|
|
}
|
2016-06-01 11:24:21 +00:00
|
|
|
|
2016-06-12 11:32:46 +00:00
|
|
|
const PROVIDERS = [UserProvider, RoomProvider, EmojiProvider, NotifProvider, CommandProvider, SpaceProvider];
|
2016-06-01 11:24:21 +00:00
|
|
|
|
2016-09-13 10:11:52 +00:00
|
|
|
// Providers will get rejected if they take longer than this.
|
|
|
|
const PROVIDER_COMPLETION_TIMEOUT = 3000;
|
|
|
|
|
2020-04-20 18:00:54 +00:00
|
|
|
export interface IProviderCompletions {
|
|
|
|
completions: ICompletion[];
|
|
|
|
provider: AutocompleteProvider;
|
2023-02-24 15:28:40 +00:00
|
|
|
command: Partial<ICommand>;
|
2020-04-20 18:00:54 +00:00
|
|
|
}
|
|
|
|
|
2017-11-02 17:51:08 +00:00
|
|
|
export default class Autocompleter {
|
2022-12-16 12:29:59 +00:00
|
|
|
public room: Room;
|
|
|
|
public providers: AutocompleteProvider[];
|
2020-04-20 18:00:54 +00:00
|
|
|
|
2022-12-16 12:29:59 +00:00
|
|
|
public constructor(room: Room, renderingType: TimelineRenderingType = TimelineRenderingType.Room) {
|
2017-11-02 17:51:08 +00:00
|
|
|
this.room = room;
|
2019-01-11 13:54:11 +00:00
|
|
|
this.providers = PROVIDERS.map((Prov) => {
|
2021-11-18 12:47:11 +00:00
|
|
|
return new Prov(room, renderingType);
|
2017-11-02 17:51:08 +00:00
|
|
|
});
|
|
|
|
}
|
2016-09-13 10:11:52 +00:00
|
|
|
|
2023-01-12 13:25:14 +00:00
|
|
|
public destroy(): void {
|
2017-11-02 17:51:08 +00:00
|
|
|
this.providers.forEach((p) => {
|
|
|
|
p.destroy();
|
|
|
|
});
|
|
|
|
}
|
2016-09-13 10:11:52 +00:00
|
|
|
|
2022-12-16 12:29:59 +00:00
|
|
|
public async getCompletions(
|
2021-05-12 11:18:56 +00:00
|
|
|
query: string,
|
|
|
|
selection: ISelectionRange,
|
|
|
|
force = false,
|
|
|
|
limit = -1,
|
|
|
|
): Promise<IProviderCompletions[]> {
|
2017-11-02 18:11:18 +00:00
|
|
|
/* Note: This intentionally waits for all providers to return,
|
2017-11-02 17:51:08 +00:00
|
|
|
otherwise, we run into a condition where new completions are displayed
|
|
|
|
while the user is interacting with the list, which makes it difficult
|
|
|
|
to predict whether an action will actually do what is intended
|
|
|
|
*/
|
2020-04-20 18:00:54 +00:00
|
|
|
// list of results from each provider, each being a list of completions or null if it times out
|
2023-02-03 15:27:47 +00:00
|
|
|
const completionsList: Array<ICompletion[] | null> = await Promise.all(
|
2023-01-12 13:25:14 +00:00
|
|
|
this.providers.map(async (provider): Promise<ICompletion[] | null> => {
|
2022-05-03 21:04:37 +00:00
|
|
|
return timeout(
|
2021-05-12 11:18:56 +00:00
|
|
|
provider.getCompletions(query, selection, force, limit),
|
|
|
|
null,
|
|
|
|
PROVIDER_COMPLETION_TIMEOUT,
|
|
|
|
);
|
2019-11-14 13:52:17 +00:00
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
// map then filter to maintain the index for the map-operation, for this.providers to line up
|
2023-02-24 15:28:40 +00:00
|
|
|
return filterBoolean(
|
|
|
|
completionsList.map((completions, i) => {
|
2019-11-14 13:52:17 +00:00
|
|
|
if (!completions || !completions.length) return;
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2017-11-02 17:51:08 +00:00
|
|
|
return {
|
2019-11-14 13:52:17 +00:00
|
|
|
completions,
|
2017-11-02 17:51:08 +00:00
|
|
|
provider: this.providers[i],
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2017-11-02 17:51:08 +00:00
|
|
|
/* the currently matched "command" the completer tried to complete
|
|
|
|
* we pass this through so that Autocomplete can figure out when to
|
|
|
|
* re-show itself once hidden.
|
|
|
|
*/
|
|
|
|
command: this.providers[i].getCurrentCommand(query, selection, force),
|
|
|
|
};
|
2023-02-24 15:28:40 +00:00
|
|
|
}),
|
|
|
|
);
|
2017-11-02 17:51:08 +00:00
|
|
|
}
|
2016-06-01 11:24:21 +00:00
|
|
|
}
|