2022-10-01 06:37:54 -07:00
import classNames from 'clsx' ;
import React , { useCallback , useEffect , useRef , useState } from 'react' ;
import 'wicg-inert' ;
import { FormattedMessage , defineMessages , useIntl } from 'react-intl' ;
import { useHistory } from 'react-router-dom' ;
import { cancelReplyCompose } from 'soapbox/actions/compose' ;
2022-11-30 10:32:35 -08:00
import { cancelEventCompose } from 'soapbox/actions/events' ;
2022-10-01 06:37:54 -07:00
import { openModal , closeModal } from 'soapbox/actions/modals' ;
2022-11-30 10:32:35 -08:00
import { useAppDispatch , usePrevious } from 'soapbox/hooks' ;
2022-11-01 10:22:29 -07:00
import { queryClient } from 'soapbox/queries/client' ;
import { IPolicy , PolicyKeys } from 'soapbox/queries/policies' ;
2022-10-01 06:37:54 -07:00
2022-11-16 05:32:32 -08:00
import type { ModalType } from 'soapbox/features/ui/components/modal-root' ;
2022-10-01 06:37:54 -07:00
import type { ReducerCompose } from 'soapbox/reducers/compose' ;
2022-11-30 10:32:35 -08:00
import type { ReducerRecord as ReducerComposeEvent } from 'soapbox/reducers/compose-event' ;
2022-10-01 06:37:54 -07:00
const messages = defineMessages ( {
confirm : { id : 'confirmations.delete.confirm' , defaultMessage : 'Delete' } ,
2022-11-30 10:32:35 -08:00
cancelEditing : { id : 'confirmations.cancel_editing.confirm' , defaultMessage : 'Cancel editing' } ,
2022-10-01 06:37:54 -07:00
} ) ;
export const checkComposeContent = ( compose? : ReturnType < typeof ReducerCompose > ) = > {
return ! ! compose && [
compose . text . length > 0 ,
compose . spoiler_text . length > 0 ,
compose . media_attachments . size > 0 ,
compose . poll !== null ,
] . some ( check = > check === true ) ;
} ;
2022-11-30 10:32:35 -08:00
export const checkEventComposeContent = ( compose? : ReturnType < typeof ReducerComposeEvent > ) = > {
return ! ! compose && [
compose . name . length > 0 ,
compose . status . length > 0 ,
compose . location !== null ,
compose . banner !== null ,
] . some ( check = > check === true ) ;
} ;
2022-10-01 06:37:54 -07:00
interface IModalRoot {
onCancel ? : ( ) = > void ,
2022-11-12 06:18:24 -08:00
onClose : ( type ? : ModalType ) = > void ,
type : ModalType ,
2022-10-01 06:37:54 -07:00
}
const ModalRoot : React.FC < IModalRoot > = ( { children , onCancel , onClose , type } ) = > {
const intl = useIntl ( ) ;
const history = useHistory ( ) ;
const dispatch = useAppDispatch ( ) ;
const [ revealed , setRevealed ] = useState ( ! ! children ) ;
const ref = useRef < HTMLDivElement > ( null ) ;
const activeElement = useRef < HTMLDivElement | null > ( revealed ? document . activeElement as HTMLDivElement | null : null ) ;
const modalHistoryKey = useRef < number > ( ) ;
2023-01-09 13:43:41 -08:00
const unlistenHistory = useRef < ReturnType < typeof history.listen > > ( ) ;
2022-10-01 06:37:54 -07:00
const prevChildren = usePrevious ( children ) ;
const prevType = usePrevious ( type ) ;
2022-11-10 12:46:51 -08:00
const visible = ! ! children ;
const handleKeyUp = ( e : KeyboardEvent ) = > {
if ( e . key === 'Escape' || e . key === 'Esc' || e . keyCode === 27 ) {
2022-10-01 06:37:54 -07:00
handleOnClose ( ) ;
}
2022-11-10 12:46:51 -08:00
} ;
2022-10-01 06:37:54 -07:00
const handleOnClose = ( ) = > {
dispatch ( ( _ , getState ) = > {
2022-11-30 10:32:35 -08:00
const compose = getState ( ) . compose . get ( 'compose-modal' ) ;
const hasComposeContent = checkComposeContent ( compose ) ;
const hasEventComposeContent = checkEventComposeContent ( getState ( ) . compose_event ) ;
2022-10-01 06:37:54 -07:00
if ( hasComposeContent && type === 'COMPOSE' ) {
2022-11-30 10:32:35 -08:00
const isEditing = compose ! . id !== null ;
2022-10-01 06:37:54 -07:00
dispatch ( openModal ( 'CONFIRM' , {
icon : require ( '@tabler/icons/trash.svg' ) ,
2022-11-30 10:32:35 -08:00
heading : isEditing
? < FormattedMessage id = 'confirmations.cancel_editing.heading' defaultMessage = 'Cancel post editing' / >
: < FormattedMessage id = 'confirmations.delete.heading' defaultMessage = 'Delete post' / > ,
message : isEditing
? < FormattedMessage id = 'confirmations.cancel_editing.message' defaultMessage = 'Are you sure you want to cancel editing this post? All changes will be lost.' / >
: < FormattedMessage id = 'confirmations.delete.message' defaultMessage = 'Are you sure you want to delete this post?' / > ,
2022-10-01 06:37:54 -07:00
confirm : intl.formatMessage ( messages . confirm ) ,
onConfirm : ( ) = > {
dispatch ( closeModal ( 'COMPOSE' ) ) ;
dispatch ( cancelReplyCompose ( ) ) ;
} ,
onCancel : ( ) = > {
dispatch ( closeModal ( 'CONFIRM' ) ) ;
} ,
} ) ) ;
2022-11-30 10:32:35 -08:00
} else if ( hasEventComposeContent && type === 'COMPOSE_EVENT' ) {
const isEditing = getState ( ) . compose_event . id !== null ;
dispatch ( openModal ( 'CONFIRM' , {
icon : require ( '@tabler/icons/trash.svg' ) ,
heading : isEditing
? < FormattedMessage id = 'confirmations.cancel_event_editing.heading' defaultMessage = 'Cancel event editing' / >
: < FormattedMessage id = 'confirmations.delete_event.heading' defaultMessage = 'Delete event' / > ,
message : isEditing
? < FormattedMessage id = 'confirmations.cancel_event_editing.message' defaultMessage = 'Are you sure you want to cancel editing this event? All changes will be lost.' / >
: < FormattedMessage id = 'confirmations.delete_event.message' defaultMessage = 'Are you sure you want to delete this event?' / > ,
confirm : intl.formatMessage ( isEditing ? messages.cancelEditing : messages.confirm ) ,
onConfirm : ( ) = > {
dispatch ( closeModal ( 'COMPOSE_EVENT' ) ) ;
dispatch ( cancelEventCompose ( ) ) ;
} ,
onCancel : ( ) = > {
dispatch ( closeModal ( 'CONFIRM' ) ) ;
} ,
} ) ) ;
} else if ( ( hasComposeContent || hasEventComposeContent ) && type === 'CONFIRM' ) {
2022-10-01 06:37:54 -07:00
dispatch ( closeModal ( 'CONFIRM' ) ) ;
2022-11-01 10:22:29 -07:00
} else if ( type === 'POLICY' ) {
// If the user has not accepted the Policy, prevent them
// from closing the Modal.
const pendingPolicy = queryClient . getQueryData ( PolicyKeys . policy ) as IPolicy ;
if ( pendingPolicy ? . pending_policy_id ) {
return ;
}
onClose ( ) ;
2022-10-01 06:37:54 -07:00
} else {
onClose ( ) ;
}
} ) ;
} ;
const handleKeyDown = useCallback ( ( e ) = > {
if ( e . key === 'Tab' ) {
const focusable = Array . from ( ref . current ! . querySelectorAll ( 'button:not([disabled]), [href], input:not([disabled]), select:not([disabled]), textarea:not([disabled]), [tabindex]:not([tabindex="-1"])' ) ) . filter ( ( x ) = > window . getComputedStyle ( x ) . display !== 'none' ) ;
const index = focusable . indexOf ( e . target ) ;
let element ;
if ( e . shiftKey ) {
element = focusable [ index - 1 ] || focusable [ focusable . length - 1 ] ;
} else {
element = focusable [ index + 1 ] || focusable [ 0 ] ;
}
if ( element ) {
( element as HTMLDivElement ) . focus ( ) ;
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
}
}
} , [ ] ) ;
const handleModalOpen = ( ) = > {
modalHistoryKey . current = Date . now ( ) ;
2022-12-03 11:49:43 -08:00
unlistenHistory . current = history . listen ( ( { state } , action ) = > {
if ( ! ( state as any ) ? . soapboxModalKey ) {
onClose ( ) ;
} else if ( action === 'POP' ) {
2022-10-01 06:37:54 -07:00
handleOnClose ( ) ;
if ( onCancel ) onCancel ( ) ;
}
} ) ;
} ;
const handleModalClose = ( type : string ) = > {
if ( unlistenHistory . current ) {
unlistenHistory . current ( ) ;
}
2022-12-03 11:49:43 -08:00
const { state } = history . location ;
if ( state && ( state as any ) . soapboxModalKey === modalHistoryKey . current ) {
history . goBack ( ) ;
2022-10-01 06:37:54 -07:00
}
} ;
const ensureHistoryBuffer = ( ) = > {
const { pathname , state } = history . location ;
if ( ! state || ( state as any ) . soapboxModalKey !== modalHistoryKey . current ) {
history . push ( pathname , { . . . ( state as any ) , soapboxModalKey : modalHistoryKey.current } ) ;
}
} ;
const getSiblings = ( ) = > {
return Array ( . . . ( ref . current ! . parentElement ! . childNodes as any as ChildNode [ ] ) ) . filter ( node = > node !== ref . current ) ;
} ;
useEffect ( ( ) = > {
2022-11-10 12:46:51 -08:00
if ( ! visible ) return ;
2022-10-01 06:37:54 -07:00
window . addEventListener ( 'keyup' , handleKeyUp , false ) ;
window . addEventListener ( 'keydown' , handleKeyDown , false ) ;
return ( ) = > {
window . removeEventListener ( 'keyup' , handleKeyUp ) ;
window . removeEventListener ( 'keydown' , handleKeyDown ) ;
} ;
2022-11-10 12:46:51 -08:00
} , [ visible ] ) ;
2022-10-01 06:37:54 -07:00
useEffect ( ( ) = > {
if ( ! ! children && ! prevChildren ) {
activeElement . current = document . activeElement as HTMLDivElement ;
getSiblings ( ) . forEach ( sibling = > ( sibling as HTMLDivElement ) . setAttribute ( 'inert' , 'true' ) ) ;
handleModalOpen ( ) ;
} else if ( ! prevChildren ) {
setRevealed ( false ) ;
}
if ( ! children && ! ! prevChildren ) {
activeElement . current ? . focus ( ) ;
activeElement . current = null ;
getSiblings ( ) . forEach ( sibling = > ( sibling as HTMLDivElement ) . removeAttribute ( 'inert' ) ) ;
handleModalClose ( prevType ! ) ;
}
if ( children ) {
requestAnimationFrame ( ( ) = > {
setRevealed ( true ) ;
} ) ;
ensureHistoryBuffer ( ) ;
}
2022-12-03 11:49:43 -08:00
} , [ children ] ) ;
2022-10-01 06:37:54 -07:00
if ( ! visible ) {
return (
< div className = 'z-50 transition-all' ref = { ref } style = { { opacity : 0 } } / >
) ;
}
return (
< div
ref = { ref }
className = { classNames ( {
'fixed top-0 left-0 z-[100] w-full h-full overflow-x-hidden overflow-y-auto' : true ,
'pointer-events-none' : ! visible ,
} ) }
style = { { opacity : revealed ? 1 : 0 } }
>
< div
role = 'presentation'
id = 'modal-overlay'
2022-12-30 15:27:52 -08:00
className = 'fixed inset-0 bg-gray-500/90 dark:bg-gray-700/90 backdrop-blur'
2022-10-01 06:37:54 -07:00
onClick = { handleOnClose }
/ >
< div
role = 'dialog'
className = { classNames ( {
'my-2 mx-auto relative pointer-events-none flex items-center' : true ,
'p-4 md:p-0' : type !== 'MEDIA' ,
} ) }
style = { { minHeight : 'calc(100% - 3.5rem)' } }
>
{ children }
< / div >
< / div >
) ;
} ;
export default ModalRoot ;