mirror of
https://github.com/langgenius/dify.git
synced 2026-03-15 11:57:10 +08:00
Problem: Model provider settings page (/plugins?action=showSettings&tab=provider) was missing plugin update indicators (red dot badge, Update button) that the /plugins page correctly displayed, because it only fetched installation data without querying for latest marketplace versions. Decision: Extract a shared usePluginsWithLatestVersion hook and migrate plugin API endpoints to oRPC contracts, ensuring both pages use identical data flows. Model: Both pages now follow the same pattern — fetch installed plugins via consoleQuery.plugins.checkInstalled, enrich with latest version metadata via usePluginsWithLatestVersion, then pass complete PluginDetail objects downstream where useDetailHeaderState computes hasNewVersion for UI indicators. Impact: - Update badge red dot and Update button now appear on provider settings page - Shared hook eliminates 15 lines of duplicate enrichment logic in plugins-panel - oRPC contracts replace legacy post() calls for plugin endpoints - Operation dropdown uses auto-width to prevent "View on Marketplace" text wrapping - Version badge aligned to use Badge component consistently across both pages - Update button tooltip added with bilingual i18n support - Deprecated Tooltip migrated to Base UI Tooltip in detail-header
137 lines
3.4 KiB
TypeScript
137 lines
3.4 KiB
TypeScript
import type { CategoryKey, TagKey } from './constants'
|
|
import type { PluginDetail } from './types'
|
|
import { useQuery } from '@tanstack/react-query'
|
|
import { useMemo } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { consoleQuery } from '@/service/client'
|
|
import {
|
|
categoryKeys,
|
|
tagKeys,
|
|
} from './constants'
|
|
import { PluginCategoryEnum, PluginSource } from './types'
|
|
|
|
export type Tag = {
|
|
name: TagKey
|
|
label: string
|
|
}
|
|
|
|
export const useTags = () => {
|
|
const { t } = useTranslation()
|
|
|
|
const tags = useMemo(() => {
|
|
return tagKeys.map((tag) => {
|
|
return {
|
|
name: tag,
|
|
label: t(`tags.${tag}`, { ns: 'pluginTags' }),
|
|
}
|
|
})
|
|
}, [t])
|
|
|
|
const tagsMap = useMemo(() => {
|
|
return tags.reduce((acc, tag) => {
|
|
acc[tag.name] = tag
|
|
return acc
|
|
}, {} as Record<string, Tag>)
|
|
}, [tags])
|
|
|
|
const getTagLabel = useMemo(() => {
|
|
return (name: string) => {
|
|
if (!tagsMap[name])
|
|
return name
|
|
return tagsMap[name].label
|
|
}
|
|
}, [tagsMap])
|
|
|
|
return {
|
|
tags,
|
|
tagsMap,
|
|
getTagLabel,
|
|
}
|
|
}
|
|
|
|
type Category = {
|
|
name: CategoryKey
|
|
label: string
|
|
}
|
|
|
|
export const useCategories = (isSingle?: boolean) => {
|
|
const { t } = useTranslation()
|
|
|
|
const categories = useMemo(() => {
|
|
return categoryKeys.map((category) => {
|
|
if (category === PluginCategoryEnum.agent) {
|
|
return {
|
|
name: PluginCategoryEnum.agent,
|
|
label: isSingle ? t('categorySingle.agent', { ns: 'plugin' }) : t('category.agents', { ns: 'plugin' }),
|
|
}
|
|
}
|
|
return {
|
|
name: category,
|
|
label: isSingle ? t(`categorySingle.${category}`, { ns: 'plugin' }) : t(`category.${category}s`, { ns: 'plugin' }),
|
|
}
|
|
})
|
|
}, [t, isSingle])
|
|
|
|
const categoriesMap = useMemo(() => {
|
|
return categories.reduce((acc, category) => {
|
|
acc[category.name] = category
|
|
return acc
|
|
}, {} as Record<string, Category>)
|
|
}, [categories])
|
|
|
|
return {
|
|
categories,
|
|
categoriesMap,
|
|
}
|
|
}
|
|
|
|
export const PLUGIN_PAGE_TABS_MAP = {
|
|
plugins: 'plugins',
|
|
marketplace: 'discover',
|
|
}
|
|
|
|
export const usePluginPageTabs = () => {
|
|
const { t } = useTranslation()
|
|
const tabs = [
|
|
{ value: PLUGIN_PAGE_TABS_MAP.plugins, text: t('menus.plugins', { ns: 'common' }) },
|
|
{ value: PLUGIN_PAGE_TABS_MAP.marketplace, text: t('menus.exploreMarketplace', { ns: 'common' }) },
|
|
]
|
|
return tabs
|
|
}
|
|
|
|
const EMPTY_PLUGINS: PluginDetail[] = []
|
|
|
|
export function usePluginsWithLatestVersion(plugins: PluginDetail[] = EMPTY_PLUGINS): PluginDetail[] {
|
|
const marketplacePluginIds = useMemo(
|
|
() => plugins
|
|
.filter(p => p.source === PluginSource.marketplace)
|
|
.map(p => p.plugin_id),
|
|
[plugins],
|
|
)
|
|
|
|
const { data: latestVersionData } = useQuery(consoleQuery.plugins.latestVersions.queryOptions({
|
|
input: { body: { plugin_ids: marketplacePluginIds } },
|
|
enabled: !!marketplacePluginIds.length,
|
|
}))
|
|
|
|
return useMemo(() => {
|
|
const versions = latestVersionData?.versions
|
|
if (!versions)
|
|
return plugins
|
|
|
|
return plugins.map((plugin) => {
|
|
const info = versions[plugin.plugin_id]
|
|
if (!info)
|
|
return plugin
|
|
return {
|
|
...plugin,
|
|
latest_version: info.version,
|
|
latest_unique_identifier: info.unique_identifier,
|
|
status: info.status,
|
|
deprecated_reason: info.deprecated_reason,
|
|
alternative_plugin_id: info.alternative_plugin_id,
|
|
}
|
|
})
|
|
}, [plugins, latestVersionData])
|
|
}
|