mirror of
				https://github.com/ChatGPTNextWeb/ChatGPT-Next-Web.git
				synced 2025-11-04 16:23:41 +08:00 
			
		
		
		
	Compare commits
	
		
			17 Commits
		
	
	
		
			feat/markd
			...
			2de9466088
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					2de9466088 | ||
| 
						 | 
					705dffc664 | ||
| 
						 | 
					02f7e6de98 | ||
| 
						 | 
					bc403c0cd8 | ||
| 
						 | 
					3809375694 | ||
| 
						 | 
					1b0de25986 | ||
| 
						 | 
					865c45dd29 | ||
| 
						 | 
					1f5d8e6d9c | ||
| 
						 | 
					c9ef6d58ed | ||
| 
						 | 
					2d7229d2b8 | ||
| 
						 | 
					f2c03fc242 | ||
| 
						 | 
					11b37c15bd | ||
| 
						 | 
					1d0038f17d | ||
| 
						 | 
					619fa519c0 | ||
| 
						 | 
					ea3ce4df35 | ||
| 
						 | 
					a6c68cf480 | ||
| 
						 | 
					cbeda625bd | 
							
								
								
									
										15
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								README.md
									
									
									
									
									
								
							@@ -22,7 +22,7 @@ English / [简体中文](./README_CN.md)
 | 
			
		||||
[![MacOS][MacOS-image]][download-url]
 | 
			
		||||
[![Linux][Linux-image]][download-url]
 | 
			
		||||
 | 
			
		||||
[NextChatAI](https://nextchat.club?utm_source=readme) / [Web App Demo](https://app.nextchat.dev) / [Desktop App](https://github.com/Yidadaa/ChatGPT-Next-Web/releases) / [Discord](https://discord.gg/YCkeafCafC) / [Enterprise Edition](#enterprise-edition) / [Twitter](https://twitter.com/NextChatDev)
 | 
			
		||||
[NextChatAI](https://nextchat.club?utm_source=readme) / [iOS APP](https://apps.apple.com/us/app/nextchat-ai/id6743085599) / [Web App Demo](https://app.nextchat.dev) / [Desktop App](https://github.com/Yidadaa/ChatGPT-Next-Web/releases) / [Enterprise Edition](#enterprise-edition) 
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
[saas-url]: https://nextchat.club?utm_source=readme
 | 
			
		||||
@@ -40,13 +40,14 @@ English / [简体中文](./README_CN.md)
 | 
			
		||||
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
## 🥳 Cheer for DeepSeek, China's AI star!
 | 
			
		||||
 > Purpose-Built UI for DeepSeek Reasoner Model
 | 
			
		||||
## 🥳 Cheer for NextChat iOS Version Online!
 | 
			
		||||
> [👉 Click Here to Install Now](https://apps.apple.com/us/app/nextchat-ai/id6743085599)
 | 
			
		||||
 | 
			
		||||
> [❤️ Source Code Coming Soon](https://github.com/ChatGPTNextWeb/NextChat-iOS)
 | 
			
		||||
 | 
			
		||||

 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
<img src="https://github.com/user-attachments/assets/f3952210-3af1-4dc0-9b81-40eaa4847d9a"/>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
## 🫣 NextChat Support MCP  ! 
 | 
			
		||||
> Before build, please set env ENABLE_MCP=true
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -198,7 +198,8 @@ export class ChatGPTApi implements LLMApi {
 | 
			
		||||
    const isDalle3 = _isDalle3(options.config.model);
 | 
			
		||||
    const isO1OrO3 =
 | 
			
		||||
      options.config.model.startsWith("o1") ||
 | 
			
		||||
      options.config.model.startsWith("o3");
 | 
			
		||||
      options.config.model.startsWith("o3") ||
 | 
			
		||||
      options.config.model.startsWith("o4-mini");
 | 
			
		||||
    if (isDalle3) {
 | 
			
		||||
      const prompt = getMessageTextContent(
 | 
			
		||||
        options.messages.slice(-1)?.pop() as any,
 | 
			
		||||
@@ -243,7 +244,7 @@ export class ChatGPTApi implements LLMApi {
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // add max_tokens to vision model
 | 
			
		||||
      if (visionModel) {
 | 
			
		||||
      if (visionModel && !isO1OrO3) {
 | 
			
		||||
        requestPayload["max_tokens"] = Math.max(modelConfig.max_tokens, 4000);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -18,6 +18,7 @@ import ReturnIcon from "../icons/return.svg";
 | 
			
		||||
import CopyIcon from "../icons/copy.svg";
 | 
			
		||||
import SpeakIcon from "../icons/speak.svg";
 | 
			
		||||
import SpeakStopIcon from "../icons/speak-stop.svg";
 | 
			
		||||
import LoadingIcon from "../icons/three-dots.svg";
 | 
			
		||||
import LoadingButtonIcon from "../icons/loading.svg";
 | 
			
		||||
import PromptIcon from "../icons/prompt.svg";
 | 
			
		||||
import MaskIcon from "../icons/mask.svg";
 | 
			
		||||
@@ -78,6 +79,8 @@ import {
 | 
			
		||||
 | 
			
		||||
import { uploadImage as uploadImageRemote } from "@/app/utils/chat";
 | 
			
		||||
 | 
			
		||||
import dynamic from "next/dynamic";
 | 
			
		||||
 | 
			
		||||
import { ChatControllerPool } from "../client/controller";
 | 
			
		||||
import { DalleQuality, DalleStyle, ModelSize } from "../typing";
 | 
			
		||||
import { Prompt, usePromptStore } from "../store/prompt";
 | 
			
		||||
@@ -122,15 +125,14 @@ import { getModelProvider } from "../utils/model";
 | 
			
		||||
import { RealtimeChat } from "@/app/components/realtime-chat";
 | 
			
		||||
import clsx from "clsx";
 | 
			
		||||
import { getAvailableClientsCount, isMcpEnabled } from "../mcp/actions";
 | 
			
		||||
import { Markdown } from "./markdown";
 | 
			
		||||
 | 
			
		||||
const localStorage = safeLocalStorage();
 | 
			
		||||
 | 
			
		||||
const ttsPlayer = createTTSPlayer();
 | 
			
		||||
 | 
			
		||||
// const Markdown = dynamic(async () => (await import("./markdown")).Markdown, {
 | 
			
		||||
//   loading: () => <LoadingIcon />,
 | 
			
		||||
// });
 | 
			
		||||
const Markdown = dynamic(async () => (await import("./markdown")).Markdown, {
 | 
			
		||||
  loading: () => <LoadingIcon />,
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const MCPAction = () => {
 | 
			
		||||
  const navigate = useNavigate();
 | 
			
		||||
@@ -1982,8 +1984,6 @@ function _Chat() {
 | 
			
		||||
                              fontFamily={fontFamily}
 | 
			
		||||
                              parentRef={scrollRef}
 | 
			
		||||
                              defaultShow={i >= messages.length - 6}
 | 
			
		||||
                              immediatelyRender={i >= messages.length - 3}
 | 
			
		||||
                              streaming={message.streaming}
 | 
			
		||||
                            />
 | 
			
		||||
                            {getMessageImages(message).length == 1 && (
 | 
			
		||||
                              <img
 | 
			
		||||
 
 | 
			
		||||
@@ -267,136 +267,6 @@ function tryWrapHtmlCode(text: string) {
 | 
			
		||||
    );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Split content into paragraphs while preserving code blocks
 | 
			
		||||
function splitContentIntoParagraphs(content: string) {
 | 
			
		||||
  // Check for unclosed code blocks
 | 
			
		||||
  const codeBlockStartCount = (content.match(/```/g) || []).length;
 | 
			
		||||
  let processedContent = content;
 | 
			
		||||
 | 
			
		||||
  // Add closing tag if there's an odd number of code block markers
 | 
			
		||||
  if (codeBlockStartCount % 2 !== 0) {
 | 
			
		||||
    processedContent = content + "\n```";
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Extract code blocks
 | 
			
		||||
  const codeBlockRegex = /```[\s\S]*?```/g;
 | 
			
		||||
  const codeBlocks: string[] = [];
 | 
			
		||||
  let codeBlockCounter = 0;
 | 
			
		||||
 | 
			
		||||
  // Replace code blocks with placeholders
 | 
			
		||||
  const contentWithPlaceholders = processedContent.replace(
 | 
			
		||||
    codeBlockRegex,
 | 
			
		||||
    (match) => {
 | 
			
		||||
      codeBlocks.push(match);
 | 
			
		||||
      const placeholder = `__CODE_BLOCK_${codeBlockCounter++}__`;
 | 
			
		||||
      return placeholder;
 | 
			
		||||
    },
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  // Split by double newlines
 | 
			
		||||
  const paragraphs = contentWithPlaceholders
 | 
			
		||||
    .split(/\n\n+/)
 | 
			
		||||
    .filter((p) => p.trim());
 | 
			
		||||
 | 
			
		||||
  // Restore code blocks
 | 
			
		||||
  return paragraphs.map((p) => {
 | 
			
		||||
    if (p.match(/__CODE_BLOCK_\d+__/)) {
 | 
			
		||||
      return p.replace(/__CODE_BLOCK_\d+__/g, (match) => {
 | 
			
		||||
        const index = parseInt(match.match(/\d+/)?.[0] || "0");
 | 
			
		||||
        return codeBlocks[index] || match;
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
    return p;
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Lazy-loaded paragraph component
 | 
			
		||||
function MarkdownParagraph({
 | 
			
		||||
  content,
 | 
			
		||||
  onLoad,
 | 
			
		||||
}: {
 | 
			
		||||
  content: string;
 | 
			
		||||
  onLoad?: () => void;
 | 
			
		||||
}) {
 | 
			
		||||
  const [isLoaded, setIsLoaded] = useState(false);
 | 
			
		||||
  const placeholderRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
  const [isVisible, setIsVisible] = useState(false);
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    let observer: IntersectionObserver;
 | 
			
		||||
    if (placeholderRef.current) {
 | 
			
		||||
      observer = new IntersectionObserver(
 | 
			
		||||
        (entries) => {
 | 
			
		||||
          if (entries[0].isIntersecting) {
 | 
			
		||||
            setIsVisible(true);
 | 
			
		||||
          }
 | 
			
		||||
        },
 | 
			
		||||
        { threshold: 0.1, rootMargin: "200px 0px" },
 | 
			
		||||
      );
 | 
			
		||||
      observer.observe(placeholderRef.current);
 | 
			
		||||
    }
 | 
			
		||||
    return () => observer?.disconnect();
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    if (isVisible && !isLoaded) {
 | 
			
		||||
      setIsLoaded(true);
 | 
			
		||||
      onLoad?.();
 | 
			
		||||
    }
 | 
			
		||||
  }, [isVisible, isLoaded, onLoad]);
 | 
			
		||||
 | 
			
		||||
  // Generate preview content
 | 
			
		||||
  const previewContent = useMemo(() => {
 | 
			
		||||
    if (content.startsWith("```")) {
 | 
			
		||||
      return "```" + (content.split("\n")[0] || "").slice(3) + "...```";
 | 
			
		||||
    }
 | 
			
		||||
    return content.length > 60 ? content.slice(0, 60) + "..." : content;
 | 
			
		||||
  }, [content]);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div className="markdown-paragraph" ref={placeholderRef}>
 | 
			
		||||
      {!isLoaded ? (
 | 
			
		||||
        <div className="markdown-paragraph-placeholder">{previewContent}</div>
 | 
			
		||||
      ) : (
 | 
			
		||||
        <_MarkDownContent content={content} />
 | 
			
		||||
      )}
 | 
			
		||||
    </div>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Memoized paragraph component to prevent unnecessary re-renders
 | 
			
		||||
const MemoizedMarkdownParagraph = React.memo(
 | 
			
		||||
  ({ content }: { content: string }) => {
 | 
			
		||||
    return <_MarkDownContent content={content} />;
 | 
			
		||||
  },
 | 
			
		||||
  (prevProps, nextProps) => prevProps.content === nextProps.content,
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
MemoizedMarkdownParagraph.displayName = "MemoizedMarkdownParagraph";
 | 
			
		||||
 | 
			
		||||
// Specialized component for streaming content
 | 
			
		||||
function StreamingMarkdownContent({ content }: { content: string }) {
 | 
			
		||||
  const paragraphs = useMemo(
 | 
			
		||||
    () => splitContentIntoParagraphs(content),
 | 
			
		||||
    [content],
 | 
			
		||||
  );
 | 
			
		||||
  const lastParagraphRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div className="markdown-streaming-content">
 | 
			
		||||
      {paragraphs.map((paragraph, index) => (
 | 
			
		||||
        <div
 | 
			
		||||
          key={`p-${index}-${paragraph.substring(0, 20)}`}
 | 
			
		||||
          className="markdown-paragraph markdown-streaming-paragraph"
 | 
			
		||||
          ref={index === paragraphs.length - 1 ? lastParagraphRef : null}
 | 
			
		||||
        >
 | 
			
		||||
          <MemoizedMarkdownParagraph content={paragraph} />
 | 
			
		||||
        </div>
 | 
			
		||||
      ))}
 | 
			
		||||
    </div>
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function _MarkDownContent(props: { content: string }) {
 | 
			
		||||
  const escapedContent = useMemo(() => {
 | 
			
		||||
    return tryWrapHtmlCode(escapeBrackets(props.content));
 | 
			
		||||
@@ -456,27 +326,9 @@ export function Markdown(
 | 
			
		||||
    fontFamily?: string;
 | 
			
		||||
    parentRef?: RefObject<HTMLDivElement>;
 | 
			
		||||
    defaultShow?: boolean;
 | 
			
		||||
    immediatelyRender?: boolean;
 | 
			
		||||
    streaming?: boolean; // Whether this is a streaming response
 | 
			
		||||
  } & React.DOMAttributes<HTMLDivElement>,
 | 
			
		||||
) {
 | 
			
		||||
  const mdRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
  const paragraphs = useMemo(
 | 
			
		||||
    () => splitContentIntoParagraphs(props.content),
 | 
			
		||||
    [props.content],
 | 
			
		||||
  );
 | 
			
		||||
  const [loadedCount, setLoadedCount] = useState(0);
 | 
			
		||||
 | 
			
		||||
  // Determine rendering strategy based on props
 | 
			
		||||
  const shouldAsyncRender =
 | 
			
		||||
    !props.immediatelyRender && !props.streaming && paragraphs.length > 1;
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    // Immediately render all paragraphs if specified
 | 
			
		||||
    if (props.immediatelyRender) {
 | 
			
		||||
      setLoadedCount(paragraphs.length);
 | 
			
		||||
    }
 | 
			
		||||
  }, [props.immediatelyRender, paragraphs.length]);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
@@ -492,24 +344,6 @@ export function Markdown(
 | 
			
		||||
    >
 | 
			
		||||
      {props.loading ? (
 | 
			
		||||
        <LoadingIcon />
 | 
			
		||||
      ) : props.streaming ? (
 | 
			
		||||
        // Use specialized component for streaming content
 | 
			
		||||
        <StreamingMarkdownContent content={props.content} />
 | 
			
		||||
      ) : shouldAsyncRender ? (
 | 
			
		||||
        <div className="markdown-content">
 | 
			
		||||
          {paragraphs.map((paragraph, index) => (
 | 
			
		||||
            <MarkdownParagraph
 | 
			
		||||
              key={index}
 | 
			
		||||
              content={paragraph}
 | 
			
		||||
              onLoad={() => setLoadedCount((prev) => prev + 1)}
 | 
			
		||||
            />
 | 
			
		||||
          ))}
 | 
			
		||||
          {loadedCount < paragraphs.length && loadedCount > 0 && (
 | 
			
		||||
            <div className="markdown-paragraph-loading">
 | 
			
		||||
              <LoadingIcon />
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
        </div>
 | 
			
		||||
      ) : (
 | 
			
		||||
        <MarkdownContent content={props.content} />
 | 
			
		||||
      )}
 | 
			
		||||
 
 | 
			
		||||
@@ -417,6 +417,14 @@ export const KnowledgeCutOffDate: Record<string, string> = {
 | 
			
		||||
  "gpt-4-turbo": "2023-12",
 | 
			
		||||
  "gpt-4-turbo-2024-04-09": "2023-12",
 | 
			
		||||
  "gpt-4-turbo-preview": "2023-12",
 | 
			
		||||
  "gpt-4.1": "2024-06",
 | 
			
		||||
  "gpt-4.1-2025-04-14": "2024-06",
 | 
			
		||||
  "gpt-4.1-mini": "2024-06",
 | 
			
		||||
  "gpt-4.1-mini-2025-04-14": "2024-06",
 | 
			
		||||
  "gpt-4.1-nano": "2024-06",
 | 
			
		||||
  "gpt-4.1-nano-2025-04-14": "2024-06",
 | 
			
		||||
  "gpt-4.5-preview": "2023-10",
 | 
			
		||||
  "gpt-4.5-preview-2025-02-27": "2023-10",
 | 
			
		||||
  "gpt-4o": "2023-10",
 | 
			
		||||
  "gpt-4o-2024-05-13": "2023-10",
 | 
			
		||||
  "gpt-4o-2024-08-06": "2023-10",
 | 
			
		||||
@@ -458,6 +466,7 @@ export const DEFAULT_TTS_VOICES = [
 | 
			
		||||
export const VISION_MODEL_REGEXES = [
 | 
			
		||||
  /vision/,
 | 
			
		||||
  /gpt-4o/,
 | 
			
		||||
  /gpt-4\.1/,
 | 
			
		||||
  /claude-3/,
 | 
			
		||||
  /gemini-1\.5/,
 | 
			
		||||
  /gemini-exp/,
 | 
			
		||||
@@ -469,6 +478,8 @@ export const VISION_MODEL_REGEXES = [
 | 
			
		||||
  /^dall-e-3$/, // Matches exactly "dall-e-3"
 | 
			
		||||
  /glm-4v/,
 | 
			
		||||
  /vl/i,
 | 
			
		||||
  /o3/,
 | 
			
		||||
  /o4-mini/,
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
export const EXCLUDE_VISION_MODEL_REGEXES = [/claude-3-5-haiku-20241022/];
 | 
			
		||||
@@ -485,6 +496,14 @@ const openaiModels = [
 | 
			
		||||
  "gpt-4-32k-0613",
 | 
			
		||||
  "gpt-4-turbo",
 | 
			
		||||
  "gpt-4-turbo-preview",
 | 
			
		||||
  "gpt-4.1",
 | 
			
		||||
  "gpt-4.1-2025-04-14",
 | 
			
		||||
  "gpt-4.1-mini",
 | 
			
		||||
  "gpt-4.1-mini-2025-04-14",
 | 
			
		||||
  "gpt-4.1-nano",
 | 
			
		||||
  "gpt-4.1-nano-2025-04-14",
 | 
			
		||||
  "gpt-4.5-preview",
 | 
			
		||||
  "gpt-4.5-preview-2025-02-27",
 | 
			
		||||
  "gpt-4o",
 | 
			
		||||
  "gpt-4o-2024-05-13",
 | 
			
		||||
  "gpt-4o-2024-08-06",
 | 
			
		||||
@@ -499,6 +518,8 @@ const openaiModels = [
 | 
			
		||||
  "o1-mini",
 | 
			
		||||
  "o1-preview",
 | 
			
		||||
  "o3-mini",
 | 
			
		||||
  "o3",
 | 
			
		||||
  "o4-mini",
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const googleModels = [
 | 
			
		||||
@@ -525,6 +546,7 @@ const googleModels = [
 | 
			
		||||
  "gemini-2.0-flash-thinking-exp-01-21",
 | 
			
		||||
  "gemini-2.0-pro-exp",
 | 
			
		||||
  "gemini-2.0-pro-exp-02-05",
 | 
			
		||||
  "gemini-2.5-pro-preview-06-05",
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const anthropicModels = [
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const ar: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 واجهت المحادثة بعض المشكلات، لا داعي للقلق:
 | 
			
		||||
    \\ 1️⃣ إذا كنت ترغب في تجربة دون إعداد، [انقر هنا لبدء المحادثة فورًا 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ إذا كنت تريد استخدام موارد OpenAI الخاصة بك، انقر [هنا](/#/settings) لتعديل الإعدادات ⚙️`
 | 
			
		||||
    \ 1️⃣ إذا كنت ترغب في تجربة دون إعداد، [انقر هنا لبدء المحادثة فورًا 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ إذا كنت تريد استخدام موارد OpenAI الخاصة بك، انقر [هنا](/#/settings) لتعديل الإعدادات ⚙️`
 | 
			
		||||
      : `😆 واجهت المحادثة بعض المشكلات، لا داعي للقلق:
 | 
			
		||||
    \ 1️⃣ إذا كنت ترغب في تجربة دون إعداد، [انقر هنا لبدء المحادثة فورًا 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ إذا كنت تستخدم إصدار النشر الخاص، انقر [هنا](/#/auth) لإدخال مفتاح الوصول 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const bn: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 কথোপকথনে কিছু সমস্যা হয়েছে, চিন্তার কিছু নেই:
 | 
			
		||||
    \\ 1️⃣ যদি আপনি শূন্য কনফিগারেশনে শুরু করতে চান, তাহলে [এখানে ক্লিক করে অবিলম্বে কথোপকথন শুরু করুন 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ যদি আপনি আপনার নিজস্ব OpenAI সম্পদ ব্যবহার করতে চান, তাহলে [এখানে ক্লিক করুন](/#/settings) সেটিংস পরিবর্তন করতে ⚙️`
 | 
			
		||||
    \ 1️⃣ যদি আপনি শূন্য কনফিগারেশনে শুরু করতে চান, তাহলে [এখানে ক্লিক করে অবিলম্বে কথোপকথন শুরু করুন 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ যদি আপনি আপনার নিজস্ব OpenAI সম্পদ ব্যবহার করতে চান, তাহলে [এখানে ক্লিক করুন](/#/settings) সেটিংস পরিবর্তন করতে ⚙️`
 | 
			
		||||
      : `😆 কথোপকথনে কিছু সমস্যা হয়েছে, চিন্তার কিছু নেই:
 | 
			
		||||
    \ 1️⃣ যদি আপনি শূন্য কনফিগারেশনে শুরু করতে চান, তাহলে [এখানে ক্লিক করে অবিলম্বে কথোপকথন শুরু করুন 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ যদি আপনি একটি প্রাইভেট ডেপ্লয়মেন্ট সংস্করণ ব্যবহার করেন, তাহলে [এখানে ক্লিক করুন](/#/auth) প্রবেশাধিকার কীগুলি প্রবেশ করতে 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const cn = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 对话遇到了一些问题,不用慌:
 | 
			
		||||
       \\ 1️⃣ 想要零配置开箱即用,[点击这里立刻开启对话 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
       \\ 2️⃣ 如果你想消耗自己的 OpenAI 资源,点击[这里](/#/settings)修改设置 ⚙️`
 | 
			
		||||
       \ 1️⃣ 想要零配置开箱即用,[点击这里立刻开启对话 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
       \ 2️⃣ 如果你想消耗自己的 OpenAI 资源,点击[这里](/#/settings)修改设置 ⚙️`
 | 
			
		||||
      : `😆 对话遇到了一些问题,不用慌:
 | 
			
		||||
       \ 1️⃣ 想要零配置开箱即用,[点击这里立刻开启对话 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
       \ 2️⃣ 如果你正在使用私有部署版本,点击[这里](/#/auth)输入访问秘钥 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const cs: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Rozhovor narazil na nějaké problémy, nebojte se:
 | 
			
		||||
    \\ 1️⃣ Pokud chcete začít bez konfigurace, [klikněte sem pro okamžitý začátek chatu 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Pokud chcete využít své vlastní zdroje OpenAI, klikněte [sem](/#/settings) a upravte nastavení ⚙️`
 | 
			
		||||
    \ 1️⃣ Pokud chcete začít bez konfigurace, [klikněte sem pro okamžitý začátek chatu 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Pokud chcete využít své vlastní zdroje OpenAI, klikněte [sem](/#/settings) a upravte nastavení ⚙️`
 | 
			
		||||
      : `😆 Rozhovor narazil na nějaké problémy, nebojte se:
 | 
			
		||||
    \ 1️⃣ Pokud chcete začít bez konfigurace, [klikněte sem pro okamžitý začátek chatu 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Pokud používáte verzi soukromého nasazení, klikněte [sem](/#/auth) a zadejte přístupový klíč 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,12 +9,12 @@ const da: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `Hov, der skete en fejl. Sådan kan du komme videre:
 | 
			
		||||
       \\ 1️⃣ Er du ny her? [Tryk for at starte nu 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
       \\ 2️⃣ Vil du bruge dine egne OpenAI-nøgler? [Tryk her](/#/settings) for at ændre indstillinger ⚙️`
 | 
			
		||||
       \ 1️⃣ Er du ny her? [Tryk for at starte nu 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
       \ 2️⃣ Vil du bruge dine egne OpenAI-nøgler? [Tryk her](/#/settings) for at ændre indstillinger ⚙️`
 | 
			
		||||
      : `Hov, der skete en fejl. Lad os løse det:
 | 
			
		||||
       \\ 1️⃣ Er du ny her? [Tryk for at starte nu 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
       \\ 2️⃣ Bruger du en privat opsætning? [Tryk her](/#/auth) for at taste din nøgle 🔑
 | 
			
		||||
       \\ 3️⃣ Vil du bruge dine egne OpenAI-nøgler? [Tryk her](/#/settings) for at ændre indstillinger ⚙️
 | 
			
		||||
       \ 1️⃣ Er du ny her? [Tryk for at starte nu 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
       \ 2️⃣ Bruger du en privat opsætning? [Tryk her](/#/auth) for at taste din nøgle 🔑
 | 
			
		||||
       \ 3️⃣ Vil du bruge dine egne OpenAI-nøgler? [Tryk her](/#/settings) for at ændre indstillinger ⚙️
 | 
			
		||||
       `,
 | 
			
		||||
  },
 | 
			
		||||
  Auth: {
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const de: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Das Gespräch hatte einige Probleme, keine Sorge:
 | 
			
		||||
    \\ 1️⃣ Wenn du ohne Konfiguration sofort starten möchtest, [klicke hier, um sofort zu chatten 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Wenn du deine eigenen OpenAI-Ressourcen verwenden möchtest, klicke [hier](/#/settings), um die Einstellungen zu ändern ⚙️`
 | 
			
		||||
    \ 1️⃣ Wenn du ohne Konfiguration sofort starten möchtest, [klicke hier, um sofort zu chatten 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Wenn du deine eigenen OpenAI-Ressourcen verwenden möchtest, klicke [hier](/#/settings), um die Einstellungen zu ändern ⚙️`
 | 
			
		||||
      : `😆 Das Gespräch hatte einige Probleme, keine Sorge:
 | 
			
		||||
    \ 1️⃣ Wenn du ohne Konfiguration sofort starten möchtest, [klicke hier, um sofort zu chatten 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Wenn du eine private Bereitstellung verwendest, klicke [hier](/#/auth), um den Zugriffsschlüssel einzugeben 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -10,8 +10,8 @@ const en: LocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Oops, there's an issue. No worries:
 | 
			
		||||
     \\ 1️⃣ New here? [Click to start chatting now 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
     \\ 2️⃣ Want to use your own OpenAI resources? [Click here](/#/settings) to change settings ⚙️`
 | 
			
		||||
     \ 1️⃣ New here? [Click to start chatting now 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
     \ 2️⃣ Want to use your own OpenAI resources? [Click here](/#/settings) to change settings ⚙️`
 | 
			
		||||
      : `😆 Oops, there's an issue. Let's fix it:
 | 
			
		||||
     \ 1️⃣ New here? [Click to start chatting now 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
     \ 2️⃣ Using a private setup? [Click here](/#/auth) to enter your key 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const es: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 La conversación encontró algunos problemas, no te preocupes:
 | 
			
		||||
    \\ 1️⃣ Si deseas comenzar sin configuración, [haz clic aquí para empezar a chatear inmediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Si deseas usar tus propios recursos de OpenAI, haz clic [aquí](/#/settings) para modificar la configuración ⚙️`
 | 
			
		||||
    \ 1️⃣ Si deseas comenzar sin configuración, [haz clic aquí para empezar a chatear inmediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Si deseas usar tus propios recursos de OpenAI, haz clic [aquí](/#/settings) para modificar la configuración ⚙️`
 | 
			
		||||
      : `😆 La conversación encontró algunos problemas, no te preocupes:
 | 
			
		||||
    \ 1️⃣ Si deseas comenzar sin configuración, [haz clic aquí para empezar a chatear inmediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Si estás utilizando una versión de implementación privada, haz clic [aquí](/#/auth) para ingresar la clave de acceso 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const fr: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 La conversation a rencontré quelques problèmes, pas de panique :
 | 
			
		||||
    \\ 1️⃣ Si vous souhaitez commencer sans configuration, [cliquez ici pour démarrer la conversation immédiatement 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Si vous souhaitez utiliser vos propres ressources OpenAI, cliquez [ici](/#/settings) pour modifier les paramètres ⚙️`
 | 
			
		||||
    \ 1️⃣ Si vous souhaitez commencer sans configuration, [cliquez ici pour démarrer la conversation immédiatement 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Si vous souhaitez utiliser vos propres ressources OpenAI, cliquez [ici](/#/settings) pour modifier les paramètres ⚙️`
 | 
			
		||||
      : `😆 La conversation a rencontré quelques problèmes, pas de panique :
 | 
			
		||||
    \ 1️⃣ Si vous souhaitez commencer sans configuration, [cliquez ici pour démarrer la conversation immédiatement 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Si vous utilisez une version déployée privée, cliquez [ici](/#/auth) pour entrer la clé d'accès 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const id: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Percakapan mengalami beberapa masalah, tidak perlu khawatir:
 | 
			
		||||
   \\ 1️⃣ Jika Anda ingin memulai tanpa konfigurasi, [klik di sini untuk mulai mengobrol segera 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
   \\ 2️⃣ Jika Anda ingin menggunakan sumber daya OpenAI Anda sendiri, klik [di sini](/#/settings) untuk mengubah pengaturan ⚙️`
 | 
			
		||||
   \ 1️⃣ Jika Anda ingin memulai tanpa konfigurasi, [klik di sini untuk mulai mengobrol segera 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
   \ 2️⃣ Jika Anda ingin menggunakan sumber daya OpenAI Anda sendiri, klik [di sini](/#/settings) untuk mengubah pengaturan ⚙️`
 | 
			
		||||
      : `😆 Percakapan mengalami beberapa masalah, tidak perlu khawatir:
 | 
			
		||||
   \ 1️⃣ Jika Anda ingin memulai tanpa konfigurasi, [klik di sini untuk mulai mengobrol segera 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
   \ 2️⃣ Jika Anda menggunakan versi penyebaran pribadi, klik [di sini](/#/auth) untuk memasukkan kunci akses 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const it: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 La conversazione ha incontrato alcuni problemi, non preoccuparti:
 | 
			
		||||
    \\ 1️⃣ Se vuoi iniziare senza configurazione, [clicca qui per iniziare a chattare immediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Se vuoi utilizzare le tue risorse OpenAI, clicca [qui](/#/settings) per modificare le impostazioni ⚙️`
 | 
			
		||||
    \ 1️⃣ Se vuoi iniziare senza configurazione, [clicca qui per iniziare a chattare immediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Se vuoi utilizzare le tue risorse OpenAI, clicca [qui](/#/settings) per modificare le impostazioni ⚙️`
 | 
			
		||||
      : `😆 La conversazione ha incontrato alcuni problemi, non preoccuparti:
 | 
			
		||||
    \ 1️⃣ Se vuoi iniziare senza configurazione, [clicca qui per iniziare a chattare immediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Se stai utilizzando una versione di distribuzione privata, clicca [qui](/#/auth) per inserire la chiave di accesso 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const jp: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 会話中に問題が発生しましたが、心配しないでください:
 | 
			
		||||
    \\ 1️⃣ 設定なしで始めたい場合は、[ここをクリックしてすぐにチャットを開始 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ 自分のOpenAIリソースを使用したい場合は、[ここをクリックして](/#/settings)設定を変更してください ⚙️`
 | 
			
		||||
    \ 1️⃣ 設定なしで始めたい場合は、[ここをクリックしてすぐにチャットを開始 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ 自分のOpenAIリソースを使用したい場合は、[ここをクリックして](/#/settings)設定を変更してください ⚙️`
 | 
			
		||||
      : `😆 会話中に問題が発生しましたが、心配しないでください:
 | 
			
		||||
    \ 1️⃣ 設定なしで始めたい場合は、[ここをクリックしてすぐにチャットを開始 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ プライベートデプロイ版を使用している場合は、[ここをクリックして](/#/auth)アクセストークンを入力してください 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const ko: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 대화 중 문제가 발생했습니다, 걱정하지 마세요:
 | 
			
		||||
    \\ 1️⃣ 제로 구성으로 시작하고 싶다면, [여기를 클릭하여 즉시 대화를 시작하세요 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ 자신의 OpenAI 리소스를 사용하고 싶다면, [여기를 클릭하여](/#/settings) 설정을 수정하세요 ⚙️`
 | 
			
		||||
    \ 1️⃣ 제로 구성으로 시작하고 싶다면, [여기를 클릭하여 즉시 대화를 시작하세요 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ 자신의 OpenAI 리소스를 사용하고 싶다면, [여기를 클릭하여](/#/settings) 설정을 수정하세요 ⚙️`
 | 
			
		||||
      : `😆 대화 중 문제가 발생했습니다, 걱정하지 마세요:
 | 
			
		||||
    \ 1️⃣ 제로 구성으로 시작하고 싶다면, [여기를 클릭하여 즉시 대화를 시작하세요 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ 개인 배포 버전을 사용하고 있다면, [여기를 클릭하여](/#/auth) 접근 키를 입력하세요 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const no: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Samtalen har støtt på noen problemer, ikke bekymre deg:
 | 
			
		||||
    \\ 1️⃣ Hvis du vil starte uten konfigurasjon, [klikk her for å begynne å chatte umiddelbart 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Hvis du vil bruke dine egne OpenAI-ressurser, klikk [her](/#/settings) for å endre innstillingene ⚙️`
 | 
			
		||||
    \ 1️⃣ Hvis du vil starte uten konfigurasjon, [klikk her for å begynne å chatte umiddelbart 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Hvis du vil bruke dine egne OpenAI-ressurser, klikk [her](/#/settings) for å endre innstillingene ⚙️`
 | 
			
		||||
      : `😆 Samtalen har støtt på noen problemer, ikke bekymre deg:
 | 
			
		||||
    \ 1️⃣ Hvis du vil starte uten konfigurasjon, [klikk her for å begynne å chatte umiddelbart 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Hvis du bruker en privat distribusjonsversjon, klikk [her](/#/auth) for å skrive inn tilgangsnøkkelen 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const pt: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 A conversa encontrou alguns problemas, não se preocupe:
 | 
			
		||||
   \\ 1️⃣ Se você quiser começar sem configuração, [clique aqui para começar a conversar imediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
   \\ 2️⃣ Se você deseja usar seus próprios recursos OpenAI, clique [aqui](/#/settings) para modificar as configurações ⚙️`
 | 
			
		||||
   \ 1️⃣ Se você quiser começar sem configuração, [clique aqui para começar a conversar imediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
   \ 2️⃣ Se você deseja usar seus próprios recursos OpenAI, clique [aqui](/#/settings) para modificar as configurações ⚙️`
 | 
			
		||||
      : `😆 A conversa encontrou alguns problemas, não se preocupe:
 | 
			
		||||
   \ 1️⃣ Se você quiser começar sem configuração, [clique aqui para começar a conversar imediatamente 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
   \ 2️⃣ Se você estiver usando uma versão de implantação privada, clique [aqui](/#/auth) para inserir a chave de acesso 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const ru: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 В разговоре возникли некоторые проблемы, не переживайте:
 | 
			
		||||
    \\ 1️⃣ Если вы хотите начать без настройки, [нажмите здесь, чтобы немедленно начать разговор 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Если вы хотите использовать свои ресурсы OpenAI, нажмите [здесь](/#/settings), чтобы изменить настройки ⚙️`
 | 
			
		||||
    \ 1️⃣ Если вы хотите начать без настройки, [нажмите здесь, чтобы немедленно начать разговор 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Если вы хотите использовать свои ресурсы OpenAI, нажмите [здесь](/#/settings), чтобы изменить настройки ⚙️`
 | 
			
		||||
      : `😆 В разговоре возникли некоторые проблемы, не переживайте:
 | 
			
		||||
    \ 1️⃣ Если вы хотите начать без настройки, [нажмите здесь, чтобы немедленно начать разговор 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Если вы используете частную версию развертывания, нажмите [здесь](/#/auth), чтобы ввести ключ доступа 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -10,8 +10,8 @@ const sk: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Rozhovor narazil na nejaké problémy, nebojte sa:
 | 
			
		||||
    \\ 1️⃣ Ak chcete začať bez konfigurácie, [kliknite sem, aby ste okamžite začali chatovať 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Ak chcete používať svoje vlastné zdroje OpenAI, kliknite [sem](/#/settings), aby ste upravili nastavenia ⚙️`
 | 
			
		||||
    \ 1️⃣ Ak chcete začať bez konfigurácie, [kliknite sem, aby ste okamžite začali chatovať 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Ak chcete používať svoje vlastné zdroje OpenAI, kliknite [sem](/#/settings), aby ste upravili nastavenia ⚙️`
 | 
			
		||||
      : `😆 Rozhovor narazil na nejaké problémy, nebojte sa:
 | 
			
		||||
    \ 1️⃣ Ak chcete začať bez konfigurácie, [kliknite sem, aby ste okamžite začali chatovať 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Ak používate verziu súkromného nasadenia, kliknite [sem](/#/auth), aby ste zadali prístupový kľúč 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const tr: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Sohbet bazı sorunlarla karşılaştı, endişelenmeyin:
 | 
			
		||||
    \\ 1️⃣ Eğer sıfır yapılandırma ile başlamak istiyorsanız, [buraya tıklayarak hemen sohbete başlayın 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Kendi OpenAI kaynaklarınızı kullanmak istiyorsanız, [buraya tıklayarak](/#/settings) ayarları değiştirin ⚙️`
 | 
			
		||||
    \ 1️⃣ Eğer sıfır yapılandırma ile başlamak istiyorsanız, [buraya tıklayarak hemen sohbete başlayın 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Kendi OpenAI kaynaklarınızı kullanmak istiyorsanız, [buraya tıklayarak](/#/settings) ayarları değiştirin ⚙️`
 | 
			
		||||
      : `😆 Sohbet bazı sorunlarla karşılaştı, endişelenmeyin:
 | 
			
		||||
    \ 1️⃣ Eğer sıfır yapılandırma ile başlamak istiyorsanız, [buraya tıklayarak hemen sohbete başlayın 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Eğer özel dağıtım sürümü kullanıyorsanız, [buraya tıklayarak](/#/auth) erişim anahtarını girin 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -8,8 +8,8 @@ const tw = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 對話遇到了一些問題,不用慌:
 | 
			
		||||
    \\ 1️⃣ 想要無須設定開箱即用,[點選這裡立刻開啟對話 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ 如果你想消耗自己的 OpenAI 資源,點選[這裡](/#/settings)修改設定 ⚙️`
 | 
			
		||||
    \ 1️⃣ 想要無須設定開箱即用,[點選這裡立刻開啟對話 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ 如果你想消耗自己的 OpenAI 資源,點選[這裡](/#/settings)修改設定 ⚙️`
 | 
			
		||||
      : `😆 對話遇到了一些問題,不用慌:
 | 
			
		||||
    \ 1️⃣ 想要無須設定開箱即用,[點選這裡立刻開啟對話 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ 如果你正在使用私有部署版本,點選[這裡](/#/auth)輸入存取金鑰 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -9,8 +9,8 @@ const vi: PartialLocaleType = {
 | 
			
		||||
  Error: {
 | 
			
		||||
    Unauthorized: isApp
 | 
			
		||||
      ? `😆 Cuộc trò chuyện gặp một số vấn đề, đừng lo lắng:
 | 
			
		||||
    \\ 1️⃣ Nếu bạn muốn bắt đầu mà không cần cấu hình, [nhấp vào đây để bắt đầu trò chuyện ngay lập tức 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \\ 2️⃣ Nếu bạn muốn sử dụng tài nguyên OpenAI của riêng mình, hãy nhấp [vào đây](/#/settings) để thay đổi cài đặt ⚙️`
 | 
			
		||||
    \ 1️⃣ Nếu bạn muốn bắt đầu mà không cần cấu hình, [nhấp vào đây để bắt đầu trò chuyện ngay lập tức 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Nếu bạn muốn sử dụng tài nguyên OpenAI của riêng mình, hãy nhấp [vào đây](/#/settings) để thay đổi cài đặt ⚙️`
 | 
			
		||||
      : `😆 Cuộc trò chuyện gặp một số vấn đề, đừng lo lắng:
 | 
			
		||||
    \ 1️⃣ Nếu bạn muốn bắt đầu mà không cần cấu hình, [nhấp vào đây để bắt đầu trò chuyện ngay lập tức 🚀](${SAAS_CHAT_UTM_URL})
 | 
			
		||||
    \ 2️⃣ Nếu bạn đang sử dụng phiên bản triển khai riêng, hãy nhấp [vào đây](/#/auth) để nhập khóa truy cập 🔑
 | 
			
		||||
 
 | 
			
		||||
@@ -99,7 +99,6 @@
 | 
			
		||||
  font-size: 14px;
 | 
			
		||||
  line-height: 1.5;
 | 
			
		||||
  word-wrap: break-word;
 | 
			
		||||
  margin-bottom: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.light {
 | 
			
		||||
@@ -359,14 +358,8 @@
 | 
			
		||||
.markdown-body kbd {
 | 
			
		||||
  display: inline-block;
 | 
			
		||||
  padding: 3px 5px;
 | 
			
		||||
  font:
 | 
			
		||||
    11px ui-monospace,
 | 
			
		||||
    SFMono-Regular,
 | 
			
		||||
    SF Mono,
 | 
			
		||||
    Menlo,
 | 
			
		||||
    Consolas,
 | 
			
		||||
    Liberation Mono,
 | 
			
		||||
    monospace;
 | 
			
		||||
  font: 11px ui-monospace, SFMono-Regular, SF Mono, Menlo, Consolas,
 | 
			
		||||
    Liberation Mono, monospace;
 | 
			
		||||
  line-height: 10px;
 | 
			
		||||
  color: var(--color-fg-default);
 | 
			
		||||
  vertical-align: middle;
 | 
			
		||||
@@ -455,28 +448,16 @@
 | 
			
		||||
.markdown-body tt,
 | 
			
		||||
.markdown-body code,
 | 
			
		||||
.markdown-body samp {
 | 
			
		||||
  font-family:
 | 
			
		||||
    ui-monospace,
 | 
			
		||||
    SFMono-Regular,
 | 
			
		||||
    SF Mono,
 | 
			
		||||
    Menlo,
 | 
			
		||||
    Consolas,
 | 
			
		||||
    Liberation Mono,
 | 
			
		||||
    monospace;
 | 
			
		||||
  font-family: ui-monospace, SFMono-Regular, SF Mono, Menlo, Consolas,
 | 
			
		||||
    Liberation Mono, monospace;
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.markdown-body pre {
 | 
			
		||||
  margin-top: 0;
 | 
			
		||||
  margin-bottom: 0;
 | 
			
		||||
  font-family:
 | 
			
		||||
    ui-monospace,
 | 
			
		||||
    SFMono-Regular,
 | 
			
		||||
    SF Mono,
 | 
			
		||||
    Menlo,
 | 
			
		||||
    Consolas,
 | 
			
		||||
    Liberation Mono,
 | 
			
		||||
    monospace;
 | 
			
		||||
  font-family: ui-monospace, SFMono-Regular, SF Mono, Menlo, Consolas,
 | 
			
		||||
    Liberation Mono, monospace;
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
  word-wrap: normal;
 | 
			
		||||
}
 | 
			
		||||
@@ -1149,87 +1130,3 @@
 | 
			
		||||
#dmermaid {
 | 
			
		||||
  display: none;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.markdown-content {
 | 
			
		||||
  width: 100%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.markdown-paragraph {
 | 
			
		||||
  transition: opacity 0.3s ease;
 | 
			
		||||
  margin-bottom: 0.5em;
 | 
			
		||||
 | 
			
		||||
  &.markdown-paragraph-visible {
 | 
			
		||||
    opacity: 1;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.markdown-paragraph-hidden {
 | 
			
		||||
    opacity: 0.7;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.markdown-paragraph-placeholder {
 | 
			
		||||
  padding: 8px;
 | 
			
		||||
  color: var(--color-fg-subtle);
 | 
			
		||||
  background-color: var(--color-canvas-subtle);
 | 
			
		||||
  border-radius: 6px;
 | 
			
		||||
  border-left: 3px solid var(--color-border-muted);
 | 
			
		||||
  white-space: nowrap;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
  text-overflow: ellipsis;
 | 
			
		||||
  font-family: var(--font-family-sans);
 | 
			
		||||
  font-size: 14px;
 | 
			
		||||
  min-height: 1.2em;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.markdown-paragraph-loading {
 | 
			
		||||
  height: 20px;
 | 
			
		||||
  background-color: var(--color-canvas-subtle);
 | 
			
		||||
  border-radius: 6px;
 | 
			
		||||
  margin-bottom: 8px;
 | 
			
		||||
  position: relative;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
 | 
			
		||||
  &::after {
 | 
			
		||||
    content: "";
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    top: 0;
 | 
			
		||||
    left: 0;
 | 
			
		||||
    width: 30%;
 | 
			
		||||
    height: 100%;
 | 
			
		||||
    background: linear-gradient(
 | 
			
		||||
      90deg,
 | 
			
		||||
      transparent,
 | 
			
		||||
      rgba(255, 255, 255, 0.1),
 | 
			
		||||
      transparent
 | 
			
		||||
    );
 | 
			
		||||
    animation: shimmer 1.5s infinite;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@keyframes shimmer {
 | 
			
		||||
  0% {
 | 
			
		||||
    transform: translateX(-100%);
 | 
			
		||||
  }
 | 
			
		||||
  100% {
 | 
			
		||||
    transform: translateX(200%);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.markdown-streaming-content {
 | 
			
		||||
  width: 100%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.markdown-streaming-paragraph {
 | 
			
		||||
  opacity: 1;
 | 
			
		||||
  animation: fadeIn 0.3s ease-in-out;
 | 
			
		||||
  margin-bottom: 0.5em;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@keyframes fadeIn {
 | 
			
		||||
  from {
 | 
			
		||||
    opacity: 0.5;
 | 
			
		||||
  }
 | 
			
		||||
  to {
 | 
			
		||||
    opacity: 1;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user