2018-04-11 22:58:04 +00:00
|
|
|
/*
|
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
2019-02-24 04:42:04 +00:00
|
|
|
Copyright 2019 Michael Telatynski <7t3chguy@gmail.com>
|
2018-04-11 22:58:04 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import React from 'react';
|
2019-08-30 09:34:59 +00:00
|
|
|
import createReactClass from 'create-react-class';
|
2019-02-24 04:42:04 +00:00
|
|
|
import PropTypes from 'prop-types';
|
2018-04-12 23:43:44 +00:00
|
|
|
import { KeyCode } from '../../Keyboard';
|
|
|
|
import dis from '../../dispatcher';
|
2019-02-13 19:32:50 +00:00
|
|
|
import { throttle } from 'lodash';
|
2018-04-12 23:43:44 +00:00
|
|
|
import AccessibleButton from '../../components/views/elements/AccessibleButton';
|
2019-09-10 08:58:11 +00:00
|
|
|
import classNames from 'classnames';
|
2018-04-11 22:58:04 +00:00
|
|
|
|
2019-08-30 09:34:59 +00:00
|
|
|
module.exports = createReactClass({
|
2018-04-11 22:58:04 +00:00
|
|
|
displayName: 'SearchBox',
|
|
|
|
|
|
|
|
propTypes: {
|
2019-02-24 04:42:04 +00:00
|
|
|
onSearch: PropTypes.func,
|
|
|
|
onCleared: PropTypes.func,
|
|
|
|
className: PropTypes.string,
|
|
|
|
placeholder: PropTypes.string.isRequired,
|
2019-03-06 14:53:52 +00:00
|
|
|
|
|
|
|
// 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: PropTypes.bool,
|
|
|
|
},
|
|
|
|
|
|
|
|
getDefaultProps: function() {
|
|
|
|
return {
|
|
|
|
enableRoomSearchFocus: false,
|
|
|
|
};
|
2018-04-11 22:58:04 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
|
|
|
searchTerm: "",
|
2019-09-10 08:58:11 +00:00
|
|
|
blurred: true,
|
2018-04-11 22:58:04 +00:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
componentDidMount: function() {
|
|
|
|
this.dispatcherRef = dis.register(this.onAction);
|
|
|
|
},
|
|
|
|
|
|
|
|
componentWillUnmount: function() {
|
|
|
|
dis.unregister(this.dispatcherRef);
|
|
|
|
},
|
|
|
|
|
|
|
|
onAction: function(payload) {
|
2019-03-06 14:53:52 +00:00
|
|
|
if (!this.props.enableRoomSearchFocus) return;
|
|
|
|
|
2018-04-11 22:58:04 +00:00
|
|
|
switch (payload.action) {
|
|
|
|
case 'view_room':
|
|
|
|
if (this.refs.search && payload.clear_search) {
|
|
|
|
this._clearSearch();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'focus_room_filter':
|
|
|
|
if (this.refs.search) {
|
|
|
|
this.refs.search.focus();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onChange: function() {
|
|
|
|
if (!this.refs.search) return;
|
|
|
|
this.setState({ searchTerm: this.refs.search.value });
|
|
|
|
this.onSearch();
|
|
|
|
},
|
|
|
|
|
2019-02-13 19:32:50 +00:00
|
|
|
onSearch: throttle(function() {
|
2019-02-07 16:31:57 +00:00
|
|
|
this.props.onSearch(this.refs.search.value);
|
2019-02-13 19:32:50 +00:00
|
|
|
}, 200, {trailing: true, leading: true}),
|
2018-04-11 22:58:04 +00:00
|
|
|
|
|
|
|
_onKeyDown: function(ev) {
|
|
|
|
switch (ev.keyCode) {
|
|
|
|
case KeyCode.ESCAPE:
|
2018-11-05 08:35:44 +00:00
|
|
|
this._clearSearch("keyboard");
|
2018-04-11 22:58:04 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2018-12-18 15:35:07 +00:00
|
|
|
_onFocus: function(ev) {
|
2019-09-10 08:58:11 +00:00
|
|
|
this.setState({blurred: false});
|
2018-12-18 15:35:07 +00:00
|
|
|
ev.target.select();
|
2019-09-10 08:57:25 +00:00
|
|
|
if (this.props.onFocus) {
|
|
|
|
this.props.onFocus(ev);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_onBlur: function(ev) {
|
2019-09-10 08:58:11 +00:00
|
|
|
this.setState({blurred: true});
|
2019-09-10 08:57:25 +00:00
|
|
|
if (this.props.onBlur) {
|
|
|
|
this.props.onBlur(ev);
|
|
|
|
}
|
2018-12-18 15:35:07 +00:00
|
|
|
},
|
|
|
|
|
2018-11-05 08:35:44 +00:00
|
|
|
_clearSearch: function(source) {
|
2018-04-11 22:58:04 +00:00
|
|
|
this.refs.search.value = "";
|
|
|
|
this.onChange();
|
2018-11-05 08:35:44 +00:00
|
|
|
if (this.props.onCleared) {
|
|
|
|
this.props.onCleared(source);
|
|
|
|
}
|
2018-04-11 22:58:04 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
render: function() {
|
2019-02-04 18:51:41 +00: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 15:17:15 +00:00
|
|
|
const clearButton = (!this.state.blurred || this.state.searchTerm) ?
|
2018-11-05 08:35:44 +00:00
|
|
|
(<AccessibleButton key="button"
|
2018-11-02 14:29:18 +00:00
|
|
|
className="mx_SearchBox_closeButton"
|
2019-02-24 04:42:04 +00:00
|
|
|
onClick={ () => {this._clearSearch("button"); } }>
|
|
|
|
</AccessibleButton>) : undefined;
|
2018-11-05 08:35:44 +00:00
|
|
|
|
2019-09-10 08:58:11 +00: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 04:44:38 +00:00
|
|
|
const className = this.props.className || "";
|
2018-04-11 22:58:04 +00:00
|
|
|
return (
|
2019-09-10 08:58:44 +00:00
|
|
|
<div className={classNames("mx_SearchBox", "mx_textinput", {"mx_SearchBox_blurred": this.state.blurred})}>
|
2018-11-05 08:35:44 +00:00
|
|
|
<input
|
|
|
|
key="searchfield"
|
|
|
|
type="text"
|
|
|
|
ref="search"
|
2019-02-24 04:44:38 +00:00
|
|
|
className={"mx_textinput_icon mx_textinput_search " + className}
|
2018-11-05 08:35:44 +00:00
|
|
|
value={ this.state.searchTerm }
|
2018-12-18 15:35:07 +00:00
|
|
|
onFocus={ this._onFocus }
|
2018-11-05 08:35:44 +00:00
|
|
|
onChange={ this.onChange }
|
|
|
|
onKeyDown={ this._onKeyDown }
|
2019-09-10 08:57:25 +00:00
|
|
|
onBlur={this._onBlur}
|
2019-09-10 08:58:11 +00:00
|
|
|
placeholder={ placeholder }
|
2018-11-05 08:35:44 +00:00
|
|
|
/>
|
|
|
|
{ clearButton }
|
2018-04-11 22:58:04 +00:00
|
|
|
</div>
|
|
|
|
);
|
2018-10-27 03:50:35 +00:00
|
|
|
},
|
2018-04-11 22:58:04 +00:00
|
|
|
});
|