docs/archives/102-web-architecture-refactor/composables-plan.md
在对核心服务进行“去单例化”重构后,应用启动时暴露出一系列与 Vue 响应式系统和组件通信相关的严重问题。这些问题起初表现为多种警告和错误:
Boolean 却收到了 Object ([Vue warn]: Invalid prop: type check failed)。此问题在 PromptPanel 等多个组件中普遍存在。useStorage 等 Composable 因其依赖的 services 对象尚未初始化,导致 watch 侦听了一个 undefined 源 ([Vue warn]: Invalid watch source: undefined)。Must be called at the top of a 'setup' function 错误。经过排查,这些看似分散的问题,都指向同一个系统性的架构缺陷:Composable 的状态封装模式不当。
许多业务逻辑 Composable(如 usePromptOptimizer, useModelManager)返回的是一个包含了多个 ref 的普通 JavaScript 对象,形如:
// 旧模式
function usePromptOptimizer() {
const isIterating = ref(false);
const someOtherState = ref('');
return { isIterating, someOtherState };
}
当在 App.vue 中使用时:
<!-- App.vue -->
<script setup>
const optimizer = usePromptOptimizer();
</script>
<template>
<!--
问题所在:optimizer.isIterating 是一个 ref 对象,
而不是它内部的值。Vue 的模板自动解包不会深入到对象的属性。
-->
<PromptPanel :is-iterating="optimizer.isIterating" />
</template>
PromptPanel 组件收到的 is-iterating prop 是一个 Ref<boolean> 对象,而非期望的 boolean 值,导致类型检查失败。这个问题是所有连锁反应的核心。
reactive 对象为了从根本上解决问题,我们采取了统一的架构决策:重构所有核心业务 Composable,使其返回一个单一的 reactive 对象。
// ✅ 新模式
function usePromptOptimizer() {
const state = reactive({
isIterating: false,
someOtherState: '',
});
// ... 逻辑代码修改 state ...
return state; // 返回一个响应式对象
}
当在 App.vue 中使用时,问题迎刃而解:
<!-- App.vue (修改后) -->
<script setup>
const optimizerState = usePromptOptimizer();
</script>
<template>
<!--
现在 optimizerState.isIterating 直接是 boolean 值,
符合子组件的 prop 预期。
-->
<PromptPanel :is-iterating="optimizerState.isIterating" />
</template>
这个模式确保了传递给子组件的是原始值,而不是 ref 包装器,同时保留了跨组件的状态响应性。
本次重构已圆满完成。
核心重构:
usePromptOptimizer: 已重构为返回 reactive 对象。useModelManager: 已重构为返回 reactive 对象。useHistoryManager: 已重构为返回 reactive 对象。useTemplateManager: 已重构为返回 reactive 对象。usePromptTester: 已重构为返回 reactive 对象。useModals: 已重构为返回 reactive 对象。辅助修复:
useStorage: ThemeToggleUI 和 LanguageSwitch 组件被修改为通过 inject 获取 services 实例,并将其传递给 useStorage,解决了依赖过早初始化的问题。App.vue: 调整了 App.vue 中的模板绑定和 computed 属性,以适应新的 reactive 状态结构,并修复了因此产生的类型错误。ModelSelect 和 DataManager 等组件中,推广了使用 inject 直接从 services 中获取依赖的模式,简化了 App.vue 的模板。最终成果:
warn 和 error。App.vue,变得更加简洁和易于维护。reactive vs. 对象包裹的 ref: 对于一组高度内聚、会被一同传递或操作的响应式状态,使用 reactive 封装是比返回一个包含多个 ref 的对象更优的模式。它能有效避免深层解包问题,并简化消费端的代码。provide/inject 是服务注入的利器: 对于全局性或跨层级的服务/依赖(如 services 对象),使用 provide/inject 是比层层传递 props 更优雅、更高效的解决方案。