mirror of
https://github.com/overleaf/overleaf.git
synced 2025-01-02 16:22:37 +00:00
b7802674d5
React `project-context` GitOrigin-RevId: 6a23437d6e6a328ff5854622ff903d348db1f8b8
113 lines
2.9 KiB
JavaScript
113 lines
2.9 KiB
JavaScript
import { useCallback } from 'react'
|
|
import PropTypes from 'prop-types'
|
|
import { useShareProjectContext } from './share-project-modal'
|
|
import Icon from '../../../shared/components/icon'
|
|
import { Button, Col, Row, OverlayTrigger, Tooltip } from 'react-bootstrap'
|
|
import { Trans, useTranslation } from 'react-i18next'
|
|
import MemberPrivileges from './member-privileges'
|
|
import { resendInvite, revokeInvite } from '../utils/api'
|
|
import { useProjectContext } from '../../../shared/context/project-context'
|
|
|
|
export default function Invite({ invite, isAdmin }) {
|
|
return (
|
|
<Row className="project-invite">
|
|
<Col xs={7}>
|
|
<div>{invite.email}</div>
|
|
|
|
<div className="small">
|
|
<Trans i18nKey="invite_not_accepted" />
|
|
.
|
|
{isAdmin && <ResendInvite invite={invite} />}
|
|
</div>
|
|
</Col>
|
|
|
|
<Col xs={3} className="text-left">
|
|
<MemberPrivileges privileges={invite.privileges} />
|
|
</Col>
|
|
|
|
{isAdmin && (
|
|
<Col xs={2} className="text-center">
|
|
<RevokeInvite invite={invite} />
|
|
</Col>
|
|
)}
|
|
</Row>
|
|
)
|
|
}
|
|
|
|
Invite.propTypes = {
|
|
invite: PropTypes.object.isRequired,
|
|
isAdmin: PropTypes.bool.isRequired,
|
|
}
|
|
|
|
function ResendInvite({ invite }) {
|
|
const { monitorRequest } = useShareProjectContext()
|
|
const project = useProjectContext()
|
|
|
|
// const buttonRef = useRef(null)
|
|
//
|
|
const handleClick = useCallback(
|
|
() =>
|
|
monitorRequest(() => resendInvite(project, invite)).finally(() => {
|
|
// NOTE: disabled as react-bootstrap v0.33.1 isn't forwarding the ref to the `button`
|
|
// if (buttonRef.current) {
|
|
// buttonRef.current.blur()
|
|
// }
|
|
document.activeElement.blur()
|
|
}),
|
|
[invite, monitorRequest, project]
|
|
)
|
|
|
|
return (
|
|
<Button
|
|
bsStyle="link"
|
|
className="btn-inline-link"
|
|
onClick={handleClick}
|
|
// ref={buttonRef}
|
|
>
|
|
<Trans i18nKey="resend" />
|
|
</Button>
|
|
)
|
|
}
|
|
ResendInvite.propTypes = {
|
|
invite: PropTypes.object.isRequired,
|
|
}
|
|
|
|
function RevokeInvite({ invite }) {
|
|
const { t } = useTranslation()
|
|
const { updateProject, monitorRequest } = useShareProjectContext()
|
|
const project = useProjectContext()
|
|
|
|
function handleClick(event) {
|
|
event.preventDefault()
|
|
|
|
monitorRequest(() => revokeInvite(project, invite)).then(() => {
|
|
updateProject({
|
|
invites: project.invites.filter(existing => existing !== invite),
|
|
})
|
|
})
|
|
}
|
|
|
|
return (
|
|
<OverlayTrigger
|
|
placement="bottom"
|
|
overlay={
|
|
<Tooltip id="tooltip-revoke-invite">
|
|
<Trans i18nKey="revoke_invite" />
|
|
</Tooltip>
|
|
}
|
|
>
|
|
<Button
|
|
type="button"
|
|
bsStyle="link"
|
|
onClick={handleClick}
|
|
aria-label={t('revoke')}
|
|
className="btn-inline-link"
|
|
>
|
|
<Icon type="times" />
|
|
</Button>
|
|
</OverlayTrigger>
|
|
)
|
|
}
|
|
RevokeInvite.propTypes = {
|
|
invite: PropTypes.object.isRequired,
|
|
}
|