mirror of
https://github.com/langgenius/dify.git
synced 2026-06-25 22:31:10 +08:00
123 lines
4.2 KiB
TypeScript
123 lines
4.2 KiB
TypeScript
'use client'
|
|
|
|
import { cn } from '@langgenius/dify-ui/cn'
|
|
import { Input } from '@langgenius/dify-ui/input'
|
|
import { Textarea } from '@langgenius/dify-ui/textarea'
|
|
import { useAtom, useAtomValue } from 'jotai'
|
|
import { useEffect, useRef } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import {
|
|
createReleaseDescriptionFieldAtom,
|
|
createReleaseHasNameConflictAtom,
|
|
createReleaseNameFieldAtom,
|
|
RELEASE_NAME_REQUIRED_ERROR,
|
|
} from '../state'
|
|
|
|
const DESCRIPTION_MAX_LENGTH = 512
|
|
const DESCRIPTION_WARN_THRESHOLD = 460
|
|
|
|
function isValidationIssue(error: unknown): error is { message: string } {
|
|
return Boolean(
|
|
error
|
|
&& typeof error === 'object'
|
|
&& 'message' in error
|
|
&& typeof error.message === 'string',
|
|
)
|
|
}
|
|
|
|
function hasReleaseNameRequiredError(errors: unknown[]) {
|
|
return errors.some((error) => {
|
|
if (error === RELEASE_NAME_REQUIRED_ERROR)
|
|
return true
|
|
|
|
if (Array.isArray(error))
|
|
return error.some(issue => isValidationIssue(issue) && issue.message === RELEASE_NAME_REQUIRED_ERROR)
|
|
|
|
return isValidationIssue(error) && error.message === RELEASE_NAME_REQUIRED_ERROR
|
|
})
|
|
}
|
|
|
|
export function ReleaseMetadataFields() {
|
|
const { t } = useTranslation('deployments')
|
|
const [releaseNameField, setReleaseNameField] = useAtom(createReleaseNameFieldAtom)
|
|
const [releaseDescriptionField, setReleaseDescriptionField] = useAtom(createReleaseDescriptionFieldAtom)
|
|
const hasReleaseNameConflict = useAtomValue(createReleaseHasNameConflictAtom)
|
|
const releaseNameInputRef = useRef<HTMLInputElement>(null)
|
|
const releaseNameErrors = releaseNameField.meta?.errors ?? []
|
|
const hasReleaseNameRequired = hasReleaseNameRequiredError(releaseNameErrors)
|
|
const releaseNameError = hasReleaseNameRequired
|
|
? t('versions.releaseNameRequired')
|
|
: hasReleaseNameConflict
|
|
? t('versions.releaseNameConflict')
|
|
: ''
|
|
|
|
useEffect(() => {
|
|
releaseNameInputRef.current?.focus()
|
|
}, [])
|
|
|
|
return (
|
|
<>
|
|
<div className="flex flex-col gap-2">
|
|
<label className="system-xs-medium-uppercase text-text-tertiary" htmlFor="release-name">
|
|
{t('versions.releaseNameLabel')}
|
|
</label>
|
|
<Input
|
|
ref={releaseNameInputRef}
|
|
id="release-name"
|
|
name="releaseName"
|
|
placeholder={t('versions.releaseNamePlaceholder')}
|
|
maxLength={128}
|
|
autoComplete="off"
|
|
value={releaseNameField.value}
|
|
aria-invalid={Boolean(releaseNameError) || undefined}
|
|
aria-describedby={releaseNameError ? 'release-name-error' : undefined}
|
|
onChange={(event) => {
|
|
setReleaseNameField(event.target.value)
|
|
}}
|
|
className="h-9"
|
|
/>
|
|
{releaseNameError && (
|
|
<div id="release-name-error" role="alert" className="system-xs-regular text-text-destructive">
|
|
{releaseNameError}
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
<div className="flex flex-col gap-2">
|
|
<div className="flex items-center justify-between gap-3">
|
|
<label className="system-xs-medium-uppercase text-text-tertiary" htmlFor="release-description">
|
|
{t('versions.releaseDescriptionLabel')}
|
|
</label>
|
|
<div className="flex items-center gap-2">
|
|
<span className="system-xs-regular text-text-quaternary">
|
|
{t('versions.optional')}
|
|
</span>
|
|
<span
|
|
className={cn(
|
|
'system-xs-regular tabular-nums',
|
|
releaseDescriptionField.value.length >= DESCRIPTION_WARN_THRESHOLD ? 'text-util-colors-warning-warning-700' : 'text-text-quaternary',
|
|
)}
|
|
>
|
|
{releaseDescriptionField.value.length}
|
|
/
|
|
{DESCRIPTION_MAX_LENGTH}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<Textarea
|
|
id="release-description"
|
|
name="releaseDescription"
|
|
placeholder={t('versions.releaseDescriptionPlaceholder')}
|
|
maxLength={DESCRIPTION_MAX_LENGTH}
|
|
autoComplete="off"
|
|
value={releaseDescriptionField.value}
|
|
onValueChange={(value) => {
|
|
setReleaseDescriptionField(value)
|
|
}}
|
|
className="min-h-24 resize-none"
|
|
/>
|
|
</div>
|
|
</>
|
|
)
|
|
}
|