mirror of
https://github.com/overleaf/overleaf.git
synced 2024-12-25 19:13:09 +00:00
b7802674d5
React `project-context` GitOrigin-RevId: 6a23437d6e6a328ff5854622ff903d348db1f8b8
149 lines
3.2 KiB
JavaScript
149 lines
3.2 KiB
JavaScript
import React, {
|
|
createContext,
|
|
useCallback,
|
|
useContext,
|
|
useEffect,
|
|
useState,
|
|
} from 'react'
|
|
import PropTypes from 'prop-types'
|
|
import ShareProjectModalContent from './share-project-modal-content'
|
|
import { useProjectContext } from '../../../shared/context/project-context'
|
|
|
|
const ShareProjectContext = createContext()
|
|
|
|
ShareProjectContext.Provider.propTypes = {
|
|
value: PropTypes.shape({
|
|
isAdmin: PropTypes.bool.isRequired,
|
|
updateProject: PropTypes.func.isRequired,
|
|
monitorRequest: PropTypes.func.isRequired,
|
|
inFlight: PropTypes.bool,
|
|
setInFlight: PropTypes.func,
|
|
error: PropTypes.oneOfType([PropTypes.bool, PropTypes.string]),
|
|
setError: PropTypes.func,
|
|
}),
|
|
}
|
|
|
|
export function useShareProjectContext() {
|
|
const context = useContext(ShareProjectContext)
|
|
|
|
if (!context) {
|
|
throw new Error(
|
|
'useShareProjectContext is only available inside ShareProjectProvider'
|
|
)
|
|
}
|
|
|
|
return context
|
|
}
|
|
|
|
const projectShape = {
|
|
_id: PropTypes.string.isRequired,
|
|
members: PropTypes.arrayOf(
|
|
PropTypes.shape({
|
|
_id: PropTypes.string.isRequired,
|
|
})
|
|
),
|
|
invites: PropTypes.arrayOf(
|
|
PropTypes.shape({
|
|
_id: PropTypes.string.isRequired,
|
|
})
|
|
),
|
|
name: PropTypes.string,
|
|
features: PropTypes.shape({
|
|
collaborators: PropTypes.number,
|
|
}),
|
|
publicAccesLevel: PropTypes.string,
|
|
tokens: PropTypes.shape({
|
|
readOnly: PropTypes.string,
|
|
readAndWrite: PropTypes.string,
|
|
}),
|
|
owner: PropTypes.shape({
|
|
email: PropTypes.string,
|
|
}),
|
|
}
|
|
|
|
const ShareProjectModal = React.memo(function ShareProjectModal({
|
|
handleHide,
|
|
show,
|
|
animation = true,
|
|
isAdmin,
|
|
}) {
|
|
const [inFlight, setInFlight] = useState(false)
|
|
const [error, setError] = useState()
|
|
|
|
const project = useProjectContext(projectShape)
|
|
|
|
// reset error when the modal is opened
|
|
useEffect(() => {
|
|
if (show) {
|
|
setError(undefined)
|
|
}
|
|
}, [show])
|
|
|
|
// close the modal if not in flight
|
|
const cancel = useCallback(() => {
|
|
if (!inFlight) {
|
|
handleHide()
|
|
}
|
|
}, [handleHide, inFlight])
|
|
|
|
// update `error` and `inFlight` while sending a request
|
|
const monitorRequest = useCallback(request => {
|
|
setError(undefined)
|
|
setInFlight(true)
|
|
|
|
const promise = request()
|
|
|
|
promise.catch(error => {
|
|
setError(
|
|
error.data?.errorReason ||
|
|
error.data?.error ||
|
|
'generic_something_went_wrong'
|
|
)
|
|
})
|
|
|
|
promise.finally(() => {
|
|
setInFlight(false)
|
|
})
|
|
|
|
return promise
|
|
}, [])
|
|
|
|
// merge the new data with the old project data
|
|
const updateProject = useCallback(data => Object.assign(project, data), [
|
|
project,
|
|
])
|
|
|
|
if (!project) {
|
|
return null
|
|
}
|
|
|
|
return (
|
|
<ShareProjectContext.Provider
|
|
value={{
|
|
isAdmin,
|
|
updateProject,
|
|
monitorRequest,
|
|
inFlight,
|
|
setInFlight,
|
|
error,
|
|
setError,
|
|
}}
|
|
>
|
|
<ShareProjectModalContent
|
|
animation={animation}
|
|
cancel={cancel}
|
|
error={error}
|
|
inFlight={inFlight}
|
|
show={show}
|
|
/>
|
|
</ShareProjectContext.Provider>
|
|
)
|
|
})
|
|
ShareProjectModal.propTypes = {
|
|
animation: PropTypes.bool,
|
|
handleHide: PropTypes.func.isRequired,
|
|
isAdmin: PropTypes.bool.isRequired,
|
|
show: PropTypes.bool.isRequired,
|
|
}
|
|
|
|
export default ShareProjectModal
|