mirror of
				https://github.com/Yidadaa/ChatGPT-Next-Web.git
				synced 2025-10-23 00:19:23 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			256 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { useNavigate } from "react-router-dom";
 | |
| 
 | |
| import { ModelType, Theme, useAppConfig } from "@/app/store/config";
 | |
| import { useChatStore } from "@/app/store/chat";
 | |
| import { ChatControllerPool } from "@/app/client/controller";
 | |
| import { useAllModels } from "@/app/utils/hooks";
 | |
| import { useEffect, useMemo, useState } from "react";
 | |
| import { isVisionModel } from "@/app/utils";
 | |
| import { showToast } from "@/app/components/ui-lib";
 | |
| import Locale from "@/app/locales";
 | |
| import { Path } from "@/app/constant";
 | |
| 
 | |
| import LightIcon from "@/app/icons/light.svg";
 | |
| import DarkIcon from "@/app/icons/dark.svg";
 | |
| import AutoIcon from "@/app/icons/auto.svg";
 | |
| import BottomIcon from "@/app/icons/bottom.svg";
 | |
| import StopIcon from "@/app/icons/pause.svg";
 | |
| import RobotIcon from "@/app/icons/robot.svg";
 | |
| import LoadingButtonIcon from "@/app/icons/loading.svg";
 | |
| import PromptIcon from "@/app/icons/prompt.svg";
 | |
| import MaskIcon from "@/app/icons/mask.svg";
 | |
| import BreakIcon from "@/app/icons/break.svg";
 | |
| import SettingsIcon from "@/app/icons/chat-settings.svg";
 | |
| import ImageIcon from "@/app/icons/image.svg";
 | |
| import AddCircleIcon from "@/app/icons/addCircle.svg";
 | |
| 
 | |
| import Popover from "@/app/components/Popover";
 | |
| 
 | |
| export function ChatActions(props: {
 | |
|   uploadImage: () => void;
 | |
|   setAttachImages: (images: string[]) => void;
 | |
|   setUploading: (uploading: boolean) => void;
 | |
|   showChatSetting: () => void;
 | |
|   scrollToBottom: () => void;
 | |
|   showPromptHints: () => void;
 | |
|   showModelSelector: (show: boolean) => void;
 | |
|   hitBottom: boolean;
 | |
|   uploading: boolean;
 | |
|   isMobileScreen: boolean;
 | |
|   className?: string;
 | |
| }) {
 | |
|   const config = useAppConfig();
 | |
|   const navigate = useNavigate();
 | |
|   const chatStore = useChatStore();
 | |
| 
 | |
|   // switch themes
 | |
|   const theme = config.theme;
 | |
|   function nextTheme() {
 | |
|     const themes = [Theme.Auto, Theme.Light, Theme.Dark];
 | |
|     const themeIndex = themes.indexOf(theme);
 | |
|     const nextIndex = (themeIndex + 1) % themes.length;
 | |
|     const nextTheme = themes[nextIndex];
 | |
|     config.update((config) => (config.theme = nextTheme));
 | |
|   }
 | |
| 
 | |
|   // stop all responses
 | |
|   const couldStop = ChatControllerPool.hasPending();
 | |
|   const stopAll = () => ChatControllerPool.stopAll();
 | |
| 
 | |
|   // switch model
 | |
|   const currentModel = chatStore.currentSession().mask.modelConfig.model;
 | |
|   const allModels = useAllModels();
 | |
|   const models = useMemo(
 | |
|     () => allModels.filter((m) => m.available),
 | |
|     [allModels],
 | |
|   );
 | |
|   const [showUploadImage, setShowUploadImage] = useState(false);
 | |
| 
 | |
|   useEffect(() => {
 | |
|     const show = isVisionModel(currentModel);
 | |
|     setShowUploadImage(show);
 | |
|     if (!show) {
 | |
|       props.setAttachImages([]);
 | |
|       props.setUploading(false);
 | |
|     }
 | |
| 
 | |
|     // if current model is not available
 | |
|     // switch to first available model
 | |
|     const isUnavaliableModel = !models.some((m) => m.name === currentModel);
 | |
|     if (isUnavaliableModel && models.length > 0) {
 | |
|       const nextModel = models[0].name as ModelType;
 | |
|       chatStore.updateCurrentSession(
 | |
|         (session) => (session.mask.modelConfig.model = nextModel),
 | |
|       );
 | |
|       showToast(nextModel);
 | |
|     }
 | |
|   }, [chatStore, currentModel, models]);
 | |
| 
 | |
|   const actions = [
 | |
|     {
 | |
|       onClick: stopAll,
 | |
|       text: Locale.Chat.InputActions.Stop,
 | |
|       isShow: couldStop,
 | |
|       icon: <StopIcon />,
 | |
|       placement: "left",
 | |
|     },
 | |
|     {
 | |
|       onClick: props.scrollToBottom,
 | |
|       text: Locale.Chat.InputActions.ToBottom,
 | |
|       isShow: !props.hitBottom,
 | |
|       icon: <BottomIcon />,
 | |
|       placement: "left",
 | |
|     },
 | |
|     {
 | |
|       onClick: props.showChatSetting,
 | |
|       text: Locale.Chat.InputActions.Settings,
 | |
|       isShow: true,
 | |
|       icon: <SettingsIcon />,
 | |
|       placement: "right",
 | |
|     },
 | |
|     {
 | |
|       onClick: props.uploadImage,
 | |
|       text: Locale.Chat.InputActions.UploadImage,
 | |
|       isShow: showUploadImage,
 | |
|       icon: props.uploading ? <LoadingButtonIcon /> : <ImageIcon />,
 | |
|       placement: "left",
 | |
|     },
 | |
|     {
 | |
|       onClick: nextTheme,
 | |
|       text: Locale.Chat.InputActions.Theme[theme],
 | |
|       isShow: true,
 | |
|       icon: (
 | |
|         <>
 | |
|           {theme === Theme.Auto ? (
 | |
|             <AutoIcon />
 | |
|           ) : theme === Theme.Light ? (
 | |
|             <LightIcon />
 | |
|           ) : theme === Theme.Dark ? (
 | |
|             <DarkIcon />
 | |
|           ) : null}
 | |
|         </>
 | |
|       ),
 | |
|       placement: "left",
 | |
|     },
 | |
|     {
 | |
|       onClick: props.showPromptHints,
 | |
|       text: Locale.Chat.InputActions.Prompt,
 | |
|       isShow: true,
 | |
|       icon: <PromptIcon />,
 | |
|       placement: "left",
 | |
|     },
 | |
|     {
 | |
|       onClick: () => {
 | |
|         navigate(Path.Masks);
 | |
|       },
 | |
|       text: Locale.Chat.InputActions.Masks,
 | |
|       isShow: true,
 | |
|       icon: <MaskIcon />,
 | |
|       placement: "left",
 | |
|     },
 | |
|     {
 | |
|       onClick: () => {
 | |
|         chatStore.updateCurrentSession((session) => {
 | |
|           if (session.clearContextIndex === session.messages.length) {
 | |
|             session.clearContextIndex = undefined;
 | |
|           } else {
 | |
|             session.clearContextIndex = session.messages.length;
 | |
|             session.memoryPrompt = ""; // will clear memory
 | |
|           }
 | |
|         });
 | |
|       },
 | |
|       text: Locale.Chat.InputActions.Clear,
 | |
|       isShow: true,
 | |
|       icon: <BreakIcon />,
 | |
|       placement: "right",
 | |
|     },
 | |
|     {
 | |
|       onClick: () => props.showModelSelector(true),
 | |
|       text: currentModel,
 | |
|       isShow: true,
 | |
|       icon: <RobotIcon />,
 | |
|       placement: "left",
 | |
|     },
 | |
|   ] as const;
 | |
| 
 | |
|   if (props.isMobileScreen) {
 | |
|     const content = (
 | |
|       <div className="w-[100%]">
 | |
|         {actions
 | |
|           .filter((v) => v.isShow)
 | |
|           .map((act) => {
 | |
|             return (
 | |
|               <div
 | |
|                 key={act.text}
 | |
|                 className={`flex items-center gap-3 p-3 bg-white hover:bg-select-btn rounded-action-btn leading-6`}
 | |
|                 onClick={act.onClick}
 | |
|               >
 | |
|                 {act.icon}
 | |
|                 <div className="flex-1 font-common text-actions-popover-menu-item">
 | |
|                   {act.text}
 | |
|                 </div>
 | |
|               </div>
 | |
|             );
 | |
|           })}
 | |
|       </div>
 | |
|     );
 | |
|     return (
 | |
|       <Popover
 | |
|         content={content}
 | |
|         trigger="click"
 | |
|         placement="lt"
 | |
|         noArrow
 | |
|         popoverClassName="border-actions-popover border-gray-200 rounded-md shadow-actions-popover w-actions-popover bg-white "
 | |
|       >
 | |
|         <AddCircleIcon />
 | |
|       </Popover>
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   const popoverClassName = `bg-chat-actions-popover-color whitespace-nowrap px-3 py-2.5 text-white text-sm-title rounded-md`;
 | |
| 
 | |
|   return (
 | |
|     <div className={`flex gap-2 item-center ${props.className}`}>
 | |
|       {actions
 | |
|         .filter((v) => v.placement === "left" && v.isShow)
 | |
|         .map((act, ind) => {
 | |
|           return (
 | |
|             <Popover
 | |
|               key={act.text}
 | |
|               content={act.text}
 | |
|               popoverClassName={`${popoverClassName}`}
 | |
|               placement={ind ? "t" : "rt"}
 | |
|             >
 | |
|               <div
 | |
|                 className="h-[32px] w-[32px] flex items-center justify-center hover:bg-gray-200 hover:rounded-action-btn"
 | |
|                 onClick={act.onClick}
 | |
|               >
 | |
|                 {act.icon}
 | |
|               </div>
 | |
|             </Popover>
 | |
|           );
 | |
|         })}
 | |
|       <div className="flex-1"></div>
 | |
|       {actions
 | |
|         .filter((v) => v.placement === "right" && v.isShow)
 | |
|         .map((act, ind, arr) => {
 | |
|           return (
 | |
|             <Popover
 | |
|               key={act.text}
 | |
|               content={act.text}
 | |
|               popoverClassName={`${popoverClassName}`}
 | |
|               placement={ind === arr.length - 1 ? "lt" : "t"}
 | |
|             >
 | |
|               <div
 | |
|                 className="h-[32px] w-[32px] flex items-center justify-center hover:bg-gray-200 hover:rounded-action-btn"
 | |
|                 onClick={act.onClick}
 | |
|               >
 | |
|                 {act.icon}
 | |
|               </div>
 | |
|             </Popover>
 | |
|           );
 | |
|         })}
 | |
|     </div>
 | |
|   );
 | |
| }
 |