Merge branch 'langgenius:main' into add-document-status-update
commit
fcbbc2d42d
@ -0,0 +1,24 @@
|
|||||||
|
from enum import StrEnum
|
||||||
|
|
||||||
|
|
||||||
|
class ResponseFormat(StrEnum):
|
||||||
|
"""Constants for model response formats"""
|
||||||
|
|
||||||
|
JSON_SCHEMA = "json_schema" # model's structured output mode. some model like gemini, gpt-4o, support this mode.
|
||||||
|
JSON = "JSON" # model's json mode. some model like claude support this mode.
|
||||||
|
JSON_OBJECT = "json_object" # json mode's another alias. some model like deepseek-chat, qwen use this alias.
|
||||||
|
|
||||||
|
|
||||||
|
class SpecialModelType(StrEnum):
|
||||||
|
"""Constants for identifying model types"""
|
||||||
|
|
||||||
|
GEMINI = "gemini"
|
||||||
|
OLLAMA = "ollama"
|
||||||
|
|
||||||
|
|
||||||
|
class SupportStructuredOutputStatus(StrEnum):
|
||||||
|
"""Constants for structured output support status"""
|
||||||
|
|
||||||
|
SUPPORTED = "supported"
|
||||||
|
UNSUPPORTED = "unsupported"
|
||||||
|
DISABLED = "disabled"
|
||||||
@ -0,0 +1,82 @@
|
|||||||
|
import { fireEvent, render, screen } from '@testing-library/react'
|
||||||
|
import ConfigSelect from './index'
|
||||||
|
|
||||||
|
jest.mock('react-sortablejs', () => ({
|
||||||
|
ReactSortable: ({ children }: { children: React.ReactNode }) => <div>{children}</div>,
|
||||||
|
}))
|
||||||
|
|
||||||
|
jest.mock('react-i18next', () => ({
|
||||||
|
useTranslation: () => ({
|
||||||
|
t: (key: string) => key,
|
||||||
|
}),
|
||||||
|
}))
|
||||||
|
|
||||||
|
describe('ConfigSelect Component', () => {
|
||||||
|
const defaultProps = {
|
||||||
|
options: ['Option 1', 'Option 2'],
|
||||||
|
onChange: jest.fn(),
|
||||||
|
}
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
jest.clearAllMocks()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders all options', () => {
|
||||||
|
render(<ConfigSelect {...defaultProps} />)
|
||||||
|
|
||||||
|
defaultProps.options.forEach((option) => {
|
||||||
|
expect(screen.getByDisplayValue(option)).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders add button', () => {
|
||||||
|
render(<ConfigSelect {...defaultProps} />)
|
||||||
|
|
||||||
|
expect(screen.getByText('appDebug.variableConfig.addOption')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles option deletion', () => {
|
||||||
|
render(<ConfigSelect {...defaultProps} />)
|
||||||
|
const optionContainer = screen.getByDisplayValue('Option 1').closest('div')
|
||||||
|
const deleteButton = optionContainer?.querySelector('div[role="button"]')
|
||||||
|
|
||||||
|
if (!deleteButton) return
|
||||||
|
fireEvent.click(deleteButton)
|
||||||
|
expect(defaultProps.onChange).toHaveBeenCalledWith(['Option 2'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles adding new option', () => {
|
||||||
|
render(<ConfigSelect {...defaultProps} />)
|
||||||
|
const addButton = screen.getByText('appDebug.variableConfig.addOption')
|
||||||
|
|
||||||
|
fireEvent.click(addButton)
|
||||||
|
|
||||||
|
expect(defaultProps.onChange).toHaveBeenCalledWith([...defaultProps.options, ''])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies focus styles on input focus', () => {
|
||||||
|
render(<ConfigSelect {...defaultProps} />)
|
||||||
|
const firstInput = screen.getByDisplayValue('Option 1')
|
||||||
|
|
||||||
|
fireEvent.focus(firstInput)
|
||||||
|
|
||||||
|
expect(firstInput.closest('div')).toHaveClass('border-components-input-border-active')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies delete hover styles', () => {
|
||||||
|
render(<ConfigSelect {...defaultProps} />)
|
||||||
|
const optionContainer = screen.getByDisplayValue('Option 1').closest('div')
|
||||||
|
const deleteButton = optionContainer?.querySelector('div[role="button"]')
|
||||||
|
|
||||||
|
if (!deleteButton) return
|
||||||
|
fireEvent.mouseEnter(deleteButton)
|
||||||
|
expect(optionContainer).toHaveClass('border-components-input-border-destructive')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders empty state correctly', () => {
|
||||||
|
render(<ConfigSelect options={[]} onChange={defaultProps.onChange} />)
|
||||||
|
|
||||||
|
expect(screen.queryByRole('textbox')).not.toBeInTheDocument()
|
||||||
|
expect(screen.getByText('appDebug.variableConfig.addOption')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
})
|
||||||
@ -0,0 +1,11 @@
|
|||||||
|
const IndeterminateIcon = () => {
|
||||||
|
return (
|
||||||
|
<div data-testid='indeterminate-icon'>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 12 12" fill="none">
|
||||||
|
<path d="M2.5 6H9.5" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default IndeterminateIcon
|
||||||
@ -1,5 +0,0 @@
|
|||||||
<svg width="12" height="12" viewBox="0 0 12 12" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
<g id="check">
|
|
||||||
<path id="Vector 1" d="M2.5 6H9.5" stroke="white" stroke-width="1.5" stroke-linecap="round"/>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 217 B |
@ -1,10 +0,0 @@
|
|||||||
.mixed {
|
|
||||||
background: var(--color-components-checkbox-bg) url(./assets/mixed.svg) center center no-repeat;
|
|
||||||
background-size: 12px 12px;
|
|
||||||
border: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.checked.disabled {
|
|
||||||
background-color: #d0d5dd;
|
|
||||||
border-color: #d0d5dd;
|
|
||||||
}
|
|
||||||
@ -0,0 +1,67 @@
|
|||||||
|
import { fireEvent, render, screen } from '@testing-library/react'
|
||||||
|
import Checkbox from './index'
|
||||||
|
|
||||||
|
describe('Checkbox Component', () => {
|
||||||
|
const mockProps = {
|
||||||
|
id: 'test',
|
||||||
|
}
|
||||||
|
|
||||||
|
it('renders unchecked checkbox by default', () => {
|
||||||
|
render(<Checkbox {...mockProps} />)
|
||||||
|
const checkbox = screen.getByTestId('checkbox-test')
|
||||||
|
expect(checkbox).toBeInTheDocument()
|
||||||
|
expect(checkbox).not.toHaveClass('bg-components-checkbox-bg')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders checked checkbox when checked prop is true', () => {
|
||||||
|
render(<Checkbox {...mockProps} checked />)
|
||||||
|
const checkbox = screen.getByTestId('checkbox-test')
|
||||||
|
expect(checkbox).toHaveClass('bg-components-checkbox-bg')
|
||||||
|
expect(screen.getByTestId('check-icon-test')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders indeterminate state correctly', () => {
|
||||||
|
render(<Checkbox {...mockProps} indeterminate />)
|
||||||
|
expect(screen.getByTestId('indeterminate-icon')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles click events when not disabled', () => {
|
||||||
|
const onCheck = jest.fn()
|
||||||
|
render(<Checkbox {...mockProps} onCheck={onCheck} />)
|
||||||
|
const checkbox = screen.getByTestId('checkbox-test')
|
||||||
|
|
||||||
|
fireEvent.click(checkbox)
|
||||||
|
expect(onCheck).toHaveBeenCalledTimes(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not handle click events when disabled', () => {
|
||||||
|
const onCheck = jest.fn()
|
||||||
|
render(<Checkbox {...mockProps} disabled onCheck={onCheck} />)
|
||||||
|
const checkbox = screen.getByTestId('checkbox-test')
|
||||||
|
|
||||||
|
fireEvent.click(checkbox)
|
||||||
|
expect(onCheck).not.toHaveBeenCalled()
|
||||||
|
expect(checkbox).toHaveClass('cursor-not-allowed')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies custom className when provided', () => {
|
||||||
|
const customClass = 'custom-class'
|
||||||
|
render(<Checkbox {...mockProps} className={customClass} />)
|
||||||
|
const checkbox = screen.getByTestId('checkbox-test')
|
||||||
|
expect(checkbox).toHaveClass(customClass)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies correct styles for disabled checked state', () => {
|
||||||
|
render(<Checkbox {...mockProps} checked disabled />)
|
||||||
|
const checkbox = screen.getByTestId('checkbox-test')
|
||||||
|
expect(checkbox).toHaveClass('bg-components-checkbox-bg-disabled-checked')
|
||||||
|
expect(checkbox).toHaveClass('cursor-not-allowed')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies correct styles for disabled unchecked state', () => {
|
||||||
|
render(<Checkbox {...mockProps} disabled />)
|
||||||
|
const checkbox = screen.getByTestId('checkbox-test')
|
||||||
|
expect(checkbox).toHaveClass('bg-components-checkbox-bg-disabled')
|
||||||
|
expect(checkbox).toHaveClass('cursor-not-allowed')
|
||||||
|
})
|
||||||
|
})
|
||||||
@ -0,0 +1,43 @@
|
|||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import { useFieldContext } from '../..'
|
||||||
|
import Checkbox from '../../../checkbox'
|
||||||
|
|
||||||
|
type CheckboxFieldProps = {
|
||||||
|
label: string;
|
||||||
|
labelClassName?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const CheckboxField = ({
|
||||||
|
label,
|
||||||
|
labelClassName,
|
||||||
|
}: CheckboxFieldProps) => {
|
||||||
|
const field = useFieldContext<boolean>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className='flex gap-2'>
|
||||||
|
<div className='flex h-6 shrink-0 items-center'>
|
||||||
|
<Checkbox
|
||||||
|
id={field.name}
|
||||||
|
checked={field.state.value}
|
||||||
|
onCheck={() => {
|
||||||
|
field.handleChange(!field.state.value)
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<label
|
||||||
|
htmlFor={field.name}
|
||||||
|
className={cn(
|
||||||
|
'system-sm-medium grow cursor-pointer pt-1 text-text-secondary',
|
||||||
|
labelClassName,
|
||||||
|
)}
|
||||||
|
onClick={() => {
|
||||||
|
field.handleChange(!field.state.value)
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default CheckboxField
|
||||||
@ -0,0 +1,49 @@
|
|||||||
|
import React from 'react'
|
||||||
|
import { useFieldContext } from '../..'
|
||||||
|
import Label from '../label'
|
||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import type { InputNumberProps } from '../../../input-number'
|
||||||
|
import { InputNumber } from '../../../input-number'
|
||||||
|
|
||||||
|
type TextFieldProps = {
|
||||||
|
label: string
|
||||||
|
isRequired?: boolean
|
||||||
|
showOptional?: boolean
|
||||||
|
tooltip?: string
|
||||||
|
className?: string
|
||||||
|
labelClassName?: string
|
||||||
|
} & Omit<InputNumberProps, 'id' | 'value' | 'onChange' | 'onBlur'>
|
||||||
|
|
||||||
|
const NumberInputField = ({
|
||||||
|
label,
|
||||||
|
isRequired,
|
||||||
|
showOptional,
|
||||||
|
tooltip,
|
||||||
|
className,
|
||||||
|
labelClassName,
|
||||||
|
...inputProps
|
||||||
|
}: TextFieldProps) => {
|
||||||
|
const field = useFieldContext<number | undefined>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('flex flex-col gap-y-0.5', className)}>
|
||||||
|
<Label
|
||||||
|
htmlFor={field.name}
|
||||||
|
label={label}
|
||||||
|
isRequired={isRequired}
|
||||||
|
showOptional={showOptional}
|
||||||
|
tooltip={tooltip}
|
||||||
|
className={labelClassName}
|
||||||
|
/>
|
||||||
|
<InputNumber
|
||||||
|
id={field.name}
|
||||||
|
value={field.state.value}
|
||||||
|
onChange={value => field.handleChange(value)}
|
||||||
|
onBlur={field.handleBlur}
|
||||||
|
{...inputProps}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default NumberInputField
|
||||||
@ -0,0 +1,34 @@
|
|||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import { useFieldContext } from '../..'
|
||||||
|
import Label from '../label'
|
||||||
|
import ConfigSelect from '@/app/components/app/configuration/config-var/config-select'
|
||||||
|
|
||||||
|
type OptionsFieldProps = {
|
||||||
|
label: string;
|
||||||
|
className?: string;
|
||||||
|
labelClassName?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const OptionsField = ({
|
||||||
|
label,
|
||||||
|
className,
|
||||||
|
labelClassName,
|
||||||
|
}: OptionsFieldProps) => {
|
||||||
|
const field = useFieldContext<string[]>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('flex flex-col gap-y-0.5', className)}>
|
||||||
|
<Label
|
||||||
|
htmlFor={field.name}
|
||||||
|
label={label}
|
||||||
|
className={labelClassName}
|
||||||
|
/>
|
||||||
|
<ConfigSelect
|
||||||
|
options={field.state.value}
|
||||||
|
onChange={value => field.handleChange(value)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default OptionsField
|
||||||
@ -0,0 +1,51 @@
|
|||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import { useFieldContext } from '../..'
|
||||||
|
import PureSelect from '../../../select/pure'
|
||||||
|
import Label from '../label'
|
||||||
|
|
||||||
|
type SelectOption = {
|
||||||
|
value: string
|
||||||
|
label: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type SelectFieldProps = {
|
||||||
|
label: string
|
||||||
|
options: SelectOption[]
|
||||||
|
isRequired?: boolean
|
||||||
|
showOptional?: boolean
|
||||||
|
tooltip?: string
|
||||||
|
className?: string
|
||||||
|
labelClassName?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const SelectField = ({
|
||||||
|
label,
|
||||||
|
options,
|
||||||
|
isRequired,
|
||||||
|
showOptional,
|
||||||
|
tooltip,
|
||||||
|
className,
|
||||||
|
labelClassName,
|
||||||
|
}: SelectFieldProps) => {
|
||||||
|
const field = useFieldContext<string>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('flex flex-col gap-y-0.5', className)}>
|
||||||
|
<Label
|
||||||
|
htmlFor={field.name}
|
||||||
|
label={label}
|
||||||
|
isRequired={isRequired}
|
||||||
|
showOptional={showOptional}
|
||||||
|
tooltip={tooltip}
|
||||||
|
className={labelClassName}
|
||||||
|
/>
|
||||||
|
<PureSelect
|
||||||
|
value={field.state.value}
|
||||||
|
options={options}
|
||||||
|
onChange={value => field.handleChange(value)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default SelectField
|
||||||
@ -0,0 +1,48 @@
|
|||||||
|
import React from 'react'
|
||||||
|
import { useFieldContext } from '../..'
|
||||||
|
import Input, { type InputProps } from '../../../input'
|
||||||
|
import Label from '../label'
|
||||||
|
import cn from '@/utils/classnames'
|
||||||
|
|
||||||
|
type TextFieldProps = {
|
||||||
|
label: string
|
||||||
|
isRequired?: boolean
|
||||||
|
showOptional?: boolean
|
||||||
|
tooltip?: string
|
||||||
|
className?: string
|
||||||
|
labelClassName?: string
|
||||||
|
} & Omit<InputProps, 'className' | 'onChange' | 'onBlur' | 'value' | 'id'>
|
||||||
|
|
||||||
|
const TextField = ({
|
||||||
|
label,
|
||||||
|
isRequired,
|
||||||
|
showOptional,
|
||||||
|
tooltip,
|
||||||
|
className,
|
||||||
|
labelClassName,
|
||||||
|
...inputProps
|
||||||
|
}: TextFieldProps) => {
|
||||||
|
const field = useFieldContext<string>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('flex flex-col gap-y-0.5', className)}>
|
||||||
|
<Label
|
||||||
|
htmlFor={field.name}
|
||||||
|
label={label}
|
||||||
|
isRequired={isRequired}
|
||||||
|
showOptional={showOptional}
|
||||||
|
tooltip={tooltip}
|
||||||
|
className={labelClassName}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
id={field.name}
|
||||||
|
value={field.state.value}
|
||||||
|
onChange={e => field.handleChange(e.target.value)}
|
||||||
|
onBlur={field.handleBlur}
|
||||||
|
{...inputProps}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default TextField
|
||||||
@ -0,0 +1,25 @@
|
|||||||
|
import { useStore } from '@tanstack/react-form'
|
||||||
|
import { useFormContext } from '../..'
|
||||||
|
import Button, { type ButtonProps } from '../../../button'
|
||||||
|
|
||||||
|
type SubmitButtonProps = Omit<ButtonProps, 'disabled' | 'loading' | 'onClick'>
|
||||||
|
|
||||||
|
const SubmitButton = ({ ...buttonProps }: SubmitButtonProps) => {
|
||||||
|
const form = useFormContext()
|
||||||
|
|
||||||
|
const [isSubmitting, canSubmit] = useStore(form.store, state => [
|
||||||
|
state.isSubmitting,
|
||||||
|
state.canSubmit,
|
||||||
|
])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
disabled={isSubmitting || !canSubmit}
|
||||||
|
loading={isSubmitting}
|
||||||
|
onClick={() => form.handleSubmit()}
|
||||||
|
{...buttonProps}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default SubmitButton
|
||||||
@ -0,0 +1,53 @@
|
|||||||
|
import { fireEvent, render, screen } from '@testing-library/react'
|
||||||
|
import Label from './label'
|
||||||
|
|
||||||
|
jest.mock('react-i18next', () => ({
|
||||||
|
useTranslation: () => ({
|
||||||
|
t: (key: string) => key,
|
||||||
|
}),
|
||||||
|
}))
|
||||||
|
|
||||||
|
describe('Label Component', () => {
|
||||||
|
const defaultProps = {
|
||||||
|
htmlFor: 'test-input',
|
||||||
|
label: 'Test Label',
|
||||||
|
}
|
||||||
|
|
||||||
|
it('renders basic label correctly', () => {
|
||||||
|
render(<Label {...defaultProps} />)
|
||||||
|
const label = screen.getByTestId('label')
|
||||||
|
expect(label).toBeInTheDocument()
|
||||||
|
expect(label).toHaveAttribute('for', 'test-input')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows optional text when showOptional is true', () => {
|
||||||
|
render(<Label {...defaultProps} showOptional />)
|
||||||
|
expect(screen.getByText('common.label.optional')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows required asterisk when isRequired is true', () => {
|
||||||
|
render(<Label {...defaultProps} isRequired />)
|
||||||
|
expect(screen.getByText('*')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders tooltip when tooltip prop is provided', () => {
|
||||||
|
const tooltipText = 'Test Tooltip'
|
||||||
|
render(<Label {...defaultProps} tooltip={tooltipText} />)
|
||||||
|
const trigger = screen.getByTestId('test-input-tooltip')
|
||||||
|
fireEvent.mouseEnter(trigger)
|
||||||
|
expect(screen.getByText(tooltipText)).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies custom className when provided', () => {
|
||||||
|
const customClass = 'custom-label'
|
||||||
|
render(<Label {...defaultProps} className={customClass} />)
|
||||||
|
const label = screen.getByTestId('label')
|
||||||
|
expect(label).toHaveClass(customClass)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not show optional text and required asterisk simultaneously', () => {
|
||||||
|
render(<Label {...defaultProps} isRequired showOptional />)
|
||||||
|
expect(screen.queryByText('common.label.optional')).not.toBeInTheDocument()
|
||||||
|
expect(screen.getByText('*')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
})
|
||||||
@ -0,0 +1,48 @@
|
|||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import Tooltip from '../../tooltip'
|
||||||
|
import { useTranslation } from 'react-i18next'
|
||||||
|
|
||||||
|
export type LabelProps = {
|
||||||
|
htmlFor: string
|
||||||
|
label: string
|
||||||
|
isRequired?: boolean
|
||||||
|
showOptional?: boolean
|
||||||
|
tooltip?: string
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const Label = ({
|
||||||
|
htmlFor,
|
||||||
|
label,
|
||||||
|
isRequired,
|
||||||
|
showOptional,
|
||||||
|
tooltip,
|
||||||
|
className,
|
||||||
|
}: LabelProps) => {
|
||||||
|
const { t } = useTranslation()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className='flex h-6 items-center'>
|
||||||
|
<label
|
||||||
|
data-testid='label'
|
||||||
|
htmlFor={htmlFor}
|
||||||
|
className={cn('system-sm-medium text-text-secondary', className)}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</label>
|
||||||
|
{!isRequired && showOptional && <div className='system-xs-regular ml-1 text-text-tertiary'>{t('common.label.optional')}</div>}
|
||||||
|
{isRequired && <div className='system-xs-regular ml-1 text-text-destructive-secondary'>*</div>}
|
||||||
|
{tooltip && (
|
||||||
|
<Tooltip
|
||||||
|
popupContent={
|
||||||
|
<div className='w-[200px]'>{tooltip}</div>
|
||||||
|
}
|
||||||
|
triggerClassName='ml-0.5 w-4 h-4'
|
||||||
|
triggerTestId={`${htmlFor}-tooltip`}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Label
|
||||||
@ -0,0 +1,35 @@
|
|||||||
|
import { withForm } from '../..'
|
||||||
|
import { demoFormOpts } from './shared-options'
|
||||||
|
import { ContactMethods } from './types'
|
||||||
|
|
||||||
|
const ContactFields = withForm({
|
||||||
|
...demoFormOpts,
|
||||||
|
render: ({ form }) => {
|
||||||
|
return (
|
||||||
|
<div className='my-2'>
|
||||||
|
<h3 className='title-lg-bold text-text-primary'>Contacts</h3>
|
||||||
|
<div className='flex flex-col gap-4'>
|
||||||
|
<form.AppField
|
||||||
|
name='contact.email'
|
||||||
|
children={field => <field.TextField label='Email' />}
|
||||||
|
/>
|
||||||
|
<form.AppField
|
||||||
|
name='contact.phone'
|
||||||
|
children={field => <field.TextField label='Phone' />}
|
||||||
|
/>
|
||||||
|
<form.AppField
|
||||||
|
name='contact.preferredContactMethod'
|
||||||
|
children={field => (
|
||||||
|
<field.SelectField
|
||||||
|
label='Preferred Contact Method'
|
||||||
|
options={ContactMethods}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
export default ContactFields
|
||||||
@ -0,0 +1,68 @@
|
|||||||
|
import { useStore } from '@tanstack/react-form'
|
||||||
|
import { useAppForm } from '../..'
|
||||||
|
import ContactFields from './contact-fields'
|
||||||
|
import { demoFormOpts } from './shared-options'
|
||||||
|
import { UserSchema } from './types'
|
||||||
|
|
||||||
|
const DemoForm = () => {
|
||||||
|
const form = useAppForm({
|
||||||
|
...demoFormOpts,
|
||||||
|
validators: {
|
||||||
|
onSubmit: ({ value }) => {
|
||||||
|
// Validate the entire form
|
||||||
|
const result = UserSchema.safeParse(value)
|
||||||
|
if (!result.success) {
|
||||||
|
const issues = result.error.issues
|
||||||
|
console.log('Validation errors:', issues)
|
||||||
|
return issues[0].message
|
||||||
|
}
|
||||||
|
return undefined
|
||||||
|
},
|
||||||
|
},
|
||||||
|
onSubmit: ({ value }) => {
|
||||||
|
console.log('Form submitted:', value)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const name = useStore(form.store, state => state.values.name)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<form
|
||||||
|
className='flex w-[400px] flex-col gap-4'
|
||||||
|
onSubmit={(e) => {
|
||||||
|
e.preventDefault()
|
||||||
|
e.stopPropagation()
|
||||||
|
form.handleSubmit()
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<form.AppField
|
||||||
|
name='name'
|
||||||
|
children={field => (
|
||||||
|
<field.TextField label='Name' />
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<form.AppField
|
||||||
|
name='surname'
|
||||||
|
children={field => (
|
||||||
|
<field.TextField label='Surname' />
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<form.AppField
|
||||||
|
name='isAcceptingTerms'
|
||||||
|
children={field => (
|
||||||
|
<field.CheckboxField label='I accept the terms and conditions.' />
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
{
|
||||||
|
!!name && (
|
||||||
|
<ContactFields form={form} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
<form.AppForm>
|
||||||
|
<form.SubmitButton>Submit</form.SubmitButton>
|
||||||
|
</form.AppForm>
|
||||||
|
</form>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default DemoForm
|
||||||
@ -0,0 +1,14 @@
|
|||||||
|
import { formOptions } from '@tanstack/react-form'
|
||||||
|
|
||||||
|
export const demoFormOpts = formOptions({
|
||||||
|
defaultValues: {
|
||||||
|
name: '',
|
||||||
|
surname: '',
|
||||||
|
isAcceptingTerms: false,
|
||||||
|
contact: {
|
||||||
|
email: '',
|
||||||
|
phone: '',
|
||||||
|
preferredContactMethod: 'email',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
@ -0,0 +1,34 @@
|
|||||||
|
import { z } from 'zod'
|
||||||
|
|
||||||
|
const ContactMethod = z.union([
|
||||||
|
z.literal('email'),
|
||||||
|
z.literal('phone'),
|
||||||
|
z.literal('whatsapp'),
|
||||||
|
z.literal('sms'),
|
||||||
|
])
|
||||||
|
|
||||||
|
export const ContactMethods = ContactMethod.options.map(({ value }) => ({
|
||||||
|
value,
|
||||||
|
label: value.charAt(0).toUpperCase() + value.slice(1),
|
||||||
|
}))
|
||||||
|
|
||||||
|
export const UserSchema = z.object({
|
||||||
|
name: z
|
||||||
|
.string()
|
||||||
|
.regex(/^[A-Z]/, 'Name must start with a capital letter')
|
||||||
|
.min(3, 'Name must be at least 3 characters long'),
|
||||||
|
surname: z
|
||||||
|
.string()
|
||||||
|
.min(3, 'Surname must be at least 3 characters long')
|
||||||
|
.regex(/^[A-Z]/, 'Surname must start with a capital letter'),
|
||||||
|
isAcceptingTerms: z.boolean().refine(val => val, {
|
||||||
|
message: 'You must accept the terms and conditions',
|
||||||
|
}),
|
||||||
|
contact: z.object({
|
||||||
|
email: z.string().email('Invalid email address'),
|
||||||
|
phone: z.string().optional(),
|
||||||
|
preferredContactMethod: ContactMethod,
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
|
||||||
|
export type User = z.infer<typeof UserSchema>
|
||||||
@ -0,0 +1,25 @@
|
|||||||
|
import { createFormHook, createFormHookContexts } from '@tanstack/react-form'
|
||||||
|
import TextField from './components/field/text'
|
||||||
|
import NumberInputField from './components/field/number-input'
|
||||||
|
import CheckboxField from './components/field/checkbox'
|
||||||
|
import SelectField from './components/field/select'
|
||||||
|
import OptionsField from './components/field/options'
|
||||||
|
import SubmitButton from './components/form/submit-button'
|
||||||
|
|
||||||
|
export const { fieldContext, useFieldContext, formContext, useFormContext }
|
||||||
|
= createFormHookContexts()
|
||||||
|
|
||||||
|
export const { useAppForm, withForm } = createFormHook({
|
||||||
|
fieldComponents: {
|
||||||
|
TextField,
|
||||||
|
NumberInputField,
|
||||||
|
CheckboxField,
|
||||||
|
SelectField,
|
||||||
|
OptionsField,
|
||||||
|
},
|
||||||
|
formComponents: {
|
||||||
|
SubmitButton,
|
||||||
|
},
|
||||||
|
fieldContext,
|
||||||
|
formContext,
|
||||||
|
})
|
||||||
@ -0,0 +1,5 @@
|
|||||||
|
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<g id="arrow-down-round-fill">
|
||||||
|
<path id="Vector" d="M6.02913 6.23572C5.08582 6.23572 4.56482 7.33027 5.15967 8.06239L7.13093 10.4885C7.57922 11.0403 8.42149 11.0403 8.86986 10.4885L10.8411 8.06239C11.4359 7.33027 10.9149 6.23572 9.97158 6.23572H6.02913Z" fill="#101828"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 380 B |
@ -0,0 +1,36 @@
|
|||||||
|
{
|
||||||
|
"icon": {
|
||||||
|
"type": "element",
|
||||||
|
"isRootNode": true,
|
||||||
|
"name": "svg",
|
||||||
|
"attributes": {
|
||||||
|
"width": "16",
|
||||||
|
"height": "16",
|
||||||
|
"viewBox": "0 0 16 16",
|
||||||
|
"fill": "none",
|
||||||
|
"xmlns": "http://www.w3.org/2000/svg"
|
||||||
|
},
|
||||||
|
"children": [
|
||||||
|
{
|
||||||
|
"type": "element",
|
||||||
|
"name": "g",
|
||||||
|
"attributes": {
|
||||||
|
"id": "arrow-down-round-fill"
|
||||||
|
},
|
||||||
|
"children": [
|
||||||
|
{
|
||||||
|
"type": "element",
|
||||||
|
"name": "path",
|
||||||
|
"attributes": {
|
||||||
|
"id": "Vector",
|
||||||
|
"d": "M6.02913 6.23572C5.08582 6.23572 4.56482 7.33027 5.15967 8.06239L7.13093 10.4885C7.57922 11.0403 8.42149 11.0403 8.86986 10.4885L10.8411 8.06239C11.4359 7.33027 10.9149 6.23572 9.97158 6.23572H6.02913Z",
|
||||||
|
"fill": "currentColor"
|
||||||
|
},
|
||||||
|
"children": []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"name": "ArrowDownRoundFill"
|
||||||
|
}
|
||||||
@ -0,0 +1,20 @@
|
|||||||
|
// GENERATE BY script
|
||||||
|
// DON NOT EDIT IT MANUALLY
|
||||||
|
|
||||||
|
import * as React from 'react'
|
||||||
|
import data from './ArrowDownRoundFill.json'
|
||||||
|
import IconBase from '@/app/components/base/icons/IconBase'
|
||||||
|
import type { IconData } from '@/app/components/base/icons/IconBase'
|
||||||
|
|
||||||
|
const Icon = (
|
||||||
|
{
|
||||||
|
ref,
|
||||||
|
...props
|
||||||
|
}: React.SVGProps<SVGSVGElement> & {
|
||||||
|
ref?: React.RefObject<React.MutableRefObject<HTMLOrSVGElement>>;
|
||||||
|
},
|
||||||
|
) => <IconBase {...props} ref={ref} data={data as IconData} />
|
||||||
|
|
||||||
|
Icon.displayName = 'ArrowDownRoundFill'
|
||||||
|
|
||||||
|
export default Icon
|
||||||
@ -0,0 +1,97 @@
|
|||||||
|
import { fireEvent, render, screen } from '@testing-library/react'
|
||||||
|
import { InputNumber } from './index'
|
||||||
|
|
||||||
|
jest.mock('react-i18next', () => ({
|
||||||
|
useTranslation: () => ({
|
||||||
|
t: (key: string) => key,
|
||||||
|
}),
|
||||||
|
}))
|
||||||
|
|
||||||
|
describe('InputNumber Component', () => {
|
||||||
|
const defaultProps = {
|
||||||
|
onChange: jest.fn(),
|
||||||
|
}
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
jest.clearAllMocks()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders input with default values', () => {
|
||||||
|
render(<InputNumber {...defaultProps} />)
|
||||||
|
const input = screen.getByRole('textbox')
|
||||||
|
expect(input).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles increment button click', () => {
|
||||||
|
render(<InputNumber {...defaultProps} value={5} />)
|
||||||
|
const incrementBtn = screen.getByRole('button', { name: /increment/i })
|
||||||
|
|
||||||
|
fireEvent.click(incrementBtn)
|
||||||
|
expect(defaultProps.onChange).toHaveBeenCalledWith(6)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles decrement button click', () => {
|
||||||
|
render(<InputNumber {...defaultProps} value={5} />)
|
||||||
|
const decrementBtn = screen.getByRole('button', { name: /decrement/i })
|
||||||
|
|
||||||
|
fireEvent.click(decrementBtn)
|
||||||
|
expect(defaultProps.onChange).toHaveBeenCalledWith(4)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('respects max value constraint', () => {
|
||||||
|
render(<InputNumber {...defaultProps} value={10} max={10} />)
|
||||||
|
const incrementBtn = screen.getByRole('button', { name: /increment/i })
|
||||||
|
|
||||||
|
fireEvent.click(incrementBtn)
|
||||||
|
expect(defaultProps.onChange).not.toHaveBeenCalled()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('respects min value constraint', () => {
|
||||||
|
render(<InputNumber {...defaultProps} value={0} min={0} />)
|
||||||
|
const decrementBtn = screen.getByRole('button', { name: /decrement/i })
|
||||||
|
|
||||||
|
fireEvent.click(decrementBtn)
|
||||||
|
expect(defaultProps.onChange).not.toHaveBeenCalled()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles direct input changes', () => {
|
||||||
|
render(<InputNumber {...defaultProps} />)
|
||||||
|
const input = screen.getByRole('textbox')
|
||||||
|
|
||||||
|
fireEvent.change(input, { target: { value: '42' } })
|
||||||
|
expect(defaultProps.onChange).toHaveBeenCalledWith(42)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles empty input', () => {
|
||||||
|
render(<InputNumber {...defaultProps} value={0} />)
|
||||||
|
const input = screen.getByRole('textbox')
|
||||||
|
|
||||||
|
fireEvent.change(input, { target: { value: '' } })
|
||||||
|
expect(defaultProps.onChange).toHaveBeenCalledWith(undefined)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles invalid input', () => {
|
||||||
|
render(<InputNumber {...defaultProps} />)
|
||||||
|
const input = screen.getByRole('textbox')
|
||||||
|
|
||||||
|
fireEvent.change(input, { target: { value: 'abc' } })
|
||||||
|
expect(defaultProps.onChange).not.toHaveBeenCalled()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('displays unit when provided', () => {
|
||||||
|
const unit = 'px'
|
||||||
|
render(<InputNumber {...defaultProps} unit={unit} />)
|
||||||
|
expect(screen.getByText(unit)).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disables controls when disabled prop is true', () => {
|
||||||
|
render(<InputNumber {...defaultProps} disabled />)
|
||||||
|
const input = screen.getByRole('textbox')
|
||||||
|
const incrementBtn = screen.getByRole('button', { name: /increment/i })
|
||||||
|
const decrementBtn = screen.getByRole('button', { name: /decrement/i })
|
||||||
|
|
||||||
|
expect(input).toBeDisabled()
|
||||||
|
expect(incrementBtn).toBeDisabled()
|
||||||
|
expect(decrementBtn).toBeDisabled()
|
||||||
|
})
|
||||||
|
})
|
||||||
@ -0,0 +1,37 @@
|
|||||||
|
import abcjs from 'abcjs'
|
||||||
|
import { useEffect, useRef } from 'react'
|
||||||
|
import 'abcjs/abcjs-audio.css'
|
||||||
|
|
||||||
|
const MarkdownMusic = ({ children }: { children: React.ReactNode }) => {
|
||||||
|
const containerRef = useRef<HTMLDivElement>(null)
|
||||||
|
const controlsRef = useRef<HTMLDivElement>(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (containerRef.current && controlsRef.current) {
|
||||||
|
if (typeof children === 'string') {
|
||||||
|
const visualObjs = abcjs.renderAbc(containerRef.current, children, {
|
||||||
|
add_classes: true, // Add classes to SVG elements for cursor tracking
|
||||||
|
responsive: 'resize', // Make notation responsive
|
||||||
|
})
|
||||||
|
const synthControl = new abcjs.synth.SynthController()
|
||||||
|
synthControl.load(controlsRef.current, {}, { displayPlay: true })
|
||||||
|
const synth = new abcjs.synth.CreateSynth()
|
||||||
|
const visualObj = visualObjs[0]
|
||||||
|
synth.init({ visualObj }).then(() => {
|
||||||
|
synthControl.setTune(visualObj, false)
|
||||||
|
})
|
||||||
|
containerRef.current.style.overflow = 'auto'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [children])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div style={{ minWidth: '100%', overflow: 'auto' }}>
|
||||||
|
<div ref={containerRef} />
|
||||||
|
<div ref={controlsRef} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
MarkdownMusic.displayName = 'MarkdownMusic'
|
||||||
|
|
||||||
|
export default MarkdownMusic
|
||||||
@ -0,0 +1,68 @@
|
|||||||
|
import React from 'react'
|
||||||
|
import classNames from '@/utils/classnames'
|
||||||
|
import type { RemixiconComponentType } from '@remixicon/react'
|
||||||
|
import Divider from '../divider'
|
||||||
|
|
||||||
|
// Updated generic type to allow enum values
|
||||||
|
type SegmentedControlProps<T extends string | number | symbol> = {
|
||||||
|
options: { Icon: RemixiconComponentType, text: string, value: T }[]
|
||||||
|
value: T
|
||||||
|
onChange: (value: T) => void
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SegmentedControl = <T extends string | number | symbol>({
|
||||||
|
options,
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
className,
|
||||||
|
}: SegmentedControlProps<T>): JSX.Element => {
|
||||||
|
const selectedOptionIndex = options.findIndex(option => option.value === value)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classNames(
|
||||||
|
'flex items-center rounded-lg bg-components-segmented-control-bg-normal gap-x-[1px] p-0.5',
|
||||||
|
className,
|
||||||
|
)}>
|
||||||
|
{options.map((option, index) => {
|
||||||
|
const { Icon } = option
|
||||||
|
const isSelected = index === selectedOptionIndex
|
||||||
|
const isNextSelected = index === selectedOptionIndex - 1
|
||||||
|
const isLast = index === options.length - 1
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
type='button'
|
||||||
|
key={String(option.value)}
|
||||||
|
className={classNames(
|
||||||
|
'flex items-center justify-center relative px-2 py-1 rounded-lg gap-x-0.5 group border-0.5 border-transparent',
|
||||||
|
isSelected
|
||||||
|
? 'border-components-segmented-control-item-active-border bg-components-segmented-control-item-active-bg shadow-xs shadow-shadow-shadow-3'
|
||||||
|
: 'hover:bg-state-base-hover',
|
||||||
|
)}
|
||||||
|
onClick={() => onChange(option.value)}
|
||||||
|
>
|
||||||
|
<span className='flex h-5 w-5 items-center justify-center'>
|
||||||
|
<Icon className={classNames(
|
||||||
|
'w-4 h-4 text-text-tertiary',
|
||||||
|
isSelected ? 'text-text-accent-light-mode-only' : 'group-hover:text-text-secondary',
|
||||||
|
)} />
|
||||||
|
</span>
|
||||||
|
<span className={classNames(
|
||||||
|
'p-0.5 text-text-tertiary system-sm-medium',
|
||||||
|
isSelected ? 'text-text-accent-light-mode-only' : 'group-hover:text-text-secondary',
|
||||||
|
)}>
|
||||||
|
{option.text}
|
||||||
|
</span>
|
||||||
|
{!isLast && !isSelected && !isNextSelected && (
|
||||||
|
<div className='absolute right-[-1px] top-0 flex h-full items-center'>
|
||||||
|
<Divider type='vertical' className='mx-0 h-3.5' />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default React.memo(SegmentedControl) as typeof SegmentedControl
|
||||||
@ -0,0 +1,77 @@
|
|||||||
|
'use client'
|
||||||
|
import type { FC } from 'react'
|
||||||
|
import React from 'react'
|
||||||
|
import { Type } from '../../../../../llm/types'
|
||||||
|
import { getFieldType } from '../../../../../llm/utils'
|
||||||
|
import type { Field as FieldType } from '../../../../../llm/types'
|
||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import TreeIndentLine from '../tree-indent-line'
|
||||||
|
import { RiMoreFill } from '@remixicon/react'
|
||||||
|
import Tooltip from '@/app/components/base/tooltip'
|
||||||
|
import type { ValueSelector } from '@/app/components/workflow/types'
|
||||||
|
import { useTranslation } from 'react-i18next'
|
||||||
|
|
||||||
|
const MAX_DEPTH = 10
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
valueSelector: ValueSelector
|
||||||
|
name: string,
|
||||||
|
payload: FieldType,
|
||||||
|
depth?: number
|
||||||
|
readonly?: boolean
|
||||||
|
onSelect?: (valueSelector: ValueSelector) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const Field: FC<Props> = ({
|
||||||
|
valueSelector,
|
||||||
|
name,
|
||||||
|
payload,
|
||||||
|
depth = 1,
|
||||||
|
readonly,
|
||||||
|
onSelect,
|
||||||
|
}) => {
|
||||||
|
const { t } = useTranslation()
|
||||||
|
const isLastFieldHighlight = readonly
|
||||||
|
const hasChildren = payload.type === Type.object && payload.properties
|
||||||
|
const isHighlight = isLastFieldHighlight && !hasChildren
|
||||||
|
if (depth > MAX_DEPTH + 1)
|
||||||
|
return null
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<Tooltip popupContent={t('app.structOutput.moreFillTip')} disabled={depth !== MAX_DEPTH + 1}>
|
||||||
|
<div
|
||||||
|
className={cn('flex items-center justify-between rounded-md pr-2', !readonly && 'hover:bg-state-base-hover', depth !== MAX_DEPTH + 1 && 'cursor-pointer')}
|
||||||
|
onClick={() => !readonly && onSelect?.([...valueSelector, name])}
|
||||||
|
>
|
||||||
|
<div className='flex grow items-stretch'>
|
||||||
|
<TreeIndentLine depth={depth} />
|
||||||
|
{depth === MAX_DEPTH + 1 ? (
|
||||||
|
<RiMoreFill className='h-3 w-3 text-text-tertiary' />
|
||||||
|
) : (<div className={cn('system-sm-medium h-6 w-0 grow truncate leading-6 text-text-secondary', isHighlight && 'text-text-accent')}>{name}</div>)}
|
||||||
|
|
||||||
|
</div>
|
||||||
|
{depth < MAX_DEPTH + 1 && (
|
||||||
|
<div className='system-xs-regular ml-2 shrink-0 text-text-tertiary'>{getFieldType(payload)}</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</Tooltip>
|
||||||
|
|
||||||
|
{depth <= MAX_DEPTH && payload.type === Type.object && payload.properties && (
|
||||||
|
<div>
|
||||||
|
{Object.keys(payload.properties).map(propName => (
|
||||||
|
<Field
|
||||||
|
key={propName}
|
||||||
|
name={propName}
|
||||||
|
payload={payload.properties?.[propName] as FieldType}
|
||||||
|
depth={depth + 1}
|
||||||
|
readonly={readonly}
|
||||||
|
valueSelector={[...valueSelector, name]}
|
||||||
|
onSelect={onSelect}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default React.memo(Field)
|
||||||
@ -0,0 +1,82 @@
|
|||||||
|
'use client'
|
||||||
|
import type { FC } from 'react'
|
||||||
|
import React, { useRef } from 'react'
|
||||||
|
import type { StructuredOutput } from '../../../../../llm/types'
|
||||||
|
import Field from './field'
|
||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import { useHover } from 'ahooks'
|
||||||
|
import type { ValueSelector } from '@/app/components/workflow/types'
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
className?: string
|
||||||
|
root: { nodeId?: string, nodeName?: string, attrName: string }
|
||||||
|
payload: StructuredOutput
|
||||||
|
readonly?: boolean
|
||||||
|
onSelect?: (valueSelector: ValueSelector) => void
|
||||||
|
onHovering?: (value: boolean) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export const PickerPanelMain: FC<Props> = ({
|
||||||
|
className,
|
||||||
|
root,
|
||||||
|
payload,
|
||||||
|
readonly,
|
||||||
|
onHovering,
|
||||||
|
onSelect,
|
||||||
|
}) => {
|
||||||
|
const ref = useRef<HTMLDivElement>(null)
|
||||||
|
useHover(ref, {
|
||||||
|
onChange: (hovering) => {
|
||||||
|
if (hovering) {
|
||||||
|
onHovering?.(true)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
setTimeout(() => {
|
||||||
|
onHovering?.(false)
|
||||||
|
}, 100)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const schema = payload.schema
|
||||||
|
const fieldNames = Object.keys(schema.properties)
|
||||||
|
return (
|
||||||
|
<div className={cn(className)} ref={ref}>
|
||||||
|
{/* Root info */}
|
||||||
|
<div className='flex items-center justify-between px-2 py-1'>
|
||||||
|
<div className='flex'>
|
||||||
|
{root.nodeName && (
|
||||||
|
<>
|
||||||
|
<div className='system-sm-medium max-w-[100px] truncate text-text-tertiary'>{root.nodeName}</div>
|
||||||
|
<div className='system-sm-medium text-text-tertiary'>.</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
<div className='system-sm-medium text-text-secondary'>{root.attrName}</div>
|
||||||
|
</div>
|
||||||
|
{/* It must be object */}
|
||||||
|
<div className='system-xs-regular ml-2 shrink-0 text-text-tertiary'>object</div>
|
||||||
|
</div>
|
||||||
|
{fieldNames.map(name => (
|
||||||
|
<Field
|
||||||
|
key={name}
|
||||||
|
name={name}
|
||||||
|
payload={schema.properties[name]}
|
||||||
|
readonly={readonly}
|
||||||
|
valueSelector={[root.nodeId!, root.attrName]}
|
||||||
|
onSelect={onSelect}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const PickerPanel: FC<Props> = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}) => {
|
||||||
|
return (
|
||||||
|
<div className={cn('w-[296px] rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur p-1 pb-0 shadow-lg backdrop-blur-[5px]', className)}>
|
||||||
|
<PickerPanelMain {...props} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default React.memo(PickerPanel)
|
||||||
@ -0,0 +1,74 @@
|
|||||||
|
'use client'
|
||||||
|
import type { FC } from 'react'
|
||||||
|
import React from 'react'
|
||||||
|
import { Type } from '../../../../../llm/types'
|
||||||
|
import { getFieldType } from '../../../../../llm/utils'
|
||||||
|
import type { Field as FieldType } from '../../../../../llm/types'
|
||||||
|
import cn from '@/utils/classnames'
|
||||||
|
import TreeIndentLine from '../tree-indent-line'
|
||||||
|
import { useTranslation } from 'react-i18next'
|
||||||
|
import { useBoolean } from 'ahooks'
|
||||||
|
import { RiArrowDropDownLine } from '@remixicon/react'
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
name: string,
|
||||||
|
payload: FieldType,
|
||||||
|
required: boolean,
|
||||||
|
depth?: number,
|
||||||
|
rootClassName?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const Field: FC<Props> = ({
|
||||||
|
name,
|
||||||
|
payload,
|
||||||
|
depth = 1,
|
||||||
|
required,
|
||||||
|
rootClassName,
|
||||||
|
}) => {
|
||||||
|
const { t } = useTranslation()
|
||||||
|
const isRoot = depth === 1
|
||||||
|
const hasChildren = payload.type === Type.object && payload.properties
|
||||||
|
const [fold, {
|
||||||
|
toggle: toggleFold,
|
||||||
|
}] = useBoolean(false)
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<div className={cn('flex pr-2')}>
|
||||||
|
<TreeIndentLine depth={depth} />
|
||||||
|
<div className='w-0 grow'>
|
||||||
|
<div className='relative flex select-none'>
|
||||||
|
{hasChildren && (
|
||||||
|
<RiArrowDropDownLine
|
||||||
|
className={cn('absolute left-[-18px] top-[50%] h-4 w-4 translate-y-[-50%] cursor-pointer bg-components-panel-bg text-text-tertiary', fold && 'rotate-[270deg] text-text-accent')}
|
||||||
|
onClick={toggleFold}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<div className={cn('system-sm-medium ml-[7px] h-6 truncate leading-6 text-text-secondary', isRoot && rootClassName)}>{name}</div>
|
||||||
|
<div className='system-xs-regular ml-3 shrink-0 leading-6 text-text-tertiary'>{getFieldType(payload)}</div>
|
||||||
|
{required && <div className='system-2xs-medium-uppercase ml-3 leading-6 text-text-warning'>{t('app.structOutput.required')}</div>}
|
||||||
|
</div>
|
||||||
|
{payload.description && (
|
||||||
|
<div className='ml-[7px] flex'>
|
||||||
|
<div className='system-xs-regular w-0 grow truncate text-text-tertiary'>{payload.description}</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{hasChildren && !fold && (
|
||||||
|
<div>
|
||||||
|
{Object.keys(payload.properties!).map(name => (
|
||||||
|
<Field
|
||||||
|
key={name}
|
||||||
|
name={name}
|
||||||
|
payload={payload.properties?.[name] as FieldType}
|
||||||
|
depth={depth + 1}
|
||||||
|
required={!!payload.required?.includes(name)}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default React.memo(Field)
|
||||||
@ -0,0 +1,39 @@
|
|||||||
|
'use client'
|
||||||
|
import type { FC } from 'react'
|
||||||
|
import React from 'react'
|
||||||
|
import type { StructuredOutput } from '../../../../../llm/types'
|
||||||
|
import Field from './field'
|
||||||
|
import { useTranslation } from 'react-i18next'
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
payload: StructuredOutput
|
||||||
|
rootClassName?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const ShowPanel: FC<Props> = ({
|
||||||
|
payload,
|
||||||
|
rootClassName,
|
||||||
|
}) => {
|
||||||
|
const { t } = useTranslation()
|
||||||
|
const schema = {
|
||||||
|
...payload,
|
||||||
|
schema: {
|
||||||
|
...payload.schema,
|
||||||
|
description: t('app.structOutput.LLMResponse'),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<div className='relative left-[-7px]'>
|
||||||
|
{Object.keys(schema.schema.properties!).map(name => (
|
||||||
|
<Field
|
||||||
|
key={name}
|
||||||
|
name={name}
|
||||||
|
payload={schema.schema.properties![name]}
|
||||||
|
required={!!schema.schema.required?.includes(name)}
|
||||||
|
rootClassName={rootClassName}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default React.memo(ShowPanel)
|
||||||
@ -0,0 +1,24 @@
|
|||||||
|
'use client'
|
||||||
|
import type { FC } from 'react'
|
||||||
|
import React from 'react'
|
||||||
|
import cn from '@/utils/classnames'
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
depth?: number,
|
||||||
|
className?: string,
|
||||||
|
}
|
||||||
|
|
||||||
|
const TreeIndentLine: FC<Props> = ({
|
||||||
|
depth = 1,
|
||||||
|
className,
|
||||||
|
}) => {
|
||||||
|
const depthArray = Array.from({ length: depth }, (_, index) => index)
|
||||||
|
return (
|
||||||
|
<div className={cn('flex', className)}>
|
||||||
|
{depthArray.map(d => (
|
||||||
|
<div key={d} className={cn('ml-2.5 mr-2.5 w-px bg-divider-regular')}></div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default React.memo(TreeIndentLine)
|
||||||
@ -0,0 +1,59 @@
|
|||||||
|
'use client'
|
||||||
|
import type { FC } from 'react'
|
||||||
|
import React from 'react'
|
||||||
|
import type { Field, StructuredOutput, TypeWithArray } from '../../../llm/types'
|
||||||
|
import { Type } from '../../../llm/types'
|
||||||
|
import { PickerPanelMain as Panel } from '@/app/components/workflow/nodes/_base/components/variable/object-child-tree-panel/picker'
|
||||||
|
import BlockIcon from '@/app/components/workflow/block-icon'
|
||||||
|
import { BlockEnum } from '@/app/components/workflow/types'
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
nodeName: string
|
||||||
|
path: string[]
|
||||||
|
varType: TypeWithArray
|
||||||
|
nodeType?: BlockEnum
|
||||||
|
}
|
||||||
|
|
||||||
|
const VarFullPathPanel: FC<Props> = ({
|
||||||
|
nodeName,
|
||||||
|
path,
|
||||||
|
varType,
|
||||||
|
nodeType = BlockEnum.LLM,
|
||||||
|
}) => {
|
||||||
|
const schema: StructuredOutput = (() => {
|
||||||
|
const schema: StructuredOutput['schema'] = {
|
||||||
|
type: Type.object,
|
||||||
|
properties: {} as { [key: string]: Field },
|
||||||
|
required: [],
|
||||||
|
additionalProperties: false,
|
||||||
|
}
|
||||||
|
let current = schema
|
||||||
|
for (let i = 1; i < path.length; i++) {
|
||||||
|
const isLast = i === path.length - 1
|
||||||
|
const name = path[i]
|
||||||
|
current.properties[name] = {
|
||||||
|
type: isLast ? varType : Type.object,
|
||||||
|
properties: {},
|
||||||
|
} as Field
|
||||||
|
current = current.properties[name] as { type: Type.object; properties: { [key: string]: Field; }; required: never[]; additionalProperties: false; }
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
schema,
|
||||||
|
}
|
||||||
|
})()
|
||||||
|
return (
|
||||||
|
<div className='w-[280px] rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur pb-0 shadow-lg backdrop-blur-[5px]'>
|
||||||
|
<div className='flex space-x-1 border-b-[0.5px] border-divider-subtle p-3 pb-2 '>
|
||||||
|
<BlockIcon size='xs' type={nodeType} />
|
||||||
|
<div className='system-xs-medium w-0 grow truncate text-text-secondary'>{nodeName}</div>
|
||||||
|
</div>
|
||||||
|
<Panel
|
||||||
|
className='px-1 pb-3 pt-2'
|
||||||
|
root={{ attrName: path[0] }}
|
||||||
|
payload={schema}
|
||||||
|
readonly
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
export default React.memo(VarFullPathPanel)
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue