Files
chaos-api/web/default/src/features/channels/components/dialogs/codex-oauth-dialog.tsx
T
QuentinHsu 2eaa943d9f perf(web): improve dialog sizing and footer layout
- migrate frontend dialogs to the shared footer API so actions stay separated from scrollable body content.
- tune dialog dimensions for model analytics, prefill groups, billing history, channel model sync, and related workflows.
- update channel terminology and dialog action translations across supported locales.
2026-06-06 21:49:33 +08:00

216 lines
6.9 KiB
TypeScript
Vendored

/*
Copyright (C) 2023-2026 QuantumNous
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
For commercial licensing, please contact support@quantumnous.com
*/
import { useEffect, useMemo, useState } from 'react'
import { ExternalLink, Copy, Check, Loader2 } from 'lucide-react'
import { useTranslation } from 'react-i18next'
import { toast } from 'sonner'
import { tryPrettyJson } from '@/lib/utils'
import { useCopyToClipboard } from '@/hooks/use-copy-to-clipboard'
import { Alert, AlertDescription } from '@/components/ui/alert'
import { Button } from '@/components/ui/button'
import { Input } from '@/components/ui/input'
import { Dialog } from '@/components/dialog'
import { completeCodexOAuth, startCodexOAuth } from '../../api'
type CodexOAuthDialogProps = {
open: boolean
onOpenChange: (open: boolean) => void
onKeyGenerated: (key: string) => void
}
export function CodexOAuthDialog({
open,
onOpenChange,
onKeyGenerated,
}: CodexOAuthDialogProps) {
const { t } = useTranslation()
const { copiedText, copyToClipboard } = useCopyToClipboard({ notify: false })
const [state, setState] = useState({
authorizeUrl: '',
callbackUrl: '',
isStarting: false,
isCompleting: false,
})
useEffect(() => {
if (!open) {
setState({
authorizeUrl: '',
callbackUrl: '',
isStarting: false,
isCompleting: false,
})
}
}, [open])
const canCopyAuthorizeUrl = Boolean(state.authorizeUrl && !state.isStarting)
const canComplete = useMemo(
() => Boolean(state.callbackUrl.trim()) && !state.isCompleting,
[state.callbackUrl, state.isCompleting]
)
const handleStart = async () => {
setState((prev) => ({ ...prev, isStarting: true }))
try {
const res = await startCodexOAuth()
if (!res.success) {
throw new Error(res.message || 'Failed to start OAuth')
}
const url = res.data?.authorize_url || ''
if (!url) {
throw new Error('Missing authorize_url in response')
}
setState((prev) => ({ ...prev, authorizeUrl: url }))
try {
window.open(url, '_blank', 'noopener,noreferrer')
toast.success(t('Opened authorization page'))
} catch (error) {
// eslint-disable-next-line no-console
console.warn('Failed to open authorization page:', error)
toast.warning(t('Please manually copy and open the authorization link'))
}
} catch (error) {
toast.error(
error instanceof Error ? error.message : t('OAuth start failed')
)
} finally {
setState((prev) => ({ ...prev, isStarting: false }))
}
}
const handleComplete = async () => {
if (!state.callbackUrl.trim()) return
setState((prev) => ({ ...prev, isCompleting: true }))
try {
const res = await completeCodexOAuth(state.callbackUrl.trim())
if (!res.success) {
throw new Error(res.message || 'OAuth failed')
}
const rawKey = res.data?.key || ''
if (!rawKey) {
throw new Error('Missing key in response')
}
onKeyGenerated(tryPrettyJson(rawKey))
toast.success(t('Credential generated'))
onOpenChange(false)
} catch (error) {
toast.error(error instanceof Error ? error.message : t('OAuth failed'))
} finally {
setState((prev) => ({ ...prev, isCompleting: false }))
}
}
return (
<Dialog
open={open}
onOpenChange={onOpenChange}
title={t('Codex Authorization')}
description={t(
'Generate a Codex OAuth credential and paste it into the channel key field.'
)}
contentClassName='sm:max-w-2xl'
contentHeight='auto'
bodyClassName='space-y-4'
footer={
<>
<Button
type='button'
variant='outline'
onClick={() => onOpenChange(false)}
disabled={state.isStarting || state.isCompleting}
>
{t('Cancel')}
</Button>
<Button onClick={handleComplete} disabled={!canComplete}>
{state.isCompleting && (
<Loader2 className='mr-2 h-4 w-4 animate-spin' />
)}
{state.isCompleting ? t('Generating...') : t('Generate credential')}
</Button>
</>
}
>
<div className='space-y-4'>
<Alert>
<AlertDescription>
{t(
'1) Click "Open authorization page" and complete login. 2) Your browser may redirect to localhost (it is OK if the page does not load). 3) Copy the full URL from the address bar and paste it below. 4) Click "Generate credential".'
)}
</AlertDescription>
</Alert>
<div className='flex flex-wrap gap-2'>
<Button onClick={handleStart} disabled={state.isStarting}>
{state.isStarting ? (
<Loader2 className='mr-2 h-4 w-4 animate-spin' />
) : (
<ExternalLink className='mr-2 h-4 w-4' />
)}
{t('Open authorization page')}
</Button>
<Button
type='button'
variant='outline'
disabled={!canCopyAuthorizeUrl}
onClick={async () => {
if (!state.authorizeUrl) return
await copyToClipboard(state.authorizeUrl)
}}
aria-label={t('Copy authorization link')}
title={t('Copy authorization link')}
>
{copiedText === state.authorizeUrl ? (
<Check className='mr-2 h-4 w-4 text-green-600' />
) : (
<Copy className='mr-2 h-4 w-4' />
)}
{t('Copy authorization link')}
</Button>
</div>
<div className='space-y-2'>
<div className='text-sm font-medium'>{t('Callback URL')}</div>
<Input
value={state.callbackUrl}
onChange={(e) =>
setState((prev) => ({ ...prev, callbackUrl: e.target.value }))
}
placeholder={t(
'Paste the full callback URL (includes code & state)'
)}
autoComplete='off'
spellCheck={false}
/>
<div className='text-muted-foreground text-xs'>
{t(
'Tip: The generated key is a JSON credential including access_token / refresh_token / account_id.'
)}
</div>
</div>
</div>
</Dialog>
)
}