mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-02 03:55:55 +00:00
feat: add new version notification dialog and version comparison logic
This commit is contained in:
@@ -35,6 +35,28 @@ import { LanguageSelector } from '@/components/ui/language-selector';
|
||||
import { Badge } from '@/components/ui/badge';
|
||||
import PasswordChangeDialog from '@/app/home/components/password-change-dialog/PasswordChangeDialog';
|
||||
import ApiIntegrationDialog from '@/app/home/components/api-integration-dialog/ApiIntegrationDialog';
|
||||
import NewVersionDialog from '@/app/home/components/new-version-dialog/NewVersionDialog';
|
||||
import { GitHubRelease } from '@/app/infra/http/CloudServiceClient';
|
||||
|
||||
// Compare two version strings, returns true if v1 > v2
|
||||
function compareVersions(v1: string, v2: string): boolean {
|
||||
// Remove 'v' prefix if present
|
||||
const clean1 = v1.replace(/^v/, '');
|
||||
const clean2 = v2.replace(/^v/, '');
|
||||
|
||||
const parts1 = clean1.split('.').map((p) => parseInt(p, 10) || 0);
|
||||
const parts2 = clean2.split('.').map((p) => parseInt(p, 10) || 0);
|
||||
|
||||
const maxLen = Math.max(parts1.length, parts2.length);
|
||||
|
||||
for (let i = 0; i < maxLen; i++) {
|
||||
const p1 = parts1[i] || 0;
|
||||
const p2 = parts2[i] || 0;
|
||||
if (p1 > p2) return true;
|
||||
if (p1 < p2) return false;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// TODO 侧边导航栏要加动画
|
||||
export default function HomeSidebar({
|
||||
@@ -58,6 +80,11 @@ export default function HomeSidebar({
|
||||
const [apiKeyDialogOpen, setApiKeyDialogOpen] = useState(false);
|
||||
const [languageSelectorOpen, setLanguageSelectorOpen] = useState(false);
|
||||
const [starCount, setStarCount] = useState<number | null>(null);
|
||||
const [latestRelease, setLatestRelease] = useState<GitHubRelease | null>(
|
||||
null,
|
||||
);
|
||||
const [hasNewVersion, setHasNewVersion] = useState(false);
|
||||
const [versionDialogOpen, setVersionDialogOpen] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
initSelect();
|
||||
@@ -75,6 +102,28 @@ export default function HomeSidebar({
|
||||
.catch((error) => {
|
||||
console.error('Failed to fetch GitHub star count:', error);
|
||||
});
|
||||
|
||||
// Fetch releases to check for new version
|
||||
getCloudServiceClientSync()
|
||||
.getLangBotReleases()
|
||||
.then((releases) => {
|
||||
if (releases && releases.length > 0) {
|
||||
// Find the latest non-prerelease, non-draft release
|
||||
const latestStable = releases.find((r) => !r.prerelease && !r.draft);
|
||||
const latest = latestStable || releases[0];
|
||||
setLatestRelease(latest);
|
||||
|
||||
// Compare versions
|
||||
const currentVersion = systemInfo?.version;
|
||||
if (currentVersion && latest.tag_name) {
|
||||
const isNewer = compareVersions(latest.tag_name, currentVersion);
|
||||
setHasNewVersion(isNewer);
|
||||
}
|
||||
}
|
||||
})
|
||||
.catch((error) => {
|
||||
console.error('Failed to fetch releases:', error);
|
||||
});
|
||||
}, []);
|
||||
|
||||
function handleChildClick(child: SidebarChildVO) {
|
||||
@@ -138,8 +187,18 @@ export default function HomeSidebar({
|
||||
{/* 文字 */}
|
||||
<div className={`${styles.langbotTextContainer}`}>
|
||||
<div className={`${styles.langbotText}`}>LangBot</div>
|
||||
<div className={`${styles.langbotVersion}`}>
|
||||
{systemInfo?.version}
|
||||
<div className="flex items-center gap-1.5">
|
||||
<div className={`${styles.langbotVersion}`}>
|
||||
{systemInfo?.version}
|
||||
</div>
|
||||
{hasNewVersion && (
|
||||
<Badge
|
||||
onClick={() => setVersionDialogOpen(true)}
|
||||
className="bg-red-500 hover:bg-red-600 text-white text-[0.6rem] px-1.5 py-0 h-4 cursor-pointer"
|
||||
>
|
||||
{t('plugins.new')}
|
||||
</Badge>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -350,6 +409,11 @@ export default function HomeSidebar({
|
||||
open={apiKeyDialogOpen}
|
||||
onOpenChange={setApiKeyDialogOpen}
|
||||
/>
|
||||
<NewVersionDialog
|
||||
open={versionDialogOpen}
|
||||
onOpenChange={setVersionDialogOpen}
|
||||
release={latestRelease}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -0,0 +1,107 @@
|
||||
'use client';
|
||||
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import ReactMarkdown from 'react-markdown';
|
||||
import remarkGfm from 'remark-gfm';
|
||||
import rehypeRaw from 'rehype-raw';
|
||||
import rehypeHighlight from 'rehype-highlight';
|
||||
import i18n from 'i18next';
|
||||
import { ExternalLink } from 'lucide-react';
|
||||
|
||||
import {
|
||||
Dialog,
|
||||
DialogContent,
|
||||
DialogHeader,
|
||||
DialogTitle,
|
||||
DialogFooter,
|
||||
} from '@/components/ui/dialog';
|
||||
import { Button } from '@/components/ui/button';
|
||||
import '@/styles/github-markdown.css';
|
||||
import { GitHubRelease } from '@/app/infra/http/CloudServiceClient';
|
||||
|
||||
interface NewVersionDialogProps {
|
||||
open: boolean;
|
||||
onOpenChange: (open: boolean) => void;
|
||||
release: GitHubRelease | null;
|
||||
}
|
||||
|
||||
export default function NewVersionDialog({
|
||||
open,
|
||||
onOpenChange,
|
||||
release,
|
||||
}: NewVersionDialogProps) {
|
||||
const { t } = useTranslation();
|
||||
|
||||
const getUpdateDocsUrl = () => {
|
||||
const language = i18n.language;
|
||||
if (language === 'zh-Hans' || language === 'zh-Hant') {
|
||||
return 'https://docs.langbot.app/zh/deploy/update.html';
|
||||
} else if (language === 'ja-JP') {
|
||||
return 'https://docs.langbot.app/ja/deploy/update.html';
|
||||
} else {
|
||||
return 'https://docs.langbot.app/en/deploy/update.html';
|
||||
}
|
||||
};
|
||||
|
||||
const handleViewUpdateGuide = () => {
|
||||
window.open(getUpdateDocsUrl(), '_blank');
|
||||
};
|
||||
|
||||
if (!release) return null;
|
||||
|
||||
return (
|
||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
||||
<DialogContent className="sm:max-w-[600px] max-h-[80vh] flex flex-col">
|
||||
<DialogHeader className="flex-shrink-0">
|
||||
<DialogTitle className="flex items-center gap-2">
|
||||
{t('version.newVersionAvailable')}
|
||||
<span className="text-primary font-mono">{release.tag_name}</span>
|
||||
</DialogTitle>
|
||||
</DialogHeader>
|
||||
<div className="flex-1 overflow-y-auto min-h-0 pr-2">
|
||||
<div className="markdown-body max-w-none text-sm">
|
||||
<ReactMarkdown
|
||||
remarkPlugins={[remarkGfm]}
|
||||
rehypePlugins={[rehypeRaw, rehypeHighlight]}
|
||||
components={{
|
||||
ul: ({ children }) => <ul className="list-disc">{children}</ul>,
|
||||
ol: ({ children }) => (
|
||||
<ol className="list-decimal">{children}</ol>
|
||||
),
|
||||
li: ({ children }) => <li className="ml-4">{children}</li>,
|
||||
a: ({ href, children }) => (
|
||||
<a
|
||||
href={href}
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
className="text-primary hover:underline"
|
||||
>
|
||||
{children}
|
||||
</a>
|
||||
),
|
||||
}}
|
||||
>
|
||||
{release.body || t('version.noReleaseNotes')}
|
||||
</ReactMarkdown>
|
||||
</div>
|
||||
</div>
|
||||
<DialogFooter className="flex-shrink-0 flex flex-col sm:flex-row gap-2 pt-2">
|
||||
<Button
|
||||
variant="outline"
|
||||
onClick={() => onOpenChange(false)}
|
||||
className="w-full sm:w-auto"
|
||||
>
|
||||
{t('common.close')}
|
||||
</Button>
|
||||
<Button
|
||||
onClick={handleViewUpdateGuide}
|
||||
className="w-full sm:w-auto flex items-center gap-2"
|
||||
>
|
||||
{t('version.viewUpdateGuide')}
|
||||
<ExternalLink className="w-4 h-4" />
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
);
|
||||
}
|
||||
@@ -84,4 +84,18 @@ export class CloudServiceClient extends BaseHttpClient {
|
||||
public getPluginMarketplaceURL(author: string, name: string): string {
|
||||
return `https://space.langbot.app/market/${author}/${name}`;
|
||||
}
|
||||
|
||||
public getLangBotReleases(): Promise<GitHubRelease[]> {
|
||||
return this.get<GitHubRelease[]>('/api/v1/dist/info/releases');
|
||||
}
|
||||
}
|
||||
|
||||
export interface GitHubRelease {
|
||||
tag_name: string;
|
||||
name: string;
|
||||
body: string;
|
||||
html_url: string;
|
||||
published_at: string;
|
||||
prerelease: boolean;
|
||||
draft: boolean;
|
||||
}
|
||||
|
||||
@@ -677,6 +677,11 @@ const enUS = {
|
||||
extraParametersDescription:
|
||||
'Will be attached to the request body, such as max_tokens, temperature, top_p, etc.',
|
||||
},
|
||||
version: {
|
||||
newVersionAvailable: 'New Version Available',
|
||||
viewUpdateGuide: 'View Update Guide',
|
||||
noReleaseNotes: 'No release notes available',
|
||||
},
|
||||
};
|
||||
|
||||
export default enUS;
|
||||
|
||||
@@ -682,6 +682,11 @@ const jaJP = {
|
||||
extraParametersDescription:
|
||||
'リクエストボディに追加されるパラメータ(max_tokens、temperature、top_p など)',
|
||||
},
|
||||
version: {
|
||||
newVersionAvailable: '新しいバージョンが利用可能',
|
||||
viewUpdateGuide: 'アップデート方法を見る',
|
||||
noReleaseNotes: 'リリースノートはありません',
|
||||
},
|
||||
};
|
||||
|
||||
export default jaJP;
|
||||
|
||||
@@ -651,6 +651,11 @@ const zhHans = {
|
||||
extraParametersDescription:
|
||||
'将在请求时附加到请求体中,如 max_tokens, temperature, top_p 等',
|
||||
},
|
||||
version: {
|
||||
newVersionAvailable: '有新版本可用',
|
||||
viewUpdateGuide: '查看更新方式',
|
||||
noReleaseNotes: '暂无更新日志',
|
||||
},
|
||||
};
|
||||
|
||||
export default zhHans;
|
||||
|
||||
@@ -649,6 +649,11 @@ const zhHant = {
|
||||
extraParametersDescription:
|
||||
'將在請求時附加到請求體中,如 max_tokens, temperature, top_p 等',
|
||||
},
|
||||
version: {
|
||||
newVersionAvailable: '有新版本可用',
|
||||
viewUpdateGuide: '查看更新方式',
|
||||
noReleaseNotes: '暫無更新日誌',
|
||||
},
|
||||
};
|
||||
|
||||
export default zhHant;
|
||||
|
||||
Reference in New Issue
Block a user