OrderButtonBar.test.tsx 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import { render, fireEvent, waitFor } from '@testing-library/react'
  2. import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
  3. import { mockShowModal, mockShowToast, mockGetNetworkType } from '~/__mocks__/taroMock'
  4. import OrderButtonBar from '@/components/order/OrderButtonBar'
  5. // Mock specific React Query hooks
  6. jest.mock('@tanstack/react-query', () => {
  7. const originalModule = jest.requireActual('@tanstack/react-query')
  8. return {
  9. ...originalModule,
  10. useMutation: jest.fn(() => ({
  11. mutate: jest.fn(),
  12. mutateAsync: jest.fn(),
  13. isLoading: false,
  14. isError: false,
  15. isSuccess: false,
  16. error: null,
  17. data: null
  18. })),
  19. useQueryClient: jest.fn(() => ({
  20. invalidateQueries: jest.fn()
  21. }))
  22. }
  23. })
  24. // Mock API client
  25. jest.mock('@/api', () => ({
  26. orderClient: {
  27. cancelOrder: {
  28. $post: jest.fn()
  29. }
  30. }
  31. }))
  32. const mockOrder = {
  33. id: 1,
  34. orderNo: 'ORDER001',
  35. payState: 0, // 未支付
  36. state: 0, // 未发货
  37. amount: 100,
  38. payAmount: 100,
  39. freightAmount: 0,
  40. discountAmount: 0,
  41. goodsDetail: JSON.stringify([
  42. { id: 1, name: '商品1', price: 50, num: 2, image: '', spec: '默认规格' }
  43. ]),
  44. recevierName: '张三',
  45. receiverMobile: '13800138000',
  46. address: '北京市朝阳区',
  47. createdAt: '2025-01-01T00:00:00Z'
  48. }
  49. const createTestQueryClient = () => new QueryClient({
  50. defaultOptions: {
  51. queries: { retry: false },
  52. mutations: { retry: false }
  53. }
  54. })
  55. const TestWrapper = ({ children }: { children: React.ReactNode }) => (
  56. <QueryClientProvider client={createTestQueryClient()}>
  57. {children}
  58. </QueryClientProvider>
  59. )
  60. describe('OrderButtonBar', () => {
  61. beforeEach(() => {
  62. jest.clearAllMocks()
  63. mockGetNetworkType.mockResolvedValue({ networkType: 'wifi' })
  64. })
  65. it('should render cancel button for unpaid order', () => {
  66. const { getByText } = render(
  67. <TestWrapper>
  68. <OrderButtonBar order={mockOrder} onViewDetail={jest.fn()} />
  69. </TestWrapper>
  70. )
  71. expect(getByText('取消订单')).toBeTruthy()
  72. expect(getByText('去支付')).toBeTruthy()
  73. expect(getByText('查看详情')).toBeTruthy()
  74. })
  75. it('should show cancel reason dialog when cancel button is clicked', async () => {
  76. mockGetNetworkType.mockResolvedValue({ networkType: 'wifi' })
  77. mockShowModal.mockResolvedValue({ confirm: true, content: '测试取消原因' })
  78. const { getByText } = render(
  79. <TestWrapper>
  80. <OrderButtonBar order={mockOrder} onViewDetail={jest.fn()} />
  81. </TestWrapper>
  82. )
  83. fireEvent.click(getByText('取消订单'))
  84. await waitFor(() => {
  85. expect(mockShowModal).toHaveBeenCalledWith({
  86. title: '取消订单',
  87. content: '请填写取消原因:',
  88. editable: true,
  89. placeholderText: '请输入取消原因(必填)'
  90. })
  91. })
  92. })
  93. it('should call API when cancel order is confirmed', async () => {
  94. const mockApiCall = require('@/api').orderClient.cancelOrder.$post as jest.Mock
  95. mockGetNetworkType.mockResolvedValue({ networkType: 'wifi' })
  96. mockShowModal
  97. .mockResolvedValueOnce({ confirm: true, content: '测试取消原因' }) // 原因输入
  98. .mockResolvedValueOnce({ confirm: true }) // 确认取消
  99. mockApiCall.mockResolvedValue({ status: 200, json: () => Promise.resolve({ success: true, message: '取消成功' }) })
  100. const { getByText } = render(
  101. <TestWrapper>
  102. <OrderButtonBar order={mockOrder} onViewDetail={jest.fn()} />
  103. </TestWrapper>
  104. )
  105. fireEvent.click(getByText('取消订单'))
  106. await waitFor(() => {
  107. expect(mockApiCall).toHaveBeenCalledWith({
  108. json: {
  109. orderId: 1,
  110. reason: '测试取消原因'
  111. }
  112. })
  113. })
  114. })
  115. it('should show error when cancel reason is empty', async () => {
  116. mockGetNetworkType.mockResolvedValue({ networkType: 'wifi' })
  117. mockShowModal.mockResolvedValue({ confirm: true, content: '' })
  118. const { getByText } = render(
  119. <TestWrapper>
  120. <OrderButtonBar order={mockOrder} onViewDetail={jest.fn()} />
  121. </TestWrapper>
  122. )
  123. fireEvent.click(getByText('取消订单'))
  124. await waitFor(() => {
  125. expect(mockShowToast).toHaveBeenCalledWith({
  126. title: '请填写取消原因',
  127. icon: 'error',
  128. duration: 2000
  129. })
  130. })
  131. })
  132. it('should handle network error gracefully', async () => {
  133. const mockApiCall = require('@/api').orderClient.cancelOrder.$post as jest.Mock
  134. mockGetNetworkType.mockResolvedValue({ networkType: 'wifi' })
  135. mockShowModal
  136. .mockResolvedValueOnce({ confirm: true, content: '测试取消原因' })
  137. .mockResolvedValueOnce({ confirm: true })
  138. mockApiCall.mockRejectedValue(new Error('网络连接失败'))
  139. const { getByText } = render(
  140. <TestWrapper>
  141. <OrderButtonBar order={mockOrder} onViewDetail={jest.fn()} />
  142. </TestWrapper>
  143. )
  144. fireEvent.click(getByText('取消订单'))
  145. await waitFor(() => {
  146. expect(mockShowToast).toHaveBeenCalledWith({
  147. title: '网络连接失败,请检查网络后重试',
  148. icon: 'error',
  149. duration: 3000
  150. })
  151. })
  152. })
  153. it('should disable cancel button during mutation', async () => {
  154. const mockApiCall = require('@/api').orderClient.cancelOrder.$post as jest.Mock
  155. mockGetNetworkType.mockResolvedValue({ networkType: 'wifi' })
  156. mockShowModal
  157. .mockResolvedValueOnce({ confirm: true, content: '测试取消原因' })
  158. .mockResolvedValueOnce({ confirm: true })
  159. // 模拟API调用延迟
  160. mockApiCall.mockImplementation(() => new Promise(resolve => {
  161. setTimeout(() => resolve({ status: 200, json: () => Promise.resolve({ success: true }) }), 100)
  162. }))
  163. const { getByText } = render(
  164. <TestWrapper>
  165. <OrderButtonBar order={mockOrder} onViewDetail={jest.fn()} />
  166. </TestWrapper>
  167. )
  168. fireEvent.click(getByText('取消订单'))
  169. await waitFor(() => {
  170. expect(getByText('取消中...')).toBeTruthy()
  171. })
  172. })
  173. it('should not show cancel button for shipped order', () => {
  174. const shippedOrder = { ...mockOrder, state: 1 } // 已发货
  175. const { queryByText } = render(
  176. <TestWrapper>
  177. <OrderButtonBar order={shippedOrder} onViewDetail={jest.fn()} />
  178. </TestWrapper>
  179. )
  180. expect(queryByText('取消订单')).toBeNull()
  181. expect(queryByText('确认收货')).toBeTruthy()
  182. })
  183. it('should use external cancel handler when provided', async () => {
  184. const mockOnCancelOrder = jest.fn()
  185. mockGetNetworkType.mockResolvedValue({ networkType: 'wifi' })
  186. const { getByText } = render(
  187. <TestWrapper>
  188. <OrderButtonBar
  189. order={mockOrder}
  190. onViewDetail={jest.fn()}
  191. onCancelOrder={mockOnCancelOrder}
  192. />
  193. </TestWrapper>
  194. )
  195. fireEvent.click(getByText('取消订单'))
  196. await waitFor(() => {
  197. expect(mockOnCancelOrder).toHaveBeenCalled()
  198. })
  199. })
  200. })