@astro-minimax/ai 0.9.0 → 0.9.3
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/README.md +108 -18
- package/dist/cache/global-cache.d.ts +6 -2
- package/dist/cache/global-cache.d.ts.map +1 -1
- package/dist/cache/global-cache.js +24 -9
- package/dist/cache/index.d.ts +7 -6
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +12 -4
- package/dist/cache/injection-cache.d.ts +36 -0
- package/dist/cache/injection-cache.d.ts.map +1 -0
- package/dist/cache/injection-cache.js +90 -0
- package/dist/cache/kv-adapter.d.ts.map +1 -1
- package/dist/cache/kv-adapter.js +2 -1
- package/dist/cache/memory-adapter.d.ts.map +1 -1
- package/dist/cache/memory-adapter.js +2 -1
- package/dist/cache/response-cache.d.ts +10 -5
- package/dist/cache/response-cache.d.ts.map +1 -1
- package/dist/cache/response-cache.js +18 -6
- package/dist/components/AIChatContainer.d.ts +2 -2
- package/dist/components/AIChatContainer.d.ts.map +1 -1
- package/dist/components/AIChatContainer.js +8 -920
- package/dist/components/ChatInput.d.ts +15 -0
- package/dist/components/ChatInput.d.ts.map +1 -0
- package/dist/components/ChatInput.js +72 -0
- package/dist/components/ChatPanel.d.ts +1 -1
- package/dist/components/ChatPanel.d.ts.map +1 -1
- package/dist/components/ChatPanel.js +210 -672
- package/dist/components/CodeBlock.d.ts +31 -0
- package/dist/components/CodeBlock.d.ts.map +1 -0
- package/dist/components/CodeBlock.js +143 -0
- package/dist/components/MarkmapBlock.d.ts +4 -0
- package/dist/components/MarkmapBlock.d.ts.map +1 -0
- package/dist/components/MarkmapBlock.js +180 -0
- package/dist/components/MermaidBlock.d.ts +4 -0
- package/dist/components/MermaidBlock.d.ts.map +1 -0
- package/dist/components/MermaidBlock.js +193 -0
- package/dist/components/MessageBubble.d.ts +21 -0
- package/dist/components/MessageBubble.d.ts.map +1 -0
- package/dist/components/MessageBubble.js +233 -0
- package/dist/components/ReasoningBlock.d.ts +6 -0
- package/dist/components/ReasoningBlock.d.ts.map +1 -0
- package/dist/components/ReasoningBlock.js +11 -0
- package/dist/components/RichText.d.ts +41 -0
- package/dist/components/RichText.d.ts.map +1 -0
- package/dist/components/RichText.js +202 -0
- package/dist/components/VizShared.d.ts +57 -0
- package/dist/components/VizShared.d.ts.map +1 -0
- package/dist/components/VizShared.js +233 -0
- package/dist/components/tool-auto-continue.d.ts +5 -0
- package/dist/components/tool-auto-continue.d.ts.map +1 -0
- package/dist/components/tool-auto-continue.js +33 -0
- package/dist/constants.d.ts +61 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +72 -0
- package/dist/data/index.d.ts +4 -3
- package/dist/data/index.d.ts.map +1 -1
- package/dist/data/index.js +4 -10
- package/dist/data/knowledge-types.d.ts +8 -0
- package/dist/data/knowledge-types.d.ts.map +1 -0
- package/dist/data/knowledge-types.js +14 -0
- package/dist/data/metadata-loader.d.ts +4 -28
- package/dist/data/metadata-loader.d.ts.map +1 -1
- package/dist/data/metadata-loader.js +11 -34
- package/dist/data/types.d.ts +17 -2
- package/dist/data/types.d.ts.map +1 -1
- package/dist/extensions/index.d.ts +5 -0
- package/dist/extensions/index.d.ts.map +1 -0
- package/dist/extensions/index.js +24 -0
- package/dist/extensions/injector.d.ts +14 -0
- package/dist/extensions/injector.d.ts.map +1 -0
- package/dist/extensions/injector.js +146 -0
- package/dist/extensions/loader.d.ts +5 -0
- package/dist/extensions/loader.d.ts.map +1 -0
- package/dist/extensions/loader.js +45 -0
- package/dist/extensions/registry.d.ts +4 -0
- package/dist/extensions/registry.d.ts.map +1 -0
- package/dist/extensions/registry.js +144 -0
- package/dist/extensions/types.d.ts +126 -0
- package/dist/extensions/types.d.ts.map +1 -0
- package/dist/extensions/types.js +0 -0
- package/dist/fact-registry/prompt-injector.d.ts +1 -1
- package/dist/fact-registry/prompt-injector.d.ts.map +1 -1
- package/dist/fact-registry/prompt-injector.js +2 -1
- package/dist/index.d.ts +3 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +3 -2
- package/dist/intelligence/citation-guard.d.ts +2 -13
- package/dist/intelligence/citation-guard.d.ts.map +1 -1
- package/dist/intelligence/citation-guard.js +52 -23
- package/dist/intelligence/evidence-analysis.d.ts +24 -16
- package/dist/intelligence/evidence-analysis.d.ts.map +1 -1
- package/dist/intelligence/evidence-analysis.js +118 -20
- package/dist/intelligence/evidence-budget.d.ts +13 -0
- package/dist/intelligence/evidence-budget.d.ts.map +1 -0
- package/dist/intelligence/evidence-budget.js +49 -0
- package/dist/intelligence/index.d.ts +10 -4
- package/dist/intelligence/index.d.ts.map +1 -1
- package/dist/intelligence/index.js +27 -3
- package/dist/intelligence/keyword-extract.d.ts +1 -1
- package/dist/intelligence/keyword-extract.d.ts.map +1 -1
- package/dist/intelligence/keyword-extract.js +5 -9
- package/dist/intelligence/request-interpretation.d.ts +40 -0
- package/dist/intelligence/request-interpretation.d.ts.map +1 -0
- package/dist/intelligence/request-interpretation.js +71 -0
- package/dist/intelligence/response-templates.d.ts +1 -0
- package/dist/intelligence/response-templates.d.ts.map +1 -1
- package/dist/intelligence/response-templates.js +13 -0
- package/dist/prompt/dynamic-layer.d.ts +1 -5
- package/dist/prompt/dynamic-layer.d.ts.map +1 -1
- package/dist/prompt/dynamic-layer.js +145 -9
- package/dist/prompt/prompt-builder.d.ts +1 -1
- package/dist/prompt/prompt-builder.d.ts.map +1 -1
- package/dist/prompt/prompt-builder.js +5 -1
- package/dist/prompt/semi-static-layer.d.ts +1 -1
- package/dist/prompt/semi-static-layer.d.ts.map +1 -1
- package/dist/prompt/semi-static-layer.js +22 -12
- package/dist/prompt/static-layer.d.ts.map +1 -1
- package/dist/prompt/static-layer.js +37 -4
- package/dist/prompt/types.d.ts +9 -4
- package/dist/prompt/types.d.ts.map +1 -1
- package/dist/provider-manager/base.d.ts +5 -1
- package/dist/provider-manager/base.d.ts.map +1 -1
- package/dist/provider-manager/base.js +22 -2
- package/dist/provider-manager/config.d.ts.map +1 -1
- package/dist/provider-manager/config.js +3 -2
- package/dist/provider-manager/index.d.ts +1 -1
- package/dist/provider-manager/index.d.ts.map +1 -1
- package/dist/provider-manager/index.js +1 -2
- package/dist/provider-manager/manager.d.ts +10 -1
- package/dist/provider-manager/manager.d.ts.map +1 -1
- package/dist/provider-manager/manager.js +26 -10
- package/dist/provider-manager/openai.d.ts +2 -2
- package/dist/provider-manager/openai.d.ts.map +1 -1
- package/dist/provider-manager/openai.js +19 -4
- package/dist/provider-manager/types.d.ts +18 -38
- package/dist/provider-manager/types.d.ts.map +1 -1
- package/dist/provider-manager/workers.d.ts +2 -2
- package/dist/provider-manager/workers.d.ts.map +1 -1
- package/dist/provider-manager/workers.js +15 -4
- package/dist/query/followup.d.ts +7 -0
- package/dist/query/followup.d.ts.map +1 -0
- package/dist/query/followup.js +46 -0
- package/dist/query/intent.d.ts +6 -0
- package/dist/query/intent.d.ts.map +1 -0
- package/dist/query/intent.js +137 -0
- package/dist/query/types.d.ts +8 -0
- package/dist/query/types.d.ts.map +1 -0
- package/dist/query/types.js +0 -0
- package/dist/search/hybrid-search.d.ts +111 -0
- package/dist/search/hybrid-search.d.ts.map +1 -0
- package/dist/search/hybrid-search.js +326 -0
- package/dist/search/index.d.ts +11 -9
- package/dist/search/index.d.ts.map +1 -1
- package/dist/search/index.js +46 -10
- package/dist/search/scoring.d.ts +18 -0
- package/dist/search/scoring.d.ts.map +1 -0
- package/dist/search/{search-utils.js → scoring.js} +14 -27
- package/dist/search/search-api.d.ts +16 -1
- package/dist/search/search-api.d.ts.map +1 -1
- package/dist/search/search-api.js +118 -15
- package/dist/search/search-index.d.ts +2 -2
- package/dist/search/search-index.d.ts.map +1 -1
- package/dist/search/search-index.js +4 -2
- package/dist/search/session-cache.d.ts +4 -10
- package/dist/search/session-cache.d.ts.map +1 -1
- package/dist/search/session-cache.js +12 -45
- package/dist/search/types.d.ts +28 -0
- package/dist/search/types.d.ts.map +1 -1
- package/dist/search/vector-reranker.d.ts +3 -3
- package/dist/search/vector-reranker.d.ts.map +1 -1
- package/dist/search/vector-reranker.js +14 -2
- package/dist/server/chat-handler.d.ts +86 -1
- package/dist/server/chat-handler.d.ts.map +1 -1
- package/dist/server/chat-handler.js +835 -401
- package/dist/server/chat-message-utils.d.ts +6 -0
- package/dist/server/chat-message-utils.d.ts.map +1 -0
- package/dist/server/chat-message-utils.js +40 -0
- package/dist/server/chat-utils.d.ts +30 -0
- package/dist/server/chat-utils.d.ts.map +1 -0
- package/dist/server/chat-utils.js +88 -0
- package/dist/server/dev-server.js +238 -101
- package/dist/server/env-config.d.ts +22 -0
- package/dist/server/env-config.d.ts.map +1 -0
- package/dist/server/env-config.js +25 -0
- package/dist/server/errors.d.ts +1 -0
- package/dist/server/errors.d.ts.map +1 -1
- package/dist/server/errors.js +14 -7
- package/dist/server/index.d.ts +2 -4
- package/dist/server/index.d.ts.map +1 -1
- package/dist/server/index.js +4 -25
- package/dist/server/metadata-init.d.ts +10 -5
- package/dist/server/metadata-init.d.ts.map +1 -1
- package/dist/server/metadata-init.js +78 -34
- package/dist/server/notify.d.ts +12 -11
- package/dist/server/notify.d.ts.map +1 -1
- package/dist/server/notify.js +46 -48
- package/dist/server/prompt-runtime.d.ts +60 -0
- package/dist/server/prompt-runtime.d.ts.map +1 -0
- package/dist/server/prompt-runtime.js +284 -0
- package/dist/server/stream-helpers.d.ts +30 -16
- package/dist/server/stream-helpers.d.ts.map +1 -1
- package/dist/server/stream-helpers.js +152 -15
- package/dist/server/types.d.ts +47 -12
- package/dist/server/types.d.ts.map +1 -1
- package/dist/structured-output/generator.d.ts +6 -0
- package/dist/structured-output/generator.d.ts.map +1 -0
- package/dist/structured-output/generator.js +164 -0
- package/dist/structured-output/index.d.ts +4 -0
- package/dist/structured-output/index.d.ts.map +1 -0
- package/dist/structured-output/index.js +6 -0
- package/dist/structured-output/schemas/evidence.d.ts +88 -0
- package/dist/structured-output/schemas/evidence.d.ts.map +1 -0
- package/dist/structured-output/schemas/evidence.js +65 -0
- package/dist/structured-output/types.d.ts +69 -0
- package/dist/structured-output/types.d.ts.map +1 -0
- package/dist/structured-output/types.js +0 -0
- package/dist/tools/action-tools.d.ts +63 -0
- package/dist/tools/action-tools.d.ts.map +1 -0
- package/dist/tools/action-tools.js +158 -0
- package/dist/tools/index.d.ts +2 -0
- package/dist/tools/index.d.ts.map +1 -0
- package/dist/tools/index.js +30 -0
- package/dist/utils/i18n.d.ts +1 -1
- package/dist/utils/i18n.d.ts.map +1 -1
- package/dist/utils/i18n.js +1 -1
- package/dist/utils/logger.d.ts +11 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +36 -0
- package/dist/utils/text.d.ts +11 -0
- package/dist/utils/text.d.ts.map +1 -0
- package/dist/utils/text.js +87 -0
- package/dist/utils/url.d.ts +19 -0
- package/dist/utils/url.d.ts.map +1 -0
- package/dist/utils/url.js +13 -0
- package/package.json +46 -12
- package/dist/intelligence/intent-detect.d.ts +0 -40
- package/dist/intelligence/intent-detect.d.ts.map +0 -1
- package/dist/intelligence/intent-detect.js +0 -93
- package/dist/providers/index.d.ts +0 -2
- package/dist/providers/index.d.ts.map +0 -1
- package/dist/providers/index.js +0 -5
- package/dist/search/search-utils.d.ts +0 -47
- package/dist/search/search-utils.d.ts.map +0 -1
- package/dist/stream/index.d.ts +0 -3
- package/dist/stream/index.d.ts.map +0 -1
- package/dist/stream/index.js +0 -8
- package/dist/stream/mock-stream.d.ts +0 -12
- package/dist/stream/mock-stream.d.ts.map +0 -1
- package/dist/stream/mock-stream.js +0 -26
- package/dist/stream/response.d.ts +0 -10
- package/dist/stream/response.d.ts.map +0 -1
- package/dist/stream/response.js +0 -21
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import type { FunctionalComponent, VNode } from "preact";
|
|
2
|
+
import { MermaidBlock } from "./MermaidBlock.js";
|
|
3
|
+
import { MarkmapBlock } from "./MarkmapBlock.js";
|
|
4
|
+
export { MermaidBlock, MarkmapBlock };
|
|
5
|
+
export interface CodeBlockProps {
|
|
6
|
+
code: string;
|
|
7
|
+
lang?: string;
|
|
8
|
+
isStreaming?: boolean;
|
|
9
|
+
}
|
|
10
|
+
export interface HighlightResult {
|
|
11
|
+
html: string;
|
|
12
|
+
loading: boolean;
|
|
13
|
+
error?: string;
|
|
14
|
+
}
|
|
15
|
+
export declare function useShikiHighlighter(code: string, lang?: string): HighlightResult;
|
|
16
|
+
export declare function CodeBlock({ code, lang, isStreaming }: CodeBlockProps): VNode;
|
|
17
|
+
export interface FollowUpSuggestion {
|
|
18
|
+
text: string;
|
|
19
|
+
icon?: string;
|
|
20
|
+
}
|
|
21
|
+
interface FollowUpSuggestionsProps {
|
|
22
|
+
suggestions: FollowUpSuggestion[];
|
|
23
|
+
onSend: (text: string) => void;
|
|
24
|
+
lang?: string;
|
|
25
|
+
}
|
|
26
|
+
export declare function generateFollowUpSuggestions(responseText: string, articleContext?: {
|
|
27
|
+
title?: string;
|
|
28
|
+
keyPoints?: string[];
|
|
29
|
+
}): FollowUpSuggestion[];
|
|
30
|
+
export declare const FollowUpSuggestions: FunctionalComponent<FollowUpSuggestionsProps>;
|
|
31
|
+
//# sourceMappingURL=CodeBlock.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"CodeBlock.d.ts","sourceRoot":"","sources":["../../src/components/CodeBlock.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,mBAAmB,EAAE,KAAK,EAAE,MAAM,QAAQ,CAAC;AAEzD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAGlD,OAAO,EAAE,YAAY,EAAE,YAAY,EAAE,CAAC;AAEtC,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,OAAO,CAAC;CACvB;AAED,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,OAAO,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AA+BD,wBAAgB,mBAAmB,CACjC,IAAI,EAAE,MAAM,EACZ,IAAI,CAAC,EAAE,MAAM,GACZ,eAAe,CA0DjB;AAaD,wBAAgB,SAAS,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,cAAc,GAAG,KAAK,CA6D5E;AAID,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED,UAAU,wBAAwB;IAChC,WAAW,EAAE,kBAAkB,EAAE,CAAC;IAClC,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IAC/B,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED,wBAAgB,2BAA2B,CACzC,YAAY,EAAE,MAAM,EACpB,cAAc,CAAC,EAAE;IAAE,KAAK,CAAC,EAAE,MAAM,CAAC;IAAC,SAAS,CAAC,EAAE,MAAM,EAAE,CAAA;CAAE,GACxD,kBAAkB,EAAE,CAwCtB;AAED,eAAO,MAAM,mBAAmB,EAAE,mBAAmB,CACnD,wBAAwB,CAkBzB,CAAC"}
|
|
@@ -0,0 +1,143 @@
|
|
|
1
|
+
import { useEffect, useState } from "preact/hooks";
|
|
2
|
+
import { h } from 'preact';
|
|
3
|
+
|
|
4
|
+
import { CopyButton } from "./VizShared.js";
|
|
5
|
+
import { MermaidBlock } from "./MermaidBlock.js";
|
|
6
|
+
import { MarkmapBlock } from "./MarkmapBlock.js";
|
|
7
|
+
import { normalizeCodeBlockLang } from "../utils/text.js";
|
|
8
|
+
let shikiModule = null;
|
|
9
|
+
let highlighterCache = null;
|
|
10
|
+
async function loadShikiHighlighter() {
|
|
11
|
+
if (highlighterCache) return highlighterCache;
|
|
12
|
+
if (!shikiModule) {
|
|
13
|
+
shikiModule = import("shiki").catch(() => null);
|
|
14
|
+
}
|
|
15
|
+
const mod = await shikiModule;
|
|
16
|
+
if (!mod || typeof mod !== "object") return null;
|
|
17
|
+
const shiki = mod;
|
|
18
|
+
if (typeof shiki.codeToHtml === "function") {
|
|
19
|
+
highlighterCache = shiki;
|
|
20
|
+
return highlighterCache;
|
|
21
|
+
}
|
|
22
|
+
return null;
|
|
23
|
+
}
|
|
24
|
+
function useShikiHighlighter(code, lang) {
|
|
25
|
+
const [html, setHtml] = useState("");
|
|
26
|
+
const [loading, setLoading] = useState(true);
|
|
27
|
+
const [error, setError] = useState();
|
|
28
|
+
const normalizedLang = normalizeCodeBlockLang(lang);
|
|
29
|
+
useEffect(() => {
|
|
30
|
+
let mounted = true;
|
|
31
|
+
setLoading(true);
|
|
32
|
+
setError(void 0);
|
|
33
|
+
loadShikiHighlighter().then(async (highlighter) => {
|
|
34
|
+
if (!mounted) return;
|
|
35
|
+
if (!highlighter) {
|
|
36
|
+
setHtml(escapeHtml(code));
|
|
37
|
+
setLoading(false);
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
try {
|
|
41
|
+
const isDark = typeof document !== "undefined" && document.documentElement.getAttribute("data-theme") === "dark";
|
|
42
|
+
const theme = isDark ? "github-dark" : "github-light";
|
|
43
|
+
const highlighted = await highlighter.codeToHtml(code, {
|
|
44
|
+
lang: normalizedLang,
|
|
45
|
+
theme
|
|
46
|
+
});
|
|
47
|
+
if (mounted) {
|
|
48
|
+
setHtml(highlighted);
|
|
49
|
+
setLoading(false);
|
|
50
|
+
}
|
|
51
|
+
} catch (err) {
|
|
52
|
+
if (mounted) {
|
|
53
|
+
setHtml(escapeHtml(code));
|
|
54
|
+
setError(err instanceof Error ? err.message : "Highlight error");
|
|
55
|
+
setLoading(false);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
}).catch((err) => {
|
|
59
|
+
if (mounted) {
|
|
60
|
+
setHtml(escapeHtml(code));
|
|
61
|
+
setError(err instanceof Error ? err.message : "Shiki load error");
|
|
62
|
+
setLoading(false);
|
|
63
|
+
}
|
|
64
|
+
});
|
|
65
|
+
return () => {
|
|
66
|
+
mounted = false;
|
|
67
|
+
};
|
|
68
|
+
}, [code, normalizedLang]);
|
|
69
|
+
return { html, loading, error };
|
|
70
|
+
}
|
|
71
|
+
function escapeHtml(text) {
|
|
72
|
+
return text.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">").replace(/"/g, """).replace(/'/g, "'");
|
|
73
|
+
}
|
|
74
|
+
function CodeBlock({ code, lang, isStreaming }) {
|
|
75
|
+
const normalizedLang = normalizeCodeBlockLang(lang);
|
|
76
|
+
if (normalizedLang === "mermaid") {
|
|
77
|
+
return /* @__PURE__ */ h(MermaidBlock, { code, isStreaming });
|
|
78
|
+
}
|
|
79
|
+
if (normalizedLang === "markmap" || normalizedLang === "mindmap") {
|
|
80
|
+
return /* @__PURE__ */ h(MarkmapBlock, { code, isStreaming });
|
|
81
|
+
}
|
|
82
|
+
const { html, loading, error } = useShikiHighlighter(
|
|
83
|
+
isStreaming ? "" : code,
|
|
84
|
+
lang
|
|
85
|
+
);
|
|
86
|
+
if (isStreaming || loading) {
|
|
87
|
+
return /* @__PURE__ */ h("div", { class: "code-block-wrapper group relative" }, /* @__PURE__ */ h("pre", { class: "bg-muted/60 overflow-x-auto rounded-md px-3 py-2 font-mono text-[12px] leading-relaxed" }, /* @__PURE__ */ h("code", null, code)), lang && /* @__PURE__ */ h("span", { class: "text-foreground-soft/50 bg-muted/80 absolute top-1 right-2 rounded px-1.5 py-0.5 text-[10px] font-medium" }, lang));
|
|
88
|
+
}
|
|
89
|
+
if (html && !error) {
|
|
90
|
+
return /* @__PURE__ */ h("div", { class: "code-block-wrapper group relative" }, /* @__PURE__ */ h(
|
|
91
|
+
"div",
|
|
92
|
+
{
|
|
93
|
+
class: "code-highlight [&_pre]:!bg-muted/60 overflow-x-auto rounded-md text-[12px] leading-relaxed [&_pre]:!m-0 [&_pre]:!p-3",
|
|
94
|
+
dangerouslySetInnerHTML: { __html: html }
|
|
95
|
+
}
|
|
96
|
+
), lang && /* @__PURE__ */ h("span", { class: "text-foreground-soft/50 bg-muted/80 absolute top-1 right-2 rounded px-1.5 py-0.5 text-[10px] font-medium" }, lang), /* @__PURE__ */ h(CopyButton, { code }));
|
|
97
|
+
}
|
|
98
|
+
return /* @__PURE__ */ h("div", { class: "code-block-wrapper group relative" }, /* @__PURE__ */ h("pre", { class: "bg-muted/60 overflow-x-auto rounded-md px-3 py-2 font-mono text-[12px] leading-relaxed" }, /* @__PURE__ */ h("code", null, code)), lang && /* @__PURE__ */ h("span", { class: "text-foreground-soft/50 bg-muted/80 absolute top-1 right-2 rounded px-1.5 py-0.5 text-[10px] font-medium" }, lang), /* @__PURE__ */ h(CopyButton, { code }));
|
|
99
|
+
}
|
|
100
|
+
function generateFollowUpSuggestions(responseText, articleContext) {
|
|
101
|
+
const suggestions = [];
|
|
102
|
+
const lowerText = responseText.toLowerCase();
|
|
103
|
+
if (lowerText.includes("```") || lowerText.includes("code") || lowerText.includes("\u4EE3\u7801")) {
|
|
104
|
+
suggestions.push({ text: "\u89E3\u91CA\u8FD9\u6BB5\u4EE3\u7801", icon: "code" });
|
|
105
|
+
}
|
|
106
|
+
if (lowerText.includes("config") || lowerText.includes("\u914D\u7F6E") || lowerText.includes("\u8BBE\u7F6E")) {
|
|
107
|
+
suggestions.push({ text: "\u8BE6\u7EC6\u914D\u7F6E\u6B65\u9AA4", icon: "config" });
|
|
108
|
+
}
|
|
109
|
+
if (lowerText.includes("\u6587\u7AE0") || lowerText.includes("article") || lowerText.includes("post")) {
|
|
110
|
+
suggestions.push({ text: "\u63A8\u8350\u76F8\u5173\u6587\u7AE0", icon: "article" });
|
|
111
|
+
}
|
|
112
|
+
if (lowerText.includes("\u5982\u4F55") || lowerText.includes("how to")) {
|
|
113
|
+
suggestions.push({ text: "\u4E3E\u4E2A\u5177\u4F53\u4F8B\u5B50", icon: "example" });
|
|
114
|
+
}
|
|
115
|
+
if (articleContext?.title) {
|
|
116
|
+
suggestions.push({
|
|
117
|
+
text: `\u8BE6\u89E3 "${articleContext.title.slice(0, 20)}..."`,
|
|
118
|
+
icon: "detail"
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
return suggestions.slice(0, 3);
|
|
122
|
+
}
|
|
123
|
+
const FollowUpSuggestions = ({ suggestions, onSend }) => {
|
|
124
|
+
if (suggestions.length === 0) return null;
|
|
125
|
+
return /* @__PURE__ */ h("div", { class: "follow-up-suggestions mt-2 flex flex-wrap gap-1.5" }, suggestions.map((s, i) => /* @__PURE__ */ h(
|
|
126
|
+
"button",
|
|
127
|
+
{
|
|
128
|
+
key: i,
|
|
129
|
+
type: "button",
|
|
130
|
+
onClick: () => onSend(s.text),
|
|
131
|
+
class: "border-border bg-muted/30 text-foreground-soft hover:border-accent/40 hover:bg-accent/10 hover:text-foreground inline-flex items-center gap-1 rounded-full border px-2.5 py-1 text-[11px] transition-colors"
|
|
132
|
+
},
|
|
133
|
+
s.text
|
|
134
|
+
)));
|
|
135
|
+
};
|
|
136
|
+
export {
|
|
137
|
+
CodeBlock,
|
|
138
|
+
FollowUpSuggestions,
|
|
139
|
+
MarkmapBlock,
|
|
140
|
+
MermaidBlock,
|
|
141
|
+
generateFollowUpSuggestions,
|
|
142
|
+
useShikiHighlighter
|
|
143
|
+
};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MarkmapBlock.d.ts","sourceRoot":"","sources":["../../src/components/MarkmapBlock.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,QAAQ,CAAC;AACpC,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAoFtD,wBAAgB,YAAY,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,cAAc,GAAG,KAAK,GAAG,IAAI,CAkKhF"}
|
|
@@ -0,0 +1,180 @@
|
|
|
1
|
+
import { useCallback, useEffect, useRef, useState } from "preact/hooks";
|
|
2
|
+
import { h, Fragment } from 'preact';
|
|
3
|
+
|
|
4
|
+
import { SkeletonLoader, useDragPanScroll, useScaledCanvas, useVizScaleControls, VizToolbar } from "./VizShared.js";
|
|
5
|
+
function useMarkmap(svgRef, code, visible) {
|
|
6
|
+
const [loading, setLoading] = useState(true);
|
|
7
|
+
const [error, setError] = useState();
|
|
8
|
+
const markmapInstanceRef = useRef(null);
|
|
9
|
+
useEffect(() => {
|
|
10
|
+
if (!code || !visible) {
|
|
11
|
+
setLoading(false);
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
const svgEl = svgRef.current;
|
|
15
|
+
if (!svgEl) return;
|
|
16
|
+
let mounted = true;
|
|
17
|
+
setLoading(true);
|
|
18
|
+
setError(void 0);
|
|
19
|
+
if (markmapInstanceRef.current?.destroy) {
|
|
20
|
+
markmapInstanceRef.current.destroy();
|
|
21
|
+
markmapInstanceRef.current = null;
|
|
22
|
+
}
|
|
23
|
+
Promise.all([
|
|
24
|
+
import("markmap-lib"),
|
|
25
|
+
import("markmap-view")
|
|
26
|
+
]).then(async ([markmapLib, markmapView]) => {
|
|
27
|
+
if (!mounted) return;
|
|
28
|
+
try {
|
|
29
|
+
const { Transformer } = markmapLib;
|
|
30
|
+
const { Markmap } = markmapView;
|
|
31
|
+
const transformer = new Transformer();
|
|
32
|
+
const { root } = transformer.transform(code);
|
|
33
|
+
const mm = new Markmap(svgEl);
|
|
34
|
+
mm.setData(root);
|
|
35
|
+
mm.fit();
|
|
36
|
+
markmapInstanceRef.current = mm;
|
|
37
|
+
if (mounted) {
|
|
38
|
+
setLoading(false);
|
|
39
|
+
}
|
|
40
|
+
} catch (err) {
|
|
41
|
+
if (mounted) {
|
|
42
|
+
setError(err instanceof Error ? err.message : "Markmap error");
|
|
43
|
+
setLoading(false);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
}).catch(() => {
|
|
47
|
+
if (mounted) {
|
|
48
|
+
setError("Markmap not available");
|
|
49
|
+
setLoading(false);
|
|
50
|
+
}
|
|
51
|
+
});
|
|
52
|
+
return () => {
|
|
53
|
+
mounted = false;
|
|
54
|
+
if (markmapInstanceRef.current?.destroy) {
|
|
55
|
+
markmapInstanceRef.current.destroy();
|
|
56
|
+
}
|
|
57
|
+
};
|
|
58
|
+
}, [code, visible]);
|
|
59
|
+
return { loading, error, markmapInstanceRef };
|
|
60
|
+
}
|
|
61
|
+
function MarkmapBlock({ code, isStreaming }) {
|
|
62
|
+
const containerRef = useRef(null);
|
|
63
|
+
const viewportRef = useRef(null);
|
|
64
|
+
const contentRef = useRef(null);
|
|
65
|
+
const svgRef = useRef(null);
|
|
66
|
+
const { scale, handleZoomIn, handleZoomOut, handleReset, handleWheelZoom } = useVizScaleControls();
|
|
67
|
+
const [showSource, setShowSource] = useState(false);
|
|
68
|
+
const [isFullscreen, setIsFullscreen] = useState(false);
|
|
69
|
+
const [isVisible, setIsVisible] = useState(true);
|
|
70
|
+
const showDiagram = !showSource;
|
|
71
|
+
useEffect(() => {
|
|
72
|
+
const container = containerRef.current;
|
|
73
|
+
if (!container) return;
|
|
74
|
+
const observer = new IntersectionObserver(
|
|
75
|
+
([entry]) => setIsVisible(entry.isIntersecting),
|
|
76
|
+
{ threshold: 0.1 }
|
|
77
|
+
);
|
|
78
|
+
observer.observe(container);
|
|
79
|
+
return () => observer.disconnect();
|
|
80
|
+
}, []);
|
|
81
|
+
const resolvedCode = isStreaming ? "" : code;
|
|
82
|
+
const { loading, error, markmapInstanceRef } = useMarkmap(svgRef, resolvedCode, isVisible);
|
|
83
|
+
const showSkeleton = isStreaming || loading && !error;
|
|
84
|
+
const handleShowSource = useCallback(() => setShowSource((v) => !v), []);
|
|
85
|
+
const handleFullscreen = useCallback(() => {
|
|
86
|
+
if (!containerRef.current) return;
|
|
87
|
+
if (document.fullscreenElement === containerRef.current) {
|
|
88
|
+
document.exitFullscreen();
|
|
89
|
+
} else {
|
|
90
|
+
containerRef.current.requestFullscreen().catch(() => {
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
}, []);
|
|
94
|
+
useEffect(() => {
|
|
95
|
+
const handleChange = () => {
|
|
96
|
+
setIsFullscreen(document.fullscreenElement === containerRef.current);
|
|
97
|
+
};
|
|
98
|
+
document.addEventListener("fullscreenchange", handleChange);
|
|
99
|
+
return () => document.removeEventListener("fullscreenchange", handleChange);
|
|
100
|
+
}, []);
|
|
101
|
+
useEffect(() => {
|
|
102
|
+
if (!isFullscreen) return;
|
|
103
|
+
const handleKeyDown = (e) => {
|
|
104
|
+
if (e.key === "Escape" && document.fullscreenElement) {
|
|
105
|
+
document.exitFullscreen();
|
|
106
|
+
}
|
|
107
|
+
};
|
|
108
|
+
document.addEventListener("keydown", handleKeyDown);
|
|
109
|
+
return () => document.removeEventListener("keydown", handleKeyDown);
|
|
110
|
+
}, [isFullscreen]);
|
|
111
|
+
useEffect(() => {
|
|
112
|
+
if (isFullscreen) return;
|
|
113
|
+
const container = containerRef.current;
|
|
114
|
+
if (!container) return;
|
|
115
|
+
container.scrollTop = 0;
|
|
116
|
+
container.scrollLeft = 0;
|
|
117
|
+
}, [isFullscreen, scale, showSource]);
|
|
118
|
+
const { scaledStyle, transformStyle } = useScaledCanvas(contentRef, scale, [resolvedCode, showSource, isFullscreen, loading, error]);
|
|
119
|
+
useEffect(() => {
|
|
120
|
+
if (!showDiagram || loading || error || !isVisible || showSkeleton) return;
|
|
121
|
+
const instance = markmapInstanceRef.current;
|
|
122
|
+
if (!instance?.fit) return;
|
|
123
|
+
const refit = () => {
|
|
124
|
+
requestAnimationFrame(() => {
|
|
125
|
+
requestAnimationFrame(() => {
|
|
126
|
+
instance.fit?.();
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
};
|
|
130
|
+
refit();
|
|
131
|
+
const frame = requestAnimationFrame(refit);
|
|
132
|
+
const settleTimer = window.setTimeout(refit, 180);
|
|
133
|
+
const lateTimer = window.setTimeout(refit, 420);
|
|
134
|
+
const resizeObserver = typeof ResizeObserver !== "undefined" ? new ResizeObserver(refit) : null;
|
|
135
|
+
if (viewportRef.current) resizeObserver?.observe(viewportRef.current);
|
|
136
|
+
if (svgRef.current) resizeObserver?.observe(svgRef.current);
|
|
137
|
+
if (contentRef.current) resizeObserver?.observe(contentRef.current);
|
|
138
|
+
return () => {
|
|
139
|
+
cancelAnimationFrame(frame);
|
|
140
|
+
window.clearTimeout(settleTimer);
|
|
141
|
+
window.clearTimeout(lateTimer);
|
|
142
|
+
resizeObserver?.disconnect();
|
|
143
|
+
};
|
|
144
|
+
}, [showDiagram, resolvedCode, isFullscreen, loading, error, isVisible, markmapInstanceRef, showSkeleton]);
|
|
145
|
+
useDragPanScroll(viewportRef, showDiagram && !showSkeleton, [resolvedCode, isFullscreen, scale, showSkeleton]);
|
|
146
|
+
return /* @__PURE__ */ h(
|
|
147
|
+
"div",
|
|
148
|
+
{
|
|
149
|
+
ref: containerRef,
|
|
150
|
+
class: `markmap-block group relative rounded-md border border-[var(--viz-border)] bg-[var(--viz-bg)] ${isFullscreen ? "flex h-full w-full flex-col overflow-hidden p-4" : "p-3"}`,
|
|
151
|
+
style: {
|
|
152
|
+
background: isFullscreen ? "var(--background)" : void 0
|
|
153
|
+
}
|
|
154
|
+
},
|
|
155
|
+
showSkeleton && /* @__PURE__ */ h(SkeletonLoader, { height: "120px" }),
|
|
156
|
+
error && !showSkeleton && /* @__PURE__ */ h(Fragment, null, /* @__PURE__ */ h("pre", { class: "overflow-x-auto rounded-md bg-muted/60 px-3 py-2 text-[12px] leading-relaxed font-mono" }, /* @__PURE__ */ h("code", { class: "text-amber-600 dark:text-amber-400" }, code)), /* @__PURE__ */ h("div", { class: "mt-1 px-1 text-[10px] text-foreground-soft" }, "Markmap: ", error)),
|
|
157
|
+
!error && /* @__PURE__ */ h(Fragment, null, showSource ? /* @__PURE__ */ h("pre", { class: `overflow-auto text-[11px] leading-relaxed font-mono text-foreground-soft ${isFullscreen ? "min-h-0 flex-1 pt-10" : ""}` }, /* @__PURE__ */ h("code", null, code)) : /* @__PURE__ */ h(
|
|
158
|
+
"div",
|
|
159
|
+
{
|
|
160
|
+
ref: viewportRef,
|
|
161
|
+
class: `overflow-auto cursor-grab active:cursor-grabbing ${isFullscreen ? "min-h-0 flex-1 pt-10" : "max-h-[400px]"}`,
|
|
162
|
+
onWheel: handleWheelZoom
|
|
163
|
+
},
|
|
164
|
+
/* @__PURE__ */ h("div", { class: "flex min-h-full min-w-full items-start justify-center" }, /* @__PURE__ */ h("div", { style: { ...scaledStyle, display: showSkeleton ? "none" : void 0 } }, /* @__PURE__ */ h("div", { ref: contentRef, class: "markmap-content transition-transform duration-200", style: transformStyle }, /* @__PURE__ */ h("svg", { ref: svgRef, class: "markmap-svg block", style: { minHeight: "200px" } }))))
|
|
165
|
+
), /* @__PURE__ */ h(
|
|
166
|
+
VizToolbar,
|
|
167
|
+
{
|
|
168
|
+
onZoomIn: handleZoomIn,
|
|
169
|
+
onZoomOut: handleZoomOut,
|
|
170
|
+
onReset: handleReset,
|
|
171
|
+
onFullscreen: handleFullscreen,
|
|
172
|
+
onShowSource: handleShowSource,
|
|
173
|
+
showSourceActive: showSource
|
|
174
|
+
}
|
|
175
|
+
))
|
|
176
|
+
);
|
|
177
|
+
}
|
|
178
|
+
export {
|
|
179
|
+
MarkmapBlock
|
|
180
|
+
};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MermaidBlock.d.ts","sourceRoot":"","sources":["../../src/components/MermaidBlock.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,QAAQ,CAAC;AACpC,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AA4DtD,wBAAgB,YAAY,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,cAAc,GAAG,KAAK,GAAG,IAAI,CAiMhF"}
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
import { useCallback, useEffect, useMemo, useRef, useState } from "preact/hooks";
|
|
2
|
+
import { h } from 'preact';
|
|
3
|
+
|
|
4
|
+
import { SkeletonLoader, useDragPanScroll, useScaledCanvas, useVizScaleControls, VizToolbar } from "./VizShared.js";
|
|
5
|
+
function useMermaid(code) {
|
|
6
|
+
const [svg, setSvg] = useState("");
|
|
7
|
+
const [loading, setLoading] = useState(true);
|
|
8
|
+
const [error, setError] = useState();
|
|
9
|
+
useEffect(() => {
|
|
10
|
+
let mounted = true;
|
|
11
|
+
setLoading(true);
|
|
12
|
+
setError(void 0);
|
|
13
|
+
import("mermaid").then(async ({ default: mermaid }) => {
|
|
14
|
+
if (!mounted) return;
|
|
15
|
+
try {
|
|
16
|
+
const isDark = typeof document !== "undefined" && document.documentElement.getAttribute("data-theme") === "dark";
|
|
17
|
+
mermaid.initialize({
|
|
18
|
+
startOnLoad: false,
|
|
19
|
+
securityLevel: "strict",
|
|
20
|
+
theme: isDark ? "dark" : "default"
|
|
21
|
+
});
|
|
22
|
+
const id = `mermaid-${Date.now()}-${Math.random().toString(36).slice(2)}`;
|
|
23
|
+
const { svg: rendered } = await mermaid.render(id, code);
|
|
24
|
+
if (mounted) {
|
|
25
|
+
setSvg(rendered);
|
|
26
|
+
setLoading(false);
|
|
27
|
+
}
|
|
28
|
+
} catch (err) {
|
|
29
|
+
if (mounted) {
|
|
30
|
+
setError(err instanceof Error ? err.message : "Mermaid error");
|
|
31
|
+
setLoading(false);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}).catch(() => {
|
|
35
|
+
if (mounted) {
|
|
36
|
+
setError("Mermaid not available");
|
|
37
|
+
setLoading(false);
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
return () => {
|
|
41
|
+
mounted = false;
|
|
42
|
+
};
|
|
43
|
+
}, [code]);
|
|
44
|
+
return { svg, loading, error };
|
|
45
|
+
}
|
|
46
|
+
function MermaidBlock({ code, isStreaming }) {
|
|
47
|
+
const { svg, loading, error } = useMermaid(isStreaming ? "" : code);
|
|
48
|
+
const containerRef = useRef(null);
|
|
49
|
+
const viewportRef = useRef(null);
|
|
50
|
+
const contentRef = useRef(null);
|
|
51
|
+
const [svgSize, setSvgSize] = useState(null);
|
|
52
|
+
const { scale, handleZoomIn, handleZoomOut, handleReset, handleWheelZoom } = useVizScaleControls();
|
|
53
|
+
const [showSource, setShowSource] = useState(false);
|
|
54
|
+
const [isFullscreen, setIsFullscreen] = useState(false);
|
|
55
|
+
const showDiagram = !showSource;
|
|
56
|
+
const contentKey = useMemo(() => {
|
|
57
|
+
if (!code) return "";
|
|
58
|
+
let hash = 0;
|
|
59
|
+
for (let i = 0; i < code.length; i++) {
|
|
60
|
+
const char = code.charCodeAt(i);
|
|
61
|
+
hash = (hash << 5) - hash + char;
|
|
62
|
+
hash = hash & hash;
|
|
63
|
+
}
|
|
64
|
+
return `mermaid-${Math.abs(hash).toString(36)}`;
|
|
65
|
+
}, [code]);
|
|
66
|
+
const handleShowSource = useCallback(() => setShowSource((v) => !v), []);
|
|
67
|
+
const handleFullscreen = useCallback(() => {
|
|
68
|
+
if (!containerRef.current) return;
|
|
69
|
+
if (document.fullscreenElement === containerRef.current) {
|
|
70
|
+
document.exitFullscreen();
|
|
71
|
+
} else {
|
|
72
|
+
containerRef.current.requestFullscreen().catch(() => {
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
}, []);
|
|
76
|
+
useEffect(() => {
|
|
77
|
+
const handleChange = () => {
|
|
78
|
+
setIsFullscreen(document.fullscreenElement === containerRef.current);
|
|
79
|
+
};
|
|
80
|
+
document.addEventListener("fullscreenchange", handleChange);
|
|
81
|
+
return () => document.removeEventListener("fullscreenchange", handleChange);
|
|
82
|
+
}, []);
|
|
83
|
+
useEffect(() => {
|
|
84
|
+
if (!isFullscreen) return;
|
|
85
|
+
const handleKeyDown = (e) => {
|
|
86
|
+
if (e.key === "Escape" && document.fullscreenElement) {
|
|
87
|
+
document.exitFullscreen();
|
|
88
|
+
}
|
|
89
|
+
};
|
|
90
|
+
document.addEventListener("keydown", handleKeyDown);
|
|
91
|
+
return () => document.removeEventListener("keydown", handleKeyDown);
|
|
92
|
+
}, [isFullscreen]);
|
|
93
|
+
useEffect(() => {
|
|
94
|
+
if (isFullscreen) return;
|
|
95
|
+
const container = containerRef.current;
|
|
96
|
+
if (!container) return;
|
|
97
|
+
container.scrollTop = 0;
|
|
98
|
+
container.scrollLeft = 0;
|
|
99
|
+
}, [isFullscreen, scale, showSource]);
|
|
100
|
+
useEffect(() => {
|
|
101
|
+
if (!svg || !showDiagram) {
|
|
102
|
+
setSvgSize(null);
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
const content = contentRef.current;
|
|
106
|
+
const svgEl = content?.querySelector("svg");
|
|
107
|
+
if (!(svgEl instanceof SVGSVGElement)) {
|
|
108
|
+
setSvgSize(null);
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
const updateSvgSize = () => {
|
|
112
|
+
const viewBox = svgEl.viewBox?.baseVal;
|
|
113
|
+
const widthAttr = Number(svgEl.getAttribute("width") || 0);
|
|
114
|
+
const heightAttr = Number(svgEl.getAttribute("height") || 0);
|
|
115
|
+
const rect = svgEl.getBoundingClientRect();
|
|
116
|
+
const width = Math.max(1, Math.ceil(viewBox?.width || widthAttr || rect.width || 1));
|
|
117
|
+
const height = Math.max(1, Math.ceil(viewBox?.height || heightAttr || rect.height || 1));
|
|
118
|
+
svgEl.style.display = "block";
|
|
119
|
+
svgEl.style.width = `${width}px`;
|
|
120
|
+
svgEl.style.height = `${height}px`;
|
|
121
|
+
svgEl.style.maxWidth = "none";
|
|
122
|
+
setSvgSize((current) => current?.width === width && current?.height === height ? current : { width, height });
|
|
123
|
+
};
|
|
124
|
+
updateSvgSize();
|
|
125
|
+
const frame = requestAnimationFrame(updateSvgSize);
|
|
126
|
+
const delayedFrame = requestAnimationFrame(() => requestAnimationFrame(updateSvgSize));
|
|
127
|
+
const settleTimer = window.setTimeout(updateSvgSize, 180);
|
|
128
|
+
const lateTimer = window.setTimeout(updateSvgSize, 420);
|
|
129
|
+
const resizeObserver = typeof ResizeObserver !== "undefined" ? new ResizeObserver(updateSvgSize) : null;
|
|
130
|
+
resizeObserver?.observe(svgEl);
|
|
131
|
+
if (content) resizeObserver?.observe(content);
|
|
132
|
+
if (viewportRef.current) resizeObserver?.observe(viewportRef.current);
|
|
133
|
+
return () => {
|
|
134
|
+
cancelAnimationFrame(frame);
|
|
135
|
+
cancelAnimationFrame(delayedFrame);
|
|
136
|
+
window.clearTimeout(settleTimer);
|
|
137
|
+
window.clearTimeout(lateTimer);
|
|
138
|
+
resizeObserver?.disconnect();
|
|
139
|
+
};
|
|
140
|
+
}, [svg, showDiagram, isFullscreen]);
|
|
141
|
+
useDragPanScroll(viewportRef, showDiagram, [svg, isFullscreen, scale]);
|
|
142
|
+
const { scaledStyle, transformStyle } = useScaledCanvas(contentRef, scale, [svg, showSource, isFullscreen]);
|
|
143
|
+
const effectiveScaledStyle = svgSize ? {
|
|
144
|
+
width: `${Math.max(1, Math.ceil(svgSize.width * scale))}px`,
|
|
145
|
+
height: `${Math.max(1, Math.ceil(svgSize.height * scale))}px`
|
|
146
|
+
} : scaledStyle;
|
|
147
|
+
if (isStreaming || loading) {
|
|
148
|
+
return /* @__PURE__ */ h("div", { class: "mermaid-block group relative overflow-hidden rounded-md border border-[var(--viz-border)] bg-[var(--viz-bg)] p-3" }, /* @__PURE__ */ h(SkeletonLoader, { height: "80px" }));
|
|
149
|
+
}
|
|
150
|
+
if (error) {
|
|
151
|
+
return /* @__PURE__ */ h("div", { class: "mermaid-block" }, /* @__PURE__ */ h("pre", { class: "overflow-x-auto rounded-md bg-muted/60 px-3 py-2 text-[12px] leading-relaxed font-mono" }, /* @__PURE__ */ h("code", { class: "text-amber-600 dark:text-amber-400" }, code)), /* @__PURE__ */ h("div", { class: "mt-1 px-1 text-[10px] text-foreground-soft" }, "Mermaid: ", error));
|
|
152
|
+
}
|
|
153
|
+
return /* @__PURE__ */ h(
|
|
154
|
+
"div",
|
|
155
|
+
{
|
|
156
|
+
ref: containerRef,
|
|
157
|
+
key: contentKey,
|
|
158
|
+
class: `mermaid-block group relative rounded-md border border-[var(--viz-border)] bg-[var(--viz-bg)] ${isFullscreen ? "flex h-full w-full flex-col overflow-hidden p-4" : "p-3"}`,
|
|
159
|
+
style: isFullscreen ? { background: "var(--background)" } : void 0
|
|
160
|
+
},
|
|
161
|
+
showSource ? /* @__PURE__ */ h("pre", { class: `overflow-auto text-[11px] leading-relaxed font-mono text-foreground-soft ${isFullscreen ? "min-h-0 flex-1 pt-10" : ""}` }, /* @__PURE__ */ h("code", null, code)) : /* @__PURE__ */ h(
|
|
162
|
+
"div",
|
|
163
|
+
{
|
|
164
|
+
ref: viewportRef,
|
|
165
|
+
class: `overflow-auto cursor-grab active:cursor-grabbing ${isFullscreen ? "min-h-0 flex-1 pt-10" : "max-h-[400px]"}`,
|
|
166
|
+
onWheel: handleWheelZoom
|
|
167
|
+
},
|
|
168
|
+
/* @__PURE__ */ h("div", { class: "flex min-h-full min-w-full items-start justify-center" }, /* @__PURE__ */ h("div", { style: effectiveScaledStyle }, /* @__PURE__ */ h(
|
|
169
|
+
"div",
|
|
170
|
+
{
|
|
171
|
+
ref: contentRef,
|
|
172
|
+
class: "origin-top-left transition-transform duration-200",
|
|
173
|
+
style: transformStyle,
|
|
174
|
+
dangerouslySetInnerHTML: { __html: svg }
|
|
175
|
+
}
|
|
176
|
+
)))
|
|
177
|
+
),
|
|
178
|
+
/* @__PURE__ */ h(
|
|
179
|
+
VizToolbar,
|
|
180
|
+
{
|
|
181
|
+
onZoomIn: handleZoomIn,
|
|
182
|
+
onZoomOut: handleZoomOut,
|
|
183
|
+
onReset: handleReset,
|
|
184
|
+
onFullscreen: handleFullscreen,
|
|
185
|
+
onShowSource: handleShowSource,
|
|
186
|
+
showSourceActive: showSource
|
|
187
|
+
}
|
|
188
|
+
)
|
|
189
|
+
);
|
|
190
|
+
}
|
|
191
|
+
export {
|
|
192
|
+
MermaidBlock
|
|
193
|
+
};
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type { UIMessage } from 'ai';
|
|
2
|
+
import type { ArticleChatContext } from "../server/types.js";
|
|
3
|
+
export declare function getActionToolConfirmations(message: UIMessage, lang: string): string[];
|
|
4
|
+
export declare function shouldSuppressAssistantFallbackText(text: string, confirmations: string[], lang: string): boolean;
|
|
5
|
+
export declare function useTypewriter(fullText: string, isStreaming: boolean): string;
|
|
6
|
+
export declare function getTextFromMessage(message: UIMessage): string;
|
|
7
|
+
export declare function BotIcon({ class: cls }: {
|
|
8
|
+
class?: string;
|
|
9
|
+
}): import("preact").JSX.Element;
|
|
10
|
+
export declare function BotAvatar(): import("preact").JSX.Element;
|
|
11
|
+
export declare function TypingDots({ statusMessage }: {
|
|
12
|
+
statusMessage?: string;
|
|
13
|
+
}): import("preact").JSX.Element;
|
|
14
|
+
export declare function AssistantMessage({ message, isStreaming, lang, articleContext, onFollowUp }: {
|
|
15
|
+
message: UIMessage;
|
|
16
|
+
isStreaming?: boolean;
|
|
17
|
+
lang?: string;
|
|
18
|
+
articleContext?: ArticleChatContext;
|
|
19
|
+
onFollowUp?: (text: string) => void;
|
|
20
|
+
}): import("preact").JSX.Element | null;
|
|
21
|
+
//# sourceMappingURL=MessageBubble.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MessageBubble.d.ts","sourceRoot":"","sources":["../../src/components/MessageBubble.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AACpC,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AAyF7D,wBAAgB,0BAA0B,CAAC,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,GAAG,MAAM,EAAE,CAOrF;AAED,wBAAgB,mCAAmC,CAAC,IAAI,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAWhH;AAOD,wBAAgB,aAAa,CAAC,QAAQ,EAAE,MAAM,EAAE,WAAW,EAAE,OAAO,GAAG,MAAM,CAoE5E;AAID,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,SAAS,GAAG,MAAM,CAQ7D;AAID,wBAAgB,OAAO,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,EAAE;IAAE,KAAK,CAAC,EAAE,MAAM,CAAA;CAAE,gCAOzD;AAED,wBAAgB,SAAS,iCAMxB;AAED,wBAAgB,UAAU,CAAC,EAAE,aAAa,EAAE,EAAE;IAAE,aAAa,CAAC,EAAE,MAAM,CAAA;CAAE,gCAavE;AAMD,wBAAgB,gBAAgB,CAAC,EAAE,OAAO,EAAE,WAAW,EAAE,IAAW,EAAE,cAAc,EAAE,UAAU,EAAE,EAAE;IAClG,OAAO,EAAE,SAAS,CAAC;IACnB,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,cAAc,CAAC,EAAE,kBAAkB,CAAC;IACpC,UAAU,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;CACrC,uCAuJA"}
|