app-picker.tsx 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React, { useMemo } from 'react'
  4. import { useState } from 'react'
  5. import {
  6. PortalToFollowElem,
  7. PortalToFollowElemContent,
  8. PortalToFollowElemTrigger,
  9. } from '@/app/components/base/portal-to-follow-elem'
  10. import type {
  11. OffsetOptions,
  12. Placement,
  13. } from '@floating-ui/react'
  14. import Input from '@/app/components/base/input'
  15. import AppIcon from '@/app/components/base/app-icon'
  16. import type { App } from '@/types/app'
  17. type Props = {
  18. appList: App[]
  19. scope: string
  20. disabled: boolean
  21. trigger: React.ReactNode
  22. placement?: Placement
  23. offset?: OffsetOptions
  24. isShow: boolean
  25. onShowChange: (isShow: boolean) => void
  26. onSelect: (app: App) => void
  27. }
  28. const AppPicker: FC<Props> = ({
  29. scope,
  30. appList,
  31. disabled,
  32. trigger,
  33. placement = 'right-start',
  34. offset = 0,
  35. isShow,
  36. onShowChange,
  37. onSelect,
  38. }) => {
  39. const [searchText, setSearchText] = useState('')
  40. const filteredAppList = useMemo(() => {
  41. return (appList || [])
  42. .filter(app => app.name.toLowerCase().includes(searchText.toLowerCase()))
  43. .filter(app => (app.mode !== 'advanced-chat' && app.mode !== 'workflow') || !!app.workflow)
  44. .filter(app => scope === 'all'
  45. || (scope === 'completion' && app.mode === 'completion')
  46. || (scope === 'workflow' && app.mode === 'workflow')
  47. || (scope === 'chat' && app.mode === 'advanced-chat')
  48. || (scope === 'chat' && app.mode === 'agent-chat')
  49. || (scope === 'chat' && app.mode === 'chat'))
  50. }, [appList, scope, searchText])
  51. const getAppType = (app: App) => {
  52. switch (app.mode) {
  53. case 'advanced-chat':
  54. return 'chatflow'
  55. case 'agent-chat':
  56. return 'agent'
  57. case 'chat':
  58. return 'chat'
  59. case 'completion':
  60. return 'completion'
  61. case 'workflow':
  62. return 'workflow'
  63. }
  64. }
  65. const handleTriggerClick = () => {
  66. if (disabled) return
  67. onShowChange(true)
  68. }
  69. return (
  70. <PortalToFollowElem
  71. placement={placement}
  72. offset={offset}
  73. open={isShow}
  74. onOpenChange={onShowChange}
  75. >
  76. <PortalToFollowElemTrigger
  77. onClick={handleTriggerClick}
  78. >
  79. {trigger}
  80. </PortalToFollowElemTrigger>
  81. <PortalToFollowElemContent className='z-[1000]'>
  82. <div className="relative w-[356px] min-h-20 rounded-xl backdrop-blur-sm bg-components-panel-bg-blur border-[0.5px] border-components-panel-border shadow-lg">
  83. <div className='p-2 pb-1'>
  84. <Input
  85. showLeftIcon
  86. showClearIcon
  87. value={searchText}
  88. onChange={e => setSearchText(e.target.value)}
  89. onClear={() => setSearchText('')}
  90. />
  91. </div>
  92. <div className='p-1'>
  93. {filteredAppList.map(app => (
  94. <div
  95. key={app.id}
  96. className='flex items-center gap-3 py-1 pl-2 pr-3 rounded-lg hover:bg-state-base-hover cursor-pointer'
  97. onClick={() => onSelect(app)}
  98. >
  99. <AppIcon
  100. className='shrink-0'
  101. size='xs'
  102. iconType={app.icon_type}
  103. icon={app.icon}
  104. background={app.icon_background}
  105. imageUrl={app.icon_url}
  106. />
  107. <div title={app.name} className='grow system-sm-medium text-components-input-text-filled'>{app.name}</div>
  108. <div className='shrink-0 text-text-tertiary system-2xs-medium-uppercase'>{getAppType(app)}</div>
  109. </div>
  110. ))}
  111. </div>
  112. </div>
  113. </PortalToFollowElemContent>
  114. </PortalToFollowElem>
  115. )
  116. }
  117. export default React.memo(AppPicker)