mirror of
https://github.com/supabase/supabase.git
synced 2026-07-05 09:05:10 +08:00
185 lines
5.5 KiB
TypeScript
185 lines
5.5 KiB
TypeScript
import { FC, useContext, useState } from 'react'
|
|
import { observer } from 'mobx-react-lite'
|
|
import * as Tooltip from '@radix-ui/react-tooltip'
|
|
import { Button, Dropdown, Divider, IconTrash, IconMail, IconMoreHorizontal } from '@supabase/ui'
|
|
|
|
import { useStore } from 'hooks'
|
|
import { timeout } from 'lib/helpers'
|
|
import { post, delete_ } from 'lib/common/fetch'
|
|
import { API_URL } from 'lib/constants'
|
|
import { PageContext } from 'pages/project/[ref]/auth/users'
|
|
import { confirmAlert } from 'components/to-be-cleaned/ModalsDeprecated/ConfirmModal'
|
|
import { User } from './Users.types'
|
|
|
|
interface Props {
|
|
user: User
|
|
canRemoveUser: boolean
|
|
}
|
|
|
|
const UserDropdown: FC<Props> = ({ user, canRemoveUser }) => {
|
|
const PageState: any = useContext(PageContext)
|
|
const { ui } = useStore()
|
|
const [loading, setLoading] = useState<boolean>(false)
|
|
|
|
async function handleResetPassword() {
|
|
try {
|
|
setLoading(true)
|
|
const response = await post(`${API_URL}/auth/${PageState.projectRef}/recover`, user)
|
|
if (response.error) {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Failed to send password recovery: ${response.error.message}`,
|
|
})
|
|
} else {
|
|
ui.setNotification({
|
|
category: 'success',
|
|
message: `Sent password recovery to ${user.email}`,
|
|
})
|
|
}
|
|
} catch (error: any) {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Send password recovery failed: ${error?.message}`,
|
|
})
|
|
} finally {
|
|
setLoading(false)
|
|
}
|
|
}
|
|
|
|
async function handleSendMagicLink() {
|
|
try {
|
|
setLoading(true)
|
|
const response = await post(`${API_URL}/auth/${PageState.projectRef}/magiclink`, user)
|
|
if (response.error) {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Failed to send magic link: ${response.error.message}`,
|
|
})
|
|
} else {
|
|
ui.setNotification({
|
|
category: 'success',
|
|
message: `Sent magic link to ${user.email}`,
|
|
})
|
|
}
|
|
} catch (error: any) {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Failed to send magic link: ${error?.message}`,
|
|
})
|
|
} finally {
|
|
setLoading(false)
|
|
}
|
|
}
|
|
|
|
async function handleSendOtp() {
|
|
try {
|
|
setLoading(true)
|
|
const response = await post(`${API_URL}/auth/${PageState.projectRef}/otp`, user)
|
|
if (response.error) {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Failed to OTP: ${response.error.message}`,
|
|
})
|
|
} else {
|
|
ui.setNotification({
|
|
category: 'success',
|
|
message: `Sent OTP to ${user.phone}`,
|
|
})
|
|
}
|
|
} catch (error: any) {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Failed to send OTP: ${error?.message}`,
|
|
})
|
|
} finally {
|
|
setLoading(false)
|
|
}
|
|
}
|
|
|
|
async function handleDelete() {
|
|
await timeout(200)
|
|
|
|
confirmAlert({
|
|
title: 'Confirm to delete',
|
|
message: `This is permanent! Are you sure you want to delete user ${user.email} ?`,
|
|
onAsyncConfirm: async () => {
|
|
setLoading(true)
|
|
const response = await delete_(`${API_URL}/auth/${PageState.projectRef}/users`, user)
|
|
if (response.error) {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Failed to delete user: ${response.error.message}`,
|
|
})
|
|
} else {
|
|
ui.setNotification({ category: 'success', message: `Successfully deleted ${user.email}` })
|
|
PageState.users = PageState.users.filter((x: any) => x.id != user.id)
|
|
PageState.totalUsers -= 1
|
|
}
|
|
setLoading(false)
|
|
},
|
|
})
|
|
}
|
|
|
|
return (
|
|
<Dropdown
|
|
size="medium"
|
|
overlay={
|
|
<>
|
|
{user.email !== null ? (
|
|
<>
|
|
<Dropdown.Item onClick={handleResetPassword} icon={<IconMail size="tiny" />}>
|
|
Send password recovery
|
|
</Dropdown.Item>
|
|
<Dropdown.Item onClick={handleSendMagicLink} icon={<IconMail size="tiny" />}>
|
|
Send magic link
|
|
</Dropdown.Item>
|
|
</>
|
|
) : null}
|
|
{user.phone !== null ? (
|
|
<Dropdown.Item onClick={handleSendOtp} icon={<IconMail size="tiny" />}>
|
|
Send OTP
|
|
</Dropdown.Item>
|
|
) : null}
|
|
<Dropdown.Seperator />
|
|
<Tooltip.Root delayDuration={0}>
|
|
<Tooltip.Trigger className="w-full">
|
|
<Dropdown.Item
|
|
onClick={handleDelete}
|
|
icon={<IconTrash size="tiny" />}
|
|
disabled={!canRemoveUser}
|
|
>
|
|
Delete user
|
|
</Dropdown.Item>
|
|
</Tooltip.Trigger>
|
|
{!canRemoveUser && (
|
|
<Tooltip.Content side="bottom">
|
|
<Tooltip.Arrow className="radix-tooltip-arrow" />
|
|
<div
|
|
className={[
|
|
'bg-scale-100 rounded py-1 px-2 leading-none shadow',
|
|
'border-scale-200 border',
|
|
].join(' ')}
|
|
>
|
|
<span className="text-scale-1200 text-xs">
|
|
You need additional permissions to delete users
|
|
</span>
|
|
</div>
|
|
</Tooltip.Content>
|
|
)}
|
|
</Tooltip.Root>
|
|
</>
|
|
}
|
|
>
|
|
<Button
|
|
as="span"
|
|
type="text"
|
|
icon={<IconMoreHorizontal />}
|
|
loading={loading}
|
|
className="hover:border-gray-500"
|
|
/>
|
|
</Dropdown>
|
|
)
|
|
}
|
|
|
|
export default observer(UserDropdown)
|