- Add CustomRequestEditor component with JSON validation and real-time formatting - Implement bidirectional sync between chat messages and custom request body - Add persistent local storage for chat messages (separate from config) - Remove redundant System Prompt field in custom mode - Refactor configuration storage to separate messages and settings New Features: • Custom request body mode with JSON editor and syntax highlighting • Real-time bidirectional synchronization between chat UI and custom request body • Persistent message storage that survives page refresh • Enhanced configuration export/import including message data • Improved parameter organization with collapsible sections Technical Changes: • Add loadMessages/saveMessages functions in configStorage • Update usePlaygroundState hook to handle message persistence • Refactor SettingsPanel to remove System Prompt in custom mode • Add STORAGE_KEYS constants for better storage key management • Implement debounced auto-save for both config and messages • Add hash-based change detection to prevent unnecessary updates UI/UX Improvements: • Disabled state styling for parameters in custom mode • Warning banners and visual feedback for mode switching • Mobile-responsive design for custom request editor • Consistent styling with existing design system
213 lines
6.0 KiB
JavaScript
213 lines
6.0 KiB
JavaScript
import { useState, useCallback, useRef, useEffect } from 'react';
|
|
import { DEFAULT_MESSAGES, DEFAULT_CONFIG, DEBUG_TABS } from '../utils/constants';
|
|
import { loadConfig, saveConfig, loadMessages, saveMessages } from '../components/playground/configStorage';
|
|
|
|
export const usePlaygroundState = () => {
|
|
// 使用 ref 缓存初始配置,只加载一次
|
|
const initialConfigRef = useRef(null);
|
|
if (!initialConfigRef.current) {
|
|
initialConfigRef.current = loadConfig();
|
|
}
|
|
const savedConfig = initialConfigRef.current;
|
|
|
|
// 加载保存的消息,如果没有则使用默认消息
|
|
const initialMessages = loadMessages() || DEFAULT_MESSAGES;
|
|
|
|
// 基础配置状态
|
|
const [inputs, setInputs] = useState(savedConfig.inputs || DEFAULT_CONFIG.inputs);
|
|
const [parameterEnabled, setParameterEnabled] = useState(
|
|
savedConfig.parameterEnabled || DEFAULT_CONFIG.parameterEnabled
|
|
);
|
|
const [showDebugPanel, setShowDebugPanel] = useState(
|
|
savedConfig.showDebugPanel || DEFAULT_CONFIG.showDebugPanel
|
|
);
|
|
const [customRequestMode, setCustomRequestMode] = useState(
|
|
savedConfig.customRequestMode || DEFAULT_CONFIG.customRequestMode
|
|
);
|
|
const [customRequestBody, setCustomRequestBody] = useState(
|
|
savedConfig.customRequestBody || DEFAULT_CONFIG.customRequestBody
|
|
);
|
|
|
|
// UI状态
|
|
const [showSettings, setShowSettings] = useState(false);
|
|
const [models, setModels] = useState([]);
|
|
const [groups, setGroups] = useState([]);
|
|
const [status, setStatus] = useState({});
|
|
|
|
// 消息相关状态 - 使用加载的消息初始化
|
|
const [message, setMessage] = useState(initialMessages);
|
|
|
|
// 调试状态
|
|
const [debugData, setDebugData] = useState({
|
|
request: null,
|
|
response: null,
|
|
timestamp: null,
|
|
previewRequest: null,
|
|
previewTimestamp: null
|
|
});
|
|
const [activeDebugTab, setActiveDebugTab] = useState(DEBUG_TABS.PREVIEW);
|
|
const [previewPayload, setPreviewPayload] = useState(null);
|
|
|
|
// 编辑状态
|
|
const [editingMessageId, setEditingMessageId] = useState(null);
|
|
const [editValue, setEditValue] = useState('');
|
|
|
|
// Refs
|
|
const sseSourceRef = useRef(null);
|
|
const chatRef = useRef(null);
|
|
const saveConfigTimeoutRef = useRef(null);
|
|
const saveMessagesTimeoutRef = useRef(null);
|
|
|
|
// 配置更新函数
|
|
const handleInputChange = useCallback((name, value) => {
|
|
setInputs(prev => ({ ...prev, [name]: value }));
|
|
}, []);
|
|
|
|
const handleParameterToggle = useCallback((paramName) => {
|
|
setParameterEnabled(prev => ({
|
|
...prev,
|
|
[paramName]: !prev[paramName]
|
|
}));
|
|
}, []);
|
|
|
|
// 消息保存函数
|
|
const debouncedSaveMessages = useCallback(() => {
|
|
if (saveMessagesTimeoutRef.current) {
|
|
clearTimeout(saveMessagesTimeoutRef.current);
|
|
}
|
|
|
|
saveMessagesTimeoutRef.current = setTimeout(() => {
|
|
saveMessages(message);
|
|
}, 1000);
|
|
}, [message]);
|
|
|
|
// 配置保存
|
|
const debouncedSaveConfig = useCallback(() => {
|
|
if (saveConfigTimeoutRef.current) {
|
|
clearTimeout(saveConfigTimeoutRef.current);
|
|
}
|
|
|
|
saveConfigTimeoutRef.current = setTimeout(() => {
|
|
const configToSave = {
|
|
inputs,
|
|
parameterEnabled,
|
|
showDebugPanel,
|
|
customRequestMode,
|
|
customRequestBody,
|
|
};
|
|
saveConfig(configToSave);
|
|
}, 1000);
|
|
}, [inputs, parameterEnabled, showDebugPanel, customRequestMode, customRequestBody]);
|
|
|
|
// 配置导入/重置
|
|
const handleConfigImport = useCallback((importedConfig) => {
|
|
if (importedConfig.inputs) {
|
|
setInputs(prev => ({ ...prev, ...importedConfig.inputs }));
|
|
}
|
|
if (importedConfig.parameterEnabled) {
|
|
setParameterEnabled(prev => ({ ...prev, ...importedConfig.parameterEnabled }));
|
|
}
|
|
if (typeof importedConfig.showDebugPanel === 'boolean') {
|
|
setShowDebugPanel(importedConfig.showDebugPanel);
|
|
}
|
|
if (importedConfig.customRequestMode) {
|
|
setCustomRequestMode(importedConfig.customRequestMode);
|
|
}
|
|
if (importedConfig.customRequestBody) {
|
|
setCustomRequestBody(importedConfig.customRequestBody);
|
|
}
|
|
// 如果导入的配置包含消息,也恢复消息
|
|
if (importedConfig.messages && Array.isArray(importedConfig.messages)) {
|
|
setMessage(importedConfig.messages);
|
|
}
|
|
}, []);
|
|
|
|
const handleConfigReset = useCallback((options = {}) => {
|
|
const { resetMessages = false } = options;
|
|
|
|
setInputs(DEFAULT_CONFIG.inputs);
|
|
setParameterEnabled(DEFAULT_CONFIG.parameterEnabled);
|
|
setShowDebugPanel(DEFAULT_CONFIG.showDebugPanel);
|
|
setCustomRequestMode(DEFAULT_CONFIG.customRequestMode);
|
|
setCustomRequestBody(DEFAULT_CONFIG.customRequestBody);
|
|
|
|
// 只有在明确指定时才重置消息
|
|
if (resetMessages) {
|
|
setMessage(DEFAULT_MESSAGES);
|
|
}
|
|
}, []);
|
|
|
|
// 监听消息变化并自动保存
|
|
useEffect(() => {
|
|
debouncedSaveMessages();
|
|
}, [debouncedSaveMessages]);
|
|
|
|
// 清理定时器
|
|
useEffect(() => {
|
|
return () => {
|
|
if (saveConfigTimeoutRef.current) {
|
|
clearTimeout(saveConfigTimeoutRef.current);
|
|
}
|
|
if (saveMessagesTimeoutRef.current) {
|
|
clearTimeout(saveMessagesTimeoutRef.current);
|
|
}
|
|
};
|
|
}, []);
|
|
|
|
return {
|
|
// 配置状态
|
|
inputs,
|
|
parameterEnabled,
|
|
showDebugPanel,
|
|
customRequestMode,
|
|
customRequestBody,
|
|
|
|
// UI状态
|
|
showSettings,
|
|
models,
|
|
groups,
|
|
status,
|
|
|
|
// 消息状态
|
|
message,
|
|
|
|
// 调试状态
|
|
debugData,
|
|
activeDebugTab,
|
|
previewPayload,
|
|
|
|
// 编辑状态
|
|
editingMessageId,
|
|
editValue,
|
|
|
|
// Refs
|
|
sseSourceRef,
|
|
chatRef,
|
|
saveConfigTimeoutRef,
|
|
|
|
// 更新函数
|
|
setInputs,
|
|
setParameterEnabled,
|
|
setShowDebugPanel,
|
|
setCustomRequestMode,
|
|
setCustomRequestBody,
|
|
setShowSettings,
|
|
setModels,
|
|
setGroups,
|
|
setStatus,
|
|
setMessage,
|
|
setDebugData,
|
|
setActiveDebugTab,
|
|
setPreviewPayload,
|
|
setEditingMessageId,
|
|
setEditValue,
|
|
|
|
// 处理函数
|
|
handleInputChange,
|
|
handleParameterToggle,
|
|
debouncedSaveConfig,
|
|
debouncedSaveMessages,
|
|
handleConfigImport,
|
|
handleConfigReset,
|
|
};
|
|
};
|