From 3b726d4ffe20b621f26c5d9ee861aec00e90244e Mon Sep 17 00:00:00 2001 From: Anthony Fu Date: Mon, 7 Dec 2020 21:44:07 +0800 Subject: [PATCH] feat: add type-level `readonly()` api (#593) * feat: add type-level `readonly()` api * Update src/reactivity/readonly.ts * chore: update tests --- README.md | 12 ++++- README.zh-CN.md | 14 +++++- src/apis/index.ts | 2 + src/reactivity/index.ts | 3 +- src/reactivity/reactive.ts | 57 +--------------------- src/reactivity/readonly.ts | 97 +++++++++++++++++++++++++++++++++++++ test-dts/readonly.test-d.ts | 48 ++++++++++++++++++ 7 files changed, 172 insertions(+), 61 deletions(-) create mode 100644 src/reactivity/readonly.ts create mode 100644 test-dts/readonly.test-d.ts diff --git a/README.md b/README.md index 885d7d92..0a1072c9 100644 --- a/README.md +++ b/README.md @@ -441,6 +441,17 @@ app2.component('Bar', Bar) // equivalent to Vue.use('Bar', Bar) +### `readonly` + +
+ +⚠️ readonly() provides **only type-level** readonly check. + + +`readonly()` is provided as API alignment with Vue 3 on type-level only. Use isReadonly() on it or it's properties can not be guaranteed. + +
+ ### `props`
@@ -467,7 +478,6 @@ defineComponent({ The following APIs introduced in Vue 3 are not available in this plugin. -- `readonly` - `defineAsyncComponent` - `onRenderTracked` - `onRenderTriggered` diff --git a/README.zh-CN.md b/README.zh-CN.md index 11455e41..ae9a04bf 100644 --- a/README.zh-CN.md +++ b/README.zh-CN.md @@ -405,7 +405,7 @@ watch(
-### shallowReadonly +### `shallowReadonly`
@@ -416,6 +416,17 @@ watch(
+### `readonly` + +
+ +⚠️ readonly() **只提供类型层面**的只读。 + + +`readonly()` 只在类型层面提供和 Vue 3 的对齐。在其返回值或其属性上使用 isReadonly() 检查的结果将无法保证。 + +
+ ### `props`
@@ -442,7 +453,6 @@ defineComponent({ 以下在 Vue 3 新引入的 API ,在本插件中暂不适用: -- `readonly` - `defineAsyncComponent` - `onRenderTracked` - `onRenderTriggered` diff --git a/src/apis/index.ts b/src/apis/index.ts index d1821026..ae33a86d 100644 --- a/src/apis/index.ts +++ b/src/apis/index.ts @@ -21,6 +21,8 @@ export { shallowReadonly, proxyRefs, ShallowUnwrapRef, + readonly, + DeepReadonly, } from '../reactivity' export * from './lifecycle' export * from './watch' diff --git a/src/reactivity/index.ts b/src/reactivity/index.ts index f6741977..b045e915 100644 --- a/src/reactivity/index.ts +++ b/src/reactivity/index.ts @@ -5,8 +5,6 @@ export { shallowReactive, toRaw, isRaw, - isReadonly, - shallowReadonly, } from './reactive' export { ref, @@ -23,5 +21,6 @@ export { proxyRefs, ShallowUnwrapRef, } from './ref' +export { readonly, isReadonly, shallowReadonly, DeepReadonly } from './readonly' export { set } from './set' export { del } from './del' diff --git a/src/reactivity/reactive.ts b/src/reactivity/reactive.ts index e594e535..9414051d 100644 --- a/src/reactivity/reactive.ts +++ b/src/reactivity/reactive.ts @@ -4,16 +4,12 @@ import { isPlainObject, def, warn, isArray, hasOwn, noopFn } from '../utils' import { isComponentInstance, defineComponentInstance } from '../utils/helper' import { RefKey } from '../utils/symbols' import { isRef, UnwrapRef } from './ref' -import { rawSet, accessModifiedSet, readonlySet } from '../utils/sets' +import { rawSet, accessModifiedSet } from '../utils/sets' export function isRaw(obj: any): boolean { return Boolean(obj?.__ob__ && obj.__ob__?.__raw__) } -export function isReadonly(obj: any): boolean { - return readonlySet.has(obj) -} - export function isReactive(obj: any): boolean { return Boolean(obj?.__ob__ && !obj.__ob__?.__raw__) } @@ -219,57 +215,6 @@ export function reactive(obj: T): UnwrapRef { return observed as UnwrapRef } -export function shallowReadonly(obj: T): Readonly -export function shallowReadonly(obj: any): any { - if (!(isPlainObject(obj) || isArray(obj)) || !Object.isExtensible(obj)) { - return obj - } - - const readonlyObj = {} - - const source = reactive({}) - const ob = (source as any).__ob__ - - for (const key of Object.keys(obj)) { - let val = obj[key] - let getter: (() => any) | undefined - let setter: ((x: any) => void) | undefined - const property = Object.getOwnPropertyDescriptor(obj, key) - if (property) { - if (property.configurable === false) { - continue - } - getter = property.get - setter = property.set - if ( - (!getter || setter) /* not only have getter */ && - arguments.length === 2 - ) { - val = obj[key] - } - } - - Object.defineProperty(readonlyObj, key, { - enumerable: true, - configurable: true, - get: function getterHandler() { - const value = getter ? getter.call(obj) : val - ob.dep.depend() - return value - }, - set(v) { - if (__DEV__) { - warn(`Set operation on key "${key}" failed: target is readonly.`) - } - }, - }) - } - - readonlySet.set(readonlyObj, true) - - return readonlyObj -} - /** * Make sure obj can't be a reactive */ diff --git a/src/reactivity/readonly.ts b/src/reactivity/readonly.ts new file mode 100644 index 00000000..10ca2416 --- /dev/null +++ b/src/reactivity/readonly.ts @@ -0,0 +1,97 @@ +import { reactive, Ref, UnwrapRef } from '.' +import { isArray, isPlainObject, warn } from '../utils' +import { readonlySet } from '../utils/sets' + +export function isReadonly(obj: any): boolean { + return readonlySet.has(obj) +} + +type Primitive = string | number | boolean | bigint | symbol | undefined | null +type Builtin = Primitive | Function | Date | Error | RegExp + +// prettier-ignore +export type DeepReadonly = T extends Builtin + ? T + : T extends Map + ? ReadonlyMap, DeepReadonly> + : T extends ReadonlyMap + ? ReadonlyMap, DeepReadonly> + : T extends WeakMap + ? WeakMap, DeepReadonly> + : T extends Set + ? ReadonlySet> + : T extends ReadonlySet + ? ReadonlySet> + : T extends WeakSet + ? WeakSet> + : T extends Promise + ? Promise> + : T extends {} + ? { readonly [K in keyof T]: DeepReadonly } + : Readonly + +// only unwrap nested ref +type UnwrapNestedRefs = T extends Ref ? T : UnwrapRef + +/** + * **In @vue/composition-api, `reactive` only provides type-level readonly check** + * + * Creates a readonly copy of the original object. Note the returned copy is not + * made reactive, but `readonly` can be called on an already reactive object. + */ +export function readonly( + target: T +): DeepReadonly> { + return target as any +} + +export function shallowReadonly(obj: T): Readonly +export function shallowReadonly(obj: any): any { + if (!(isPlainObject(obj) || isArray(obj)) || !Object.isExtensible(obj)) { + return obj + } + + const readonlyObj = {} + + const source = reactive({}) + const ob = (source as any).__ob__ + + for (const key of Object.keys(obj)) { + let val = obj[key] + let getter: (() => any) | undefined + let setter: ((x: any) => void) | undefined + const property = Object.getOwnPropertyDescriptor(obj, key) + if (property) { + if (property.configurable === false) { + continue + } + getter = property.get + setter = property.set + if ( + (!getter || setter) /* not only have getter */ && + arguments.length === 2 + ) { + val = obj[key] + } + } + + Object.defineProperty(readonlyObj, key, { + enumerable: true, + configurable: true, + get: function getterHandler() { + const value = getter ? getter.call(obj) : val + ob.dep.depend() + return value + }, + set(v) { + if (__DEV__) { + warn(`Set operation on key "${key}" failed: target is readonly.`) + } + }, + }) + } + + readonlySet.set(readonlyObj, true) + + return readonlyObj +} diff --git a/test-dts/readonly.test-d.ts b/test-dts/readonly.test-d.ts new file mode 100644 index 00000000..1ca26a56 --- /dev/null +++ b/test-dts/readonly.test-d.ts @@ -0,0 +1,48 @@ +import { expectType, readonly, ref } from './index' + +describe('readonly', () => { + it('nested', () => { + const r = readonly({ + obj: { k: 'v' }, + arr: [1, 2, '3'], + objInArr: [{ foo: 'bar' }], + }) + + // @ts-expect-error + r.obj = {} + // @ts-expect-error + r.obj.k = 'x' + + // @ts-expect-error + r.arr.push(42) + // @ts-expect-error + r.objInArr[0].foo = 'bar2' + }) + + it('with ref', () => { + const r = readonly( + ref({ + obj: { k: 'v' }, + arr: [1, 2, '3'], + objInArr: [{ foo: 'bar' }], + }) + ) + + console.log(r.value) + + expectType(r.value.obj.k) + + // @ts-expect-error + r.value = {} + + // @ts-expect-error + r.value.obj = {} + // @ts-expect-error + r.value.obj.k = 'x' + + // @ts-expect-error + r.value.arr.push(42) + // @ts-expect-error + r.value.objInArr[0].foo = 'bar2' + }) +})