select-metadata.tsx 3.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React, { useMemo, useState } from 'react'
  4. import type { MetadataItem } from '../types'
  5. import SearchInput from '@/app/components/base/search-input'
  6. import { RiAddLine, RiArrowRightUpLine } from '@remixicon/react'
  7. import { useTranslation } from 'react-i18next'
  8. import { getIcon } from '../utils/get-icon'
  9. const i18nPrefix = 'dataset.metadata.selectMetadata'
  10. type Props = {
  11. list: MetadataItem[]
  12. onSelect: (data: MetadataItem) => void
  13. onNew: () => void
  14. onManage: () => void
  15. }
  16. const SelectMetadata: FC<Props> = ({
  17. list: notFilteredList,
  18. onSelect,
  19. onNew,
  20. onManage,
  21. }) => {
  22. const { t } = useTranslation()
  23. const [query, setQuery] = useState('')
  24. const list = useMemo(() => {
  25. if (!query) return notFilteredList
  26. return notFilteredList.filter((item) => {
  27. return item.name.toLowerCase().includes(query.toLowerCase())
  28. })
  29. }, [query, notFilteredList])
  30. return (
  31. <div className='w-[320px] pt-2 pb-0 rounded-xl bg-components-panel-bg-blur border-[0.5px] border-components-panel-border shadow-lg backdrop-blur-[5px]'>
  32. <SearchInput
  33. className='mx-2'
  34. value={query}
  35. onChange={setQuery}
  36. placeholder={t(`${i18nPrefix}.search`)}
  37. />
  38. <div className='mt-2'>
  39. {list.map((item) => {
  40. const Icon = getIcon(item.type)
  41. return (
  42. <div
  43. key={item.id}
  44. className='mx-1 flex items-center h-6 px-3 justify-between rounded-md hover:bg-state-base-hover cursor-pointer'
  45. onClick={() => onSelect({
  46. id: item.id,
  47. name: item.name,
  48. type: item.type,
  49. })}
  50. >
  51. <div className='w-0 grow flex items-center h-full text-text-secondary'>
  52. <Icon className='shrink-0 mr-[5px] size-3.5' />
  53. <div className='w-0 grow truncate system-sm-medium'>{item.name}</div>
  54. </div>
  55. <div className='ml-1 shrink-0 system-xs-regular text-text-tertiary'>
  56. {item.type}
  57. </div>
  58. </div>
  59. )
  60. })}
  61. </div>
  62. <div className='mt-1 flex justify-between p-1 border-t border-divider-subtle'>
  63. <div className='flex items-center h-6 px-3 text-text-secondary rounded-md hover:bg-state-base-hover cursor-pointer space-x-1' onClick={onNew}>
  64. <RiAddLine className='size-3.5' />
  65. <div className='system-sm-medium'>{t(`${i18nPrefix}.newAction`)}</div>
  66. </div>
  67. <div className='flex items-center h-6 text-text-secondary '>
  68. <div className='mr-[3px] w-px h-3 bg-divider-regular'></div>
  69. <div className='flex h-full items-center px-1.5 hover:bg-state-base-hover rounded-md cursor-pointer' onClick={onManage}>
  70. <div className='mr-1 system-sm-medium'>{t(`${i18nPrefix}.manageAction`)}</div>
  71. <RiArrowRightUpLine className='size-3.5' />
  72. </div>
  73. </div>
  74. </div>
  75. </div>
  76. )
  77. }
  78. export default React.memo(SelectMetadata)