lbry-desktop/ui/component/wunderbar/view.jsx

248 lines
7.2 KiB
React
Raw Normal View History

2018-03-26 14:32:43 -07:00
// @flow
import { URL, URL_LOCAL, URL_DEV } from 'config';
2020-07-27 16:04:12 -04:00
import { SEARCH_TYPES } from 'constants/search';
2019-03-28 12:53:13 -04:00
import * as PAGES from 'constants/pages';
2018-11-25 20:21:25 -05:00
import * as ICONS from 'constants/icons';
import React from 'react';
2018-03-26 14:32:43 -07:00
import classnames from 'classnames';
2019-07-11 14:16:39 -04:00
import { withRouter } from 'react-router';
2018-03-26 14:32:43 -07:00
import Icon from 'component/common/icon';
2018-03-27 14:25:23 -07:00
import Autocomplete from './internal/autocomplete';
2019-07-11 14:16:39 -04:00
import Tag from 'component/tag';
import { isURIValid, normalizeURI } from 'lbry-redux';
import { formatLbryUrlForWeb } from '../../util/url';
const WEB_DEV_PREFIX = `${URL_DEV}/`;
const WEB_LOCAL_PREFIX = `${URL_LOCAL}/`;
const WEB_PROD_PREFIX = `${URL}/`;
const SEARCH_PREFIX = `$/${PAGES.SEARCH}q=`;
const INVALID_URL_ERROR = "Invalid LBRY URL entered. Only A-Z, a-z, 0-9, and '-' allowed.";
const L_KEY_CODE = 76;
const ESC_KEY_CODE = 27;
2018-10-04 01:59:47 -04:00
2018-03-26 14:32:43 -07:00
type Props = {
2019-03-28 12:53:13 -04:00
searchQuery: ?string,
2018-03-26 14:32:43 -07:00
updateSearchQuery: string => void,
2019-02-18 12:24:56 -05:00
onSearch: string => void,
2019-03-28 12:53:13 -04:00
onSubmit: string => void,
navigateToUri: string => void,
doSearch: string => void,
2018-03-26 14:32:43 -07:00
suggestions: Array<string>,
2018-05-16 14:32:25 -04:00
doFocus: () => void,
doBlur: () => void,
2018-10-04 01:59:47 -04:00
focused: boolean,
doShowSnackBar: string => void,
2019-07-11 14:16:39 -04:00
history: { push: string => void },
2018-03-26 14:32:43 -07:00
};
2019-03-28 12:53:13 -04:00
type State = {
query: ?string,
};
class WunderBar extends React.PureComponent<Props, State> {
2018-10-04 01:59:47 -04:00
constructor() {
super();
2017-05-03 23:44:08 -04:00
2019-03-28 12:53:13 -04:00
this.state = {
query: null,
};
2018-03-26 14:32:43 -07:00
(this: any).handleSubmit = this.handleSubmit.bind(this);
(this: any).handleChange = this.handleChange.bind(this);
2018-10-05 14:20:28 -04:00
(this: any).handleKeyDown = this.handleKeyDown.bind(this);
2018-10-04 01:59:47 -04:00
}
componentDidMount() {
2018-10-05 14:20:28 -04:00
window.addEventListener('keydown', this.handleKeyDown);
2018-10-04 01:59:47 -04:00
}
componentWillUnmount() {
window.removeEventListener('keydown', this.handleKeyDown);
}
2018-03-26 14:32:43 -07:00
getSuggestionIcon = (type: string) => {
switch (type) {
2019-07-11 14:16:39 -04:00
case SEARCH_TYPES.FILE:
return ICONS.FILE;
2019-07-11 14:16:39 -04:00
case SEARCH_TYPES.CHANNEL:
2019-01-22 15:36:28 -05:00
return ICONS.CHANNEL;
2019-07-11 14:16:39 -04:00
case SEARCH_TYPES.TAG:
return ICONS.TAG;
2018-03-26 14:32:43 -07:00
default:
2018-11-25 20:21:25 -05:00
return ICONS.SEARCH;
}
2018-03-26 14:32:43 -07:00
};
2017-05-03 23:44:08 -04:00
2018-10-04 01:59:47 -04:00
handleKeyDown(event: SyntheticKeyboardEvent<*>) {
const { ctrlKey, metaKey, keyCode } = event;
const { doFocus, doBlur, focused } = this.props;
2019-01-19 13:54:06 -05:00
if (this.input) {
if (focused && keyCode === ESC_KEY_CODE) {
this.input.blur();
doBlur();
return;
}
2018-10-04 01:59:47 -04:00
2019-03-28 12:53:13 -04:00
// @if TARGET='app'
2019-01-19 13:54:06 -05:00
const shouldFocus =
2019-05-07 17:38:29 -04:00
process.platform === 'darwin' ? keyCode === L_KEY_CODE && metaKey : keyCode === L_KEY_CODE && ctrlKey;
2018-10-04 01:59:47 -04:00
2019-01-19 13:54:06 -05:00
if (shouldFocus) {
this.input.focus();
doFocus();
}
2019-03-28 12:53:13 -04:00
// @endif
2018-10-04 01:59:47 -04:00
}
}
2018-03-26 14:32:43 -07:00
handleChange(e: SyntheticInputEvent<*>) {
const { value } = e.target;
2019-03-28 12:53:13 -04:00
const { updateSearchQuery } = this.props;
2018-03-26 14:32:43 -07:00
updateSearchQuery(value);
}
onSubmitWebUri(uri: string, prefix: string) {
// Allow copying a lbry.tv url and pasting it into the search bar
const { doSearch, navigateToUri, updateSearchQuery } = this.props;
let query = uri.slice(prefix.length);
query = query.replace(/:/g, '#');
if (query.includes(SEARCH_PREFIX)) {
query = query.slice(SEARCH_PREFIX.length);
doSearch(query);
} else {
// TODO - double check this code path
let path = `lbry://${query}`;
const uri = formatLbryUrlForWeb(path);
navigateToUri(uri);
updateSearchQuery('');
2017-05-03 23:44:08 -04:00
}
}
onClickSuggestion(query: string, suggestion: { value: string, type: string }): void {
const { navigateToUri, doSearch, doShowSnackBar } = this.props;
if (suggestion.type === SEARCH_TYPES.SEARCH) {
doSearch(query);
} else if (suggestion.type === SEARCH_TYPES.TAG) {
const encodedSuggestion = encodeURIComponent(suggestion.value);
const uri = `/$/${PAGES.DISCOVER}?t=${encodedSuggestion}`;
navigateToUri(uri);
} else if (isURIValid(query)) {
let uri = normalizeURI(query);
uri = formatLbryUrlForWeb(uri);
navigateToUri(uri);
} else {
doShowSnackBar(INVALID_URL_ERROR);
}
}
onSubmitRawString(st: string): void {
const { navigateToUri, doSearch, doShowSnackBar } = this.props;
2018-03-26 14:32:43 -07:00
// Currently no suggestion is highlighted. The user may have started
// typing, then lost focus and came back later on the same page
try {
if (isURIValid(st)) {
const uri = normalizeURI(st);
navigateToUri(uri);
} else {
doShowSnackBar(INVALID_URL_ERROR);
}
2018-03-26 14:32:43 -07:00
} catch (e) {
doSearch(st);
}
}
handleSubmit(value: string, suggestion?: { value: string, type: string }) {
let query = value.trim();
this.input && this.input.blur();
const includesLbryTvProd = query.includes(WEB_PROD_PREFIX);
const includesLbryTvLocal = query.includes(WEB_LOCAL_PREFIX);
const includesLbryTvDev = query.includes(WEB_DEV_PREFIX);
const wasCopiedFromWeb = includesLbryTvDev || includesLbryTvLocal || includesLbryTvProd;
if (wasCopiedFromWeb) {
let prefix = WEB_PROD_PREFIX;
if (includesLbryTvLocal) prefix = WEB_LOCAL_PREFIX;
if (includesLbryTvDev) prefix = WEB_DEV_PREFIX;
this.onSubmitWebUri(query, prefix);
} else if (suggestion) {
this.onClickSuggestion(query, suggestion);
} else {
this.onSubmitRawString(query);
2018-01-04 00:05:20 -05:00
}
2017-05-03 23:44:08 -04:00
}
2018-03-26 14:32:43 -07:00
input: ?HTMLInputElement;
2018-01-04 00:05:20 -05:00
render() {
2019-03-28 12:53:13 -04:00
const { suggestions, doFocus, doBlur, searchQuery } = this.props;
2018-03-26 14:32:43 -07:00
2017-05-03 23:44:08 -04:00
return (
<div
// @if TARGET='app'
onDoubleClick={e => {
e.stopPropagation();
}}
// @endif
className="wunderbar"
>
2018-11-25 20:21:25 -05:00
<Icon icon={ICONS.SEARCH} />
2018-03-26 14:32:43 -07:00
<Autocomplete
autoHighlight
2018-06-20 18:58:55 -04:00
wrapperStyle={{ flex: 1, position: 'relative' }}
2019-03-28 12:53:13 -04:00
value={searchQuery}
2018-03-26 14:32:43 -07:00
items={suggestions}
getItemValue={item => item.value}
onChange={this.handleChange}
onSelect={this.handleSubmit}
2018-05-16 14:32:25 -04:00
inputProps={{
onFocus: doFocus,
onBlur: doBlur,
}}
2018-03-26 14:32:43 -07:00
renderInput={props => (
<input
{...props}
2018-10-04 01:59:47 -04:00
ref={el => {
props.ref(el);
this.input = el;
}}
2018-03-26 14:32:43 -07:00
className="wunderbar__input"
2020-07-22 16:34:20 -04:00
placeholder={__('Search')}
2018-03-26 14:32:43 -07:00
/>
)}
renderItem={({ value, type, shorthand }, isHighlighted) => (
2018-03-26 14:32:43 -07:00
<div
2019-07-11 14:16:39 -04:00
// Use value + type for key because there might be suggestions with same value but different type
key={`${value}-${type}`}
2018-03-26 14:32:43 -07:00
className={classnames('wunderbar__suggestion', {
'wunderbar__active-suggestion': isHighlighted,
})}
>
<Icon icon={this.getSuggestionIcon(type)} />
2019-07-11 14:16:39 -04:00
<span className="wunderbar__suggestion-label">
2020-03-09 10:50:07 -04:00
{type === SEARCH_TYPES.TAG ? <Tag name={value} /> : shorthand || value}
2019-07-11 14:16:39 -04:00
</span>
2018-06-19 13:59:55 -04:00
{isHighlighted && (
2018-03-26 14:32:43 -07:00
<span className="wunderbar__suggestion-label--action">
2018-06-19 13:59:55 -04:00
{type === SEARCH_TYPES.SEARCH && __('Search')}
{type === SEARCH_TYPES.CHANNEL && __('View channel')}
{type === SEARCH_TYPES.FILE && __('View file')}
2019-07-11 14:16:39 -04:00
{type === SEARCH_TYPES.TAG && __('View Tag')}
2018-03-26 14:32:43 -07:00
</span>
)}
</div>
)}
2017-06-06 17:19:12 -04:00
/>
2017-05-03 23:44:08 -04:00
</div>
);
}
}
2019-07-11 14:16:39 -04:00
export default withRouter(WunderBar);