2.0变量 , 向量总结正式推送

This commit is contained in:
RT15548
2026-02-16 00:30:59 +08:00
parent 17b1fe9091
commit cd9fe53f84
75 changed files with 48287 additions and 12186 deletions

View File

@@ -199,13 +199,6 @@ const DEFAULT_JSON_TEMPLATES = {
]
}
}
}`,
worldNewsRefresh: `{
"world": {
"news": [
{ "title": "新闻标题", "time": "时间(可选)", "content": "新闻内容" }
]
}
}`,
localMapGen: `{
"review": {
@@ -268,7 +261,7 @@ const DEFAULT_PROMPTS = {
stranger: {
u1: v => `你是TRPG数据整理助手。从剧情文本中提取{{user}}遇到的陌生人/NPC整理为JSON数组。`,
a1: () => `明白。请提供【世界观】和【剧情经历】我将提取角色并以JSON数组输出。`,
u2: v => `### 上下文\n\n**1. 世界观:**\n${worldInfo}\n\n**2. {{user}}经历:**\n${history(v.historyCount)}${v.storyOutline ? `\n\n**剧情大纲:**\n${wrap('story_outline', v.storyOutline)}` : ''}${nameList(v.existingContacts, v.existingStrangers)}\n\n### 输出要求\n\n1. 返回一个合法 JSON 数组,使用标准 JSON 语法(键名和字符串都用半角双引号 "\n2. 只提取有具体称呼的角色\n3. 每个角色只需 name / location / info 三个字段\n4. 文本内容中如需使用引号,请使用单引号或中文引号「」或"",不要使用半角双引号 "\n5. 无新角色返回 []\n\n\n模板:${JSON_TEMPLATES.stranger}`,
u2: v => `### 上下文\n\n**1. 世界观:**\n${worldInfo}\n\n**2. {{user}}经历:**\n${history(v.historyCount)}${v.storyOutline ? `\n\n**剧情大纲:**\n${wrap('story_outline', v.storyOutline)}` : ''}${nameList(v.existingContacts, v.existingStrangers)}\n\n### 输出要求\n\n1. 返回一个合法 JSON 数组,使用标准 JSON 语法(键名和字符串都用半角双引号 "\n2. 只提取有具体称呼的角色\n3. 每个角色只需 name / location / info 三个字段\n4. 文本内容中如需使用引号,请使用单引号或中文引号「」或"",不要使用半角双引号 "\n5. 无新角色返回 []\n\n\n模板:${JSON_TEMPLATES.npc}`,
a2: () => `了解开始生成JSON:`
},
worldGenStep1: {
@@ -380,12 +373,6 @@ const DEFAULT_PROMPTS = {
u2: v => `【世界观设定】:\n${worldInfo}\n\n【{{user}}历史】:\n${history(v.historyCount)}\n\n【当前世界状态JSON】可能包含 meta/world/maps 等字段):\n${v.currentWorldData || '{}'}\n\n【JSON模板辅助模式\n${JSON_TEMPLATES.worldSimAssist}`,
a2: () => `开始按 worldSimAssist 模板输出JSON:`
},
worldNewsRefresh: {
u1: v => `你是世界新闻编辑。基于世界观设定与{{user}}近期经历,为世界生成「最新资讯」。\n\n要求:\n1) 只输出 world.news不要输出 maps/meta/其他字段)。\n2) news 至少 ${randomRange(2, 4)} 条;语气轻松、中性,夹带少量日常生活细节;可以包含与主剧情相关的跟进报道。\n3) 只输出符合模板的 JSON禁止解释文字。\n\n- 使用标准 JSON 语法:所有键名与字符串都使用半角双引号\n- 文本内容如需使用引号,请使用单引号或中文引号「」/“”,不要使用半角双引号`,
a1: () => `明白。我将只更新 world.news不改动世界其它字段。请提供当前世界数据。`,
u2: v => `【世界观设定】:\n${worldInfo}\n\n【{{user}}历史】:\n${history(v.historyCount)}\n\n【当前世界状态JSON】可能包含 meta/world/maps 等字段):\n${v.currentWorldData || '{}'}\n\n【JSON模板】\n${JSON_TEMPLATES.worldNewsRefresh}`,
a2: () => `OK, worldNewsRefresh JSON generate start:`
},
localMapGen: {
u1: v => `你是TRPG局部场景生成器。你的任务是根据聊天历史推断{{user}}当前或将要前往的位置(视经历的最后一条消息而定),并为该位置生成详细的局部地图/室内场景。
@@ -602,7 +589,6 @@ export const buildExtractStrangersMessages = v => build('stranger', v);
export const buildWorldGenStep1Messages = v => build('worldGenStep1', v);
export const buildWorldGenStep2Messages = v => build('worldGenStep2', v);
export const buildWorldSimMessages = v => build(v?.mode === 'assist' ? 'worldSimAssist' : 'worldSim', v);
export const buildWorldNewsRefreshMessages = v => build('worldNewsRefresh', v);
export const buildSceneSwitchMessages = v => build('sceneSwitch', v);
export const buildLocalMapGenMessages = v => build('localMapGen', v);
export const buildLocalMapRefreshMessages = v => build('localMapRefresh', v);

File diff suppressed because one or more lines are too long

View File

@@ -21,7 +21,7 @@
*/
// ==================== 1. 导入与常量 ====================
import { extension_settings, saveMetadataDebounced, writeExtensionField } from "../../../../../extensions.js";
import { extension_settings, saveMetadataDebounced } from "../../../../../extensions.js";
import { chat_metadata, name1, processCommands, eventSource, event_types as st_event_types } from "../../../../../../script.js";
import { loadWorldInfo, saveWorldInfo, world_names, world_info } from "../../../../../world-info.js";
import { getContext } from "../../../../../st-context.js";
@@ -33,7 +33,7 @@ import { promptManager } from "../../../../../openai.js";
import {
buildSmsMessages, buildSummaryMessages, buildSmsHistoryContent, buildExistingSummaryContent,
buildNpcGenerationMessages, formatNpcToWorldbookContent, buildExtractStrangersMessages,
buildWorldGenStep1Messages, buildWorldGenStep2Messages, buildWorldSimMessages, buildWorldNewsRefreshMessages, buildSceneSwitchMessages,
buildWorldGenStep1Messages, buildWorldGenStep2Messages, buildWorldSimMessages, buildSceneSwitchMessages,
buildInviteMessages, buildLocalMapGenMessages, buildLocalMapRefreshMessages, buildLocalSceneGenMessages,
buildOverlayHtml, MOBILE_LAYOUT_STYLE, DESKTOP_LAYOUT_STYLE, getPromptConfigPayload, setPromptConfig
} from "./story-outline-prompt.js";
@@ -48,86 +48,6 @@ const DEBUG_KEY = 'LittleWhiteBox_StoryOutline_Debug';
let overlayCreated = false, frameReady = false, pendingMsgs = [], presetCleanup = null, step1Cache = null;
// ==================== PromptConfig (global + character card) ====================
// Global: server storage key `promptConfig` (old behavior)
// Character: character-card extension field (see scheduled-tasks implementation)
const PROMPTS_MODULE_NAME = 'xiaobaix-story-outline-prompts';
let promptConfigGlobal = { jsonTemplates: {}, promptSources: {} };
let promptConfigCharacter = { jsonTemplates: {}, promptSources: {} };
let promptConfigCharacterId = null;
function normalizePromptConfig(cfg) {
const src = (cfg && typeof cfg === 'object') ? cfg : {};
const out = {
jsonTemplates: { ...(src.jsonTemplates || {}) },
promptSources: {},
};
const ps = src.promptSources || src.prompts || {};
Object.entries(ps).forEach(([k, v]) => {
if (!v || typeof v !== 'object' || Array.isArray(v)) return;
out.promptSources[k] = { ...v };
});
return out;
}
function mergePromptConfig(globalCfg, charCfg) {
const g = normalizePromptConfig(globalCfg);
const c = normalizePromptConfig(charCfg);
const mergedPromptSources = { ...(g.promptSources || {}) };
Object.entries(c.promptSources || {}).forEach(([key, parts]) => {
const base = mergedPromptSources[key];
mergedPromptSources[key] = (base && typeof base === 'object' && !Array.isArray(base))
? { ...base, ...(parts || {}) }
: { ...(parts || {}) };
});
return {
jsonTemplates: { ...(g.jsonTemplates || {}), ...(c.jsonTemplates || {}) },
promptSources: mergedPromptSources,
};
}
function getCharacterPromptConfig() {
const ctx = getContext?.();
const charId = ctx?.characterId ?? null;
const char = (charId != null) ? ctx?.characters?.[charId] : null;
const cfg = char?.data?.extensions?.[PROMPTS_MODULE_NAME]?.promptConfig || null;
return normalizePromptConfig(cfg);
}
async function saveCharacterPromptConfig(cfg) {
const ctx = getContext?.();
const charId = ctx?.characterId ?? null;
if (charId == null) return;
const payload = { promptConfig: normalizePromptConfig(cfg) };
await writeExtensionField(Number(charId), PROMPTS_MODULE_NAME, payload);
// Keep in-memory character extension in sync (same pattern as scheduled-tasks).
try {
const char = ctx?.characters?.[charId];
if (char) {
if (!char.data) char.data = {};
if (!char.data.extensions) char.data.extensions = {};
char.data.extensions[PROMPTS_MODULE_NAME] = payload;
}
} catch { }
}
function getPromptConfigPayloadWithStores() {
const base = getPromptConfigPayload?.() || {};
return {
...base,
stores: {
global: normalizePromptConfig(promptConfigGlobal),
character: normalizePromptConfig(promptConfigCharacter),
},
};
}
// ==================== 2. 通用工具 ====================
/** 移动端检测 */
@@ -691,40 +611,17 @@ function postFrame(payload) {
const flushPending = () => { if (!frameReady) return; const f = document.getElementById("xiaobaix-story-outline-iframe"); pendingMsgs.forEach(p => { if (f) postToIframe(f, p, "LittleWhiteBox"); }); pendingMsgs = []; };
async function syncPromptConfigForCurrentCharacter() {
const ctx = getContext?.();
const charId = ctx?.characterId ?? null;
if (promptConfigCharacterId !== charId) {
promptConfigCharacterId = charId;
promptConfigCharacter = getCharacterPromptConfig();
}
try {
const cfg = await StoryOutlineStorage?.get?.('promptConfig', null);
promptConfigGlobal = normalizePromptConfig(cfg);
} catch {
promptConfigGlobal = { jsonTemplates: {}, promptSources: {} };
}
const merged = mergePromptConfig(promptConfigGlobal, promptConfigCharacter);
setPromptConfig?.(merged, false);
postFrame({ type: "PROMPT_CONFIG_UPDATED", promptConfig: getPromptConfigPayloadWithStores() });
}
/** 发送设置到iframe */
function sendSettings() {
const store = getOutlineStore(), { name: charName, desc: charDesc } = getCharInfo();
syncPromptConfigForCurrentCharacter().catch(() => { });
postFrame({
type: "LOAD_SETTINGS", globalSettings: getGlobalSettings(), commSettings: getCommSettings(),
stage: store?.stage ?? 0, deviationScore: store?.deviationScore ?? 0,
simulationTarget: store?.simulationTarget ?? 5, playerLocation: store?.playerLocation ?? '家',
dataChecked: store?.dataChecked || {}, outlineData: store?.outlineData || {}, promptConfig: getPromptConfigPayloadWithStores(),
dataChecked: store?.dataChecked || {}, outlineData: store?.outlineData || {}, promptConfig: getPromptConfigPayload?.(),
characterCardName: charName, characterCardDescription: charDesc,
characterContactSmsHistory: getCharSmsHistory()
});
}
const loadAndSend = () => { const s = getOutlineStore(); if (s?.mapData) postFrame({ type: "LOAD_MAP_DATA", mapData: s.mapData }); sendSettings(); };
@@ -814,7 +711,6 @@ const V = {
wg1: d => !!d && typeof d === 'object', // 只要是对象就行,后续会 normalize
wg2: d => !!((d?.world && (d?.maps || d?.world?.maps)?.outdoor) || (d?.outdoor && d?.inside)),
wga: d => !!((d?.world && d?.maps?.outdoor) || d?.outdoor), ws: d => !!d, w: o => !!o && typeof o === 'object',
wn: d => Array.isArray(d?.world?.news),
lm: o => !!o?.inside?.name && !!o?.inside?.description
};
@@ -1198,34 +1094,6 @@ async function handleSimWorld({ requestId, currentData, isAuto }) {
} catch (e) { replyErr('SIMULATE_WORLD_RESULT', requestId, `推演失败: ${e.message}`); }
}
async function handleRefreshWorldNews({ requestId }) {
try {
const store = getOutlineStore();
const od = store?.outlineData;
if (!od) return replyErr('REFRESH_WORLD_NEWS_RESULT', requestId, '未找到世界数据,请先生成世界');
// Store may persist maps either under `maps` or as `outdoor/indoor` (iframe SAVE_ALL_DATA format).
const maps = od?.maps || { outdoor: od?.outdoor || null, indoor: od?.indoor || null };
const snapshot = {
meta: od?.meta || {},
world: od?.world || {},
maps,
...(od?.timeline ? { timeline: od.timeline } : {}),
};
const msgs = buildWorldNewsRefreshMessages(getCommonPromptVars({
currentWorldData: JSON.stringify(snapshot, null, 2),
}));
const data = await callLLMJson({ messages: msgs, validate: V.wn });
if (!Array.isArray(data?.world?.news)) return replyErr('REFRESH_WORLD_NEWS_RESULT', requestId, '世界新闻刷新失败:无法解析 JSON 数据');
reply('REFRESH_WORLD_NEWS_RESULT', requestId, { success: true, news: data.world.news });
} catch (e) {
replyErr('REFRESH_WORLD_NEWS_RESULT', requestId, `世界新闻刷新失败: ${e.message}`);
}
}
function handleSaveSettings(d) {
if (d.globalSettings) saveGlobalSettings(d.globalSettings);
if (d.commSettings) saveCommSettings(d.commSettings);
@@ -1247,56 +1115,39 @@ function handleSaveSettings(d) {
}
async function handleSavePrompts(d) {
const scope = d?.scope === 'character' ? 'character' : 'global';
// Back-compat: full payload (old iframe) -> treat as global save (old server storage behavior).
// Back-compat: full payload (old iframe)
if (d?.promptConfig) {
promptConfigGlobal = normalizePromptConfig(d.promptConfig);
try { await StoryOutlineStorage?.set?.('promptConfig', promptConfigGlobal); } catch { }
// Re-read current character config (if any) and apply merged.
promptConfigCharacter = getCharacterPromptConfig();
const merged = mergePromptConfig(promptConfigGlobal, promptConfigCharacter);
setPromptConfig?.(merged, false);
postFrame({ type: "PROMPT_CONFIG_UPDATED", promptConfig: getPromptConfigPayloadWithStores() });
const payload = setPromptConfig?.(d.promptConfig, false) || d.promptConfig;
try { await StoryOutlineStorage?.set?.('promptConfig', payload); } catch { }
postFrame({ type: "PROMPT_CONFIG_UPDATED", promptConfig: getPromptConfigPayload?.() });
return;
}
// New: incremental update by key
const key = d?.key;
if (!key) return;
// Always merge against the latest global config from server storage.
try { promptConfigGlobal = normalizePromptConfig(await StoryOutlineStorage?.get?.('promptConfig', null)); } catch { }
// Always merge against the current character-card config.
promptConfigCharacterId = getContext?.()?.characterId ?? null;
promptConfigCharacter = getCharacterPromptConfig();
let current = null;
try { current = await StoryOutlineStorage?.get?.('promptConfig', null); } catch { }
const next = (current && typeof current === 'object') ? {
jsonTemplates: { ...(current.jsonTemplates || {}) },
promptSources: { ...(current.promptSources || {}) },
} : { jsonTemplates: {}, promptSources: {} };
const applyDelta = (cfg) => {
const next = normalizePromptConfig(cfg);
if (d?.reset) {
delete next.promptSources[key];
delete next.jsonTemplates[key];
return next;
}
if (d?.reset) {
delete next.promptSources[key];
delete next.jsonTemplates[key];
} else {
if (d?.prompt && typeof d.prompt === 'object') next.promptSources[key] = d.prompt;
if ('jsonTemplate' in (d || {})) {
if (d.jsonTemplate == null) delete next.jsonTemplates[key];
else next.jsonTemplates[key] = String(d.jsonTemplate ?? '');
}
return next;
};
if (scope === 'character') {
promptConfigCharacter = applyDelta(promptConfigCharacter);
await saveCharacterPromptConfig(promptConfigCharacter);
} else {
promptConfigGlobal = applyDelta(promptConfigGlobal);
try { await StoryOutlineStorage?.set?.('promptConfig', promptConfigGlobal); } catch { }
}
const merged = mergePromptConfig(promptConfigGlobal, promptConfigCharacter);
setPromptConfig?.(merged, false);
postFrame({ type: "PROMPT_CONFIG_UPDATED", promptConfig: getPromptConfigPayloadWithStores() });
const payload = setPromptConfig?.(next, false) || next;
try { await StoryOutlineStorage?.set?.('promptConfig', payload); } catch { }
postFrame({ type: "PROMPT_CONFIG_UPDATED", promptConfig: getPromptConfigPayload?.() });
}
function handleSaveContacts(d) {
@@ -1357,7 +1208,6 @@ const handlers = {
GENERATE_WORLD: handleGenWorld,
RETRY_WORLD_GEN_STEP2: handleRetryStep2,
SIMULATE_WORLD: handleSimWorld,
REFRESH_WORLD_NEWS: handleRefreshWorldNews,
GENERATE_LOCAL_MAP: handleGenLocalMap,
REFRESH_LOCAL_MAP: handleRefreshLocalMap,
GENERATE_LOCAL_SCENE: handleGenLocalScene
@@ -1520,7 +1370,10 @@ document.addEventListener('xiaobaixEnabledChanged', e => {
async function initPromptConfigFromServer() {
try {
await syncPromptConfigForCurrentCharacter();
const cfg = await StoryOutlineStorage?.get?.('promptConfig', null);
if (!cfg) return;
setPromptConfig?.(cfg, false);
postFrame({ type: "PROMPT_CONFIG_UPDATED", promptConfig: getPromptConfigPayload?.() });
} catch { }
}