modal-context.tsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. 'use client'
  2. import type { Dispatch, SetStateAction } from 'react'
  3. import { useCallback, useState } from 'react'
  4. import { createContext, useContext, useContextSelector } from 'use-context-selector'
  5. import { useRouter, useSearchParams } from 'next/navigation'
  6. import AccountSetting from '@/app/components/header/account-setting'
  7. import ApiBasedExtensionModal from '@/app/components/header/account-setting/api-based-extension-page/modal'
  8. import ModerationSettingModal from '@/app/components/app/configuration/toolbox/moderation/moderation-setting-modal'
  9. import ExternalDataToolModal from '@/app/components/app/configuration/tools/external-data-tool-modal'
  10. import AnnotationFullModal from '@/app/components/billing/annotation-full/modal'
  11. import ModelModal from '@/app/components/header/account-setting/model-provider-page/model-modal'
  12. import ExternalAPIModal from '@/app/components/datasets/external-api/external-api-modal'
  13. import type {
  14. ConfigurationMethodEnum,
  15. CustomConfigurationModelFixedFields,
  16. ModelLoadBalancingConfigEntry,
  17. ModelProvider,
  18. } from '@/app/components/header/account-setting/model-provider-page/declarations'
  19. import Pricing from '@/app/components/billing/pricing'
  20. import type { ModerationConfig } from '@/models/debug'
  21. import type {
  22. ApiBasedExtension,
  23. ExternalDataTool,
  24. } from '@/models/common'
  25. import type { CreateExternalAPIReq } from '@/app/components/datasets/external-api/declarations'
  26. import ModelLoadBalancingEntryModal from '@/app/components/header/account-setting/model-provider-page/model-modal/model-load-balancing-entry-modal'
  27. import type { ModelLoadBalancingModalProps } from '@/app/components/header/account-setting/model-provider-page/provider-added-card/model-load-balancing-modal'
  28. import ModelLoadBalancingModal from '@/app/components/header/account-setting/model-provider-page/provider-added-card/model-load-balancing-modal'
  29. export type ModalState<T> = {
  30. payload: T
  31. onCancelCallback?: () => void
  32. onSaveCallback?: (newPayload: T) => void
  33. onRemoveCallback?: (newPayload: T) => void
  34. onEditCallback?: (newPayload: T) => void
  35. onValidateBeforeSaveCallback?: (newPayload: T) => boolean
  36. isEditMode?: boolean
  37. datasetBindings?: { id: string; name: string }[]
  38. }
  39. export type ModelModalType = {
  40. currentProvider: ModelProvider
  41. currentConfigurationMethod: ConfigurationMethodEnum
  42. currentCustomConfigurationModelFixedFields?: CustomConfigurationModelFixedFields
  43. }
  44. export type LoadBalancingEntryModalType = ModelModalType & {
  45. entry?: ModelLoadBalancingConfigEntry
  46. index?: number
  47. }
  48. export type ModalContextState = {
  49. setShowAccountSettingModal: Dispatch<SetStateAction<ModalState<string> | null>>
  50. setShowApiBasedExtensionModal: Dispatch<SetStateAction<ModalState<ApiBasedExtension> | null>>
  51. setShowModerationSettingModal: Dispatch<SetStateAction<ModalState<ModerationConfig> | null>>
  52. setShowExternalDataToolModal: Dispatch<SetStateAction<ModalState<ExternalDataTool> | null>>
  53. setShowPricingModal: () => void
  54. setShowAnnotationFullModal: () => void
  55. setShowModelModal: Dispatch<SetStateAction<ModalState<ModelModalType> | null>>
  56. setShowExternalKnowledgeAPIModal: Dispatch<SetStateAction<ModalState<CreateExternalAPIReq> | null>>
  57. setShowModelLoadBalancingModal: Dispatch<SetStateAction<ModelLoadBalancingModalProps | null>>
  58. setShowModelLoadBalancingEntryModal: Dispatch<SetStateAction<ModalState<LoadBalancingEntryModalType> | null>>
  59. }
  60. const ModalContext = createContext<ModalContextState>({
  61. setShowAccountSettingModal: () => { },
  62. setShowApiBasedExtensionModal: () => { },
  63. setShowModerationSettingModal: () => { },
  64. setShowExternalDataToolModal: () => { },
  65. setShowPricingModal: () => { },
  66. setShowAnnotationFullModal: () => { },
  67. setShowModelModal: () => { },
  68. setShowExternalKnowledgeAPIModal: () => { },
  69. setShowModelLoadBalancingModal: () => { },
  70. setShowModelLoadBalancingEntryModal: () => { },
  71. })
  72. export const useModalContext = () => useContext(ModalContext)
  73. // Adding a dangling comma to avoid the generic parsing issue in tsx, see:
  74. // https://github.com/microsoft/TypeScript/issues/15713
  75. // eslint-disable-next-line @typescript-eslint/comma-dangle
  76. export const useModalContextSelector = <T,>(selector: (state: ModalContextState) => T): T =>
  77. useContextSelector(ModalContext, selector)
  78. type ModalContextProviderProps = {
  79. children: React.ReactNode
  80. }
  81. export const ModalContextProvider = ({
  82. children,
  83. }: ModalContextProviderProps) => {
  84. const [showAccountSettingModal, setShowAccountSettingModal] = useState<ModalState<string> | null>(null)
  85. const [showApiBasedExtensionModal, setShowApiBasedExtensionModal] = useState<ModalState<ApiBasedExtension> | null>(null)
  86. const [showModerationSettingModal, setShowModerationSettingModal] = useState<ModalState<ModerationConfig> | null>(null)
  87. const [showExternalDataToolModal, setShowExternalDataToolModal] = useState<ModalState<ExternalDataTool> | null>(null)
  88. const [showModelModal, setShowModelModal] = useState<ModalState<ModelModalType> | null>(null)
  89. const [showExternalKnowledgeAPIModal, setShowExternalKnowledgeAPIModal] = useState<ModalState<CreateExternalAPIReq> | null>(null)
  90. const [showModelLoadBalancingModal, setShowModelLoadBalancingModal] = useState<ModelLoadBalancingModalProps | null>(null)
  91. const [showModelLoadBalancingEntryModal, setShowModelLoadBalancingEntryModal] = useState<ModalState<LoadBalancingEntryModalType> | null>(null)
  92. const searchParams = useSearchParams()
  93. const router = useRouter()
  94. const [showPricingModal, setShowPricingModal] = useState(searchParams.get('show-pricing') === '1')
  95. const [showAnnotationFullModal, setShowAnnotationFullModal] = useState(false)
  96. const handleCancelAccountSettingModal = () => {
  97. setShowAccountSettingModal(null)
  98. if (showAccountSettingModal?.onCancelCallback)
  99. showAccountSettingModal?.onCancelCallback()
  100. }
  101. const handleCancelModerationSettingModal = () => {
  102. setShowModerationSettingModal(null)
  103. if (showModerationSettingModal?.onCancelCallback)
  104. showModerationSettingModal.onCancelCallback()
  105. }
  106. const handleCancelExternalDataToolModal = () => {
  107. setShowExternalDataToolModal(null)
  108. if (showExternalDataToolModal?.onCancelCallback)
  109. showExternalDataToolModal.onCancelCallback()
  110. }
  111. const handleCancelModelModal = useCallback(() => {
  112. setShowModelModal(null)
  113. if (showModelModal?.onCancelCallback)
  114. showModelModal.onCancelCallback()
  115. }, [showModelModal])
  116. const handleSaveModelModal = useCallback(() => {
  117. if (showModelModal?.onSaveCallback)
  118. showModelModal.onSaveCallback(showModelModal.payload)
  119. setShowModelModal(null)
  120. }, [showModelModal])
  121. const handleCancelExternalApiModal = useCallback(() => {
  122. setShowExternalKnowledgeAPIModal(null)
  123. if (showExternalKnowledgeAPIModal?.onCancelCallback)
  124. showExternalKnowledgeAPIModal.onCancelCallback()
  125. }, [showExternalKnowledgeAPIModal])
  126. const handleSaveExternalApiModal = useCallback(async (updatedFormValue: CreateExternalAPIReq) => {
  127. if (showExternalKnowledgeAPIModal?.onSaveCallback)
  128. showExternalKnowledgeAPIModal.onSaveCallback(updatedFormValue)
  129. setShowExternalKnowledgeAPIModal(null)
  130. }, [showExternalKnowledgeAPIModal])
  131. const handleEditExternalApiModal = useCallback(async (updatedFormValue: CreateExternalAPIReq) => {
  132. if (showExternalKnowledgeAPIModal?.onEditCallback)
  133. showExternalKnowledgeAPIModal.onEditCallback(updatedFormValue)
  134. setShowExternalKnowledgeAPIModal(null)
  135. }, [showExternalKnowledgeAPIModal])
  136. const handleCancelModelLoadBalancingEntryModal = useCallback(() => {
  137. showModelLoadBalancingEntryModal?.onCancelCallback?.()
  138. setShowModelLoadBalancingEntryModal(null)
  139. }, [showModelLoadBalancingEntryModal])
  140. const handleSaveModelLoadBalancingEntryModal = useCallback((entry: ModelLoadBalancingConfigEntry) => {
  141. showModelLoadBalancingEntryModal?.onSaveCallback?.({
  142. ...showModelLoadBalancingEntryModal.payload,
  143. entry,
  144. })
  145. setShowModelLoadBalancingEntryModal(null)
  146. }, [showModelLoadBalancingEntryModal])
  147. const handleRemoveModelLoadBalancingEntry = useCallback(() => {
  148. showModelLoadBalancingEntryModal?.onRemoveCallback?.(showModelLoadBalancingEntryModal.payload)
  149. setShowModelLoadBalancingEntryModal(null)
  150. }, [showModelLoadBalancingEntryModal])
  151. const handleSaveApiBasedExtension = (newApiBasedExtension: ApiBasedExtension) => {
  152. if (showApiBasedExtensionModal?.onSaveCallback)
  153. showApiBasedExtensionModal.onSaveCallback(newApiBasedExtension)
  154. setShowApiBasedExtensionModal(null)
  155. }
  156. const handleSaveModeration = (newModerationConfig: ModerationConfig) => {
  157. if (showModerationSettingModal?.onSaveCallback)
  158. showModerationSettingModal.onSaveCallback(newModerationConfig)
  159. setShowModerationSettingModal(null)
  160. }
  161. const handleSaveExternalDataTool = (newExternalDataTool: ExternalDataTool) => {
  162. if (showExternalDataToolModal?.onSaveCallback)
  163. showExternalDataToolModal.onSaveCallback(newExternalDataTool)
  164. setShowExternalDataToolModal(null)
  165. }
  166. const handleValidateBeforeSaveExternalDataTool = (newExternalDataTool: ExternalDataTool) => {
  167. if (showExternalDataToolModal?.onValidateBeforeSaveCallback)
  168. return showExternalDataToolModal?.onValidateBeforeSaveCallback(newExternalDataTool)
  169. return true
  170. }
  171. return (
  172. <ModalContext.Provider value={{
  173. setShowAccountSettingModal,
  174. setShowApiBasedExtensionModal,
  175. setShowModerationSettingModal,
  176. setShowExternalDataToolModal,
  177. setShowPricingModal: () => setShowPricingModal(true),
  178. setShowAnnotationFullModal: () => setShowAnnotationFullModal(true),
  179. setShowModelModal,
  180. setShowExternalKnowledgeAPIModal,
  181. setShowModelLoadBalancingModal,
  182. setShowModelLoadBalancingEntryModal,
  183. }}>
  184. <>
  185. {children}
  186. {
  187. !!showAccountSettingModal && (
  188. <AccountSetting
  189. activeTab={showAccountSettingModal.payload}
  190. onCancel={handleCancelAccountSettingModal}
  191. />
  192. )
  193. }
  194. {
  195. !!showApiBasedExtensionModal && (
  196. <ApiBasedExtensionModal
  197. data={showApiBasedExtensionModal.payload}
  198. onCancel={() => setShowApiBasedExtensionModal(null)}
  199. onSave={handleSaveApiBasedExtension}
  200. />
  201. )
  202. }
  203. {
  204. !!showModerationSettingModal && (
  205. <ModerationSettingModal
  206. data={showModerationSettingModal.payload}
  207. onCancel={handleCancelModerationSettingModal}
  208. onSave={handleSaveModeration}
  209. />
  210. )
  211. }
  212. {
  213. !!showExternalDataToolModal && (
  214. <ExternalDataToolModal
  215. data={showExternalDataToolModal.payload}
  216. onCancel={handleCancelExternalDataToolModal}
  217. onSave={handleSaveExternalDataTool}
  218. onValidateBeforeSave={handleValidateBeforeSaveExternalDataTool}
  219. />
  220. )
  221. }
  222. {
  223. !!showPricingModal && (
  224. <Pricing onCancel={() => {
  225. if (searchParams.get('show-pricing') === '1')
  226. router.push(location.pathname, { forceOptimisticNavigation: true } as any)
  227. setShowPricingModal(false)
  228. }} />
  229. )
  230. }
  231. {
  232. showAnnotationFullModal && (
  233. <AnnotationFullModal
  234. show={showAnnotationFullModal}
  235. onHide={() => setShowAnnotationFullModal(false)} />
  236. )
  237. }
  238. {
  239. !!showModelModal && (
  240. <ModelModal
  241. provider={showModelModal.payload.currentProvider}
  242. configurateMethod={showModelModal.payload.currentConfigurationMethod}
  243. currentCustomConfigurationModelFixedFields={showModelModal.payload.currentCustomConfigurationModelFixedFields}
  244. onCancel={handleCancelModelModal}
  245. onSave={handleSaveModelModal}
  246. />
  247. )
  248. }
  249. {
  250. !!showExternalKnowledgeAPIModal && (
  251. <ExternalAPIModal
  252. data={showExternalKnowledgeAPIModal.payload}
  253. datasetBindings={showExternalKnowledgeAPIModal.datasetBindings ?? []}
  254. onSave={handleSaveExternalApiModal}
  255. onCancel={handleCancelExternalApiModal}
  256. onEdit={handleEditExternalApiModal}
  257. isEditMode={showExternalKnowledgeAPIModal.isEditMode ?? false}
  258. />
  259. )
  260. }
  261. {
  262. Boolean(showModelLoadBalancingModal) && (
  263. <ModelLoadBalancingModal {...showModelLoadBalancingModal!} />
  264. )
  265. }
  266. {
  267. !!showModelLoadBalancingEntryModal && (
  268. <ModelLoadBalancingEntryModal
  269. provider={showModelLoadBalancingEntryModal.payload.currentProvider}
  270. configurationMethod={showModelLoadBalancingEntryModal.payload.currentConfigurationMethod}
  271. currentCustomConfigurationModelFixedFields={showModelLoadBalancingEntryModal.payload.currentCustomConfigurationModelFixedFields}
  272. entry={showModelLoadBalancingEntryModal.payload.entry}
  273. onCancel={handleCancelModelLoadBalancingEntryModal}
  274. onSave={handleSaveModelLoadBalancingEntryModal}
  275. onRemove={handleRemoveModelLoadBalancingEntry}
  276. />
  277. )
  278. }
  279. </>
  280. </ModalContext.Provider>
  281. )
  282. }
  283. export default ModalContext