matrix-react-sdk
Version:
SDK for matrix.org using React
264 lines (255 loc) • 39.8 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.generateCompletionDomId = exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireWildcard(require("react"));
var _classnames = _interopRequireDefault(require("classnames"));
var _lodash = require("lodash");
var _Autocompleter = _interopRequireDefault(require("../../../autocomplete/Autocompleter"));
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
var _RoomContext = _interopRequireDefault(require("../../../contexts/RoomContext"));
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 2017-2024 New Vector 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 MAX_PROVIDER_MATCHES = 20;
const generateCompletionDomId = n => `mx_Autocomplete_Completion_${n}`;
exports.generateCompletionDomId = generateCompletionDomId;
class Autocomplete extends _react.default.PureComponent {
constructor(props, context) {
super(props, context);
(0, _defineProperty2.default)(this, "autocompleter", void 0);
(0, _defineProperty2.default)(this, "queryRequested", void 0);
(0, _defineProperty2.default)(this, "debounceCompletionsRequest", void 0);
(0, _defineProperty2.default)(this, "containerRef", /*#__PURE__*/(0, _react.createRef)());
(0, _defineProperty2.default)(this, "hide", () => {
this.setState({
hide: true,
selectionOffset: 1,
completions: [],
completionList: []
});
});
(0, _defineProperty2.default)(this, "onConfirmCompletion", () => {
this.onCompletionClicked(this.state.selectionOffset);
});
(0, _defineProperty2.default)(this, "onCompletionClicked", selectionOffset => {
const count = this.countCompletions();
if (count === 0 || selectionOffset < 1 || selectionOffset > count) {
return false;
}
this.props.onConfirm(this.state.completionList[selectionOffset - 1]);
this.hide();
return true;
});
this.state = {
// list of completionResults, each containing completions
completions: [],
// array of completions, so we can look up current selection by offset quickly
completionList: [],
// how far down the completion list we are (THIS IS 1-INDEXED!)
selectionOffset: 1,
// whether we should show completions if they're available
shouldShowCompletions: true,
hide: false,
forceComplete: false
};
}
componentDidMount() {
this.autocompleter = new _Autocompleter.default(this.props.room, this.context.timelineRenderingType);
this.applyNewProps();
}
applyNewProps(oldQuery, oldRoom) {
if (oldRoom && this.props.room.roomId !== oldRoom.roomId) {
this.autocompleter?.destroy();
this.autocompleter = new _Autocompleter.default(this.props.room);
}
// Query hasn't changed so don't try to complete it
if (oldQuery === this.props.query) {
return;
}
this.complete(this.props.query, this.props.selection);
}
componentWillUnmount() {
this.autocompleter?.destroy();
}
complete(query, selection) {
this.queryRequested = query;
if (this.debounceCompletionsRequest) {
clearTimeout(this.debounceCompletionsRequest);
}
if (query === "") {
this.setState({
// Clear displayed completions
completions: [],
completionList: [],
// Reset selected completion
selectionOffset: 1,
// Hide the autocomplete box
hide: true
});
return Promise.resolve();
}
let autocompleteDelay = _SettingsStore.default.getValue("autocompleteDelay");
// Don't debounce if we are already showing completions
if (this.state.completions.length > 0 || this.state.forceComplete) {
autocompleteDelay = 0;
}
return new Promise(resolve => {
this.debounceCompletionsRequest = window.setTimeout(() => {
resolve(this.processQuery(query, selection));
}, autocompleteDelay);
});
}
async processQuery(query, selection) {
return this.autocompleter?.getCompletions(query, selection, this.state.forceComplete, MAX_PROVIDER_MATCHES).then(completions => {
// Only ever process the completions for the most recent query being processed
if (query !== this.queryRequested) {
return;
}
this.processCompletions(completions);
});
}
processCompletions(completions) {
const completionList = (0, _lodash.flatMap)(completions, provider => provider.completions);
// Reset selection when completion list becomes empty.
let selectionOffset = 1;
if (completionList.length > 0) {
/* If the currently selected completion is still in the completion list,
try to find it and jump to it. If not, select composer.
*/
const currentSelection = this.state.selectionOffset <= 1 ? null : this.state.completionList[this.state.selectionOffset - 1].completion;
selectionOffset = completionList.findIndex(completion => completion.completion === currentSelection);
if (selectionOffset === -1) {
selectionOffset = 1;
} else {
selectionOffset++; // selectionOffset is 1-indexed!
}
}
let hide = true;
// If `completion.command.command` is truthy, then a provider has matched with the query
const anyMatches = completions.some(completion => !!completion.command.command);
if (anyMatches) {
hide = false;
if (this.props.onSelectionChange) {
this.props.onSelectionChange(selectionOffset - 1);
}
}
this.setState({
completions,
completionList,
selectionOffset,
hide,
// Force complete is turned off each time since we can't edit the query in that case
forceComplete: false
});
}
hasSelection() {
return this.countCompletions() > 0 && this.state.selectionOffset !== 0;
}
countCompletions() {
return this.state.completionList.length;
}
// called from MessageComposerInput
moveSelection(delta) {
const completionCount = this.countCompletions();
if (completionCount === 0) return; // there are no items to move the selection through
// Note: selectionOffset 0 represents the unsubstituted text, while 1 means first pill selected
const index = (this.state.selectionOffset + delta + completionCount - 1) % completionCount;
this.setSelection(1 + index);
}
onEscape(e) {
const completionCount = this.countCompletions();
if (completionCount === 0) {
// autocomplete is already empty, so don't preventDefault
return;
}
e.preventDefault();
// selectionOffset = 0, so we don't end up completing when autocomplete is hidden
this.hide();
}
forceComplete() {
return new Promise(resolve => {
this.setState({
forceComplete: true,
hide: false
}, () => {
this.complete(this.props.query, this.props.selection).then(() => {
resolve(this.countCompletions());
});
});
});
}
setSelection(selectionOffset) {
this.setState({
selectionOffset,
hide: false
});
if (this.props.onSelectionChange) {
this.props.onSelectionChange(selectionOffset - 1);
}
}
componentDidUpdate(prevProps) {
this.applyNewProps(prevProps.query, prevProps.room);
// this is the selected completion, so scroll it into view if needed
const selectedCompletion = this.refs[`completion${this.state.selectionOffset}`];
if (selectedCompletion) {
selectedCompletion.scrollIntoView({
behavior: "auto",
block: "nearest"
});
} else if (this.containerRef.current) {
this.containerRef.current.scrollTo({
top: 0
});
}
}
render() {
let position = 1;
const renderedCompletions = this.state.completions.map((completionResult, i) => {
const completions = completionResult.completions.map((completion, j) => {
const selected = position === this.state.selectionOffset;
const className = (0, _classnames.default)("mx_Autocomplete_Completion", {
selected
});
const componentPosition = position;
position++;
const onClick = () => {
this.onCompletionClicked(componentPosition);
};
return /*#__PURE__*/_react.default.cloneElement(completion.component, {
"key": j,
"ref": `completion${componentPosition}`,
"id": generateCompletionDomId(componentPosition - 1),
// 0 index the completion IDs
className,
onClick,
"aria-selected": selected
});
});
return completions.length > 0 ? /*#__PURE__*/_react.default.createElement("div", {
key: i,
className: "mx_Autocomplete_ProviderSection",
role: "presentation"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Autocomplete_provider_name"
}, completionResult.provider.getName()), completionResult.provider.renderCompletions(completions)) : null;
}).filter(completion => !!completion);
return !this.state.hide && renderedCompletions.length > 0 ? /*#__PURE__*/_react.default.createElement("div", {
id: "mx_Autocomplete",
className: "mx_Autocomplete",
ref: this.containerRef,
role: "listbox"
}, renderedCompletions) : null;
}
}
exports.default = Autocomplete;
(0, _defineProperty2.default)(Autocomplete, "contextType", _RoomContext.default);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_classnames","_interopRequireDefault","_lodash","_Autocompleter","_SettingsStore","_RoomContext","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","MAX_PROVIDER_MATCHES","generateCompletionDomId","exports","Autocomplete","React","PureComponent","constructor","props","context","_defineProperty2","createRef","setState","hide","selectionOffset","completions","completionList","onCompletionClicked","state","count","countCompletions","onConfirm","shouldShowCompletions","forceComplete","componentDidMount","autocompleter","Autocompleter","room","timelineRenderingType","applyNewProps","oldQuery","oldRoom","roomId","destroy","query","complete","selection","componentWillUnmount","queryRequested","debounceCompletionsRequest","clearTimeout","Promise","resolve","autocompleteDelay","SettingsStore","getValue","length","window","setTimeout","processQuery","getCompletions","then","processCompletions","flatMap","provider","currentSelection","completion","findIndex","anyMatches","some","command","onSelectionChange","hasSelection","moveSelection","delta","completionCount","index","setSelection","onEscape","preventDefault","componentDidUpdate","prevProps","selectedCompletion","refs","scrollIntoView","behavior","block","containerRef","current","scrollTo","top","render","position","renderedCompletions","map","completionResult","j","selected","className","classNames","componentPosition","onClick","cloneElement","component","createElement","key","role","getName","renderCompletions","filter","id","ref","RoomContext"],"sources":["../../../../src/components/views/rooms/Autocomplete.tsx"],"sourcesContent":["/*\nCopyright 2017-2024 New Vector 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, { createRef, KeyboardEvent } from \"react\";\nimport classNames from \"classnames\";\nimport { flatMap } from \"lodash\";\nimport { Room } from \"matrix-js-sdk/src/matrix\";\n\nimport Autocompleter, { ICompletion, ISelectionRange, IProviderCompletions } from \"../../../autocomplete/Autocompleter\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport RoomContext from \"../../../contexts/RoomContext\";\n\nconst MAX_PROVIDER_MATCHES = 20;\n\nexport const generateCompletionDomId = (n: number): string => `mx_Autocomplete_Completion_${n}`;\n\ninterface IProps {\n    // the query string for which to show autocomplete suggestions\n    query: string;\n    // method invoked with range and text content when completion is confirmed\n    onConfirm: (completion: ICompletion) => void;\n    // method invoked when selected (if any) completion changes\n    onSelectionChange?: (partIndex: number) => void;\n    selection: ISelectionRange;\n    // The room in which we're autocompleting\n    room: Room;\n}\n\ninterface IState {\n    completions: IProviderCompletions[];\n    completionList: ICompletion[];\n    selectionOffset: number;\n    shouldShowCompletions: boolean;\n    hide: boolean;\n    forceComplete: boolean;\n}\n\nexport default class Autocomplete extends React.PureComponent<IProps, IState> {\n    public autocompleter?: Autocompleter;\n    public queryRequested?: string;\n    public debounceCompletionsRequest?: number;\n    private containerRef = createRef<HTMLDivElement>();\n\n    public static contextType = RoomContext;\n    public declare context: React.ContextType<typeof RoomContext>;\n\n    public constructor(props: IProps, context: React.ContextType<typeof RoomContext>) {\n        super(props, context);\n\n        this.state = {\n            // list of completionResults, each containing completions\n            completions: [],\n\n            // array of completions, so we can look up current selection by offset quickly\n            completionList: [],\n\n            // how far down the completion list we are (THIS IS 1-INDEXED!)\n            selectionOffset: 1,\n\n            // whether we should show completions if they're available\n            shouldShowCompletions: true,\n\n            hide: false,\n\n            forceComplete: false,\n        };\n    }\n\n    public componentDidMount(): void {\n        this.autocompleter = new Autocompleter(this.props.room, this.context.timelineRenderingType);\n        this.applyNewProps();\n    }\n\n    private applyNewProps(oldQuery?: string, oldRoom?: Room): void {\n        if (oldRoom && this.props.room.roomId !== oldRoom.roomId) {\n            this.autocompleter?.destroy();\n            this.autocompleter = new Autocompleter(this.props.room);\n        }\n\n        // Query hasn't changed so don't try to complete it\n        if (oldQuery === this.props.query) {\n            return;\n        }\n\n        this.complete(this.props.query, this.props.selection);\n    }\n\n    public componentWillUnmount(): void {\n        this.autocompleter?.destroy();\n    }\n\n    private complete(query: string, selection: ISelectionRange): Promise<void> {\n        this.queryRequested = query;\n        if (this.debounceCompletionsRequest) {\n            clearTimeout(this.debounceCompletionsRequest);\n        }\n        if (query === \"\") {\n            this.setState({\n                // Clear displayed completions\n                completions: [],\n                completionList: [],\n                // Reset selected completion\n                selectionOffset: 1,\n                // Hide the autocomplete box\n                hide: true,\n            });\n            return Promise.resolve();\n        }\n        let autocompleteDelay = SettingsStore.getValue(\"autocompleteDelay\");\n\n        // Don't debounce if we are already showing completions\n        if (this.state.completions.length > 0 || this.state.forceComplete) {\n            autocompleteDelay = 0;\n        }\n\n        return new Promise((resolve) => {\n            this.debounceCompletionsRequest = window.setTimeout(() => {\n                resolve(this.processQuery(query, selection));\n            }, autocompleteDelay);\n        });\n    }\n\n    private async processQuery(query: string, selection: ISelectionRange): Promise<void> {\n        return this.autocompleter\n            ?.getCompletions(query, selection, this.state.forceComplete, MAX_PROVIDER_MATCHES)\n            .then((completions) => {\n                // Only ever process the completions for the most recent query being processed\n                if (query !== this.queryRequested) {\n                    return;\n                }\n                this.processCompletions(completions);\n            });\n    }\n\n    private processCompletions(completions: IProviderCompletions[]): void {\n        const completionList = flatMap(completions, (provider) => provider.completions);\n\n        // Reset selection when completion list becomes empty.\n        let selectionOffset = 1;\n        if (completionList.length > 0) {\n            /* If the currently selected completion is still in the completion list,\n             try to find it and jump to it. If not, select composer.\n             */\n            const currentSelection =\n                this.state.selectionOffset <= 1\n                    ? null\n                    : this.state.completionList[this.state.selectionOffset - 1].completion;\n            selectionOffset = completionList.findIndex((completion) => completion.completion === currentSelection);\n            if (selectionOffset === -1) {\n                selectionOffset = 1;\n            } else {\n                selectionOffset++; // selectionOffset is 1-indexed!\n            }\n        }\n\n        let hide = true;\n        // If `completion.command.command` is truthy, then a provider has matched with the query\n        const anyMatches = completions.some((completion) => !!completion.command.command);\n        if (anyMatches) {\n            hide = false;\n            if (this.props.onSelectionChange) {\n                this.props.onSelectionChange(selectionOffset - 1);\n            }\n        }\n\n        this.setState({\n            completions,\n            completionList,\n            selectionOffset,\n            hide,\n            // Force complete is turned off each time since we can't edit the query in that case\n            forceComplete: false,\n        });\n    }\n\n    public hasSelection(): boolean {\n        return this.countCompletions() > 0 && this.state.selectionOffset !== 0;\n    }\n\n    public countCompletions(): number {\n        return this.state.completionList.length;\n    }\n\n    // called from MessageComposerInput\n    public moveSelection(delta: number): void {\n        const completionCount = this.countCompletions();\n        if (completionCount === 0) return; // there are no items to move the selection through\n\n        // Note: selectionOffset 0 represents the unsubstituted text, while 1 means first pill selected\n        const index = (this.state.selectionOffset + delta + completionCount - 1) % completionCount;\n        this.setSelection(1 + index);\n    }\n\n    public onEscape(e: KeyboardEvent): boolean | undefined {\n        const completionCount = this.countCompletions();\n        if (completionCount === 0) {\n            // autocomplete is already empty, so don't preventDefault\n            return;\n        }\n\n        e.preventDefault();\n\n        // selectionOffset = 0, so we don't end up completing when autocomplete is hidden\n        this.hide();\n    }\n\n    private hide = (): void => {\n        this.setState({\n            hide: true,\n            selectionOffset: 1,\n            completions: [],\n            completionList: [],\n        });\n    };\n\n    public forceComplete(): Promise<number> {\n        return new Promise((resolve) => {\n            this.setState(\n                {\n                    forceComplete: true,\n                    hide: false,\n                },\n                () => {\n                    this.complete(this.props.query, this.props.selection).then(() => {\n                        resolve(this.countCompletions());\n                    });\n                },\n            );\n        });\n    }\n\n    public onConfirmCompletion = (): void => {\n        this.onCompletionClicked(this.state.selectionOffset);\n    };\n\n    private onCompletionClicked = (selectionOffset: number): boolean => {\n        const count = this.countCompletions();\n        if (count === 0 || selectionOffset < 1 || selectionOffset > count) {\n            return false;\n        }\n\n        this.props.onConfirm(this.state.completionList[selectionOffset - 1]);\n        this.hide();\n\n        return true;\n    };\n\n    private setSelection(selectionOffset: number): void {\n        this.setState({ selectionOffset, hide: false });\n        if (this.props.onSelectionChange) {\n            this.props.onSelectionChange(selectionOffset - 1);\n        }\n    }\n\n    public componentDidUpdate(prevProps: IProps): void {\n        this.applyNewProps(prevProps.query, prevProps.room);\n        // this is the selected completion, so scroll it into view if needed\n        const selectedCompletion = this.refs[`completion${this.state.selectionOffset}`] as HTMLElement;\n\n        if (selectedCompletion) {\n            selectedCompletion.scrollIntoView({\n                behavior: \"auto\",\n                block: \"nearest\",\n            });\n        } else if (this.containerRef.current) {\n            this.containerRef.current.scrollTo({ top: 0 });\n        }\n    }\n\n    public render(): React.ReactNode {\n        let position = 1;\n        const renderedCompletions = this.state.completions\n            .map((completionResult, i) => {\n                const completions = completionResult.completions.map((completion, j) => {\n                    const selected = position === this.state.selectionOffset;\n                    const className = classNames(\"mx_Autocomplete_Completion\", { selected });\n                    const componentPosition = position;\n                    position++;\n\n                    const onClick = (): void => {\n                        this.onCompletionClicked(componentPosition);\n                    };\n\n                    return React.cloneElement(completion.component, {\n                        \"key\": j,\n                        \"ref\": `completion${componentPosition}`,\n                        \"id\": generateCompletionDomId(componentPosition - 1), // 0 index the completion IDs\n                        className,\n                        onClick,\n                        \"aria-selected\": selected,\n                    });\n                });\n\n                return completions.length > 0 ? (\n                    <div key={i} className=\"mx_Autocomplete_ProviderSection\" role=\"presentation\">\n                        <div className=\"mx_Autocomplete_provider_name\">{completionResult.provider.getName()}</div>\n                        {completionResult.provider.renderCompletions(completions)}\n                    </div>\n                ) : null;\n            })\n            .filter((completion) => !!completion);\n\n        return !this.state.hide && renderedCompletions.length > 0 ? (\n            <div id=\"mx_Autocomplete\" className=\"mx_Autocomplete\" ref={this.containerRef} role=\"listbox\">\n                {renderedCompletions}\n            </div>\n        ) : null;\n    }\n}\n"],"mappings":"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","ignoreList":[]}