Fix dropdown menu positions when scrolling (#22916)
* Update react-overlays to latest version * Fix breaking changes in dropdown menus * Use react-overlays built-in arrow positioning feature * Re-implemented `.dropdown-menu__arrow` to have a defined width and height to improve positioning * Moved wrapping div (`.dropdown-menu` from `DropdownMenu` to `Dropdown`) * Wrap button in a span to solve issue with ref * Temporarily remove animations * Fix breaking changes in emoji picker * Wrap EmojiPickerMenu in a div where react-overlays’ ref is added * Fix breaking changes in language dropdown * Fix breaking changes in privacy dropdown * Fix breaking changes in search form * Add animations back using `@keyframes` * Fix arrow color in light theme * Fix linting issue * Remove unused `mounted` state * Remove `placement` state from components and redux And remove the placement state from props of the menu components. * Remove abolution position to fix flip issue * Remove z-index to fix modals and overlay positions * Fix lint issues * Set placement in privacy and language components Copy the placement state into the `PrivacyDropdown` and `LanguageDropdown` components, to apply correct styling to the buttons depending on which placement the Overlay has. * Move `placement` state to correct component
This commit is contained in:
		
					parent
					
						
							
								ae62e5fa53
							
						
					
				
			
			
				commit
				
					
						fd33bcb3b2
					
				
			
		
					 13 changed files with 301 additions and 271 deletions
				
			
		|  | @ -2,7 +2,7 @@ import React from 'react'; | |||
| import PropTypes from 'prop-types'; | ||||
| import { defineMessages, injectIntl, FormattedMessage } from 'react-intl'; | ||||
| import { EmojiPicker as EmojiPickerAsync } from '../../ui/util/async-components'; | ||||
| import Overlay from 'react-overlays/lib/Overlay'; | ||||
| import Overlay from 'react-overlays/Overlay'; | ||||
| import classNames from 'classnames'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import { supportsPassiveEvents } from 'detect-passive-events'; | ||||
|  | @ -154,9 +154,6 @@ class EmojiPickerMenu extends React.PureComponent { | |||
|     onClose: PropTypes.func.isRequired, | ||||
|     onPick: PropTypes.func.isRequired, | ||||
|     style: PropTypes.object, | ||||
|     placement: PropTypes.string, | ||||
|     arrowOffsetLeft: PropTypes.string, | ||||
|     arrowOffsetTop: PropTypes.string, | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     skinTone: PropTypes.number.isRequired, | ||||
|     onSkinTone: PropTypes.func.isRequired, | ||||
|  | @ -324,14 +321,13 @@ class EmojiPickerDropdown extends React.PureComponent { | |||
|   state = { | ||||
|     active: false, | ||||
|     loading: false, | ||||
|     placement: null, | ||||
|   }; | ||||
| 
 | ||||
|   setRef = (c) => { | ||||
|     this.dropdown = c; | ||||
|   } | ||||
| 
 | ||||
|   onShowDropdown = ({ target }) => { | ||||
|   onShowDropdown = () => { | ||||
|     this.setState({ active: true }); | ||||
| 
 | ||||
|     if (!EmojiPicker) { | ||||
|  | @ -346,9 +342,6 @@ class EmojiPickerDropdown extends React.PureComponent { | |||
|         this.setState({ loading: false, active: false }); | ||||
|       }); | ||||
|     } | ||||
| 
 | ||||
|     const { top } = target.getBoundingClientRect(); | ||||
|     this.setState({ placement: top * 2 < innerHeight ? 'bottom' : 'top' }); | ||||
|   } | ||||
| 
 | ||||
|   onHideDropdown = () => { | ||||
|  | @ -382,7 +375,7 @@ class EmojiPickerDropdown extends React.PureComponent { | |||
|   render () { | ||||
|     const { intl, onPickEmoji, onSkinTone, skinTone, frequentlyUsedEmojis, button } = this.props; | ||||
|     const title = intl.formatMessage(messages.emoji); | ||||
|     const { active, loading, placement } = this.state; | ||||
|     const { active, loading } = this.state; | ||||
| 
 | ||||
|     return ( | ||||
|       <div className='emoji-picker-dropdown' onKeyDown={this.handleKeyDown}> | ||||
|  | @ -394,16 +387,22 @@ class EmojiPickerDropdown extends React.PureComponent { | |||
|           />} | ||||
|         </div> | ||||
| 
 | ||||
|         <Overlay show={active} placement={placement} target={this.findTarget}> | ||||
|           <EmojiPickerMenu | ||||
|             custom_emojis={this.props.custom_emojis} | ||||
|             loading={loading} | ||||
|             onClose={this.onHideDropdown} | ||||
|             onPick={onPickEmoji} | ||||
|             onSkinTone={onSkinTone} | ||||
|             skinTone={skinTone} | ||||
|             frequentlyUsedEmojis={frequentlyUsedEmojis} | ||||
|           /> | ||||
|         <Overlay show={active} placement={'bottom'} target={this.findTarget} popperConfig={{ strategy: 'fixed' }}> | ||||
|           {({ props, placement })=> ( | ||||
|             <div {...props} style={{ ...props.style, width: 299 }}> | ||||
|               <div className={`dropdown-animation ${placement}`}> | ||||
|                 <EmojiPickerMenu | ||||
|                   custom_emojis={this.props.custom_emojis} | ||||
|                   loading={loading} | ||||
|                   onClose={this.onHideDropdown} | ||||
|                   onPick={onPickEmoji} | ||||
|                   onSkinTone={onSkinTone} | ||||
|                   skinTone={skinTone} | ||||
|                   frequentlyUsedEmojis={frequentlyUsedEmojis} | ||||
|                 /> | ||||
|               </div> | ||||
|             </div> | ||||
|           )} | ||||
|         </Overlay> | ||||
|       </div> | ||||
|     ); | ||||
|  |  | |||
|  | @ -2,9 +2,7 @@ import React from 'react'; | |||
| import PropTypes from 'prop-types'; | ||||
| import { injectIntl, defineMessages } from 'react-intl'; | ||||
| import TextIconButton from './text_icon_button'; | ||||
| import Overlay from 'react-overlays/lib/Overlay'; | ||||
| import Motion from 'mastodon/features/ui/util/optional_motion'; | ||||
| import spring from 'react-motion/lib/spring'; | ||||
| import Overlay from 'react-overlays/Overlay'; | ||||
| import { supportsPassiveEvents } from 'detect-passive-events'; | ||||
| import classNames from 'classnames'; | ||||
| import { languages as preloadedLanguages } from 'mastodon/initial_state'; | ||||
|  | @ -22,10 +20,8 @@ const listenerOptions = supportsPassiveEvents ? { passive: true } : false; | |||
| class LanguageDropdownMenu extends React.PureComponent { | ||||
| 
 | ||||
|   static propTypes = { | ||||
|     style: PropTypes.object, | ||||
|     value: PropTypes.string.isRequired, | ||||
|     frequentlyUsedLanguages: PropTypes.arrayOf(PropTypes.string).isRequired, | ||||
|     placement: PropTypes.string.isRequired, | ||||
|     onClose: PropTypes.func.isRequired, | ||||
|     onChange: PropTypes.func.isRequired, | ||||
|     languages: PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.string)), | ||||
|  | @ -37,7 +33,6 @@ class LanguageDropdownMenu extends React.PureComponent { | |||
|   }; | ||||
| 
 | ||||
|   state = { | ||||
|     mounted: false, | ||||
|     searchValue: '', | ||||
|   }; | ||||
| 
 | ||||
|  | @ -50,7 +45,6 @@ class LanguageDropdownMenu extends React.PureComponent { | |||
|   componentDidMount () { | ||||
|     document.addEventListener('click', this.handleDocumentClick, false); | ||||
|     document.addEventListener('touchend', this.handleDocumentClick, listenerOptions); | ||||
|     this.setState({ mounted: true }); | ||||
| 
 | ||||
|     // Because of https://github.com/react-bootstrap/react-bootstrap/issues/2614 we need
 | ||||
|     // to wait for a frame before focusing
 | ||||
|  | @ -222,29 +216,22 @@ class LanguageDropdownMenu extends React.PureComponent { | |||
|   } | ||||
| 
 | ||||
|   render () { | ||||
|     const { style, placement, intl } = this.props; | ||||
|     const { mounted, searchValue } = this.state; | ||||
|     const { intl } = this.props; | ||||
|     const { searchValue } = this.state; | ||||
|     const isSearching = searchValue !== ''; | ||||
|     const results = this.search(); | ||||
| 
 | ||||
|     return ( | ||||
|       <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 }) => ( | ||||
|           // It should not be transformed when mounting because the resulting
 | ||||
|           // size will be used to determine the coordinate of the menu by
 | ||||
|           // react-overlays
 | ||||
|           <div className={`language-dropdown__dropdown ${placement}`} style={{ ...style, opacity: opacity, transform: mounted ? `scale(${scaleX}, ${scaleY})` : null }} ref={this.setRef}> | ||||
|             <div className='emoji-mart-search'> | ||||
|               <input type='search' value={searchValue} onChange={this.handleSearchChange} onKeyDown={this.handleSearchKeyDown} placeholder={intl.formatMessage(messages.search)} /> | ||||
|               <button type='button' className='emoji-mart-search-icon' disabled={!isSearching} aria-label={intl.formatMessage(messages.clear)} onClick={this.handleClear}>{!isSearching ? loupeIcon : deleteIcon}</button> | ||||
|             </div> | ||||
|       <div ref={this.setRef}> | ||||
|         <div className='emoji-mart-search'> | ||||
|           <input type='search' value={searchValue} onChange={this.handleSearchChange} onKeyDown={this.handleSearchKeyDown} placeholder={intl.formatMessage(messages.search)} /> | ||||
|           <button type='button' className='emoji-mart-search-icon' disabled={!isSearching} aria-label={intl.formatMessage(messages.clear)} onClick={this.handleClear}>{!isSearching ? loupeIcon : deleteIcon}</button> | ||||
|         </div> | ||||
| 
 | ||||
|             <div className='language-dropdown__dropdown__results emoji-mart-scroll' role='listbox' ref={this.setListRef}> | ||||
|               {results.map(this.renderItem)} | ||||
|             </div> | ||||
|           </div> | ||||
|         )} | ||||
|       </Motion> | ||||
|         <div className='language-dropdown__dropdown__results emoji-mart-scroll' role='listbox' ref={this.setListRef}> | ||||
|           {results.map(this.renderItem)} | ||||
|         </div> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|  | @ -266,14 +253,11 @@ class LanguageDropdown extends React.PureComponent { | |||
|     placement: 'bottom', | ||||
|   }; | ||||
| 
 | ||||
|   handleToggle = ({ target }) => { | ||||
|     const { top } = target.getBoundingClientRect(); | ||||
| 
 | ||||
|   handleToggle = () => { | ||||
|     if (this.state.open && this.activeElement) { | ||||
|       this.activeElement.focus({ preventScroll: true }); | ||||
|     } | ||||
| 
 | ||||
|     this.setState({ placement: top * 2 < innerHeight ? 'bottom' : 'top' }); | ||||
|     this.setState({ open: !this.state.open }); | ||||
|   } | ||||
| 
 | ||||
|  | @ -293,13 +277,25 @@ class LanguageDropdown extends React.PureComponent { | |||
|     onChange(value); | ||||
|   } | ||||
| 
 | ||||
|   setTargetRef = c => { | ||||
|     this.target = c; | ||||
|   } | ||||
| 
 | ||||
|   findTarget = () => { | ||||
|     return this.target; | ||||
|   } | ||||
| 
 | ||||
|   handleOverlayEnter = (state) => { | ||||
|     this.setState({ placement: state.placement }); | ||||
|   } | ||||
| 
 | ||||
|   render () { | ||||
|     const { value, intl, frequentlyUsedLanguages } = this.props; | ||||
|     const { open, placement } = this.state; | ||||
| 
 | ||||
|     return ( | ||||
|       <div className={classNames('privacy-dropdown', { active: open })}> | ||||
|         <div className='privacy-dropdown__value'> | ||||
|       <div className={classNames('privacy-dropdown', placement, { active: open })}> | ||||
|         <div className='privacy-dropdown__value' ref={this.setTargetRef} > | ||||
|           <TextIconButton | ||||
|             className='privacy-dropdown__value-icon' | ||||
|             label={value && value.toUpperCase()} | ||||
|  | @ -309,15 +305,20 @@ class LanguageDropdown extends React.PureComponent { | |||
|           /> | ||||
|         </div> | ||||
| 
 | ||||
|         <Overlay show={open} placement={placement} target={this}> | ||||
|           <LanguageDropdownMenu | ||||
|             value={value} | ||||
|             frequentlyUsedLanguages={frequentlyUsedLanguages} | ||||
|             onClose={this.handleClose} | ||||
|             onChange={this.handleChange} | ||||
|             placement={placement} | ||||
|             intl={intl} | ||||
|           /> | ||||
|         <Overlay show={open} placement={'bottom'} flip target={this.findTarget} popperConfig={{ strategy: 'fixed', onFirstUpdate: this.handleOverlayEnter }}> | ||||
|           {({ props, placement }) => ( | ||||
|             <div {...props} style={{ ...props.style, width: 280 }}> | ||||
|               <div className={`dropdown-animation language-dropdown__dropdown ${placement}`} > | ||||
|                 <LanguageDropdownMenu | ||||
|                   value={value} | ||||
|                   frequentlyUsedLanguages={frequentlyUsedLanguages} | ||||
|                   onClose={this.handleClose} | ||||
|                   onChange={this.handleChange} | ||||
|                   intl={intl} | ||||
|                 /> | ||||
|               </div> | ||||
|             </div> | ||||
|           )} | ||||
|         </Overlay> | ||||
|       </div> | ||||
|     ); | ||||
|  |  | |||
|  | @ -2,9 +2,7 @@ import React from 'react'; | |||
| import PropTypes from 'prop-types'; | ||||
| import { injectIntl, defineMessages } from 'react-intl'; | ||||
| import IconButton from '../../../components/icon_button'; | ||||
| import Overlay from 'react-overlays/lib/Overlay'; | ||||
| import Motion from '../../ui/util/optional_motion'; | ||||
| import spring from 'react-motion/lib/spring'; | ||||
| import Overlay from 'react-overlays/Overlay'; | ||||
| import { supportsPassiveEvents } from 'detect-passive-events'; | ||||
| import classNames from 'classnames'; | ||||
| import Icon from 'mastodon/components/icon'; | ||||
|  | @ -29,15 +27,10 @@ class PrivacyDropdownMenu extends React.PureComponent { | |||
|     style: PropTypes.object, | ||||
|     items: PropTypes.array.isRequired, | ||||
|     value: PropTypes.string.isRequired, | ||||
|     placement: PropTypes.string.isRequired, | ||||
|     onClose: PropTypes.func.isRequired, | ||||
|     onChange: PropTypes.func.isRequired, | ||||
|   }; | ||||
| 
 | ||||
|   state = { | ||||
|     mounted: false, | ||||
|   }; | ||||
| 
 | ||||
|   handleDocumentClick = e => { | ||||
|     if (this.node && !this.node.contains(e.target)) { | ||||
|       this.props.onClose(); | ||||
|  | @ -101,7 +94,6 @@ class PrivacyDropdownMenu extends React.PureComponent { | |||
|     document.addEventListener('click', this.handleDocumentClick, false); | ||||
|     document.addEventListener('touchend', this.handleDocumentClick, listenerOptions); | ||||
|     if (this.focusedItem) this.focusedItem.focus({ preventScroll: true }); | ||||
|     this.setState({ mounted: true }); | ||||
|   } | ||||
| 
 | ||||
|   componentWillUnmount () { | ||||
|  | @ -118,31 +110,23 @@ class PrivacyDropdownMenu extends React.PureComponent { | |||
|   } | ||||
| 
 | ||||
|   render () { | ||||
|     const { mounted } = this.state; | ||||
|     const { style, items, placement, value } = this.props; | ||||
|     const { style, items, value } = this.props; | ||||
| 
 | ||||
|     return ( | ||||
|       <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 }) => ( | ||||
|           // It should not be transformed when mounting because the resulting
 | ||||
|           // size will be used to determine the coordinate of the menu by
 | ||||
|           // react-overlays
 | ||||
|           <div className={`privacy-dropdown__dropdown ${placement}`} style={{ ...style, opacity: opacity, transform: mounted ? `scale(${scaleX}, ${scaleY})` : null }} role='listbox' ref={this.setRef}> | ||||
|             {items.map(item => ( | ||||
|               <div role='option' tabIndex='0' key={item.value} data-index={item.value} onKeyDown={this.handleKeyDown} onClick={this.handleClick} className={classNames('privacy-dropdown__option', { active: item.value === value })} aria-selected={item.value === value} ref={item.value === value ? this.setFocusRef : null}> | ||||
|                 <div className='privacy-dropdown__option__icon'> | ||||
|                   <Icon id={item.icon} fixedWidth /> | ||||
|                 </div> | ||||
|       <div style={{ ...style }} role='listbox' ref={this.setRef}> | ||||
|         {items.map(item => ( | ||||
|           <div role='option' tabIndex='0' key={item.value} data-index={item.value} onKeyDown={this.handleKeyDown} onClick={this.handleClick} className={classNames('privacy-dropdown__option', { active: item.value === value })} aria-selected={item.value === value} ref={item.value === value ? this.setFocusRef : null}> | ||||
|             <div className='privacy-dropdown__option__icon'> | ||||
|               <Icon id={item.icon} fixedWidth /> | ||||
|             </div> | ||||
| 
 | ||||
|                 <div className='privacy-dropdown__option__content'> | ||||
|                   <strong>{item.text}</strong> | ||||
|                   {item.meta} | ||||
|                 </div> | ||||
|               </div> | ||||
|             ))} | ||||
|             <div className='privacy-dropdown__option__content'> | ||||
|               <strong>{item.text}</strong> | ||||
|               {item.meta} | ||||
|             </div> | ||||
|           </div> | ||||
|         )} | ||||
|       </Motion> | ||||
|         ))} | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|  | @ -168,7 +152,7 @@ class PrivacyDropdown extends React.PureComponent { | |||
|     placement: 'bottom', | ||||
|   }; | ||||
| 
 | ||||
|   handleToggle = ({ target }) => { | ||||
|   handleToggle = () => { | ||||
|     if (this.props.isUserTouching && this.props.isUserTouching()) { | ||||
|       if (this.state.open) { | ||||
|         this.props.onModalClose(); | ||||
|  | @ -179,11 +163,9 @@ class PrivacyDropdown extends React.PureComponent { | |||
|         }); | ||||
|       } | ||||
|     } else { | ||||
|       const { top } = target.getBoundingClientRect(); | ||||
|       if (this.state.open && this.activeElement) { | ||||
|         this.activeElement.focus({ preventScroll: true }); | ||||
|       } | ||||
|       this.setState({ placement: top * 2 < innerHeight ? 'bottom' : 'top' }); | ||||
|       this.setState({ open: !this.state.open }); | ||||
|     } | ||||
|   } | ||||
|  | @ -247,6 +229,18 @@ class PrivacyDropdown extends React.PureComponent { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   setTargetRef = c => { | ||||
|     this.target = c; | ||||
|   } | ||||
| 
 | ||||
|   findTarget = () => { | ||||
|     return this.target; | ||||
|   } | ||||
| 
 | ||||
|   handleOverlayEnter = (state) => { | ||||
|     this.setState({ placement: state.placement }); | ||||
|   } | ||||
| 
 | ||||
|   render () { | ||||
|     const { value, container, disabled, intl } = this.props; | ||||
|     const { open, placement } = this.state; | ||||
|  | @ -255,7 +249,7 @@ class PrivacyDropdown extends React.PureComponent { | |||
| 
 | ||||
|     return ( | ||||
|       <div className={classNames('privacy-dropdown', placement, { active: open })} onKeyDown={this.handleKeyDown}> | ||||
|         <div className={classNames('privacy-dropdown__value', { active: this.options.indexOf(valueOption) === (placement === 'bottom' ? 0 : (this.options.length - 1)) })}> | ||||
|         <div className={classNames('privacy-dropdown__value', { active: this.options.indexOf(valueOption) === (placement === 'bottom' ? 0 : (this.options.length - 1)) })} ref={this.setTargetRef}> | ||||
|           <IconButton | ||||
|             className='privacy-dropdown__value-icon' | ||||
|             icon={valueOption.icon} | ||||
|  | @ -272,14 +266,19 @@ class PrivacyDropdown extends React.PureComponent { | |||
|           /> | ||||
|         </div> | ||||
| 
 | ||||
|         <Overlay show={open} placement={placement} target={this} container={container}> | ||||
|           <PrivacyDropdownMenu | ||||
|             items={this.options} | ||||
|             value={value} | ||||
|             onClose={this.handleClose} | ||||
|             onChange={this.handleChange} | ||||
|             placement={placement} | ||||
|           /> | ||||
|         <Overlay show={open} placement={'bottom'} flip target={this.findTarget} container={container} popperConfig={{ strategy: 'fixed', onFirstUpdate: this.handleOverlayEnter }}> | ||||
|           {({ props, placement }) => ( | ||||
|             <div {...props} style={{ ...props.style, width: 350, maxWidth: '100vw' }}> | ||||
|               <div className={`dropdown-animation privacy-dropdown__dropdown ${placement}`}> | ||||
|                 <PrivacyDropdownMenu | ||||
|                   items={this.options} | ||||
|                   value={value} | ||||
|                   onClose={this.handleClose} | ||||
|                   onChange={this.handleChange} | ||||
|                 /> | ||||
|               </div> | ||||
|             </div> | ||||
|           )} | ||||
|         </Overlay> | ||||
|       </div> | ||||
|     ); | ||||
|  |  | |||
|  | @ -1,9 +1,7 @@ | |||
| import React from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import { defineMessages, injectIntl, FormattedMessage } from 'react-intl'; | ||||
| import Overlay from 'react-overlays/lib/Overlay'; | ||||
| import Motion from '../../ui/util/optional_motion'; | ||||
| import spring from 'react-motion/lib/spring'; | ||||
| import Overlay from 'react-overlays/Overlay'; | ||||
| import { searchEnabled } from '../../../initial_state'; | ||||
| import Icon from 'mastodon/components/icon'; | ||||
| 
 | ||||
|  | @ -14,31 +12,20 @@ const messages = defineMessages({ | |||
| 
 | ||||
| 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 ( | ||||
|       <div style={{ ...style, position: 'absolute', width: 285, zIndex: 2 }}> | ||||
|         <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 }) => ( | ||||
|             <div className='search-popout' style={{ opacity: opacity, transform: `scale(${scaleX}, ${scaleY})` }}> | ||||
|               <h4><FormattedMessage id='search_popout.search_format' defaultMessage='Advanced search format' /></h4> | ||||
|       <div className='search-popout'> | ||||
|         <h4><FormattedMessage id='search_popout.search_format' defaultMessage='Advanced search format' /></h4> | ||||
| 
 | ||||
|               <ul> | ||||
|                 <li><em>#example</em> <FormattedMessage id='search_popout.tips.hashtag' defaultMessage='hashtag' /></li> | ||||
|                 <li><em>@username@domain</em> <FormattedMessage id='search_popout.tips.user' defaultMessage='user' /></li> | ||||
|                 <li><em>URL</em> <FormattedMessage id='search_popout.tips.user' defaultMessage='user' /></li> | ||||
|                 <li><em>URL</em> <FormattedMessage id='search_popout.tips.status' defaultMessage='status' /></li> | ||||
|               </ul> | ||||
|         <ul> | ||||
|           <li><em>#example</em> <FormattedMessage id='search_popout.tips.hashtag' defaultMessage='hashtag' /></li> | ||||
|           <li><em>@username@domain</em> <FormattedMessage id='search_popout.tips.user' defaultMessage='user' /></li> | ||||
|           <li><em>URL</em> <FormattedMessage id='search_popout.tips.user' defaultMessage='user' /></li> | ||||
|           <li><em>URL</em> <FormattedMessage id='search_popout.tips.status' defaultMessage='status' /></li> | ||||
|         </ul> | ||||
| 
 | ||||
|               {extraInformation} | ||||
|             </div> | ||||
|           )} | ||||
|         </Motion> | ||||
|         {extraInformation} | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | @ -115,6 +102,10 @@ class Search extends React.PureComponent { | |||
|     this.setState({ expanded: false }); | ||||
|   } | ||||
| 
 | ||||
|   findTarget = () => { | ||||
|     return this.searchForm; | ||||
|   } | ||||
| 
 | ||||
|   render () { | ||||
|     const { intl, value, submitted } = this.props; | ||||
|     const { expanded } = this.state; | ||||
|  | @ -140,8 +131,14 @@ class Search extends React.PureComponent { | |||
|           <Icon id='search' className={hasValue ? '' : 'active'} /> | ||||
|           <Icon id='times-circle' className={hasValue ? 'active' : ''} aria-label={intl.formatMessage(messages.placeholder)} /> | ||||
|         </div> | ||||
|         <Overlay show={expanded && !hasValue} placement='bottom' target={this} container={this}> | ||||
|           <SearchPopout /> | ||||
|         <Overlay show={expanded && !hasValue} placement='bottom' target={this.findTarget} popperConfig={{ strategy: 'fixed' }}> | ||||
|           {({ props, placement }) => ( | ||||
|             <div {...props} style={{ ...props.style, width: 285, zIndex: 2 }}> | ||||
|               <div className={`dropdown-animation ${placement}`}> | ||||
|                 <SearchPopout /> | ||||
|               </div> | ||||
|             </div> | ||||
|           )} | ||||
|         </Overlay> | ||||
|       </div> | ||||
|     ); | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue