@lobehub/chat 1.96.19 → 1.97.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env.example +7 -0
- package/CHANGELOG.md +50 -0
- package/apps/desktop/package.json +1 -1
- package/apps/desktop/src/main/core/Browser.ts +2 -1
- package/apps/desktop/src/main/utils/next-electron-rsc.ts +15 -8
- package/changelog/v1.json +18 -0
- package/locales/ar/discover.json +452 -12
- package/locales/ar/metadata.json +4 -0
- package/locales/ar/plugin.json +89 -2
- package/locales/ar/setting.json +1 -0
- package/locales/bg-BG/discover.json +452 -12
- package/locales/bg-BG/metadata.json +4 -0
- package/locales/bg-BG/plugin.json +89 -2
- package/locales/bg-BG/setting.json +1 -0
- package/locales/de-DE/discover.json +452 -12
- package/locales/de-DE/metadata.json +4 -0
- package/locales/de-DE/plugin.json +89 -2
- package/locales/de-DE/setting.json +1 -0
- package/locales/en-US/discover.json +452 -12
- package/locales/en-US/metadata.json +4 -0
- package/locales/en-US/plugin.json +89 -2
- package/locales/en-US/setting.json +1 -0
- package/locales/es-ES/discover.json +452 -12
- package/locales/es-ES/metadata.json +4 -0
- package/locales/es-ES/plugin.json +89 -2
- package/locales/es-ES/setting.json +1 -0
- package/locales/fa-IR/discover.json +452 -12
- package/locales/fa-IR/metadata.json +4 -0
- package/locales/fa-IR/plugin.json +89 -2
- package/locales/fa-IR/setting.json +1 -0
- package/locales/fr-FR/discover.json +452 -12
- package/locales/fr-FR/metadata.json +4 -0
- package/locales/fr-FR/plugin.json +89 -2
- package/locales/fr-FR/setting.json +1 -0
- package/locales/it-IT/discover.json +452 -12
- package/locales/it-IT/metadata.json +4 -0
- package/locales/it-IT/plugin.json +89 -2
- package/locales/it-IT/setting.json +1 -0
- package/locales/ja-JP/discover.json +452 -12
- package/locales/ja-JP/metadata.json +4 -0
- package/locales/ja-JP/plugin.json +89 -2
- package/locales/ja-JP/setting.json +1 -0
- package/locales/ko-KR/discover.json +452 -12
- package/locales/ko-KR/metadata.json +4 -0
- package/locales/ko-KR/plugin.json +89 -2
- package/locales/ko-KR/setting.json +1 -0
- package/locales/nl-NL/discover.json +452 -12
- package/locales/nl-NL/metadata.json +4 -0
- package/locales/nl-NL/plugin.json +89 -2
- package/locales/nl-NL/setting.json +1 -0
- package/locales/pl-PL/discover.json +452 -12
- package/locales/pl-PL/metadata.json +4 -0
- package/locales/pl-PL/plugin.json +89 -2
- package/locales/pl-PL/setting.json +1 -0
- package/locales/pt-BR/discover.json +452 -12
- package/locales/pt-BR/metadata.json +4 -0
- package/locales/pt-BR/plugin.json +89 -2
- package/locales/pt-BR/setting.json +1 -0
- package/locales/ru-RU/discover.json +452 -12
- package/locales/ru-RU/metadata.json +4 -0
- package/locales/ru-RU/plugin.json +89 -2
- package/locales/ru-RU/setting.json +1 -0
- package/locales/tr-TR/discover.json +452 -12
- package/locales/tr-TR/metadata.json +4 -0
- package/locales/tr-TR/plugin.json +89 -2
- package/locales/tr-TR/setting.json +1 -0
- package/locales/vi-VN/discover.json +452 -12
- package/locales/vi-VN/metadata.json +4 -0
- package/locales/vi-VN/plugin.json +89 -2
- package/locales/vi-VN/setting.json +1 -0
- package/locales/zh-CN/discover.json +452 -12
- package/locales/zh-CN/metadata.json +8 -4
- package/locales/zh-CN/plugin.json +89 -2
- package/locales/zh-CN/setting.json +1 -0
- package/locales/zh-TW/discover.json +452 -12
- package/locales/zh-TW/metadata.json +4 -0
- package/locales/zh-TW/plugin.json +89 -2
- package/locales/zh-TW/setting.json +1 -0
- package/next.config.ts +70 -19
- package/package.json +6 -2
- package/scripts/buildSitemapIndex/index.ts +9 -4
- package/src/app/(backend)/trpc/lambda/[trpc]/route.ts +5 -0
- package/src/app/[variants]/(main)/_layout/Mobile/index.tsx +5 -4
- package/src/app/[variants]/(main)/chat/(workspace)/@conversation/features/ChatList/WelcomeChatItem/InboxWelcome/AgentsSuggest.tsx +31 -45
- package/src/app/[variants]/(main)/discover/(detail)/_layout/Desktop.tsx +8 -6
- package/src/app/[variants]/(main)/discover/(detail)/_layout/Mobile/Header.tsx +3 -2
- package/src/app/[variants]/(main)/discover/(detail)/_layout/Mobile/index.tsx +5 -1
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/Client.tsx +40 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/DetailProvider.tsx +19 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Capabilities/Block.tsx +27 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Capabilities/Knowledge.tsx +33 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Capabilities/KnowledgeItem.tsx +58 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Capabilities/PluginItem.tsx +68 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Capabilities/Plugins.tsx +32 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Capabilities/index.tsx +37 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Nav.tsx +121 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Overview/TagList.tsx +47 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Overview/index.tsx +96 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/Related/index.tsx +31 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/SystemRole/TagList.tsx +47 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/SystemRole/index.tsx +54 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Details/index.tsx +49 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Header.tsx +176 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/{[slug]/features → [...slugs]/features/Sidebar/ActionButton}/AddAgent.tsx +22 -21
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Sidebar/ActionButton/index.tsx +31 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Sidebar/Related/Item.tsx +57 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Sidebar/Related/index.tsx +43 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Sidebar/Summary/index.tsx +38 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Sidebar/TocList/index.tsx +77 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/features/Sidebar/index.tsx +46 -0
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[...slugs]/loading.tsx +48 -0
- package/src/app/[variants]/(main)/discover/(detail)/{plugin/[slug] → assistant/[...slugs]}/page.tsx +39 -42
- package/src/app/[variants]/(main)/discover/(detail)/features/Breadcrumb.tsx +56 -0
- package/src/app/[variants]/(main)/discover/(detail)/features/DetailLayout.tsx +0 -1
- package/src/app/[variants]/(main)/discover/(detail)/features/MakedownRender.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(detail)/features/ShareButton.tsx +4 -3
- package/src/app/[variants]/(main)/discover/(detail)/features/Toc/Heading.tsx +108 -0
- package/src/app/[variants]/(main)/discover/(detail)/features/Toc/index.tsx +92 -0
- package/src/app/[variants]/(main)/discover/(detail)/features/Toc/useToc.tsx +66 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/Client.tsx +43 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Details/Related/index.tsx +32 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Details/Versions/index.tsx +76 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Details/index.tsx +59 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Sidebar/ActionButton/index.tsx +84 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Sidebar/ConnectionTypeAlert.tsx +35 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Sidebar/Related/Item.tsx +57 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Sidebar/Related/index.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Sidebar/ServerConfig.tsx +36 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Sidebar/TocList/index.tsx +98 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/features/Sidebar/index.tsx +58 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/loading.tsx +1 -0
- package/src/app/[variants]/(main)/discover/(detail)/mcp/[slug]/page.tsx +103 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/Client.tsx +40 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/DetailProvider.tsx +19 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Details/Nav.tsx +90 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Details/Overview/ProviderList/index.tsx +179 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Details/Overview/index.tsx +22 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/{ParameterList → Details/Parameter}/ParameterItem.tsx +1 -1
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/{ParameterList → Details/Parameter}/index.tsx +11 -11
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Details/Related/index.tsx +31 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Details/index.tsx +47 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Header.tsx +84 -59
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Sidebar/ActionButton/ChatWithModel.tsx +92 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Sidebar/ActionButton/index.tsx +32 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Sidebar/Related/Item.tsx +60 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Sidebar/Related/index.tsx +43 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Sidebar/RelatedProviders/Item.tsx +60 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Sidebar/RelatedProviders/index.tsx +34 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Sidebar/index.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/loading.tsx +1 -0
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/page.tsx +22 -45
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/Client.tsx +40 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/DetailProvider.tsx +19 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/Guide/index.tsx +25 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/Nav.tsx +99 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/Overview/ModelList/index.tsx +142 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/Overview/index.tsx +23 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/Related/index.tsx +22 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/index.tsx +47 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Header.tsx +99 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/{[slug]/features → [...slugs]/features/Sidebar/ActionButton}/ProviderConfig.tsx +9 -14
- package/src/app/[variants]/(main)/discover/(detail)/provider/{[slug]/features/Actions.tsx → [...slugs]/features/Sidebar/ActionButton/index.tsx} +14 -18
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Sidebar/Related/Item.tsx +60 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Sidebar/Related/index.tsx +34 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Sidebar/RelatedModels/Item.tsx +60 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Sidebar/RelatedModels/index.tsx +43 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Sidebar/index.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/loading.tsx +1 -0
- package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/page.tsx +103 -0
- package/src/app/[variants]/(main)/discover/(list)/(home)/Client.tsx +24 -21
- package/src/app/[variants]/(main)/discover/(list)/(home)/loading.tsx +1 -1
- package/src/app/[variants]/(main)/discover/(list)/(home)/page.tsx +13 -38
- package/src/app/[variants]/(main)/discover/(list)/_layout/Desktop/Nav.tsx +15 -8
- package/src/app/[variants]/(main)/discover/(list)/_layout/Mobile/Header.tsx +1 -1
- package/src/app/[variants]/(main)/discover/(list)/_layout/Mobile/Nav.tsx +2 -1
- package/src/app/[variants]/(main)/discover/(list)/assistant/Client.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(list)/assistant/features/Category/index.tsx +84 -0
- package/src/app/[variants]/(main)/discover/(list)/assistant/features/Category/useCategory.tsx +112 -0
- package/src/app/[variants]/(main)/discover/(list)/assistant/features/List/Item.tsx +189 -0
- package/src/app/[variants]/(main)/discover/(list)/assistant/features/List/TokenTag.tsx +70 -0
- package/src/app/[variants]/(main)/discover/(list)/assistant/features/List/index.tsx +33 -0
- package/src/app/[variants]/(main)/discover/(list)/assistant/page.tsx +46 -0
- package/src/app/[variants]/(main)/discover/(list)/features/Pagination.tsx +67 -0
- package/src/app/[variants]/(main)/discover/(list)/features/SortButton/index.tsx +184 -0
- package/src/app/[variants]/(main)/discover/(list)/mcp/Client.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(list)/mcp/features/Category/index.tsx +83 -0
- package/src/app/[variants]/(main)/discover/(list)/mcp/features/List/ConnectionTypeTag.tsx +51 -0
- package/src/app/[variants]/(main)/discover/(list)/mcp/features/List/Item.tsx +225 -0
- package/src/app/[variants]/(main)/discover/(list)/mcp/features/List/MetaInfo.tsx +33 -0
- package/src/app/[variants]/(main)/discover/(list)/mcp/features/List/index.tsx +33 -0
- package/src/app/[variants]/(main)/discover/(list)/mcp/page.tsx +46 -0
- package/src/app/[variants]/(main)/discover/(list)/model/Client.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(list)/{models → model}/_layout/Desktop.tsx +1 -7
- package/src/app/[variants]/(main)/discover/(list)/model/features/Category/index.tsx +82 -0
- package/src/app/[variants]/(main)/discover/(list)/model/features/Category/useCategory.tsx +41 -0
- package/src/app/[variants]/(main)/discover/(list)/model/features/List/Item.tsx +190 -0
- package/src/app/[variants]/(main)/discover/(list)/model/features/List/ModelTypeIcon.tsx +39 -0
- package/src/app/[variants]/(main)/discover/(list)/model/features/List/index.tsx +33 -0
- package/src/app/[variants]/(main)/discover/(list)/model/loading.tsx +1 -0
- package/src/app/[variants]/(main)/discover/(list)/model/page.tsx +44 -0
- package/src/app/[variants]/(main)/discover/(list)/provider/Client.tsx +43 -0
- package/src/app/[variants]/(main)/discover/(list)/provider/features/List/Item.tsx +136 -0
- package/src/app/[variants]/(main)/discover/(list)/provider/features/List/index.tsx +33 -0
- package/src/app/[variants]/(main)/discover/(list)/provider/loading.tsx +1 -0
- package/src/app/[variants]/(main)/discover/(list)/provider/page.tsx +44 -0
- package/src/app/[variants]/(main)/discover/_layout/Desktop/Header.tsx +1 -1
- package/src/app/[variants]/(main)/discover/components/CategoryContainer.tsx +7 -5
- package/src/app/[variants]/(main)/discover/components/CategoryMenu.tsx +28 -30
- package/src/app/[variants]/(main)/discover/components/ListLoading.tsx +56 -46
- package/src/app/[variants]/(main)/discover/components/Statistic.tsx +5 -6
- package/src/app/[variants]/(main)/discover/features/Search.tsx +62 -0
- package/src/app/[variants]/(main)/discover/features/Title.tsx +83 -0
- package/src/app/[variants]/(main)/discover/features/__tests__/calculateScore.test.ts +185 -0
- package/src/app/[variants]/(main)/discover/features/useNav.tsx +32 -12
- package/src/app/robots.tsx +7 -5
- package/src/app/sitemap.tsx +81 -13
- package/src/components/CopyableLabel/index.tsx +37 -0
- package/src/components/Descriptions/index.tsx +119 -0
- package/src/components/InlineTable/index.tsx +69 -0
- package/src/{features/PluginSettings/PluginSettingRender.tsx → components/JSONSchemaConfig/ItemRender.tsx} +3 -3
- package/src/components/MCPDepsIcon/Java.tsx +23 -0
- package/src/components/MCPDepsIcon/PowerShell.tsx +23 -0
- package/src/components/MCPDepsIcon/Terminal.tsx +23 -0
- package/src/components/MCPDepsIcon/UV.tsx +23 -0
- package/src/components/MCPDepsIcon/index.tsx +72 -0
- package/src/components/MCPStdioCommandInput/index.tsx +47 -0
- package/src/components/OfficialIcon.tsx +23 -0
- package/src/components/Plugins/MCPTag.tsx +18 -0
- package/src/components/Plugins/PluginTag.tsx +50 -0
- package/src/components/PublishedTime.tsx +71 -0
- package/src/config/__tests__/app.test.ts +6 -2
- package/src/const/discover.ts +8 -34
- package/src/database/models/message.ts +1 -0
- package/src/database/models/plugin.ts +13 -3
- package/src/envs/app.ts +1 -1
- package/src/features/AgentSetting/AgentPlugin/index.tsx +2 -2
- package/src/features/ChatInput/ActionBar/Tools/ToolItem.tsx +1 -1
- package/src/features/ChatInput/ActionBar/Tools/useControls.tsx +5 -5
- package/src/features/Conversation/Messages/Assistant/Tool/Inspector/index.tsx +25 -11
- package/src/features/MCP/MCPInstallProgress/InstallError/ErrorDetails.tsx +110 -0
- package/src/features/MCP/MCPInstallProgress/InstallError/index.tsx +51 -0
- package/src/features/MCP/MCPInstallProgress/MCPConfigForm.tsx +151 -0
- package/src/features/MCP/MCPInstallProgress/MCPDependenciesGuide.tsx +217 -0
- package/src/features/MCP/MCPInstallProgress/index.tsx +118 -0
- package/src/features/MCP/Scores.tsx +268 -0
- package/src/features/MCP/calculateScore.ts +324 -0
- package/src/features/MCP/useScoreList.ts +126 -0
- package/src/features/MCP/utils.ts +225 -0
- package/src/features/MCPPluginDetail/CollapseDesc.tsx +44 -0
- package/src/features/MCPPluginDetail/CollapseLayout.tsx +34 -0
- package/src/features/MCPPluginDetail/Deployment/Platform/index.tsx +47 -0
- package/src/features/MCPPluginDetail/Deployment/index.tsx +322 -0
- package/src/features/MCPPluginDetail/DetailProvider.tsx +19 -0
- package/src/features/MCPPluginDetail/Header.tsx +218 -0
- package/src/features/MCPPluginDetail/Nav.tsx +196 -0
- package/src/features/MCPPluginDetail/Overview/TagList.tsx +47 -0
- package/src/features/MCPPluginDetail/Overview/index.tsx +57 -0
- package/src/features/MCPPluginDetail/Schema/Block.tsx +50 -0
- package/src/features/MCPPluginDetail/Schema/Prompts.tsx +125 -0
- package/src/features/MCPPluginDetail/Schema/Resources.tsx +70 -0
- package/src/features/MCPPluginDetail/Schema/Tools.tsx +146 -0
- package/src/features/MCPPluginDetail/Schema/index.tsx +63 -0
- package/src/features/MCPPluginDetail/Schema/style.ts +9 -0
- package/src/features/MCPPluginDetail/Schema/types.ts +4 -0
- package/src/features/MCPPluginDetail/Score/GithubBadge/index.tsx +82 -0
- package/src/features/MCPPluginDetail/Score/ScoreItem.tsx +34 -0
- package/src/features/MCPPluginDetail/Score/ScoreList.tsx +24 -0
- package/src/features/MCPPluginDetail/Score/TotalScore.tsx +289 -0
- package/src/features/MCPPluginDetail/Score/index.tsx +88 -0
- package/src/features/PluginAvatar/index.tsx +1 -1
- package/src/features/PluginDetailModal/Meta.tsx +3 -4
- package/src/features/PluginDevModal/MCPManifestForm/MCPTypeSelect.tsx +3 -4
- package/src/features/PluginDevModal/MCPManifestForm/index.tsx +4 -41
- package/src/features/PluginDevModal/PluginPreview/EmptyState.tsx +5 -7
- package/src/features/PluginDevModal/PluginPreview/index.tsx +6 -6
- package/src/features/PluginSettings/index.tsx +2 -2
- package/src/features/PluginStore/AddPluginButton.tsx +1 -1
- package/src/features/PluginStore/Content.tsx +59 -0
- package/src/features/PluginStore/InstalledList/Detail/CustomPluginEmptyState.tsx +81 -0
- package/src/features/PluginStore/InstalledList/Detail/index.tsx +21 -0
- package/src/features/PluginStore/InstalledList/EditCustomPlugin.tsx +52 -0
- package/src/features/PluginStore/{PluginItem → InstalledList/List/Item}/Action.tsx +33 -13
- package/src/features/PluginStore/InstalledList/List/Item/index.tsx +62 -0
- package/src/features/PluginStore/InstalledList/List/index.tsx +77 -0
- package/src/features/PluginStore/InstalledList/index.tsx +85 -0
- package/src/features/PluginStore/Loading.tsx +2 -2
- package/src/features/PluginStore/McpList/Detail/Loading.tsx +44 -0
- package/src/features/PluginStore/McpList/Detail/Settings/index.tsx +381 -0
- package/src/features/PluginStore/McpList/Detail/index.tsx +71 -0
- package/src/features/PluginStore/McpList/List/Action.tsx +94 -0
- package/src/features/PluginStore/McpList/List/Item.tsx +84 -0
- package/src/features/PluginStore/McpList/List/index.tsx +97 -0
- package/src/features/PluginStore/McpList/index.tsx +54 -0
- package/src/features/PluginStore/PluginList/Detail/DetailProvider.tsx +19 -0
- package/src/features/PluginStore/PluginList/Detail/EmptyState.tsx +58 -0
- package/src/features/PluginStore/PluginList/Detail/Header.tsx +132 -0
- package/src/features/PluginStore/PluginList/Detail/InstallDetail/Nav.tsx +75 -0
- package/src/features/PluginStore/PluginList/Detail/InstallDetail/Settings.tsx +19 -0
- package/src/features/PluginStore/PluginList/Detail/InstallDetail/Tools.tsx +109 -0
- package/src/features/PluginStore/PluginList/Detail/InstallDetail/index.tsx +24 -0
- package/src/features/PluginStore/PluginList/Detail/Loading.tsx +44 -0
- package/src/features/PluginStore/PluginList/Detail/TagList.tsx +37 -0
- package/src/features/PluginStore/PluginList/Detail/index.tsx +39 -0
- package/src/features/PluginStore/PluginList/Detail/useCategory.tsx +76 -0
- package/src/features/PluginStore/PluginList/List/Action.tsx +72 -0
- package/src/features/PluginStore/PluginList/List/Item.tsx +94 -0
- package/src/features/PluginStore/PluginList/List/index.tsx +91 -0
- package/src/features/PluginStore/PluginList/index.tsx +47 -0
- package/src/features/PluginStore/Search/index.tsx +29 -0
- package/src/features/PluginStore/VirtuosoLoading.tsx +16 -0
- package/src/features/PluginStore/index.tsx +7 -31
- package/src/features/PluginTag/index.tsx +6 -4
- package/src/hooks/useInterceptingRoutes.test.ts +6 -0
- package/src/hooks/useMCPCategory.tsx +133 -0
- package/src/hooks/useQuery.ts +8 -0
- package/src/hooks/useQueryRoute.test.ts +7 -0
- package/src/hooks/useQueryRoute.ts +5 -3
- package/src/libs/mcp/client.ts +265 -13
- package/src/libs/mcp/types.ts +99 -0
- package/src/libs/trpc/client/desktop.ts +2 -3
- package/src/libs/trpc/lambda/context.ts +36 -5
- package/src/libs/trpc/lambda/index.ts +1 -1
- package/src/libs/trpc/lambda/init.ts +8 -1
- package/src/locales/default/discover.ts +457 -12
- package/src/locales/default/metadata.ts +9 -4
- package/src/locales/default/plugin.ts +89 -1
- package/src/locales/default/setting.ts +1 -0
- package/src/locales/resources.ts +1 -1
- package/src/middleware.ts +13 -3
- package/src/server/ld.ts +4 -3
- package/src/server/modules/AssistantStore/index.test.ts +10 -8
- package/src/server/modules/AssistantStore/index.ts +37 -17
- package/src/server/modules/PluginStore/index.test.ts +1 -1
- package/src/server/modules/PluginStore/index.ts +24 -1
- package/src/server/routers/desktop/mcp.ts +36 -2
- package/src/server/routers/edge/index.ts +0 -6
- package/src/server/routers/{edge → lambda}/config/index.ts +1 -1
- package/src/server/routers/lambda/index.ts +4 -0
- package/src/server/routers/lambda/market/index.ts +621 -0
- package/src/server/routers/lambda/plugin.ts +2 -0
- package/src/server/routers/tools/mcp.ts +22 -0
- package/src/server/services/discover/index.test.ts +573 -250
- package/src/server/services/discover/index.ts +1096 -241
- package/src/server/services/mcp/deps/MCPSystemDepsCheckService.ts +238 -0
- package/src/server/services/mcp/deps/checkers/ManualInstallationChecker.ts +20 -0
- package/src/server/services/mcp/deps/checkers/NpmInstallationChecker.ts +51 -0
- package/src/server/services/mcp/deps/checkers/PythonInstallationChecker.ts +69 -0
- package/src/server/services/mcp/deps/index.ts +14 -0
- package/src/server/services/mcp/deps/types.ts +49 -0
- package/src/server/services/mcp/index.ts +251 -33
- package/src/server/sitemap.test.ts +203 -46
- package/src/server/sitemap.ts +159 -52
- package/src/services/__tests__/global.test.ts +4 -7
- package/src/services/__tests__/tool.test.ts +1 -29
- package/src/services/discover.ts +365 -0
- package/src/services/global.ts +3 -3
- package/src/services/mcp.ts +131 -10
- package/src/services/plugin/_deprecated.ts +1 -1
- package/src/services/plugin/type.ts +3 -1
- package/src/services/tool.ts +9 -6
- package/src/store/chat/slices/plugin/action.test.ts +2 -3
- package/src/store/chat/slices/plugin/action.ts +22 -4
- package/src/store/discover/index.ts +1 -0
- package/src/store/discover/slices/assistant/action.ts +73 -0
- package/src/store/discover/slices/assistant/index.ts +1 -0
- package/src/store/discover/slices/mcp/action.ts +70 -0
- package/src/store/discover/slices/mcp/index.ts +1 -0
- package/src/store/discover/slices/model/action.ts +70 -0
- package/src/store/discover/slices/model/index.ts +1 -0
- package/src/store/discover/slices/plugin/action.ts +76 -0
- package/src/store/discover/slices/plugin/index.ts +1 -0
- package/src/store/discover/slices/provider/action.ts +61 -0
- package/src/store/discover/slices/provider/index.ts +1 -0
- package/src/store/discover/store.ts +39 -0
- package/src/store/tool/initialState.ts +8 -2
- package/src/store/tool/selectors/index.ts +2 -1
- package/src/store/tool/selectors/tool.test.ts +3 -1
- package/src/store/tool/selectors/tool.ts +14 -2
- package/src/store/tool/slices/mcpStore/action.ts +496 -0
- package/src/store/tool/slices/mcpStore/initialState.ts +40 -0
- package/src/store/tool/slices/mcpStore/selectors.ts +62 -0
- package/src/store/tool/slices/{store → oldStore}/action.test.ts +24 -19
- package/src/store/tool/slices/oldStore/action.ts +269 -0
- package/src/store/tool/slices/oldStore/index.ts +3 -0
- package/src/store/tool/slices/oldStore/initialState.ts +54 -0
- package/src/store/tool/slices/{store → oldStore}/selectors.test.ts +6 -3
- package/src/store/tool/slices/{store → oldStore}/selectors.ts +18 -5
- package/src/store/tool/slices/plugin/action.test.ts +2 -1
- package/src/store/tool/slices/plugin/action.ts +22 -4
- package/src/store/tool/slices/plugin/selectors.test.ts +12 -4
- package/src/store/tool/slices/plugin/selectors.ts +20 -9
- package/src/store/tool/store.ts +5 -2
- package/src/types/discover/assistants.ts +72 -0
- package/src/types/discover/index.ts +41 -0
- package/src/types/discover/mcp.ts +59 -0
- package/src/types/discover/models.ts +51 -0
- package/src/types/discover/plugins.ts +52 -0
- package/src/types/discover/providers.ts +47 -0
- package/src/types/message/tools.ts +3 -0
- package/src/types/plugins/index.ts +53 -0
- package/src/types/plugins/mcp.ts +188 -0
- package/src/types/plugins/mcpDeps.ts +30 -0
- package/src/types/tool/index.ts +10 -0
- package/src/types/tool/plugin.ts +3 -2
- package/src/types/tool/tool.ts +4 -1
- package/src/utils/client/cookie.ts +5 -2
- package/src/utils/locale.ts +2 -17
- package/src/utils/object.ts +10 -0
- package/src/utils/server/pageProps.ts +9 -0
- package/src/utils/toolCall.ts +5 -1
- package/src/utils/toolManifest.ts +1 -2
- package/vercel.json +1 -1
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/Actions.tsx +0 -35
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/ConversationExample/TopicList.tsx +0 -75
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/ConversationExample/index.tsx +0 -105
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/Header.tsx +0 -115
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/InfoSidebar/SuggestionItem.tsx +0 -62
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/InfoSidebar/ToolItem.tsx +0 -19
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/InfoSidebar/index.tsx +0 -60
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/SystemRole.tsx +0 -35
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/features/Temp.tsx +0 -44
- package/src/app/[variants]/(main)/discover/(detail)/assistant/[slug]/page.tsx +0 -124
- package/src/app/[variants]/(main)/discover/(detail)/loading.tsx +0 -38
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/Actions.tsx +0 -40
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/ChatWithModel.tsx +0 -93
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/InfoSidebar/SuggestionItem.tsx +0 -74
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/InfoSidebar/index.tsx +0 -45
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/ProviderList/ProviderItem.tsx +0 -139
- package/src/app/[variants]/(main)/discover/(detail)/model/[...slugs]/features/ProviderList/index.tsx +0 -45
- package/src/app/[variants]/(main)/discover/(detail)/plugin/[slug]/features/Actions.tsx +0 -35
- package/src/app/[variants]/(main)/discover/(detail)/plugin/[slug]/features/Header.tsx +0 -119
- package/src/app/[variants]/(main)/discover/(detail)/plugin/[slug]/features/InfoSidebar/SuggestionItem.tsx +0 -64
- package/src/app/[variants]/(main)/discover/(detail)/plugin/[slug]/features/InfoSidebar/index.tsx +0 -45
- package/src/app/[variants]/(main)/discover/(detail)/plugin/[slug]/features/InstallPlugin.tsx +0 -83
- package/src/app/[variants]/(main)/discover/(detail)/plugin/[slug]/features/ParameterList.tsx +0 -95
- package/src/app/[variants]/(main)/discover/(detail)/plugin/[slug]/features/Schema.tsx +0 -23
- package/src/app/[variants]/(main)/discover/(detail)/provider/[slug]/features/Header.tsx +0 -73
- package/src/app/[variants]/(main)/discover/(detail)/provider/[slug]/features/InfoSidebar/SuggestionItem.tsx +0 -77
- package/src/app/[variants]/(main)/discover/(detail)/provider/[slug]/features/InfoSidebar/index.tsx +0 -45
- package/src/app/[variants]/(main)/discover/(detail)/provider/[slug]/features/ModelList/ModelItem.tsx +0 -152
- package/src/app/[variants]/(main)/discover/(detail)/provider/[slug]/features/ModelList/index.tsx +0 -60
- package/src/app/[variants]/(main)/discover/(detail)/provider/[slug]/page.tsx +0 -126
- package/src/app/[variants]/(main)/discover/(list)/(home)/features/AssistantList.tsx +0 -33
- package/src/app/[variants]/(main)/discover/(list)/(home)/features/ModelList.tsx +0 -19
- package/src/app/[variants]/(main)/discover/(list)/(home)/features/PluginList.tsx +0 -25
- package/src/app/[variants]/(main)/discover/(list)/assistants/[slug]/page.tsx +0 -75
- package/src/app/[variants]/(main)/discover/(list)/assistants/features/Card.tsx +0 -195
- package/src/app/[variants]/(main)/discover/(list)/assistants/features/Category.tsx +0 -48
- package/src/app/[variants]/(main)/discover/(list)/assistants/features/List.tsx +0 -98
- package/src/app/[variants]/(main)/discover/(list)/assistants/features/useCategory.tsx +0 -116
- package/src/app/[variants]/(main)/discover/(list)/assistants/page.tsx +0 -64
- package/src/app/[variants]/(main)/discover/(list)/loading.tsx +0 -39
- package/src/app/[variants]/(main)/discover/(list)/models/[slug]/page.tsx +0 -78
- package/src/app/[variants]/(main)/discover/(list)/models/features/Card.tsx +0 -118
- package/src/app/[variants]/(main)/discover/(list)/models/features/Category.tsx +0 -67
- package/src/app/[variants]/(main)/discover/(list)/models/features/List.tsx +0 -71
- package/src/app/[variants]/(main)/discover/(list)/models/loading.tsx +0 -1
- package/src/app/[variants]/(main)/discover/(list)/models/page.tsx +0 -73
- package/src/app/[variants]/(main)/discover/(list)/plugins/[slug]/page.tsx +0 -75
- package/src/app/[variants]/(main)/discover/(list)/plugins/features/Card.tsx +0 -161
- package/src/app/[variants]/(main)/discover/(list)/plugins/features/Category.tsx +0 -45
- package/src/app/[variants]/(main)/discover/(list)/plugins/features/List.tsx +0 -97
- package/src/app/[variants]/(main)/discover/(list)/plugins/features/useCategory.tsx +0 -80
- package/src/app/[variants]/(main)/discover/(list)/plugins/page.tsx +0 -64
- package/src/app/[variants]/(main)/discover/(list)/providers/features/Card.tsx +0 -119
- package/src/app/[variants]/(main)/discover/(list)/providers/features/List.tsx +0 -67
- package/src/app/[variants]/(main)/discover/(list)/providers/loading.tsx +0 -1
- package/src/app/[variants]/(main)/discover/(list)/providers/page.tsx +0 -64
- package/src/app/[variants]/(main)/discover/features/StoreSearchBar.tsx +0 -87
- package/src/app/[variants]/(main)/discover/loading.tsx +0 -3
- package/src/app/[variants]/(main)/discover/search/_layout/Desktop.tsx +0 -42
- package/src/app/[variants]/(main)/discover/search/_layout/Mobile/Header.tsx +0 -31
- package/src/app/[variants]/(main)/discover/search/_layout/Mobile/Nav.tsx +0 -56
- package/src/app/[variants]/(main)/discover/search/_layout/Mobile/index.tsx +0 -32
- package/src/app/[variants]/(main)/discover/search/features/AssistantsResult.tsx +0 -27
- package/src/app/[variants]/(main)/discover/search/features/Category.tsx +0 -41
- package/src/app/[variants]/(main)/discover/search/features/ModelsResult.tsx +0 -27
- package/src/app/[variants]/(main)/discover/search/features/PluginsResult.tsx +0 -27
- package/src/app/[variants]/(main)/discover/search/features/ProvidersResult.tsx +0 -26
- package/src/app/[variants]/(main)/discover/search/layout.tsx +0 -12
- package/src/app/[variants]/(main)/discover/search/loading.tsx +0 -1
- package/src/app/[variants]/(main)/discover/search/page.tsx +0 -82
- package/src/features/PluginStore/InstalledPluginList.tsx +0 -59
- package/src/features/PluginStore/OnlineList.tsx +0 -87
- package/src/features/PluginStore/PluginItem/EditCustomPlugin.tsx +0 -55
- package/src/features/PluginStore/PluginItem/PluginTag.tsx +0 -29
- package/src/features/PluginStore/PluginItem/index.tsx +0 -83
- package/src/server/routers/edge/market/index.ts +0 -108
- package/src/services/__tests__/assistant.test.ts +0 -87
- package/src/services/assistant.ts +0 -25
- package/src/store/tool/slices/store/action.ts +0 -113
- package/src/store/tool/slices/store/initialState.ts +0 -17
- package/src/types/discover.ts +0 -179
- package/src/types/requestCache.ts +0 -3
- /package/src/app/[variants]/(main)/discover/(list)/{assistants → assistant}/_layout/Desktop.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{assistants → assistant}/_layout/Mobile.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{assistants → assistant}/layout.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{assistants → assistant}/loading.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{plugins → mcp}/_layout/Desktop.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{plugins → mcp}/_layout/Mobile.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{plugins → mcp}/layout.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{plugins → mcp}/loading.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{models → model}/_layout/Mobile.tsx +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{models → model}/features/const.ts +0 -0
- /package/src/app/[variants]/(main)/discover/(list)/{models → model}/layout.tsx +0 -0
- /package/src/{features/PluginStore/PluginItem → components/Plugins}/PluginAvatar.tsx +0 -0
- /package/src/server/routers/{edge → lambda}/config/__snapshots__/index.test.ts.snap +0 -0
- /package/src/server/routers/{edge → lambda}/config/index.test.ts +0 -0
- /package/src/store/tool/slices/{store → mcpStore}/index.ts +0 -0
package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/Nav.tsx
ADDED
@@ -0,0 +1,99 @@
|
|
1
|
+
'use client';
|
2
|
+
|
3
|
+
import { Icon, Tabs } from '@lobehub/ui';
|
4
|
+
import { createStyles } from 'antd-style';
|
5
|
+
import { BookOpenIcon, BrainCircuitIcon, ListIcon } from 'lucide-react';
|
6
|
+
import Link from 'next/link';
|
7
|
+
import { memo } from 'react';
|
8
|
+
import { useTranslation } from 'react-i18next';
|
9
|
+
import { Flexbox } from 'react-layout-kit';
|
10
|
+
import urlJoin from 'url-join';
|
11
|
+
|
12
|
+
import { SOCIAL_URL } from '@/const/branding';
|
13
|
+
import { ProviderNavKey } from '@/types/discover';
|
14
|
+
|
15
|
+
import { useDetailContext } from '../DetailProvider';
|
16
|
+
|
17
|
+
const useStyles = createStyles(({ css, token }) => {
|
18
|
+
return {
|
19
|
+
link: css`
|
20
|
+
color: ${token.colorTextDescription};
|
21
|
+
|
22
|
+
&:hover {
|
23
|
+
color: ${token.colorInfo};
|
24
|
+
}
|
25
|
+
`,
|
26
|
+
nav: css`
|
27
|
+
border-block-end: 1px solid ${token.colorBorder};
|
28
|
+
`,
|
29
|
+
};
|
30
|
+
});
|
31
|
+
|
32
|
+
const Nav = memo<{
|
33
|
+
activeTab?: ProviderNavKey;
|
34
|
+
mobile?: boolean;
|
35
|
+
setActiveTab?: (tab: ProviderNavKey) => void;
|
36
|
+
}>(({ mobile, setActiveTab, activeTab = ProviderNavKey.Overview }) => {
|
37
|
+
const { t } = useTranslation('discover');
|
38
|
+
const { identifier } = useDetailContext();
|
39
|
+
const { styles } = useStyles();
|
40
|
+
|
41
|
+
const nav = (
|
42
|
+
<Tabs
|
43
|
+
activeKey={activeTab}
|
44
|
+
compact={mobile}
|
45
|
+
items={[
|
46
|
+
{
|
47
|
+
icon: <Icon icon={BookOpenIcon} size={16} />,
|
48
|
+
key: ProviderNavKey.Overview,
|
49
|
+
label: t('providers.details.overview.title'),
|
50
|
+
},
|
51
|
+
{
|
52
|
+
icon: <Icon icon={BrainCircuitIcon} size={16} />,
|
53
|
+
key: ProviderNavKey.Guide,
|
54
|
+
label: t('providers.details.guide.title'),
|
55
|
+
},
|
56
|
+
{
|
57
|
+
icon: <Icon icon={ListIcon} size={16} />,
|
58
|
+
key: ProviderNavKey.Related,
|
59
|
+
label: t('providers.details.related.title'),
|
60
|
+
},
|
61
|
+
]}
|
62
|
+
onChange={(key) => setActiveTab?.(key as ProviderNavKey)}
|
63
|
+
/>
|
64
|
+
);
|
65
|
+
|
66
|
+
return mobile ? (
|
67
|
+
nav
|
68
|
+
) : (
|
69
|
+
<Flexbox align={'center'} className={styles.nav} horizontal justify={'space-between'}>
|
70
|
+
{nav}
|
71
|
+
<Flexbox gap={12} horizontal>
|
72
|
+
<Link className={styles.link} href={SOCIAL_URL.discord} target={'_blank'}>
|
73
|
+
{t('mcp.details.nav.needHelp')}
|
74
|
+
</Link>
|
75
|
+
{identifier && (
|
76
|
+
<Link
|
77
|
+
className={styles.link}
|
78
|
+
href={urlJoin(
|
79
|
+
'https://github.com/lobehub/lobe-chat/tree/main/src/config/modelProviders',
|
80
|
+
`${identifier}.ts`,
|
81
|
+
)}
|
82
|
+
target={'_blank'}
|
83
|
+
>
|
84
|
+
{t('mcp.details.nav.viewSourceCode')}
|
85
|
+
</Link>
|
86
|
+
)}
|
87
|
+
<Link
|
88
|
+
className={styles.link}
|
89
|
+
href={'https://github.com/lobehub/lobe-chat/issues/new/choose'}
|
90
|
+
target={'_blank'}
|
91
|
+
>
|
92
|
+
{t('mcp.details.nav.reportIssue')}
|
93
|
+
</Link>
|
94
|
+
</Flexbox>
|
95
|
+
</Flexbox>
|
96
|
+
);
|
97
|
+
});
|
98
|
+
|
99
|
+
export default Nav;
|
@@ -0,0 +1,142 @@
|
|
1
|
+
'use client';
|
2
|
+
|
3
|
+
import { ModelIcon } from '@lobehub/icons';
|
4
|
+
import { ActionIcon, Block, Tooltip } from '@lobehub/ui';
|
5
|
+
import { useTheme } from 'antd-style';
|
6
|
+
import { ChevronRightIcon } from 'lucide-react';
|
7
|
+
import Link from 'next/link';
|
8
|
+
import { memo } from 'react';
|
9
|
+
import { useTranslation } from 'react-i18next';
|
10
|
+
import { Flexbox } from 'react-layout-kit';
|
11
|
+
import urlJoin from 'url-join';
|
12
|
+
|
13
|
+
import InlineTable from '@/components/InlineTable';
|
14
|
+
import { ModelInfoTags } from '@/components/ModelSelect';
|
15
|
+
import { formatPriceByCurrency, formatTokenNumber } from '@/utils/format';
|
16
|
+
|
17
|
+
import { useDetailContext } from '../../../DetailProvider';
|
18
|
+
|
19
|
+
const ModelList = memo(() => {
|
20
|
+
const { models = [] } = useDetailContext();
|
21
|
+
const { t } = useTranslation('discover');
|
22
|
+
const theme = useTheme();
|
23
|
+
|
24
|
+
return (
|
25
|
+
<Block variant={'outlined'}>
|
26
|
+
<InlineTable
|
27
|
+
columns={[
|
28
|
+
{
|
29
|
+
dataIndex: 'id',
|
30
|
+
key: 'model',
|
31
|
+
render: (_, record) => {
|
32
|
+
return (
|
33
|
+
<Link href={urlJoin('/discover/model', record.id)} style={{ color: 'inherit' }}>
|
34
|
+
<Flexbox align="center" gap={8} horizontal>
|
35
|
+
<ModelIcon model={record.id} size={24} type={'avatar'} />
|
36
|
+
<Flexbox style={{ overflow: 'hidden' }}>
|
37
|
+
<div style={{ fontWeight: 500 }}>{record.displayName}</div>
|
38
|
+
<div style={{ color: theme.colorTextSecondary, fontSize: 12 }}>
|
39
|
+
{record.id}
|
40
|
+
</div>
|
41
|
+
</Flexbox>
|
42
|
+
</Flexbox>
|
43
|
+
</Link>
|
44
|
+
);
|
45
|
+
},
|
46
|
+
sorter: (a, b) => a.displayName.localeCompare(b.displayName),
|
47
|
+
title: t('providers.modelName'),
|
48
|
+
width: 200,
|
49
|
+
},
|
50
|
+
{
|
51
|
+
dataIndex: 'abilities',
|
52
|
+
key: 'abilities',
|
53
|
+
render: (_, record) => {
|
54
|
+
if (!record?.abilities || !Object.values(record?.abilities).includes(true))
|
55
|
+
return '--';
|
56
|
+
return <ModelInfoTags {...record?.abilities} />;
|
57
|
+
},
|
58
|
+
title: t('models.abilities'),
|
59
|
+
width: 120,
|
60
|
+
},
|
61
|
+
{
|
62
|
+
dataIndex: 'contextWindowTokens',
|
63
|
+
key: 'contextLength',
|
64
|
+
render: (_, record) =>
|
65
|
+
record.contextWindowTokens ? formatTokenNumber(record.contextWindowTokens) : '--',
|
66
|
+
sorter: (a, b) => (a.contextWindowTokens || 0) - (b.contextWindowTokens || 0),
|
67
|
+
title: t('models.contentLength'),
|
68
|
+
width: 120,
|
69
|
+
},
|
70
|
+
{
|
71
|
+
dataIndex: 'maxOutput',
|
72
|
+
key: 'maxOutput',
|
73
|
+
render: (_, record) => (record.maxOutput ? formatTokenNumber(record.maxOutput) : '--'),
|
74
|
+
sorter: (a, b) => (a.maxOutput || 0) - (b.maxOutput || 0),
|
75
|
+
title: (
|
76
|
+
<Tooltip title={t('models.providerInfo.maxOutputTooltip')}>
|
77
|
+
{t('models.providerInfo.maxOutput')}
|
78
|
+
</Tooltip>
|
79
|
+
),
|
80
|
+
width: 120,
|
81
|
+
},
|
82
|
+
{
|
83
|
+
dataIndex: 'inputPrice',
|
84
|
+
key: 'inputPrice',
|
85
|
+
render: (_, record) =>
|
86
|
+
record.pricing?.input
|
87
|
+
? '$' + formatPriceByCurrency(record.pricing.input, record.pricing?.currency)
|
88
|
+
: '--',
|
89
|
+
sorter: (a, b) => (a.pricing?.input || 0) - (b.pricing?.input || 0),
|
90
|
+
title: (
|
91
|
+
<Tooltip title={t('models.providerInfo.inputTooltip')}>
|
92
|
+
{t('models.providerInfo.input')}
|
93
|
+
</Tooltip>
|
94
|
+
),
|
95
|
+
width: 100,
|
96
|
+
},
|
97
|
+
{
|
98
|
+
dataIndex: 'outputPrice',
|
99
|
+
key: 'outputPrice',
|
100
|
+
render: (_, record) =>
|
101
|
+
record.pricing?.output
|
102
|
+
? '$' + formatPriceByCurrency(record.pricing.output, record.pricing?.currency)
|
103
|
+
: '--',
|
104
|
+
sorter: (a, b) => (a.pricing?.output || 0) - (b.pricing?.output || 0),
|
105
|
+
title: (
|
106
|
+
<Tooltip title={t('models.providerInfo.outputTooltip')}>
|
107
|
+
{t('models.providerInfo.output')}
|
108
|
+
</Tooltip>
|
109
|
+
),
|
110
|
+
width: 100,
|
111
|
+
},
|
112
|
+
{
|
113
|
+
align: 'right',
|
114
|
+
dataIndex: 'action',
|
115
|
+
key: 'action',
|
116
|
+
render: (_, record) => {
|
117
|
+
return (
|
118
|
+
<Flexbox align="center" gap={4} horizontal justify={'flex-end'}>
|
119
|
+
<Link href={urlJoin('/discover/model', record.id)} style={{ color: 'inherit' }}>
|
120
|
+
<ActionIcon
|
121
|
+
color={theme.colorTextDescription}
|
122
|
+
icon={ChevronRightIcon}
|
123
|
+
size={'small'}
|
124
|
+
variant={'filled'}
|
125
|
+
/>
|
126
|
+
</Link>
|
127
|
+
</Flexbox>
|
128
|
+
);
|
129
|
+
},
|
130
|
+
title: '',
|
131
|
+
width: 60,
|
132
|
+
},
|
133
|
+
]}
|
134
|
+
dataSource={models}
|
135
|
+
rowKey="id"
|
136
|
+
scroll={{ x: 900 }}
|
137
|
+
/>
|
138
|
+
</Block>
|
139
|
+
);
|
140
|
+
});
|
141
|
+
|
142
|
+
export default ModelList;
|
@@ -0,0 +1,23 @@
|
|
1
|
+
import { Tag } from '@lobehub/ui';
|
2
|
+
import { memo } from 'react';
|
3
|
+
import { useTranslation } from 'react-i18next';
|
4
|
+
import { Flexbox } from 'react-layout-kit';
|
5
|
+
|
6
|
+
import Title from '@/app/[variants]/(main)/discover/features/Title';
|
7
|
+
|
8
|
+
import { useDetailContext } from '../../DetailProvider';
|
9
|
+
import ModelList from './ModelList';
|
10
|
+
|
11
|
+
const Overview = memo(() => {
|
12
|
+
const { t } = useTranslation('discover');
|
13
|
+
const { models = [] } = useDetailContext();
|
14
|
+
|
15
|
+
return (
|
16
|
+
<Flexbox gap={16}>
|
17
|
+
<Title tag={<Tag>{models.length}</Tag>}>{t('providers.supportedModels')}</Title>
|
18
|
+
<ModelList />
|
19
|
+
</Flexbox>
|
20
|
+
);
|
21
|
+
});
|
22
|
+
|
23
|
+
export default Overview;
|
@@ -0,0 +1,22 @@
|
|
1
|
+
import { memo } from 'react';
|
2
|
+
import { useTranslation } from 'react-i18next';
|
3
|
+
import { Flexbox } from 'react-layout-kit';
|
4
|
+
|
5
|
+
import List from '../../../../../../(list)/provider/features/List';
|
6
|
+
import Title from '../../../../../../features/Title';
|
7
|
+
import { useDetailContext } from '../../DetailProvider';
|
8
|
+
|
9
|
+
const Related = memo(() => {
|
10
|
+
const { t } = useTranslation('discover');
|
11
|
+
const { related } = useDetailContext();
|
12
|
+
return (
|
13
|
+
<Flexbox gap={16}>
|
14
|
+
<Title more={t('assistants.details.related.more')} moreLink={'/discover/provider'}>
|
15
|
+
{t('assistants.details.related.listTitle')}
|
16
|
+
</Title>
|
17
|
+
<List data={related} rows={2} />
|
18
|
+
</Flexbox>
|
19
|
+
);
|
20
|
+
});
|
21
|
+
|
22
|
+
export default Related;
|
package/src/app/[variants]/(main)/discover/(detail)/provider/[...slugs]/features/Details/index.tsx
ADDED
@@ -0,0 +1,47 @@
|
|
1
|
+
'use client';
|
2
|
+
|
3
|
+
import { useResponsive } from 'antd-style';
|
4
|
+
import { useQueryState } from 'nuqs';
|
5
|
+
import { memo } from 'react';
|
6
|
+
import { Flexbox } from 'react-layout-kit';
|
7
|
+
|
8
|
+
import { ProviderNavKey } from '@/types/discover';
|
9
|
+
|
10
|
+
import Sidebar from '../Sidebar';
|
11
|
+
import Guide from './Guide';
|
12
|
+
import Nav from './Nav';
|
13
|
+
import Overview from './Overview';
|
14
|
+
import Related from './Related';
|
15
|
+
|
16
|
+
const Details = memo<{ mobile?: boolean }>(({ mobile: isMobile }) => {
|
17
|
+
const { mobile = isMobile } = useResponsive();
|
18
|
+
const [activeTab, setActiveTab] = useQueryState('activeTab', {
|
19
|
+
clearOnDefault: true,
|
20
|
+
defaultValue: ProviderNavKey.Overview,
|
21
|
+
});
|
22
|
+
|
23
|
+
return (
|
24
|
+
<Flexbox gap={24}>
|
25
|
+
<Nav activeTab={activeTab as ProviderNavKey} mobile={mobile} setActiveTab={setActiveTab} />
|
26
|
+
<Flexbox
|
27
|
+
gap={48}
|
28
|
+
horizontal={!mobile}
|
29
|
+
style={mobile ? { flexDirection: 'column-reverse' } : undefined}
|
30
|
+
>
|
31
|
+
<Flexbox
|
32
|
+
style={{
|
33
|
+
overflow: 'hidden',
|
34
|
+
}}
|
35
|
+
width={'100%'}
|
36
|
+
>
|
37
|
+
{activeTab === ProviderNavKey.Overview && <Overview />}
|
38
|
+
{activeTab === ProviderNavKey.Guide && <Guide />}
|
39
|
+
{activeTab === ProviderNavKey.Related && <Related />}
|
40
|
+
</Flexbox>
|
41
|
+
<Sidebar mobile={mobile} />
|
42
|
+
</Flexbox>
|
43
|
+
</Flexbox>
|
44
|
+
);
|
45
|
+
});
|
46
|
+
|
47
|
+
export default Details;
|
@@ -0,0 +1,99 @@
|
|
1
|
+
'use client';
|
2
|
+
|
3
|
+
import { Github, ProviderCombine } from '@lobehub/icons';
|
4
|
+
import { ActionIcon } from '@lobehub/ui';
|
5
|
+
import { createStyles, useResponsive } from 'antd-style';
|
6
|
+
import { GlobeIcon } from 'lucide-react';
|
7
|
+
import Link from 'next/link';
|
8
|
+
import { memo } from 'react';
|
9
|
+
import { useTranslation } from 'react-i18next';
|
10
|
+
import { Flexbox } from 'react-layout-kit';
|
11
|
+
import urlJoin from 'url-join';
|
12
|
+
|
13
|
+
import { useDetailContext } from './DetailProvider';
|
14
|
+
|
15
|
+
const useStyles = createStyles(({ css, token }) => {
|
16
|
+
return {
|
17
|
+
desc: css`
|
18
|
+
color: ${token.colorTextSecondary};
|
19
|
+
`,
|
20
|
+
time: css`
|
21
|
+
font-size: 12px;
|
22
|
+
color: ${token.colorTextDescription};
|
23
|
+
`,
|
24
|
+
version: css`
|
25
|
+
font-family: ${token.fontFamilyCode};
|
26
|
+
font-size: 13px;
|
27
|
+
`,
|
28
|
+
};
|
29
|
+
});
|
30
|
+
|
31
|
+
const Header = memo<{ mobile?: boolean }>(({ mobile: isMobile }) => {
|
32
|
+
const { t } = useTranslation('providers');
|
33
|
+
const { identifier, url, modelsUrl, name } = useDetailContext();
|
34
|
+
const { theme } = useStyles();
|
35
|
+
const { mobile = isMobile } = useResponsive();
|
36
|
+
|
37
|
+
return (
|
38
|
+
<Flexbox gap={12}>
|
39
|
+
<Flexbox
|
40
|
+
align={'flex-start'}
|
41
|
+
gap={8}
|
42
|
+
horizontal
|
43
|
+
justify={'space-between'}
|
44
|
+
style={{
|
45
|
+
overflow: 'hidden',
|
46
|
+
position: 'relative',
|
47
|
+
}}
|
48
|
+
>
|
49
|
+
<Flexbox align={'flex-start'} width={'100%'}>
|
50
|
+
<ProviderCombine provider={identifier} size={mobile ? 32 : 48} />
|
51
|
+
<Flexbox align={'center'} gap={4} horizontal>
|
52
|
+
{Boolean(url || modelsUrl) ? (
|
53
|
+
<Link href={url || (modelsUrl as string)} target={'_blank'}>
|
54
|
+
@{name}
|
55
|
+
</Link>
|
56
|
+
) : (
|
57
|
+
<span>@{name}</span>
|
58
|
+
)}
|
59
|
+
</Flexbox>
|
60
|
+
</Flexbox>
|
61
|
+
<Flexbox align={'center'} horizontal>
|
62
|
+
{Boolean(url || modelsUrl) && (
|
63
|
+
<Link
|
64
|
+
href={(url || modelsUrl) as string}
|
65
|
+
onClick={(e) => e.stopPropagation()}
|
66
|
+
target={'_blank'}
|
67
|
+
>
|
68
|
+
<ActionIcon color={theme.colorTextDescription} icon={GlobeIcon} />
|
69
|
+
</Link>
|
70
|
+
)}
|
71
|
+
|
72
|
+
<Link
|
73
|
+
href={urlJoin(
|
74
|
+
'https://github.com/lobehub/lobe-chat-agents/tree/main/locales',
|
75
|
+
identifier as string,
|
76
|
+
)}
|
77
|
+
onClick={(e) => e.stopPropagation()}
|
78
|
+
target={'_blank'}
|
79
|
+
>
|
80
|
+
<ActionIcon fill={theme.colorTextDescription} icon={Github} />
|
81
|
+
</Link>
|
82
|
+
</Flexbox>
|
83
|
+
</Flexbox>
|
84
|
+
|
85
|
+
<Flexbox
|
86
|
+
align={'center'}
|
87
|
+
gap={mobile ? 12 : 24}
|
88
|
+
horizontal
|
89
|
+
style={{
|
90
|
+
color: theme.colorTextSecondary,
|
91
|
+
}}
|
92
|
+
>
|
93
|
+
{t(`${identifier}.description`)}
|
94
|
+
</Flexbox>
|
95
|
+
</Flexbox>
|
96
|
+
);
|
97
|
+
});
|
98
|
+
|
99
|
+
export default Header;
|
@@ -5,13 +5,13 @@ import { Dropdown } from 'antd';
|
|
5
5
|
import { createStyles } from 'antd-style';
|
6
6
|
import { ChevronDownIcon, SquareArrowOutUpRight } from 'lucide-react';
|
7
7
|
import Link from 'next/link';
|
8
|
-
import { useRouter } from '
|
8
|
+
import { useRouter } from 'nextjs-toploader/app';
|
9
9
|
import { memo } from 'react';
|
10
10
|
import { useTranslation } from 'react-i18next';
|
11
|
-
import { FlexboxProps } from 'react-layout-kit';
|
12
11
|
|
13
12
|
import { isDeprecatedEdition } from '@/const/version';
|
14
|
-
|
13
|
+
|
14
|
+
import { useDetailContext } from '../../DetailProvider';
|
15
15
|
|
16
16
|
const useStyles = createStyles(({ css }) => ({
|
17
17
|
button: css`
|
@@ -21,15 +21,10 @@ const useStyles = createStyles(({ css }) => ({
|
|
21
21
|
`,
|
22
22
|
}));
|
23
23
|
|
24
|
-
|
25
|
-
data: DiscoverProviderItem;
|
26
|
-
identifier: string;
|
27
|
-
}
|
28
|
-
|
29
|
-
const ProviderConfig = memo<ProviderConfigProps>(({ data, identifier }) => {
|
24
|
+
const ProviderConfig = memo(() => {
|
30
25
|
const { styles } = useStyles();
|
31
26
|
const { t } = useTranslation('discover');
|
32
|
-
|
27
|
+
const { url, modelsUrl, identifier } = useDetailContext();
|
33
28
|
const router = useRouter();
|
34
29
|
const openSettings = () => {
|
35
30
|
router.push(isDeprecatedEdition ? '/settings/llm' : `/settings/provider/${identifier}`);
|
@@ -38,20 +33,20 @@ const ProviderConfig = memo<ProviderConfigProps>(({ data, identifier }) => {
|
|
38
33
|
const icon = <Icon icon={SquareArrowOutUpRight} size={16} />;
|
39
34
|
|
40
35
|
const items = [
|
41
|
-
|
36
|
+
url && {
|
42
37
|
icon,
|
43
38
|
key: 'officialSite',
|
44
39
|
label: (
|
45
|
-
<Link href={
|
40
|
+
<Link href={url} target={'_blank'}>
|
46
41
|
{t('providers.officialSite')}
|
47
42
|
</Link>
|
48
43
|
),
|
49
44
|
},
|
50
|
-
|
45
|
+
modelsUrl && {
|
51
46
|
icon,
|
52
47
|
key: 'modelSite',
|
53
48
|
label: (
|
54
|
-
<Link href={
|
49
|
+
<Link href={modelsUrl} target={'_blank'}>
|
55
50
|
{t('providers.modelSite')}
|
56
51
|
</Link>
|
57
52
|
),
|
@@ -4,46 +4,42 @@ import { ModelTag, ProviderIcon } from '@lobehub/icons';
|
|
4
4
|
import { Tag } from '@lobehub/ui';
|
5
5
|
import { memo } from 'react';
|
6
6
|
import { useTranslation } from 'react-i18next';
|
7
|
-
import { Flexbox
|
7
|
+
import { Flexbox } from 'react-layout-kit';
|
8
8
|
import urlJoin from 'url-join';
|
9
9
|
|
10
10
|
import { OFFICIAL_URL } from '@/const/url';
|
11
|
-
import { DiscoverProviderItem } from '@/types/discover';
|
12
11
|
|
13
|
-
import ShareButton from '
|
12
|
+
import ShareButton from '../../../../../features/ShareButton';
|
13
|
+
import { useDetailContext } from '../../DetailProvider';
|
14
14
|
import ProviderConfig from './ProviderConfig';
|
15
15
|
|
16
|
-
|
17
|
-
|
18
|
-
identifier: string;
|
19
|
-
}
|
20
|
-
|
21
|
-
const ProviderActions = memo<ModelActionsProps>(({ identifier, data }) => {
|
16
|
+
const ActionButton = memo(() => {
|
17
|
+
const { models = [], identifier, name } = useDetailContext();
|
22
18
|
const { t } = useTranslation('providers');
|
23
19
|
return (
|
24
20
|
<Flexbox align={'center'} gap={8} horizontal width={'100%'}>
|
25
|
-
<ProviderConfig
|
21
|
+
<ProviderConfig />
|
26
22
|
<ShareButton
|
27
23
|
meta={{
|
28
24
|
avatar: <ProviderIcon provider={identifier} size={64} type={'avatar'} />,
|
29
|
-
desc:
|
25
|
+
desc: t(`${identifier}.description`),
|
30
26
|
tags: (
|
31
27
|
<Flexbox align={'center'} gap={4} horizontal justify={'center'} wrap={'wrap'}>
|
32
|
-
{
|
28
|
+
{models
|
33
29
|
.slice(0, 4)
|
34
30
|
.filter(Boolean)
|
35
|
-
.map((
|
36
|
-
<ModelTag key={
|
31
|
+
.map((item) => (
|
32
|
+
<ModelTag key={item.id} model={item.id} style={{ margin: 0 }} />
|
37
33
|
))}
|
38
|
-
{
|
34
|
+
{models.length > 3 && <Tag>+{models.length - 3}</Tag>}
|
39
35
|
</Flexbox>
|
40
36
|
),
|
41
|
-
title:
|
42
|
-
url: urlJoin(OFFICIAL_URL, '/discover/provider', identifier),
|
37
|
+
title: name,
|
38
|
+
url: urlJoin(OFFICIAL_URL, '/discover/provider', identifier as string),
|
43
39
|
}}
|
44
40
|
/>
|
45
41
|
</Flexbox>
|
46
42
|
);
|
47
43
|
});
|
48
44
|
|
49
|
-
export default
|
45
|
+
export default ActionButton;
|
@@ -0,0 +1,60 @@
|
|
1
|
+
import { ProviderIcon } from '@lobehub/icons';
|
2
|
+
import { Block, Text } from '@lobehub/ui';
|
3
|
+
import { createStyles } from 'antd-style';
|
4
|
+
import { memo } from 'react';
|
5
|
+
import { useTranslation } from 'react-i18next';
|
6
|
+
import { Flexbox } from 'react-layout-kit';
|
7
|
+
|
8
|
+
import { DiscoverProviderItem } from '@/types/discover';
|
9
|
+
|
10
|
+
const useStyles = createStyles(({ css, token }) => {
|
11
|
+
return {
|
12
|
+
desc: css`
|
13
|
+
flex: 1;
|
14
|
+
margin: 0 !important;
|
15
|
+
font-size: 14px !important;
|
16
|
+
color: ${token.colorTextSecondary};
|
17
|
+
`,
|
18
|
+
title: css`
|
19
|
+
margin: 0 !important;
|
20
|
+
font-size: 14px !important;
|
21
|
+
font-weight: 500 !important;
|
22
|
+
|
23
|
+
&:hover {
|
24
|
+
color: ${token.colorLink};
|
25
|
+
}
|
26
|
+
`,
|
27
|
+
};
|
28
|
+
});
|
29
|
+
|
30
|
+
const RelatedItem = memo<DiscoverProviderItem>(({ identifier, name }) => {
|
31
|
+
const { styles } = useStyles();
|
32
|
+
const { t } = useTranslation('providers');
|
33
|
+
return (
|
34
|
+
<Block gap={12} horizontal key={identifier} padding={12} variant={'outlined'}>
|
35
|
+
<ProviderIcon provider={identifier} size={40} style={{ flex: 'none' }} type={'avatar'} />
|
36
|
+
<Flexbox
|
37
|
+
flex={1}
|
38
|
+
gap={6}
|
39
|
+
style={{
|
40
|
+
overflow: 'hidden',
|
41
|
+
}}
|
42
|
+
>
|
43
|
+
<Text as={'h2'} className={styles.title} ellipsis>
|
44
|
+
{name}
|
45
|
+
</Text>
|
46
|
+
<Text
|
47
|
+
as={'p'}
|
48
|
+
className={styles.desc}
|
49
|
+
ellipsis={{
|
50
|
+
rows: 2,
|
51
|
+
}}
|
52
|
+
>
|
53
|
+
{t(`${identifier}.description`)}
|
54
|
+
</Text>
|
55
|
+
</Flexbox>
|
56
|
+
</Block>
|
57
|
+
);
|
58
|
+
});
|
59
|
+
|
60
|
+
export default RelatedItem;
|
@@ -0,0 +1,34 @@
|
|
1
|
+
import Link from 'next/link';
|
2
|
+
import { memo } from 'react';
|
3
|
+
import { useTranslation } from 'react-i18next';
|
4
|
+
import { Flexbox } from 'react-layout-kit';
|
5
|
+
import urlJoin from 'url-join';
|
6
|
+
|
7
|
+
import Title from '../../../../../../features/Title';
|
8
|
+
import { useDetailContext } from '../../DetailProvider';
|
9
|
+
import Item from './Item';
|
10
|
+
|
11
|
+
const Related = memo(() => {
|
12
|
+
const { t } = useTranslation('discover');
|
13
|
+
const { related } = useDetailContext();
|
14
|
+
|
15
|
+
return (
|
16
|
+
<Flexbox gap={16}>
|
17
|
+
<Title more={t('providers.details.related.more')} moreLink={'/discover/provider'}>
|
18
|
+
{t('providers.details.related.listTitle')}
|
19
|
+
</Title>
|
20
|
+
<Flexbox gap={8}>
|
21
|
+
{related?.map((item, index) => {
|
22
|
+
const link = urlJoin('/discover/provider', item.identifier);
|
23
|
+
return (
|
24
|
+
<Link href={link} key={index} style={{ color: 'inherit', overflow: 'hidden' }}>
|
25
|
+
<Item {...item} />
|
26
|
+
</Link>
|
27
|
+
);
|
28
|
+
})}
|
29
|
+
</Flexbox>
|
30
|
+
</Flexbox>
|
31
|
+
);
|
32
|
+
});
|
33
|
+
|
34
|
+
export default Related;
|