Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

index.tsx 7.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React, { useMemo } from 'react'
  4. import {
  5. RiArrowRightUpLine,
  6. RiBugLine,
  7. RiErrorWarningLine,
  8. RiHardDrive3Line,
  9. RiLoginCircleLine,
  10. RiVerifiedBadgeLine,
  11. } from '@remixicon/react'
  12. import { useTranslation } from 'react-i18next'
  13. import { usePluginPageContext } from '../plugin-page/context'
  14. import { Github } from '../../base/icons/src/public/common'
  15. import Badge from '../../base/badge'
  16. import { type PluginDetail, PluginSource, PluginType } from '../types'
  17. import CornerMark from '../card/base/corner-mark'
  18. import Description from '../card/base/description'
  19. import OrgInfo from '../card/base/org-info'
  20. import Title from '../card/base/title'
  21. import Action from './action'
  22. import cn from '@/utils/classnames'
  23. import { API_PREFIX, MARKETPLACE_URL_PREFIX } from '@/config'
  24. import { useSingleCategories } from '../hooks'
  25. import { useRenderI18nObject } from '@/hooks/use-i18n'
  26. import useRefreshPluginList from '@/app/components/plugins/install-plugin/hooks/use-refresh-plugin-list'
  27. import { useAppContext } from '@/context/app-context'
  28. import { gte } from 'semver'
  29. import Tooltip from '@/app/components/base/tooltip'
  30. type Props = {
  31. className?: string
  32. plugin: PluginDetail
  33. }
  34. const PluginItem: FC<Props> = ({
  35. className,
  36. plugin,
  37. }) => {
  38. const { t } = useTranslation()
  39. const { categoriesMap } = useSingleCategories()
  40. const currentPluginID = usePluginPageContext(v => v.currentPluginID)
  41. const setCurrentPluginID = usePluginPageContext(v => v.setCurrentPluginID)
  42. const { refreshPluginList } = useRefreshPluginList()
  43. const {
  44. source,
  45. tenant_id,
  46. installation_id,
  47. plugin_unique_identifier,
  48. endpoints_active,
  49. meta,
  50. plugin_id,
  51. } = plugin
  52. const { category, author, name, label, description, icon, verified, meta: declarationMeta } = plugin.declaration
  53. const orgName = useMemo(() => {
  54. return [PluginSource.github, PluginSource.marketplace].includes(source) ? author : ''
  55. }, [source, author])
  56. const { langeniusVersionInfo } = useAppContext()
  57. const isDifyVersionCompatible = useMemo(() => {
  58. if (!langeniusVersionInfo.current_version)
  59. return true
  60. return gte(langeniusVersionInfo.current_version, declarationMeta.minimum_dify_version ?? '0.0.0')
  61. }, [declarationMeta.minimum_dify_version, langeniusVersionInfo.current_version])
  62. const handleDelete = () => {
  63. refreshPluginList({ category } as any)
  64. }
  65. const getValueFromI18nObject = useRenderI18nObject()
  66. const title = getValueFromI18nObject(label)
  67. const descriptionText = getValueFromI18nObject(description)
  68. return (
  69. <div
  70. className={cn(
  71. 'rounded-xl border-[1.5px] border-background-section-burn p-1',
  72. currentPluginID === plugin_id && 'border-components-option-card-option-selected-border',
  73. source === PluginSource.debugging
  74. ? 'bg-[repeating-linear-gradient(-45deg,rgba(16,24,40,0.04),rgba(16,24,40,0.04)_5px,rgba(0,0,0,0.02)_5px,rgba(0,0,0,0.02)_10px)]'
  75. : 'bg-background-section-burn',
  76. )}
  77. onClick={() => {
  78. setCurrentPluginID(plugin.plugin_id)
  79. }}
  80. >
  81. <div className={cn('hover-bg-components-panel-on-panel-item-bg relative rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-on-panel-item-bg p-4 pb-3 shadow-xs', className)}>
  82. <CornerMark text={categoriesMap[category].label} />
  83. {/* Header */}
  84. <div className="flex">
  85. <div className='flex h-10 w-10 items-center justify-center overflow-hidden rounded-xl border-[1px] border-components-panel-border-subtle'>
  86. <img
  87. className='h-full w-full'
  88. src={`${API_PREFIX}/workspaces/current/plugin/icon?tenant_id=${tenant_id}&filename=${icon}`}
  89. alt={`plugin-${plugin_unique_identifier}-logo`}
  90. />
  91. </div>
  92. <div className="ml-3 w-0 grow">
  93. <div className="flex h-5 items-center">
  94. <Title title={title} />
  95. {verified && <RiVerifiedBadgeLine className="ml-0.5 h-4 w-4 shrink-0 text-text-accent" />}
  96. {!isDifyVersionCompatible && <Tooltip popupContent={
  97. t('plugin.difyVersionNotCompatible', { minimalDifyVersion: declarationMeta.minimum_dify_version })
  98. }><RiErrorWarningLine color='red' className="ml-0.5 h-4 w-4 shrink-0 text-text-accent" /></Tooltip>}
  99. <Badge className='ml-1 shrink-0' text={source === PluginSource.github ? plugin.meta!.version : plugin.version} />
  100. </div>
  101. <div className='flex items-center justify-between'>
  102. <Description text={descriptionText} descriptionLineRows={1}></Description>
  103. <div onClick={e => e.stopPropagation()}>
  104. <Action
  105. pluginUniqueIdentifier={plugin_unique_identifier}
  106. installationId={installation_id}
  107. author={author}
  108. pluginName={name}
  109. usedInApps={5}
  110. isShowFetchNewVersion={source === PluginSource.github}
  111. isShowInfo={source === PluginSource.github}
  112. isShowDelete
  113. meta={meta}
  114. onDelete={handleDelete}
  115. category={category}
  116. />
  117. </div>
  118. </div>
  119. </div>
  120. </div>
  121. </div>
  122. <div className='mb-1 mt-1.5 flex h-4 items-center justify-between px-4'>
  123. <div className='flex items-center'>
  124. <OrgInfo
  125. className="mt-0.5"
  126. orgName={orgName}
  127. packageName={name}
  128. packageNameClassName='w-auto max-w-[150px]'
  129. />
  130. {category === PluginType.extension && (
  131. <>
  132. <div className='system-xs-regular mx-2 text-text-quaternary'>·</div>
  133. <div className='system-xs-regular flex space-x-1 text-text-tertiary'>
  134. <RiLoginCircleLine className='h-4 w-4' />
  135. <span>{t('plugin.endpointsEnabled', { num: endpoints_active })}</span>
  136. </div>
  137. </>
  138. )}
  139. </div>
  140. <div className='flex items-center'>
  141. {source === PluginSource.github
  142. && <>
  143. <a href={`https://github.com/${meta!.repo}`} target='_blank' className='flex items-center gap-1'>
  144. <div className='system-2xs-medium-uppercase text-text-tertiary'>{t('plugin.from')}</div>
  145. <div className='flex items-center space-x-0.5 text-text-secondary'>
  146. <Github className='h-3 w-3' />
  147. <div className='system-2xs-semibold-uppercase'>GitHub</div>
  148. <RiArrowRightUpLine className='h-3 w-3' />
  149. </div>
  150. </a>
  151. </>
  152. }
  153. {source === PluginSource.marketplace
  154. && <>
  155. <a href={`${MARKETPLACE_URL_PREFIX}/plugins/${author}/${name}`} target='_blank' className='flex items-center gap-0.5'>
  156. <div className='system-2xs-medium-uppercase text-text-tertiary'>{t('plugin.from')} <span className='text-text-secondary'>marketplace</span></div>
  157. <RiArrowRightUpLine className='h-3 w-3 text-text-tertiary' />
  158. </a>
  159. </>
  160. }
  161. {source === PluginSource.local
  162. && <>
  163. <div className='flex items-center gap-1'>
  164. <RiHardDrive3Line className='h-3 w-3 text-text-tertiary' />
  165. <div className='system-2xs-medium-uppercase text-text-tertiary'>Local Plugin</div>
  166. </div>
  167. </>
  168. }
  169. {source === PluginSource.debugging
  170. && <>
  171. <div className='flex items-center gap-1'>
  172. <RiBugLine className='h-3 w-3 text-text-warning' />
  173. <div className='system-2xs-medium-uppercase text-text-warning'>Debugging Plugin</div>
  174. </div>
  175. </>
  176. }
  177. </div>
  178. </div>
  179. </div>
  180. )
  181. }
  182. export default React.memo(PluginItem)