feat: 批量转移服务器给其他用户

This commit is contained in:
naiba
2025-06-16 23:44:30 +08:00
parent bf7c42e4e7
commit 5f854c3dd0
6 changed files with 141 additions and 19 deletions
+100
View File
@@ -0,0 +1,100 @@
import { batchMoveServer } from "@/api/server"
import { Button, ButtonProps } from "@/components/ui/button"
import {
Dialog,
DialogClose,
DialogContent,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "@/components/ui/dialog"
import { Input } from "@/components/ui/input"
import { Label } from "@/components/ui/label"
import { ScrollArea } from "@/components/ui/scroll-area"
import { IconButton } from "@/components/xui/icon-button"
import { useState } from "react"
import { useTranslation } from "react-i18next"
import { toast } from "sonner"
import { Textarea } from "./ui/textarea"
interface BatchMoveServerIconProps extends ButtonProps {
serverIds: number[]
}
export const BatchMoveServerIcon: React.FC<BatchMoveServerIconProps> = ({ serverIds, ...props }) => {
const { t } = useTranslation()
const [open, setOpen] = useState(false)
const [toUserId, setToUserId] = useState<number | undefined>(undefined)
const onSubmit = async () => {
try {
await batchMoveServer({
ids: serverIds,
to_user: toUserId!
})
} catch (e) {
console.error(e)
toast(t("Error"), {
description: t("Results.UnExpectedError"),
})
return
}
toast(t("Done"))
setOpen(false)
}
return serverIds.length < 1 ? (
<IconButton
{...props}
icon="user-pen"
onClick={() => {
toast(t("Error"), {
description: t("Results.NoRowsAreSelected"),
})
}}
/>
) : (
<Dialog open={open} onOpenChange={setOpen}>
<DialogTrigger asChild>
<IconButton {...props} icon="user-pen" />
</DialogTrigger>
<DialogContent className="sm:max-w-xl">
<ScrollArea className="max-h-[calc(100dvh-5rem)] p-3">
<div className="items-center mx-1">
<DialogHeader>
<DialogTitle>{t("BatchMoveServer")}</DialogTitle>
<DialogDescription />
</DialogHeader>
<div className="flex flex-col gap-3 mt-4">
<Label>{t("Servers")}</Label>
<Textarea disabled>
{serverIds.join(", ")}
</Textarea>
<Label>{t("ToUser")}</Label>
<Input
type="number"
placeholder="User ID"
value={toUserId}
onChange={(e) => {
setToUserId(parseInt(e.target.value, 10))
}}
/>
<DialogFooter className="justify-end">
<DialogClose asChild>
<Button type="button" className="my-2" variant="secondary">
{t("Cancel")}
</Button>
</DialogClose>
<Button disabled={!toUserId || toUserId == 0} type="submit" className="my-2" onClick={onSubmit}>
{t("Move")}
</Button>
</DialogFooter>
</div>
</div>
</ScrollArea>
</DialogContent>
</Dialog>
)
}
+5
View File
@@ -16,6 +16,7 @@ import {
Terminal,
Trash2,
Upload,
UserPen,
} from "lucide-react"
import { forwardRef } from "react"
@@ -37,6 +38,7 @@ export interface IconButtonProps extends ButtonProps {
| "expand"
| "cog"
| "minus"
| "user-pen"
}
export const IconButton = forwardRef<HTMLButtonElement, IconButtonProps>((props, ref) => {
@@ -97,6 +99,9 @@ export const IconButton = forwardRef<HTMLButtonElement, IconButtonProps>((props,
case "minus": {
return <Minus />
}
case "user-pen": {
return <UserPen />
}
}
})()}
</Button>