dify/web/app/components/tools/workflow-tool/method-selector.spec.tsx
qiuqiua 9ef6b90843
feat: sync main branch (#31938)
Signed-off-by: majiayu000 <1835304752@qq.com>
Signed-off-by: dependabot[bot] <support@github.com>
Signed-off-by: NeatGuyCoding <15627489+NeatGuyCoding@users.noreply.github.com>
Signed-off-by: -LAN- <laipz8200@outlook.com>
Signed-off-by: yihong0618 <zouzou0208@gmail.com>
Co-authored-by: QuantumGhost <obelisk.reg+git@gmail.com>
Co-authored-by: 盐粒 Yanli <yanli@dify.ai>
Co-authored-by: wangxiaolei <fatelei@gmail.com>
Co-authored-by: Stephen Zhou <38493346+hyoban@users.noreply.github.com>
Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com>
Co-authored-by: Cursx <33718736+Cursx@users.noreply.github.com>
Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com>
Co-authored-by: lif <1835304752@qq.com>
Co-authored-by: 非法操作 <hjlarry@163.com>
Co-authored-by: Asuka Minato <i@asukaminato.eu.org>
Co-authored-by: fenglin <790872612@qq.com>
Co-authored-by: qiaofenglin <qiaofenglin@baidu.com>
Co-authored-by: -LAN- <laipz8200@outlook.com>
Co-authored-by: TomoOkuyama <49631611+TomoOkuyama@users.noreply.github.com>
Co-authored-by: Tomo Okuyama <tomo.okuyama@intersystems.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: zyssyz123 <916125788@qq.com>
Co-authored-by: hj24 <mambahj24@gmail.com>
Co-authored-by: Coding On Star <447357187@qq.com>
Co-authored-by: CodingOnStar <hanxujiang@dify.ai>
Co-authored-by: yyh <92089059+lyzno1@users.noreply.github.com>
Co-authored-by: Xiangxuan Qu <fghpdf@outlook.com>
Co-authored-by: fghpdf <fghpdf@users.noreply.github.com>
Co-authored-by: coopercoder <whitetiger0127@163.com>
Co-authored-by: zhaiguangpeng <zhaiguangpeng@didiglobal.com>
Co-authored-by: Junyan Qin (Chin) <rockchinq@gmail.com>
Co-authored-by: E.G <146701565+GlobalStar117@users.noreply.github.com>
Co-authored-by: GlobalStar117 <GlobalStar117@users.noreply.github.com>
Co-authored-by: Claude Haiku 4.5 <noreply@anthropic.com>
Co-authored-by: CodingOnStar <hanxujiang@dify.com>
Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com>
Co-authored-by: heyszt <270985384@qq.com>
Co-authored-by: NeatGuyCoding <15627489+NeatGuyCoding@users.noreply.github.com>
Co-authored-by: Yeuoly <45712896+Yeuoly@users.noreply.github.com>
Co-authored-by: zxhlyh <jasonapring2015@outlook.com>
Co-authored-by: moonpanda <chuanzegao@163.com>
Co-authored-by: warlocgao <warlocgao@tencent.com>
Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
Co-authored-by: claude[bot] <41898282+claude[bot]@users.noreply.github.com>
Co-authored-by: KVOJJJin <jzongcode@gmail.com>
Co-authored-by: eux <euxx@users.noreply.github.com>
Co-authored-by: bangjiehan <bangjiehan@gmail.com>
Co-authored-by: FFXN <31929997+FFXN@users.noreply.github.com>
Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com>
Co-authored-by: Nie Ronghua <nieronghua@sf-express.com>
Co-authored-by: JQSevenMiao <141806521+JQSevenMiao@users.noreply.github.com>
Co-authored-by: jiasiqi <jiasiqi3@tal.com>
Co-authored-by: Seokrin Taron Sung <sungsjade@gmail.com>
Co-authored-by: CrabSAMA <40541269+CrabSAMA@users.noreply.github.com>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
Co-authored-by: yihong <zouzou0208@gmail.com>
Co-authored-by: Joel <iamjoel007@gmail.com>
Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com>
Co-authored-by: yessenia <yessenia.contact@gmail.com>
Co-authored-by: Jax <anobaka@qq.com>
Co-authored-by: niveshdandyan <155956228+niveshdandyan@users.noreply.github.com>
Co-authored-by: OSS Contributor <oss-contributor@example.com>
Co-authored-by: niveshdandyan <niveshdandyan@users.noreply.github.com>
Co-authored-by: Sean Kenneth Doherty <Smaster7772@gmail.com>
2026-02-04 19:04:24 +08:00

318 lines
10 KiB
TypeScript

import type { ComponentProps } from 'react'
import { render, screen, waitFor } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { beforeEach, describe, expect, it, vi } from 'vitest'
import MethodSelector from './method-selector'
// Test utilities
const defaultProps: ComponentProps<typeof MethodSelector> = {
value: 'llm',
onChange: vi.fn(),
}
const renderComponent = (props: Partial<ComponentProps<typeof MethodSelector>> = {}) => {
const mergedProps = { ...defaultProps, ...props }
return render(<MethodSelector {...mergedProps} />)
}
describe('MethodSelector', () => {
beforeEach(() => {
vi.clearAllMocks()
})
// Rendering tests
describe('Rendering', () => {
it('should render without crashing', () => {
renderComponent()
// Should display the current method text
expect(screen.getByText('tools.createTool.toolInput.methodParameter')).toBeInTheDocument()
})
it('should render with llm value selected', () => {
renderComponent({ value: 'llm' })
expect(screen.getByText('tools.createTool.toolInput.methodParameter')).toBeInTheDocument()
})
it('should render with form value selected', () => {
renderComponent({ value: 'form' })
expect(screen.getByText('tools.createTool.toolInput.methodSetting')).toBeInTheDocument()
})
it('should render with undefined value', () => {
renderComponent({ value: undefined })
// When value is undefined, it should show the form method text (else branch)
expect(screen.getByText('tools.createTool.toolInput.methodSetting')).toBeInTheDocument()
})
it('should render arrow down icon', () => {
renderComponent()
// The arrow icon is rendered with remixicon
const arrowIcon = document.querySelector('.remixicon')
expect(arrowIcon).toBeInTheDocument()
})
})
// Props tests
describe('Props', () => {
it('should display methodParameter when value is llm', () => {
renderComponent({ value: 'llm' })
expect(screen.getByText('tools.createTool.toolInput.methodParameter')).toBeInTheDocument()
})
it('should display methodSetting when value is form', () => {
renderComponent({ value: 'form' })
expect(screen.getByText('tools.createTool.toolInput.methodSetting')).toBeInTheDocument()
})
it('should handle empty string value as non-llm', () => {
renderComponent({ value: '' })
expect(screen.getByText('tools.createTool.toolInput.methodSetting')).toBeInTheDocument()
})
})
// User Interactions
describe('User Interactions', () => {
it('should open dropdown when trigger is clicked', async () => {
const user = userEvent.setup()
renderComponent()
// Click the trigger to open dropdown
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
// Dropdown should now show both options with tips
await waitFor(() => {
expect(screen.getByText('tools.createTool.toolInput.methodParameterTip')).toBeInTheDocument()
expect(screen.getByText('tools.createTool.toolInput.methodSettingTip')).toBeInTheDocument()
})
})
it('should call onChange with llm when llm option is clicked', async () => {
const user = userEvent.setup()
const onChange = vi.fn()
renderComponent({ value: 'form', onChange })
// Open dropdown
const trigger = screen.getByText('tools.createTool.toolInput.methodSetting')
await user.click(trigger)
// Wait for dropdown to open
await waitFor(() => {
expect(screen.getByText('tools.createTool.toolInput.methodParameterTip')).toBeInTheDocument()
})
// Click the llm option (by finding the method parameter option in dropdown)
const llmOption = screen.getAllByText('tools.createTool.toolInput.methodParameter')[0]
await user.click(llmOption)
expect(onChange).toHaveBeenCalledWith('llm')
})
it('should call onChange with form when form option is clicked', async () => {
const user = userEvent.setup()
const onChange = vi.fn()
renderComponent({ value: 'llm', onChange })
// Open dropdown
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
// Wait for dropdown to open
await waitFor(() => {
expect(screen.getByText('tools.createTool.toolInput.methodSettingTip')).toBeInTheDocument()
})
// Click the form option (by finding the method setting option in dropdown)
const formOption = screen.getAllByText('tools.createTool.toolInput.methodSetting')[0]
await user.click(formOption)
expect(onChange).toHaveBeenCalledWith('form')
})
it('should toggle dropdown open state', async () => {
const user = userEvent.setup()
renderComponent()
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
// First click - open
await user.click(trigger)
await waitFor(() => {
expect(screen.getByText('tools.createTool.toolInput.methodParameterTip')).toBeInTheDocument()
})
// Second click - close
await user.click(trigger)
await waitFor(() => {
expect(screen.queryByText('tools.createTool.toolInput.methodParameterTip')).not.toBeInTheDocument()
})
})
})
// Styling tests
describe('Styling', () => {
it('should apply hover styles to trigger', () => {
renderComponent()
const trigger = document.querySelector('.hover\\:bg-background-section-burn')
expect(trigger).toBeInTheDocument()
})
it('should apply open state styles when dropdown is open', async () => {
const user = userEvent.setup()
renderComponent()
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
await waitFor(() => {
const openTrigger = document.querySelector('.\\!bg-background-section-burn')
expect(openTrigger).toBeInTheDocument()
})
})
it('should show checkmark for selected llm option', async () => {
const user = userEvent.setup()
renderComponent({ value: 'llm' })
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
await waitFor(() => {
// Check icon should be visible for llm option
const checkIcon = document.querySelector('.text-text-accent')
expect(checkIcon).toBeInTheDocument()
})
})
it('should show checkmark for selected form option', async () => {
const user = userEvent.setup()
renderComponent({ value: 'form' })
const trigger = screen.getByText('tools.createTool.toolInput.methodSetting')
await user.click(trigger)
await waitFor(() => {
// Check icon should be visible for form option
const checkIcon = document.querySelector('.text-text-accent')
expect(checkIcon).toBeInTheDocument()
})
})
})
// Dropdown content tests
describe('Dropdown Content', () => {
it('should render both method options in dropdown', async () => {
const user = userEvent.setup()
renderComponent()
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
await waitFor(() => {
// Should show both option titles and descriptions
expect(screen.getByText('tools.createTool.toolInput.methodParameterTip')).toBeInTheDocument()
expect(screen.getByText('tools.createTool.toolInput.methodSettingTip')).toBeInTheDocument()
})
})
it('should have proper dropdown styling', async () => {
const user = userEvent.setup()
renderComponent()
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
await waitFor(() => {
const dropdown = document.querySelector('.w-\\[320px\\]')
expect(dropdown).toBeInTheDocument()
expect(dropdown).toHaveClass('rounded-lg')
expect(dropdown).toHaveClass('shadow-lg')
})
})
it('should have hover styles on dropdown options', async () => {
const user = userEvent.setup()
renderComponent()
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
await waitFor(() => {
const options = document.querySelectorAll('.hover\\:bg-components-panel-on-panel-item-bg-hover')
expect(options.length).toBeGreaterThanOrEqual(2)
})
})
})
// Edge Cases
describe('Edge Cases', () => {
it('should handle rapid clicks on trigger', async () => {
const user = userEvent.setup()
renderComponent()
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
// Rapid clicks
await user.click(trigger)
await user.click(trigger)
await user.click(trigger)
// Should not crash and should be in a consistent state
expect(trigger).toBeInTheDocument()
})
it('should handle selecting the already selected value', async () => {
const user = userEvent.setup()
const onChange = vi.fn()
renderComponent({ value: 'llm', onChange })
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
await waitFor(() => {
expect(screen.getByText('tools.createTool.toolInput.methodParameterTip')).toBeInTheDocument()
})
// Click the llm option in the dropdown (the one with the tip text nearby)
const llmOptionContainer = screen.getByText('tools.createTool.toolInput.methodParameterTip').closest('.cursor-pointer')
expect(llmOptionContainer).toBeInTheDocument()
await user.click(llmOptionContainer!)
// Should call onChange
expect(onChange).toHaveBeenCalledWith('llm')
})
})
// Accessibility
describe('Accessibility', () => {
it('should have clickable trigger area', () => {
renderComponent()
const trigger = document.querySelector('.cursor-pointer')
expect(trigger).toBeInTheDocument()
})
it('should have clickable dropdown options', async () => {
const user = userEvent.setup()
renderComponent()
const trigger = screen.getByText('tools.createTool.toolInput.methodParameter')
await user.click(trigger)
await waitFor(() => {
const options = document.querySelectorAll('.cursor-pointer')
expect(options.length).toBeGreaterThanOrEqual(2)
})
})
})
})