2023-05-07 21:43:25 +02:00
import PropTypes from 'prop-types' ;
2023-05-28 16:38:10 +02:00
import { PureComponent } from 'react' ;
2023-08-28 13:18:39 +02:00
import { defineMessages , injectIntl , FormattedMessage , FormattedList } from 'react-intl' ;
2023-05-28 16:38:10 +02:00
2023-07-30 18:42:35 +02:00
import classNames from 'classnames' ;
import ImmutablePropTypes from 'react-immutable-proptypes' ;
2017-12-27 01:54:28 +01:00
2023-05-09 03:11:56 +02:00
import { Icon } from 'flavours/glitch/components/icon' ;
2023-09-06 17:23:58 +02:00
import { domain , searchEnabled } from 'flavours/glitch/initial_state' ;
2023-05-28 16:38:10 +02:00
import { focusRoot } from 'flavours/glitch/utils/dom_helpers' ;
2023-07-30 18:42:35 +02:00
import { HASHTAG _REGEX } from 'flavours/glitch/utils/hashtags' ;
2017-12-27 01:54:28 +01:00
const messages = defineMessages ( {
2019-04-19 21:05:18 +02:00
placeholder : { id : 'search.placeholder' , defaultMessage : 'Search' } ,
2022-10-29 13:32:49 +02:00
placeholderSignedIn : { id : 'search.search_or_paste' , defaultMessage : 'Search or paste URL' } ,
2017-12-27 01:54:28 +01:00
} ) ;
2019-04-19 20:57:43 +02:00
// The component.
2023-05-28 14:18:23 +02:00
class Search extends PureComponent {
2017-12-27 01:54:28 +01:00
2019-05-25 21:27:00 +02:00
static contextTypes = {
router : PropTypes . object . isRequired ,
2022-10-29 13:32:49 +02:00
identity : PropTypes . object . isRequired ,
2019-05-25 21:27:00 +02:00
} ;
2019-04-19 20:57:43 +02:00
static propTypes = {
value : PropTypes . string . isRequired ,
2023-07-30 18:42:35 +02:00
recent : ImmutablePropTypes . orderedSet ,
2019-04-19 20:57:43 +02:00
submitted : PropTypes . bool ,
onChange : PropTypes . func . isRequired ,
onSubmit : PropTypes . func . isRequired ,
2023-07-30 18:42:35 +02:00
onOpenURL : PropTypes . func . isRequired ,
onClickSearchResult : PropTypes . func . isRequired ,
onForgetSearchResult : PropTypes . func . isRequired ,
2019-04-19 20:57:43 +02:00
onClear : PropTypes . func . isRequired ,
onShow : PropTypes . func . isRequired ,
2019-05-25 21:27:00 +02:00
openInRoute : PropTypes . bool ,
2019-04-19 20:57:43 +02:00
intl : PropTypes . object . isRequired ,
2019-10-01 19:19:10 +02:00
singleColumn : PropTypes . bool ,
2019-04-19 20:57:43 +02:00
} ;
state = {
expanded : false ,
2023-07-30 18:42:35 +02:00
selectedOption : - 1 ,
options : [ ] ,
2019-04-19 20:57:43 +02:00
} ;
2017-12-27 01:54:28 +01:00
2023-08-28 13:18:39 +02:00
defaultOptions = [
{ label : < > < mark > has : < / mark > < FormattedList type = 'disjunction' value = { [ 'media' , 'poll' , 'embed' ] } / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'has:' ) } } ,
{ label : < > < mark > is : < / mark > < FormattedList type = 'disjunction' value = { [ 'reply' , 'sensitive' ] } / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'is:' ) } } ,
{ label : < > < mark > language : < / mark > < FormattedMessage id = 'search_popout.language_code' defaultMessage = 'ISO language code' / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'language:' ) } } ,
{ label : < > < mark > from : < / mark > < FormattedMessage id = 'search_popout.user' defaultMessage = 'user' / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'from:' ) } } ,
{ label : < > < mark > before : < / mark > < FormattedMessage id = 'search_popout.specific_date' defaultMessage = 'specific date' / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'before:' ) } } ,
{ label : < > < mark > during : < / mark > < FormattedMessage id = 'search_popout.specific_date' defaultMessage = 'specific date' / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'during:' ) } } ,
{ label : < > < mark > after : < / mark > < FormattedMessage id = 'search_popout.specific_date' defaultMessage = 'specific date' / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'after:' ) } } ,
2023-09-06 06:46:26 +02:00
{ label : < > < mark > in : < / mark > < FormattedList type = 'disjunction' value = { [ 'all' , 'library' ] } / > < / > , action : e => { e . preventDefault ( ) ; this . _insertText ( 'in:' ) } }
2023-08-28 13:18:39 +02:00
] ;
2019-10-01 19:19:10 +02:00
setRef = c => {
this . searchForm = c ;
2023-02-03 20:52:07 +01:00
} ;
2019-10-01 19:19:10 +02:00
2023-07-30 18:42:35 +02:00
handleChange = ( { target } ) => {
2017-12-27 01:54:28 +01:00
const { onChange } = this . props ;
2023-07-30 18:42:35 +02:00
onChange ( target . value ) ;
this . _calculateOptions ( target . value ) ;
2023-02-03 20:52:07 +01:00
} ;
2017-12-27 01:54:28 +01:00
2023-07-30 18:42:35 +02:00
handleClear = e => {
2017-12-27 01:54:28 +01:00
const {
onClear ,
submitted ,
2018-01-07 00:34:01 +01:00
value ,
2017-12-27 01:54:28 +01:00
} = this . props ;
2023-07-30 18:42:35 +02:00
2017-12-27 01:54:28 +01:00
e . preventDefault ( ) ; // Prevents focus change ??
2023-07-30 18:42:35 +02:00
if ( value . length > 0 || submitted ) {
2017-12-27 01:54:28 +01:00
onClear ( ) ;
2023-07-30 18:42:35 +02:00
this . setState ( { options : [ ] , selectedOption : - 1 } )
2017-12-27 01:54:28 +01:00
}
2023-02-03 20:52:07 +01:00
} ;
2019-04-19 20:57:43 +02:00
2019-04-19 21:05:18 +02:00
handleBlur = ( ) => {
2023-07-30 18:42:35 +02:00
this . setState ( { expanded : false , selectedOption : - 1 } ) ;
2023-02-03 20:52:07 +01:00
} ;
2017-12-27 01:54:28 +01:00
2019-04-19 20:57:43 +02:00
handleFocus = ( ) => {
2023-07-30 18:42:35 +02:00
const { onShow , singleColumn } = this . props ;
this . setState ( { expanded : true , selectedOption : - 1 } ) ;
onShow ( ) ;
2019-10-01 19:19:10 +02:00
2023-07-30 18:42:35 +02:00
if ( this . searchForm && ! singleColumn ) {
2019-10-01 19:19:10 +02:00
const { left , right } = this . searchForm . getBoundingClientRect ( ) ;
2023-07-30 18:42:35 +02:00
2019-10-01 19:19:10 +02:00
if ( left < 0 || right > ( window . innerWidth || document . documentElement . clientWidth ) ) {
this . searchForm . scrollIntoView ( ) ;
}
2017-12-27 01:54:28 +01:00
}
2023-02-03 20:52:07 +01:00
} ;
2017-12-27 01:54:28 +01:00
2023-07-30 18:42:35 +02:00
handleKeyDown = ( e ) => {
const { selectedOption } = this . state ;
2023-09-01 15:13:27 +02:00
const options = searchEnabled ? this . _getOptions ( ) . concat ( this . defaultOptions ) : this . _getOptions ( ) ;
2023-07-30 18:42:35 +02:00
switch ( e . key ) {
case 'Escape' :
e . preventDefault ( ) ;
focusRoot ( ) ;
break ;
case 'ArrowDown' :
e . preventDefault ( ) ;
if ( options . length > 0 ) {
this . setState ( { selectedOption : Math . min ( selectedOption + 1 , options . length - 1 ) } ) ;
}
break ;
case 'ArrowUp' :
e . preventDefault ( ) ;
if ( options . length > 0 ) {
this . setState ( { selectedOption : Math . max ( selectedOption - 1 , - 1 ) } ) ;
}
break ;
2017-12-27 01:54:28 +01:00
case 'Enter' :
2023-07-30 18:42:35 +02:00
e . preventDefault ( ) ;
2019-05-25 21:27:00 +02:00
2023-07-30 18:42:35 +02:00
if ( selectedOption === - 1 ) {
this . _submit ( ) ;
} else if ( options . length > 0 ) {
2023-08-28 13:18:39 +02:00
options [ selectedOption ] . action ( e ) ;
2023-07-30 18:42:35 +02:00
}
break ;
case 'Delete' :
if ( selectedOption > - 1 && options . length > 0 ) {
const search = options [ selectedOption ] ;
if ( typeof search . forget === 'function' ) {
e . preventDefault ( ) ;
search . forget ( e ) ;
}
2017-12-27 01:54:28 +01:00
}
break ;
}
2023-02-03 20:52:07 +01:00
} ;
2017-12-27 01:54:28 +01:00
2023-01-11 21:58:46 +01:00
findTarget = ( ) => {
return this . searchForm ;
2023-02-03 20:52:07 +01:00
} ;
2023-01-11 21:58:46 +01:00
2023-07-30 18:42:35 +02:00
handleHashtagClick = ( ) => {
const { router } = this . context ;
const { value , onClickSearchResult } = this . props ;
const query = value . trim ( ) . replace ( /^#/ , '' ) ;
router . history . push ( ` /tags/ ${ query } ` ) ;
onClickSearchResult ( query , 'hashtag' ) ;
2023-08-28 13:18:39 +02:00
this . _unfocus ( ) ;
2023-07-30 18:42:35 +02:00
} ;
handleAccountClick = ( ) => {
const { router } = this . context ;
const { value , onClickSearchResult } = this . props ;
const query = value . trim ( ) . replace ( /^@/ , '' ) ;
router . history . push ( ` /@ ${ query } ` ) ;
onClickSearchResult ( query , 'account' ) ;
2023-08-28 13:18:39 +02:00
this . _unfocus ( ) ;
2023-07-30 18:42:35 +02:00
} ;
handleURLClick = ( ) => {
const { router } = this . context ;
const { onOpenURL } = this . props ;
onOpenURL ( router . history ) ;
2023-08-28 13:18:39 +02:00
this . _unfocus ( ) ;
2023-07-30 18:42:35 +02:00
} ;
handleStatusSearch = ( ) => {
this . _submit ( 'statuses' ) ;
} ;
handleAccountSearch = ( ) => {
this . _submit ( 'accounts' ) ;
} ;
handleRecentSearchClick = search => {
const { router } = this . context ;
if ( search . get ( 'type' ) === 'account' ) {
router . history . push ( ` /@ ${ search . get ( 'q' ) } ` ) ;
} else if ( search . get ( 'type' ) === 'hashtag' ) {
router . history . push ( ` /tags/ ${ search . get ( 'q' ) } ` ) ;
}
2023-08-28 13:18:39 +02:00
this . _unfocus ( ) ;
2023-07-30 18:42:35 +02:00
} ;
handleForgetRecentSearchClick = search => {
const { onForgetSearchResult } = this . props ;
onForgetSearchResult ( search . get ( 'q' ) ) ;
} ;
_unfocus ( ) {
document . querySelector ( '.ui' ) . parentElement . focus ( ) ;
}
2023-08-28 13:18:39 +02:00
_insertText ( text ) {
const { value , onChange } = this . props ;
if ( value === '' ) {
onChange ( text ) ;
} else if ( value [ value . length - 1 ] === ' ' ) {
onChange ( ` ${ value } ${ text } ` ) ;
} else {
onChange ( ` ${ value } ${ text } ` ) ;
}
}
2023-07-30 18:42:35 +02:00
_submit ( type ) {
const { onSubmit , openInRoute } = this . props ;
const { router } = this . context ;
onSubmit ( type ) ;
if ( openInRoute ) {
router . history . push ( '/search' ) ;
}
2023-08-28 13:18:39 +02:00
this . _unfocus ( ) ;
2023-07-30 18:42:35 +02:00
}
_getOptions ( ) {
const { options } = this . state ;
if ( options . length > 0 ) {
return options ;
}
const { recent } = this . props ;
return recent . toArray ( ) . map ( search => ( {
label : search . get ( 'type' ) === 'account' ? ` @ ${ search . get ( 'q' ) } ` : ` # ${ search . get ( 'q' ) } ` ,
action : ( ) => this . handleRecentSearchClick ( search ) ,
forget : e => {
e . stopPropagation ( ) ;
this . handleForgetRecentSearchClick ( search ) ;
} ,
} ) ) ;
}
_calculateOptions ( value ) {
const trimmedValue = value . trim ( ) ;
const options = [ ] ;
if ( trimmedValue . length > 0 ) {
const couldBeURL = trimmedValue . startsWith ( 'https://' ) && ! trimmedValue . includes ( ' ' ) ;
if ( couldBeURL ) {
options . push ( { key : 'open-url' , label : < FormattedMessage id = 'search.quick_action.open_url' defaultMessage = 'Open URL in Mastodon' / > , action : this . handleURLClick } ) ;
}
const couldBeHashtag = ( trimmedValue . startsWith ( '#' ) && trimmedValue . length > 1 ) || trimmedValue . match ( HASHTAG _REGEX ) ;
if ( couldBeHashtag ) {
options . push ( { key : 'go-to-hashtag' , label : < FormattedMessage id = 'search.quick_action.go_to_hashtag' defaultMessage = 'Go to hashtag {x}' values = { { x : < mark > # { trimmedValue . replace ( /^#/ , '' ) } < / mark > } } / > , action : this . handleHashtagClick } ) ;
}
const couldBeUsername = trimmedValue . match ( /^@?[a-z0-9_-]+(@[^\s]+)?$/i ) ;
if ( couldBeUsername ) {
options . push ( { key : 'go-to-account' , label : < FormattedMessage id = 'search.quick_action.go_to_account' defaultMessage = 'Go to profile {x}' values = { { x : < mark > @ { trimmedValue . replace ( /^@/ , '' ) } < / mark > } } / > , action : this . handleAccountClick } ) ;
}
const couldBeStatusSearch = searchEnabled ;
if ( couldBeStatusSearch ) {
options . push ( { key : 'status-search' , label : < FormattedMessage id = 'search.quick_action.status_search' defaultMessage = 'Posts matching {x}' values = { { x : < mark > { trimmedValue } < / mark > } } / > , action : this . handleStatusSearch } ) ;
}
const couldBeUserSearch = true ;
if ( couldBeUserSearch ) {
options . push ( { key : 'account-search' , label : < FormattedMessage id = 'search.quick_action.account_search' defaultMessage = 'Profiles matching {x}' values = { { x : < mark > { trimmedValue } < / mark > } } / > , action : this . handleAccountSearch } ) ;
}
}
this . setState ( { options } ) ;
}
2017-12-27 01:54:28 +01:00
render ( ) {
2023-07-30 18:42:35 +02:00
const { intl , value , submitted , recent } = this . props ;
const { expanded , options , selectedOption } = this . state ;
2022-10-29 13:32:49 +02:00
const { signedIn } = this . context . identity ;
2023-07-30 18:42:35 +02:00
2019-06-27 22:30:55 +02:00
const hasValue = value . length > 0 || submitted ;
2017-12-27 01:54:28 +01:00
return (
2023-07-30 18:42:35 +02:00
< div className = { classNames ( 'search' , { active : expanded } ) } >
2022-12-15 16:20:21 +01:00
< input
ref = { this . setRef }
className = 'search__input'
type = 'text'
placeholder = { intl . formatMessage ( signedIn ? messages . placeholderSignedIn : messages . placeholder ) }
aria - label = { intl . formatMessage ( signedIn ? messages . placeholderSignedIn : messages . placeholder ) }
value = { value || '' }
onChange = { this . handleChange }
2023-07-30 18:42:35 +02:00
onKeyDown = { this . handleKeyDown }
2022-12-15 16:20:21 +01:00
onFocus = { this . handleFocus }
onBlur = { this . handleBlur }
/ >
2023-04-04 16:33:44 +02:00
< div role = 'button' tabIndex = { 0 } className = 'search__icon' onClick = { this . handleClear } >
2019-09-09 15:28:45 +02:00
< Icon id = 'search' className = { hasValue ? '' : 'active' } / >
< Icon id = 'times-circle' className = { hasValue ? 'active' : '' } / >
2017-12-27 01:54:28 +01:00
< / div >
2023-07-30 18:42:35 +02:00
< div className = 'search__popout' >
{ options . length === 0 && (
< >
< h4 > < FormattedMessage id = 'search_popout.recent' defaultMessage = 'Recent searches' / > < / h4 >
< div className = 'search__popout__menu' >
{ recent . size > 0 ? this . _getOptions ( ) . map ( ( { label , action , forget } , i ) => (
< button key = { label } onMouseDown = { action } className = { classNames ( 'search__popout__menu__item search__popout__menu__item--flex' , { selected : selectedOption === i } ) } >
< span > { label } < / span >
< button className = 'icon-button' onMouseDown = { forget } > < Icon id = 'times' / > < / button >
< / button >
) ) : (
< div className = 'search__popout__menu__message' >
< FormattedMessage id = 'search.no_recent_searches' defaultMessage = 'No recent searches' / >
< / div >
) }
< / div >
< / >
) }
{ options . length > 0 && (
< >
< h4 > < FormattedMessage id = 'search_popout.quick_actions' defaultMessage = 'Quick actions' / > < / h4 >
< div className = 'search__popout__menu' >
{ options . map ( ( { key , label , action } , i ) => (
< button key = { key } onMouseDown = { action } className = { classNames ( 'search__popout__menu__item' , { selected : selectedOption === i } ) } >
{ label }
< / button >
) ) }
2023-01-11 21:58:46 +01:00
< / div >
2023-07-30 18:42:35 +02:00
< / >
2023-01-11 21:58:46 +01:00
) }
2023-08-28 13:18:39 +02:00
2023-09-06 17:23:58 +02:00
< h4 > < FormattedMessage id = 'search_popout.options' defaultMessage = 'Search options' / > < / h4 >
{ searchEnabled ? (
< div className = 'search__popout__menu' >
{ this . defaultOptions . map ( ( { key , label , action } , i ) => (
< button key = { key } onMouseDown = { action } className = { classNames ( 'search__popout__menu__item' , { selected : selectedOption === ( options . length + i ) } ) } >
{ label }
< / button >
) ) }
< / div >
) : (
< div className = 'search__popout__menu__message' >
< FormattedMessage id = 'search_popout.full_text_search_disabled_message' defaultMessage = 'Not available on {domain}.' values = { { domain } } / >
< / div >
2023-09-01 15:13:27 +02:00
) }
2023-07-30 18:42:35 +02:00
< / div >
2017-12-27 01:54:28 +01:00
< / div >
) ;
}
}
2023-03-24 23:15:25 +01:00
export default injectIntl ( Search ) ;