select-metadata.tsx 3.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  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] rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur pb-0 pt-2 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 h-6 cursor-pointer items-center justify-between rounded-md px-3 hover:bg-state-base-hover'
  45. onClick={() => onSelect({
  46. id: item.id,
  47. name: item.name,
  48. type: item.type,
  49. })}
  50. >
  51. <div className='flex h-full w-0 grow items-center text-text-secondary'>
  52. <Icon className='mr-[5px] size-3.5 shrink-0' />
  53. <div className='system-sm-medium w-0 grow truncate'>{item.name}</div>
  54. </div>
  55. <div className='system-xs-regular ml-1 shrink-0 text-text-tertiary'>
  56. {item.type}
  57. </div>
  58. </div>
  59. )
  60. })}
  61. </div>
  62. <div className='mt-1 flex justify-between border-t border-divider-subtle p-1'>
  63. <div className='flex h-6 cursor-pointer items-center space-x-1 rounded-md px-3 text-text-secondary hover:bg-state-base-hover' onClick={onNew}>
  64. <RiAddLine className='size-3.5' />
  65. <div className='system-sm-medium'>{t(`${i18nPrefix}.newAction`)}</div>
  66. </div>
  67. <div className='flex h-6 items-center text-text-secondary '>
  68. <div className='mr-[3px] h-3 w-px bg-divider-regular'></div>
  69. <div className='flex h-full cursor-pointer items-center rounded-md px-1.5 hover:bg-state-base-hover' onClick={onManage}>
  70. <div className='system-sm-medium mr-1'>{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)