2018-04-12 01:58:04 +03:00
|
|
|
/*
|
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
2019-02-24 07:42:04 +03:00
|
|
|
Copyright 2019 Michael Telatynski <7t3chguy@gmail.com>
|
2018-04-12 01:58:04 +03: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-29 15:11:58 +03:00
|
|
|
import React, { createRef } from 'react';
|
2019-12-16 20:14:03 +03:00
|
|
|
import { Key } from '../../Keyboard';
|
2020-05-14 05:41:41 +03:00
|
|
|
import dis from '../../dispatcher/dispatcher';
|
2021-06-29 15:11:58 +03:00
|
|
|
import { throttle } from 'lodash';
|
2018-04-13 02:43:44 +03:00
|
|
|
import AccessibleButton from '../../components/views/elements/AccessibleButton';
|
2019-09-10 11:58:11 +03:00
|
|
|
import classNames from 'classnames';
|
2021-06-29 15:11:58 +03:00
|
|
|
import { replaceableComponent } from "../../utils/replaceableComponent";
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
interface IProps {
|
|
|
|
onSearch?: (query: string) => void;
|
|
|
|
onCleared?: (source?: string) => void;
|
|
|
|
onKeyDown?: (ev: React.KeyboardEvent) => void;
|
|
|
|
onFocus?: (ev: React.FocusEvent) => void;
|
|
|
|
onBlur?: (ev: React.FocusEvent) => void;
|
|
|
|
className?: string;
|
|
|
|
placeholder: string;
|
|
|
|
blurredPlaceholder?: string;
|
|
|
|
autoFocus?: boolean;
|
|
|
|
initialValue?: string;
|
|
|
|
collapsed?: boolean;
|
|
|
|
|
|
|
|
// If true, the search box will focus and clear itself
|
|
|
|
// on room search focus action (it would be nicer to take
|
|
|
|
// this functionality out, but not obvious how that would work)
|
|
|
|
enableRoomSearchFocus?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface IState {
|
|
|
|
searchTerm: string;
|
|
|
|
blurred: boolean;
|
|
|
|
}
|
|
|
|
|
2021-03-09 05:35:10 +03:00
|
|
|
@replaceableComponent("structures.SearchBox")
|
2021-09-12 09:15:46 +03:00
|
|
|
export default class SearchBox extends React.Component<IProps, IState> {
|
|
|
|
private dispatcherRef: string;
|
|
|
|
private search = createRef<HTMLInputElement>();
|
2019-03-06 17:53:52 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
static defaultProps: Partial<IProps> = {
|
2020-08-29 14:14:16 +03:00
|
|
|
enableRoomSearchFocus: false,
|
|
|
|
};
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
constructor(props: IProps) {
|
2020-08-29 14:14:16 +03:00
|
|
|
super(props);
|
|
|
|
|
|
|
|
this.state = {
|
2021-09-12 09:15:46 +03:00
|
|
|
searchTerm: props.initialValue || "",
|
2019-09-10 11:58:11 +03:00
|
|
|
blurred: true,
|
2018-04-12 01:58:04 +03:00
|
|
|
};
|
2020-08-29 14:14:16 +03:00
|
|
|
}
|
2019-12-08 15:16:17 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
public componentDidMount(): void {
|
2018-04-12 01:58:04 +03:00
|
|
|
this.dispatcherRef = dis.register(this.onAction);
|
2020-08-29 14:14:16 +03:00
|
|
|
}
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
public componentWillUnmount(): void {
|
2018-04-12 01:58:04 +03:00
|
|
|
dis.unregister(this.dispatcherRef);
|
2020-08-29 14:14:16 +03:00
|
|
|
}
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
private onAction = (payload): void => {
|
2019-03-06 17:53:52 +03:00
|
|
|
if (!this.props.enableRoomSearchFocus) return;
|
|
|
|
|
2018-04-12 01:58:04 +03:00
|
|
|
switch (payload.action) {
|
|
|
|
case 'view_room':
|
2021-09-12 09:15:46 +03:00
|
|
|
if (this.search.current && payload.clear_search) {
|
|
|
|
this.clearSearch();
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'focus_room_filter':
|
2021-09-12 09:15:46 +03:00
|
|
|
if (this.search.current) {
|
|
|
|
this.search.current.focus();
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2020-08-29 14:14:16 +03:00
|
|
|
};
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
private onChange = (): void => {
|
|
|
|
if (!this.search.current) return;
|
|
|
|
this.setState({ searchTerm: this.search.current.value });
|
2018-04-12 01:58:04 +03:00
|
|
|
this.onSearch();
|
2020-08-29 14:14:16 +03:00
|
|
|
};
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
private onSearch = throttle((): void => {
|
|
|
|
this.props.onSearch(this.search.current.value);
|
2021-06-29 15:11:58 +03:00
|
|
|
}, 200, { trailing: true, leading: true });
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
private onKeyDown = (ev: React.KeyboardEvent): void => {
|
2019-12-16 20:14:03 +03:00
|
|
|
switch (ev.key) {
|
|
|
|
case Key.ESCAPE:
|
2021-09-12 09:15:46 +03:00
|
|
|
this.clearSearch("keyboard");
|
2018-04-12 01:58:04 +03:00
|
|
|
break;
|
|
|
|
}
|
2019-10-23 20:31:00 +03:00
|
|
|
if (this.props.onKeyDown) this.props.onKeyDown(ev);
|
2020-08-29 14:14:16 +03:00
|
|
|
};
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
private onFocus = (ev: React.FocusEvent): void => {
|
2021-06-29 15:11:58 +03:00
|
|
|
this.setState({ blurred: false });
|
2021-09-12 09:15:46 +03:00
|
|
|
(ev.target as HTMLInputElement).select();
|
2019-09-10 11:57:25 +03:00
|
|
|
if (this.props.onFocus) {
|
|
|
|
this.props.onFocus(ev);
|
|
|
|
}
|
2020-08-29 14:14:16 +03:00
|
|
|
};
|
2019-09-10 11:57:25 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
private onBlur = (ev: React.FocusEvent): void => {
|
2021-06-29 15:11:58 +03:00
|
|
|
this.setState({ blurred: true });
|
2019-09-10 11:57:25 +03:00
|
|
|
if (this.props.onBlur) {
|
|
|
|
this.props.onBlur(ev);
|
|
|
|
}
|
2020-08-29 14:14:16 +03:00
|
|
|
};
|
2018-12-18 18:35:07 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
private clearSearch(source?: string): void {
|
|
|
|
this.search.current.value = "";
|
2018-04-12 01:58:04 +03:00
|
|
|
this.onChange();
|
2018-11-05 11:35:44 +03:00
|
|
|
if (this.props.onCleared) {
|
|
|
|
this.props.onCleared(source);
|
|
|
|
}
|
2020-08-29 14:14:16 +03:00
|
|
|
}
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2021-09-12 09:15:46 +03:00
|
|
|
public render(): JSX.Element {
|
2019-02-04 21:51:41 +03:00
|
|
|
// check for collapsed here and
|
|
|
|
// not at parent so we keep
|
|
|
|
// searchTerm in our state
|
|
|
|
// when collapsing and expanding
|
|
|
|
if (this.props.collapsed) {
|
|
|
|
return null;
|
|
|
|
}
|
2019-09-12 18:17:15 +03:00
|
|
|
const clearButton = (!this.state.blurred || this.state.searchTerm) ?
|
2020-01-07 13:50:02 +03:00
|
|
|
(<AccessibleButton
|
|
|
|
key="button"
|
|
|
|
tabIndex={-1}
|
|
|
|
className="mx_SearchBox_closeButton"
|
2021-09-12 09:15:46 +03:00
|
|
|
onClick={() => {this.clearSearch("button"); }}
|
2021-07-23 12:23:45 +03:00
|
|
|
/>) : undefined;
|
2018-11-05 11:35:44 +03:00
|
|
|
|
2019-09-10 11:58:11 +03:00
|
|
|
// show a shorter placeholder when blurred, if requested
|
|
|
|
// this is used for the room filter field that has
|
|
|
|
// the explore button next to it when blurred
|
|
|
|
const placeholder = this.state.blurred ?
|
|
|
|
(this.props.blurredPlaceholder || this.props.placeholder) :
|
|
|
|
this.props.placeholder;
|
2019-02-24 07:44:38 +03:00
|
|
|
const className = this.props.className || "";
|
2018-04-12 01:58:04 +03:00
|
|
|
return (
|
2021-06-29 15:11:58 +03:00
|
|
|
<div className={classNames("mx_SearchBox", "mx_textinput", { "mx_SearchBox_blurred": this.state.blurred })}>
|
2018-11-05 11:35:44 +03:00
|
|
|
<input
|
|
|
|
key="searchfield"
|
|
|
|
type="text"
|
2021-09-12 09:15:46 +03:00
|
|
|
ref={this.search}
|
2019-02-24 07:44:38 +03:00
|
|
|
className={"mx_textinput_icon mx_textinput_search " + className}
|
2021-07-20 00:43:11 +03:00
|
|
|
value={this.state.searchTerm}
|
2021-09-12 09:15:46 +03:00
|
|
|
onFocus={this.onFocus}
|
2021-07-20 00:43:11 +03:00
|
|
|
onChange={this.onChange}
|
2021-09-12 09:15:46 +03:00
|
|
|
onKeyDown={this.onKeyDown}
|
|
|
|
onBlur={this.onBlur}
|
2021-07-20 00:43:11 +03:00
|
|
|
placeholder={placeholder}
|
2020-01-29 16:24:45 +03:00
|
|
|
autoComplete="off"
|
2021-03-24 17:00:52 +03:00
|
|
|
autoFocus={this.props.autoFocus}
|
2018-11-05 11:35:44 +03:00
|
|
|
/>
|
|
|
|
{ clearButton }
|
2018-04-12 01:58:04 +03:00
|
|
|
</div>
|
|
|
|
);
|
2020-08-29 14:14:16 +03:00
|
|
|
}
|
|
|
|
}
|