Files
vue-desktop/src/common/hooks/useObservableVue.ts

88 lines
2.3 KiB
TypeScript
Raw Normal View History

2025-09-24 11:30:06 +08:00
import { reactive, onBeforeUnmount, type Reactive } from 'vue'
import type { IObservable } from '@/core/state/IObservable.ts'
/**
* Vue Hook: useObservable
* ObservableImpl + Vue
* @example
* interface AppState {
* count: number
* user: { name: string; age: number }
* items: number[]
* }
*
* // 创建 ObservableImpl
* const obs = new ObservableImpl<AppState>({
* count: 0,
* user: { name: 'Alice', age: 20 },
* items: []
* })
*
* export default defineComponent({
* setup() {
* // 深层解构 Hook
* const { count, user, items } = useObservable(obs)
*
* const increment = () => {
* count += 1 // 触发 ObservableImpl 通知 + Vue 更新
* }
*
* const changeAge = () => {
* user.age = 30 // 深层对象也能触发通知
* }
*
* const addItem = () => {
* items.push(42) // 数组方法拦截,触发通知
* }
*
* return { count, user, items, increment, changeAge, addItem }
* }
* })
*/
export function useObservable<T extends object>(observable: IObservable<T>): Reactive<T> {
// 创建 Vue 响应式对象
const state = reactive({} as T)
/**
* ObservableImpl Proxy Vue
*
*/
function mapKeys(obj: any, proxy: any) {
(Object.keys(proxy) as (keyof typeof proxy)[]).forEach(key => {
const value = proxy[key]
if (typeof value === 'object' && value !== null) {
// 递归创建子对象 Proxy
obj[key] = reactive({} as typeof value)
mapKeys(obj[key], value)
} else {
// 基本类型通过 getter/setter 同步
Object.defineProperty(obj, key, {
enumerable: true,
configurable: true,
get() {
return proxy[key]
},
set(val) {
proxy[key] = val
},
})
}
})
}
// 获取 ObservableImpl 的 Proxy
const refsProxy = observable.toRefsProxy()
mapKeys(state, refsProxy)
// 订阅 ObservableImpl保持响应式同步
const unsubscribe = observable.subscribe(() => {
// 空实现即可getter/setter 已同步
})
onBeforeUnmount(() => {
unsubscribe()
})
return state
}