2017-12-26 16:54:28 -08:00
// Package imports.
import classNames from 'classnames' ;
import PropTypes from 'prop-types' ;
import React from 'react' ;
2019-04-19 20:57:43 +02:00
import { connect } from 'react-redux' ;
2019-04-19 21:05:18 +02:00
import spring from 'react-motion/lib/spring' ;
2017-12-26 16:54:28 -08:00
import {
2019-04-19 20:57:43 +02:00
injectIntl ,
2019-04-19 21:05:18 +02:00
FormattedMessage ,
2017-12-26 16:54:28 -08:00
defineMessages ,
} from 'react-intl' ;
import Overlay from 'react-overlays/lib/Overlay' ;
// Components.
import Icon from 'flavours/glitch/components/icon' ;
// Utils.
import { focusRoot } from 'flavours/glitch/util/dom_helpers' ;
2019-04-19 21:05:18 +02:00
import { searchEnabled } from 'flavours/glitch/util/initial_state' ;
import Motion from 'flavours/glitch/util/optional_motion' ;
2017-12-26 16:54:28 -08:00
const messages = defineMessages ( {
2019-04-19 21:05:18 +02:00
placeholder : { id : 'search.placeholder' , defaultMessage : 'Search' } ,
2017-12-26 16:54:28 -08:00
} ) ;
2019-04-19 21:05:18 +02:00
class SearchPopout extends React . PureComponent {
static propTypes = {
style : PropTypes . object ,
} ;
render ( ) {
const { style } = this . props ;
const extraInformation = searchEnabled ? < FormattedMessage id = 'search_popout.tips.full_text' defaultMessage = 'Simple text returns statuses you have written, favourited, boosted, or have been mentioned in, as well as matching usernames, display names, and hashtags.' / > : < FormattedMessage id = 'search_popout.tips.text' defaultMessage = 'Simple text returns matching display names, usernames and hashtags' / > ;
return (
2019-06-06 06:40:17 -05:00
< div style = { { ... style , position : 'absolute' , width : 285 , zIndex : 2 } } >
2019-04-19 21:05:18 +02:00
< Motion defaultStyle = { { opacity : 0 , scaleX : 0.85 , scaleY : 0.75 } } style = { { opacity : spring ( 1 , { damping : 35 , stiffness : 400 } ) , scaleX : spring ( 1 , { damping : 35 , stiffness : 400 } ) , scaleY : spring ( 1 , { damping : 35 , stiffness : 400 } ) } } >
{ ( { opacity , scaleX , scaleY } ) => (
2019-06-27 22:30:55 +02:00
< div className = 'search-popout' style = { { opacity : opacity , transform : ` scale( ${ scaleX } , ${ scaleY } ) ` } } >
2019-04-19 21:05:18 +02:00
< h4 > < FormattedMessage id = 'search_popout.search_format' defaultMessage = 'Advanced search format' / > < / h 4 >
< ul >
< li > < em > # example < /em> <FormattedMessage id='search_popout.tips.hashtag' defaultMessage='hashtag' / > < / l i >
< li > < em > @ username @ domain < /em> <FormattedMessage id='search_popout.tips.user' defaultMessage='user' / > < / l i >
< li > < em > URL < /em> <FormattedMessage id='search_popout.tips.user' defaultMessage='user' / > < / l i >
< li > < em > URL < /em> <FormattedMessage id='search_popout.tips.status' defaultMessage='status' / > < / l i >
< / u l >
{ extraInformation }
< / d i v >
) }
< / M o t i o n >
< / d i v >
) ;
}
}
2019-04-19 20:57:43 +02:00
// The component.
export default @ injectIntl
2019-04-20 17:50:12 +02:00
class Search extends React . PureComponent {
2017-12-26 16:54:28 -08:00
2019-05-25 21:27:00 +02:00
static contextTypes = {
router : PropTypes . object . isRequired ,
} ;
2019-04-19 20:57:43 +02:00
static propTypes = {
value : PropTypes . string . isRequired ,
submitted : PropTypes . bool ,
onChange : PropTypes . func . isRequired ,
onSubmit : PropTypes . func . isRequired ,
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-02 02:19:10 +09:00
singleColumn : PropTypes . bool ,
2019-04-19 20:57:43 +02:00
} ;
state = {
expanded : false ,
} ;
2017-12-26 16:54:28 -08:00
2019-10-02 02:19:10 +09:00
setRef = c => {
this . searchForm = c ;
}
2019-04-19 20:57:43 +02:00
handleChange = ( e ) => {
2017-12-26 16:54:28 -08:00
const { onChange } = this . props ;
if ( onChange ) {
2019-04-19 20:57:43 +02:00
onChange ( e . target . value ) ;
2017-12-26 16:54:28 -08:00
}
2019-04-19 20:57:43 +02:00
}
2017-12-26 16:54:28 -08:00
2019-04-19 20:57:43 +02:00
handleClear = ( e ) => {
2017-12-26 16:54:28 -08:00
const {
onClear ,
submitted ,
2018-01-06 15:34:01 -08:00
value ,
2017-12-26 16:54:28 -08:00
} = this . props ;
e . preventDefault ( ) ; // Prevents focus change ??
2018-01-06 15:34:01 -08:00
if ( onClear && ( submitted || value && value . length ) ) {
2017-12-26 16:54:28 -08:00
onClear ( ) ;
}
2019-04-19 20:57:43 +02:00
}
2019-04-19 21:05:18 +02:00
handleBlur = ( ) => {
2019-04-19 20:57:43 +02:00
this . setState ( { expanded : false } ) ;
}
2017-12-26 16:54:28 -08:00
2019-04-19 20:57:43 +02:00
handleFocus = ( ) => {
2017-12-26 16:54:28 -08:00
this . setState ( { expanded : true } ) ;
2019-10-02 02:19:10 +09:00
this . props . onShow ( ) ;
if ( this . searchForm && ! this . props . singleColumn ) {
const { left , right } = this . searchForm . getBoundingClientRect ( ) ;
if ( left < 0 || right > ( window . innerWidth || document . documentElement . clientWidth ) ) {
this . searchForm . scrollIntoView ( ) ;
}
2017-12-26 16:54:28 -08:00
}
2019-04-19 20:57:43 +02:00
}
2017-12-26 16:54:28 -08:00
2019-04-19 20:57:43 +02:00
handleKeyUp = ( e ) => {
2017-12-26 16:54:28 -08:00
const { onSubmit } = this . props ;
switch ( e . key ) {
case 'Enter' :
2019-05-25 21:27:00 +02:00
onSubmit ( ) ;
if ( this . props . openInRoute ) {
this . context . router . history . push ( '/search' ) ;
2017-12-26 16:54:28 -08:00
}
break ;
case 'Escape' :
focusRoot ( ) ;
}
}
render ( ) {
2019-04-19 20:57:43 +02:00
const { intl , value , submitted } = this . props ;
2017-12-26 16:54:28 -08:00
const { expanded } = this . state ;
2019-06-27 22:30:55 +02:00
const hasValue = value . length > 0 || submitted ;
2017-12-26 16:54:28 -08:00
return (
2019-06-27 22:30:55 +02:00
< div className = 'search' >
2017-12-26 16:54:28 -08:00
< label >
2019-04-19 20:57:43 +02:00
< span style = { { display : 'none' } } > { intl . formatMessage ( messages . placeholder ) } < / s p a n >
2017-12-26 16:54:28 -08:00
< input
2019-10-02 02:19:10 +09:00
ref = { this . setRef }
2019-06-27 22:30:55 +02:00
className = 'search__input'
2017-12-26 16:54:28 -08:00
type = 'text'
placeholder = { intl . formatMessage ( messages . placeholder ) }
2017-12-27 14:28:41 -08:00
value = { value || '' }
2019-04-19 20:57:43 +02:00
onChange = { this . handleChange }
onKeyUp = { this . handleKeyUp }
onFocus = { this . handleFocus }
onBlur = { this . handleBlur }
2017-12-26 16:54:28 -08:00
/ >
< / l a b e l >
2019-06-27 22:30:55 +02:00
2017-12-26 16:54:28 -08:00
< div
aria - label = { intl . formatMessage ( messages . placeholder ) }
2019-06-27 22:30:55 +02:00
className = 'search__icon'
2019-04-19 20:57:43 +02:00
onClick = { this . handleClear }
2017-12-26 16:54:28 -08:00
role = 'button'
tabIndex = '0'
>
2019-09-09 15:28:45 +02:00
< Icon id = 'search' className = { hasValue ? '' : 'active' } / >
< Icon id = 'times-circle' className = { hasValue ? 'active' : '' } / >
2017-12-26 16:54:28 -08:00
< / d i v >
2019-06-27 22:30:55 +02:00
< Overlay show = { expanded && ! hasValue } placement = 'bottom' target = { this } >
2019-04-19 21:05:18 +02:00
< SearchPopout / >
2019-04-19 20:57:43 +02:00
< / O v e r l a y >
2017-12-26 16:54:28 -08:00
< / d i v >
) ;
}
}