| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | //  Package imports.
 | 
					
						
							|  |  |  | import PropTypes from 'prop-types'; | 
					
						
							| 
									
										
										
										
											2023-05-28 07:56:24 -05:00
										 |  |  | import { PureComponent } from 'react'; | 
					
						
							| 
									
										
										
										
											2023-05-28 09:38:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | import classNames from 'classnames'; | 
					
						
							| 
									
										
										
										
											2023-05-28 09:38:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-22 08:48:01 -05:00
										 |  |  | import { supportsPassiveEvents } from 'detect-passive-events'; | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | //  Components.
 | 
					
						
							| 
									
										
										
										
											2023-05-08 20:11:56 -05:00
										 |  |  | import { Icon } from 'flavours/glitch/components/icon'; | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-22 08:48:01 -05:00
										 |  |  | const listenerOptions = supportsPassiveEvents ? { passive: true, capture: true } : true; | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | //  The component.
 | 
					
						
							| 
									
										
										
										
											2023-05-28 07:18:23 -05:00
										 |  |  | export default class ComposerOptionsDropdownContent extends PureComponent { | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   static propTypes = { | 
					
						
							|  |  |  |     items: PropTypes.arrayOf(PropTypes.shape({ | 
					
						
							|  |  |  |       icon: PropTypes.string, | 
					
						
							|  |  |  |       meta: PropTypes.node, | 
					
						
							|  |  |  |       name: PropTypes.string.isRequired, | 
					
						
							|  |  |  |       text: PropTypes.node, | 
					
						
							|  |  |  |     })), | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     onChange: PropTypes.func.isRequired, | 
					
						
							|  |  |  |     onClose: PropTypes.func.isRequired, | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |     style: PropTypes.object, | 
					
						
							|  |  |  |     value: PropTypes.string, | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |     renderItemContents: PropTypes.func, | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     openedViaKeyboard: PropTypes.bool, | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |     closeOnChange: PropTypes.bool, | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   static defaultProps = { | 
					
						
							|  |  |  |     style: {}, | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |     closeOnChange: true, | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   state = { | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     value: this.props.openedViaKeyboard ? this.props.items[0].name : undefined, | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //  When the document is clicked elsewhere, we close the dropdown.
 | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |   handleDocumentClick = (e) => { | 
					
						
							|  |  |  |     if (this.node && !this.node.contains(e.target)) { | 
					
						
							|  |  |  |       this.props.onClose(); | 
					
						
							| 
									
										
										
										
											2023-05-22 08:48:01 -05:00
										 |  |  |       e.stopPropagation(); | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-03 13:52:07 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   //  Stores our node in `this.node`.
 | 
					
						
							| 
									
										
										
										
											2023-01-11 14:58:46 -06:00
										 |  |  |   setRef = (node) => { | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |     this.node = node; | 
					
						
							| 
									
										
										
										
											2023-02-03 13:52:07 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   //  On mounting, we add our listeners.
 | 
					
						
							|  |  |  |   componentDidMount () { | 
					
						
							| 
									
										
										
										
											2023-05-22 08:48:01 -05:00
										 |  |  |     document.addEventListener('click', this.handleDocumentClick, { capture: true }); | 
					
						
							|  |  |  |     document.addEventListener('touchend', this.handleDocumentClick, listenerOptions); | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     if (this.focusedItem) { | 
					
						
							| 
									
										
										
										
											2020-04-28 06:19:39 -05:00
										 |  |  |       this.focusedItem.focus({ preventScroll: true }); | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2020-04-28 06:19:39 -05:00
										 |  |  |       this.node.firstChild.focus({ preventScroll: true }); | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //  On unmounting, we remove our listeners.
 | 
					
						
							|  |  |  |   componentWillUnmount () { | 
					
						
							| 
									
										
										
										
											2023-05-22 08:48:01 -05:00
										 |  |  |     document.removeEventListener('click', this.handleDocumentClick, { capture: true }); | 
					
						
							|  |  |  |     document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions); | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 06:41:17 -06:00
										 |  |  |   handleClick = (e) => { | 
					
						
							|  |  |  |     const i = Number(e.currentTarget.getAttribute('data-index')); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     const { | 
					
						
							|  |  |  |       onChange, | 
					
						
							|  |  |  |       onClose, | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |       closeOnChange, | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |       items, | 
					
						
							|  |  |  |     } = this.props; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 14:43:25 -05:00
										 |  |  |     const { name } = items[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     e.preventDefault();  //  Prevents change in focus on click
 | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |     if (closeOnChange) { | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |       onClose(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     onChange(name); | 
					
						
							| 
									
										
										
										
											2023-02-03 13:52:07 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Handle changes differently whether the dropdown is a list of options or actions
 | 
					
						
							|  |  |  |   handleChange = (name) => { | 
					
						
							|  |  |  |     if (this.props.value) { | 
					
						
							|  |  |  |       this.props.onChange(name); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       this.setState({ value: name }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-03 13:52:07 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 06:41:17 -06:00
										 |  |  |   handleKeyDown = (e) => { | 
					
						
							|  |  |  |     const index = Number(e.currentTarget.getAttribute('data-index')); | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     const { items } = this.props; | 
					
						
							| 
									
										
										
										
											2020-04-21 08:13:26 -05:00
										 |  |  |     let element = null; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch(e.key) { | 
					
						
							|  |  |  |     case 'Escape': | 
					
						
							|  |  |  |       this.props.onClose(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 'Enter': | 
					
						
							|  |  |  |     case ' ': | 
					
						
							|  |  |  |       this.handleClick(e); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 'ArrowDown': | 
					
						
							| 
									
										
										
										
											2020-04-21 08:13:26 -05:00
										 |  |  |       element = this.node.childNodes[index + 1] || this.node.firstChild; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case 'ArrowUp': | 
					
						
							| 
									
										
										
										
											2020-04-21 08:13:26 -05:00
										 |  |  |       element = this.node.childNodes[index - 1] || this.node.lastChild; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case 'Tab': | 
					
						
							|  |  |  |       if (e.shiftKey) { | 
					
						
							|  |  |  |         element = this.node.childNodes[index - 1] || this.node.lastChild; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         element = this.node.childNodes[index + 1] || this.node.firstChild; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 'Home': | 
					
						
							|  |  |  |       element = this.node.firstChild; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case 'End': | 
					
						
							|  |  |  |       element = this.node.lastChild; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-21 08:13:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (element) { | 
					
						
							|  |  |  |       element.focus(); | 
					
						
							| 
									
										
										
										
											2023-05-07 14:43:25 -05:00
										 |  |  |       this.handleChange(items[Number(element.getAttribute('data-index'))].name); | 
					
						
							| 
									
										
										
										
											2020-04-21 08:13:26 -05:00
										 |  |  |       e.preventDefault(); | 
					
						
							|  |  |  |       e.stopPropagation(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-03 13:52:07 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   setFocusRef = c => { | 
					
						
							|  |  |  |     this.focusedItem = c; | 
					
						
							| 
									
										
										
										
											2023-02-03 13:52:07 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 06:41:17 -06:00
										 |  |  |   renderItem = (item, i) => { | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |     const { name, icon, meta, text } = item; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const active = (name === (this.props.value || this.state.value)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-06 06:30:37 -06:00
										 |  |  |     const computedClass = classNames('privacy-dropdown__option', { active }); | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |     let contents = this.props.renderItemContents && this.props.renderItemContents(item, i); | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |     if (!contents) { | 
					
						
							|  |  |  |       contents = ( | 
					
						
							| 
									
										
										
										
											2023-05-28 07:56:24 -05:00
										 |  |  |         <> | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |           {icon && <Icon className='icon' fixedWidth id={icon} />} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-06 06:30:37 -06:00
										 |  |  |           <div className='privacy-dropdown__option__content'> | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |             <strong>{text}</strong> | 
					
						
							|  |  |  |             {meta} | 
					
						
							|  |  |  |           </div> | 
					
						
							| 
									
										
										
										
											2023-05-28 07:56:24 -05:00
										 |  |  |         </> | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |       ); | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ( | 
					
						
							|  |  |  |       <div | 
					
						
							|  |  |  |         className={computedClass} | 
					
						
							| 
									
										
										
										
											2022-02-09 06:41:17 -06:00
										 |  |  |         onClick={this.handleClick} | 
					
						
							|  |  |  |         onKeyDown={this.handleKeyDown} | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |         role='option' | 
					
						
							| 
									
										
										
										
											2023-05-07 14:43:25 -05:00
										 |  |  |         aria-selected={active} | 
					
						
							| 
									
										
										
										
											2023-04-04 09:33:44 -05:00
										 |  |  |         tabIndex={0} | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |         key={name} | 
					
						
							| 
									
										
										
										
											2022-02-09 06:41:17 -06:00
										 |  |  |         data-index={i} | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |         ref={active ? this.setFocusRef : null} | 
					
						
							|  |  |  |       > | 
					
						
							| 
									
										
										
										
											2022-02-09 07:39:12 -06:00
										 |  |  |         {contents} | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  |       </div> | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2023-02-03 13:52:07 -06:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-08-06 07:18:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |   //  Rendering.
 | 
					
						
							|  |  |  |   render () { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       items, | 
					
						
							|  |  |  |       style, | 
					
						
							|  |  |  |     } = this.props; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //  The result.
 | 
					
						
							|  |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2023-01-11 14:58:46 -06:00
										 |  |  |       <div style={{ ...style }} role='listbox' ref={this.setRef}> | 
					
						
							|  |  |  |         {!!items && items.map((item, i) => this.renderItem(item, i))} | 
					
						
							|  |  |  |       </div> | 
					
						
							| 
									
										
										
										
											2019-04-21 10:17:10 -05:00
										 |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |