2019-07-25 23:27:29 +08:00
|
|
|
/*
|
2022-04-06 19:40:13 +08:00
|
|
|
Copyright 2019, 2022 The Matrix.org Foundation C.I.C.
|
2019-07-25 23:27:29 +08: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.
|
|
|
|
*/
|
|
|
|
|
2022-04-06 19:40:13 +08:00
|
|
|
import { Room, MatrixClient } from "matrix-js-sdk/src/matrix";
|
|
|
|
|
2022-12-30 04:52:51 +08:00
|
|
|
import AutocompleteWrapperModel, { UpdateCallback } from "../../src/editor/autocomplete";
|
|
|
|
import { Caret } from "../../src/editor/caret";
|
|
|
|
import { PillPart, Part, PartCreator } from "../../src/editor/parts";
|
2022-05-09 20:39:32 +08:00
|
|
|
import DocumentPosition from "../../src/editor/position";
|
2019-07-25 23:27:29 +08:00
|
|
|
|
|
|
|
class MockAutoComplete {
|
2022-04-06 19:40:13 +08:00
|
|
|
public _updateCallback;
|
|
|
|
public _partCreator;
|
|
|
|
public _completions;
|
2022-12-30 04:52:51 +08:00
|
|
|
public _part: Part | null;
|
2022-04-06 19:40:13 +08:00
|
|
|
|
2022-12-30 04:52:51 +08:00
|
|
|
constructor(updateCallback: UpdateCallback, partCreator: PartCreator, completions: PillPart[]) {
|
2019-07-25 23:27:29 +08:00
|
|
|
this._updateCallback = updateCallback;
|
|
|
|
this._partCreator = partCreator;
|
|
|
|
this._completions = completions;
|
|
|
|
this._part = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
close() {
|
2021-06-29 20:11:58 +08:00
|
|
|
this._updateCallback({ close: true });
|
2019-07-25 23:27:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
tryComplete(close = true) {
|
|
|
|
const matches = this._completions.filter((o) => {
|
2022-12-30 04:52:51 +08:00
|
|
|
return this._part && o.resourceId.startsWith(this._part.text);
|
2019-07-25 23:27:29 +08:00
|
|
|
});
|
2022-12-30 04:52:51 +08:00
|
|
|
if (matches.length === 1 && this._part && this._part.text.length > 1) {
|
2019-07-25 23:27:29 +08:00
|
|
|
const match = matches[0];
|
|
|
|
let pill;
|
|
|
|
if (match.resourceId[0] === "@") {
|
2022-12-30 04:52:51 +08:00
|
|
|
pill = this._partCreator.userPill(match.text, match.resourceId);
|
2019-07-25 23:27:29 +08:00
|
|
|
} else {
|
|
|
|
pill = this._partCreator.roomPill(match.resourceId);
|
|
|
|
}
|
2021-06-29 20:11:58 +08:00
|
|
|
this._updateCallback({ replaceParts: [pill], close });
|
2019-07-25 23:27:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// called by EditorModel when typing into pill-candidate part
|
2022-12-30 04:52:51 +08:00
|
|
|
onPartUpdate(part: Part, pos: DocumentPosition) {
|
2019-07-25 23:27:29 +08:00
|
|
|
this._part = part;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MockClient & MockRoom are only used for avatars in room and user pills,
|
|
|
|
// which is not tested
|
|
|
|
class MockRoom {
|
|
|
|
getMember() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-30 04:52:51 +08:00
|
|
|
export function createPartCreator(completions: PillPart[] = []) {
|
|
|
|
const autoCompleteCreator = (partCreator: PartCreator) => {
|
|
|
|
return (updateCallback: UpdateCallback) =>
|
2022-04-06 19:40:13 +08:00
|
|
|
new MockAutoComplete(updateCallback, partCreator, completions) as unknown as AutocompleteWrapperModel;
|
2019-07-25 23:27:29 +08:00
|
|
|
};
|
2022-04-06 19:40:13 +08:00
|
|
|
const room = new MockRoom() as unknown as Room;
|
|
|
|
const client = {
|
|
|
|
getRooms: jest.fn().mockReturnValue([]),
|
|
|
|
getRoom: jest.fn().mockReturnValue(null),
|
|
|
|
} as unknown as MatrixClient;
|
|
|
|
return new PartCreator(room, client, autoCompleteCreator);
|
2019-07-25 23:27:29 +08:00
|
|
|
}
|
2020-01-22 19:56:27 +08:00
|
|
|
|
|
|
|
export function createRenderer() {
|
2022-12-30 04:52:51 +08:00
|
|
|
const render = (c: Caret) => {
|
2020-01-22 19:56:27 +08:00
|
|
|
render.caret = c;
|
|
|
|
render.count += 1;
|
|
|
|
};
|
|
|
|
render.count = 0;
|
2022-12-30 04:52:51 +08:00
|
|
|
render.caret = null as unknown as Caret;
|
2020-01-22 19:56:27 +08:00
|
|
|
return render;
|
|
|
|
}
|
2022-12-30 04:52:51 +08:00
|
|
|
|
|
|
|
// in many tests we need to narrow the caret type
|
|
|
|
export function isDocumentPosition(caret: Caret): caret is DocumentPosition {
|
|
|
|
return caret instanceof DocumentPosition;
|
|
|
|
}
|