base.tsx 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. import { useCallback, useEffect, useMemo, useState } from 'react'
  2. import { useTranslation } from 'react-i18next'
  3. import { useContext } from 'use-context-selector'
  4. import useSWR from 'swr'
  5. import cn from 'classnames'
  6. import s from './base.module.css'
  7. import WorkspaceSelector from './workspace-selector'
  8. import SearchInput from './search-input'
  9. import PageSelector from './page-selector'
  10. import { preImportNotionPages } from '@/service/datasets'
  11. import { NotionConnector } from '@/app/components/datasets/create/step-one'
  12. import type { DataSourceNotionPageMap, DataSourceNotionWorkspace, NotionPage } from '@/models/common'
  13. import { ToastContext } from '@/app/components/base/toast'
  14. import { useModalContext } from '@/context/modal-context'
  15. type NotionPageSelectorProps = {
  16. value?: string[]
  17. onSelect: (selectedPages: NotionPage[]) => void
  18. canPreview?: boolean
  19. previewPageId?: string
  20. onPreview?: (selectedPage: NotionPage) => void
  21. datasetId?: string
  22. countLimit: number
  23. countUsed: number
  24. }
  25. const NotionPageSelector = ({
  26. value,
  27. onSelect,
  28. canPreview,
  29. previewPageId,
  30. onPreview,
  31. datasetId = '',
  32. countLimit,
  33. countUsed,
  34. }: NotionPageSelectorProps) => {
  35. const { t } = useTranslation()
  36. const { notify } = useContext(ToastContext)
  37. const { data, mutate } = useSWR({ url: '/notion/pre-import/pages', datasetId }, preImportNotionPages)
  38. const [prevData, setPrevData] = useState(data)
  39. const [searchValue, setSearchValue] = useState('')
  40. const [currentWorkspaceId, setCurrentWorkspaceId] = useState('')
  41. const { setShowAccountSettingModal } = useModalContext()
  42. const notionWorkspaces = useMemo(() => {
  43. return data?.notion_info || []
  44. }, [data?.notion_info])
  45. const firstWorkspaceId = notionWorkspaces[0]?.workspace_id
  46. const currentWorkspace = notionWorkspaces.find(workspace => workspace.workspace_id === currentWorkspaceId)
  47. const getPagesMapAndSelectedPagesId: [DataSourceNotionPageMap, Set<string>] = useMemo(() => {
  48. const selectedPagesId = new Set<string>()
  49. const pagesMap = notionWorkspaces.reduce((prev: DataSourceNotionPageMap, next: DataSourceNotionWorkspace) => {
  50. next.pages.forEach((page) => {
  51. if (page.is_bound)
  52. selectedPagesId.add(page.page_id)
  53. prev[page.page_id] = {
  54. ...page,
  55. workspace_id: next.workspace_id,
  56. }
  57. })
  58. return prev
  59. }, {})
  60. return [pagesMap, selectedPagesId]
  61. }, [notionWorkspaces])
  62. const defaultSelectedPagesId = [...Array.from(getPagesMapAndSelectedPagesId[1]), ...(value || [])]
  63. const [selectedPagesId, setSelectedPagesId] = useState<Set<string>>(new Set(defaultSelectedPagesId))
  64. if (prevData !== data) {
  65. setPrevData(data)
  66. setSelectedPagesId(new Set(defaultSelectedPagesId))
  67. }
  68. const handleSearchValueChange = useCallback((value: string) => {
  69. setSearchValue(value)
  70. }, [])
  71. const handleSelectWorkspace = useCallback((workspaceId: string) => {
  72. setCurrentWorkspaceId(workspaceId)
  73. }, [])
  74. const handleSelecPages = (newSelectedPagesId: Set<string>) => {
  75. const selectedPages = Array.from(newSelectedPagesId).map(pageId => getPagesMapAndSelectedPagesId[0][pageId])
  76. if (selectedPages.length > countLimit - countUsed) {
  77. notify({ type: 'error', message: t('datasetCreation.stepOne.overCountLimit', { countLimit }) })
  78. return false
  79. }
  80. setSelectedPagesId(new Set(Array.from(newSelectedPagesId)))
  81. onSelect(selectedPages)
  82. }
  83. const handlePreviewPage = (previewPageId: string) => {
  84. if (onPreview)
  85. onPreview(getPagesMapAndSelectedPagesId[0][previewPageId])
  86. }
  87. useEffect(() => {
  88. setCurrentWorkspaceId(firstWorkspaceId)
  89. }, [firstWorkspaceId])
  90. return (
  91. <div className='bg-gray-25 border border-gray-200 rounded-xl'>
  92. {
  93. data?.notion_info?.length
  94. ? (
  95. <>
  96. <div className='flex items-center pl-[10px] pr-2 h-11 bg-white border-b border-b-gray-200 rounded-t-xl'>
  97. <WorkspaceSelector
  98. value={currentWorkspaceId || firstWorkspaceId}
  99. items={notionWorkspaces}
  100. onSelect={handleSelectWorkspace}
  101. />
  102. <div className='mx-1 w-[1px] h-3 bg-gray-200' />
  103. <div
  104. className={cn(s['setting-icon'], 'w-6 h-6 cursor-pointer')}
  105. onClick={() => setShowAccountSettingModal({ payload: 'data-source', onCancelCallback: mutate })}
  106. />
  107. <div className='grow' />
  108. <SearchInput
  109. value={searchValue}
  110. onChange={handleSearchValueChange}
  111. />
  112. </div>
  113. <div className='rounded-b-xl overflow-hidden'>
  114. <PageSelector
  115. value={selectedPagesId}
  116. searchValue={searchValue}
  117. list={currentWorkspace?.pages || []}
  118. pagesMap={getPagesMapAndSelectedPagesId[0]}
  119. onSelect={handleSelecPages}
  120. canPreview={canPreview}
  121. previewPageId={previewPageId}
  122. onPreview={handlePreviewPage}
  123. />
  124. </div>
  125. </>
  126. )
  127. : (
  128. <NotionConnector onSetting={() => setShowAccountSettingModal({ payload: 'data-source', onCancelCallback: mutate })} />
  129. )
  130. }
  131. </div>
  132. )
  133. }
  134. export default NotionPageSelector