Skip to content

Commit

Permalink
fix(react): fix useAttach not work with react18 strict mode (#3284)
Browse files Browse the repository at this point in the history
  • Loading branch information
janryWang authored Jul 19, 2022
1 parent 8569c0f commit 9df806b
Show file tree
Hide file tree
Showing 9 changed files with 121 additions and 69 deletions.
8 changes: 5 additions & 3 deletions packages/react/src/__tests__/field.spec.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -131,7 +131,7 @@ test('ReactiveField', () => {
render(<ReactiveField field={null}>{() => <div></div>}</ReactiveField>)
})

test('useAttach', () => {
test('useAttach basic', async () => {
const form = createForm()
const MyComponent = (props: any) => {
return (
Expand All @@ -143,8 +143,10 @@ test('useAttach', () => {
const { rerender } = render(<MyComponent name="aa" />)
expect(form.query('aa').take().mounted).toBeTruthy()
rerender(<MyComponent name="bb" />)
expect(form.query('aa').take().mounted).toBeFalsy()
expect(form.query('bb').take().mounted).toBeTruthy()
await waitFor(() => {
expect(form.query('aa').take().mounted).toBeFalsy()
expect(form.query('bb').take().mounted).toBeTruthy()
})
})

test('useAttach with array field', async () => {
Expand Down
5 changes: 2 additions & 3 deletions packages/react/src/hooks/useAttach.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,11 @@
import { useEffect } from 'react'

import { unstable_useCompatEffect } from '@formily/reactive-react'
interface IRecycleTarget {
onMount: () => void
onUnmount: () => void
}

export const useAttach = <T extends IRecycleTarget>(target: T): T => {
useEffect(() => {
unstable_useCompatEffect(() => {
target.onMount()
return () => target.onUnmount()
}, [target])
Expand Down
19 changes: 7 additions & 12 deletions packages/react/src/hooks/useFormEffects.ts
Original file line number Diff line number Diff line change
@@ -1,22 +1,17 @@
import { useLayoutEffect, useMemo } from 'react'
import { unstable_useCompatFactory } from '@formily/reactive-react'
import { Form } from '@formily/core'
import { uid } from '@formily/shared'
import { useForm } from './useForm'

export const useFormEffects = (effects?: (form: Form) => void) => {
const form = useForm()
const ref = useMemo(() => {
unstable_useCompatFactory(() => {
const id = uid()
form.addEffects(id, effects)
const request = setTimeout(() => {
form.removeEffects(id)
}, 100)
return { id, request }
}, [])
useLayoutEffect(() => {
clearTimeout(ref.request)
return () => {
form.removeEffects(ref.id)
return {
dispose() {
form.removeEffects(id)
},
}
}, [])
})
}
15 changes: 13 additions & 2 deletions packages/reactive-react/src/hooks/index.ts
Original file line number Diff line number Diff line change
@@ -1,2 +1,13 @@
export * from './useForceUpdate'
export * from './useObserver'
import { useForceUpdate } from './useForceUpdate'
import { useCompatEffect } from './useCompatEffect'
import { useCompatFactory } from './useCompatFactory'
import { useDidUpdate } from './useDidUpdate'
import { useLayoutEffect } from './useLayoutEffect'
import { useObserver } from './useObserver'

export const unstable_useForceUpdate = useForceUpdate
export const unstable_useCompatEffect = useCompatEffect
export const unstable_useCompatFactory = useCompatFactory
export const unstable_useDidUpdate = useDidUpdate
export const unstable_useLayoutEffect = useLayoutEffect
export const unstable_useObserver = useObserver
39 changes: 39 additions & 0 deletions packages/reactive-react/src/hooks/useCompatEffect.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
import { useEffect, useRef, EffectCallback, DependencyList } from 'react'
import { immediate } from '../shared'

const isArr = Array.isArray

const isEqualDeps = (target: any, source: any) => {
const arrA = isArr(target)
const arrB = isArr(source)
if (arrA !== arrB) return false
if (arrA) {
if (target.length !== source.length) return false
return target.every((val, index) => val === source[index])
}
return target === source
}

export const useCompatEffect = (
effect: EffectCallback,
deps?: DependencyList
) => {
const depsRef = useRef<DependencyList>(null)
const mountedRef = useRef(false)
useEffect(() => {
mountedRef.current = true
const dispose = effect()
return () => {
mountedRef.current = false
if (!isEqualDeps(depsRef.current, deps)) {
if (dispose) dispose()
return
}
immediate(() => {
if (mountedRef.current) return
if (dispose) dispose()
})
}
}, deps)
depsRef.current = deps
}
42 changes: 42 additions & 0 deletions packages/reactive-react/src/hooks/useCompatFactory.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import React from 'react'
import { GarbageCollector } from '../shared'
import { useCompatEffect } from './useCompatEffect'

class ObjectToBeRetainedByReact {}

function objectToBeRetainedByReactFactory() {
return new ObjectToBeRetainedByReact()
}

export const useCompatFactory = <T extends { dispose: () => void }>(
factory: () => T
): T => {
const instRef = React.useRef<T>(null)
const gcRef = React.useRef<GarbageCollector>()
const [objectRetainedByReact] = React.useState(
objectToBeRetainedByReactFactory
)
if (!instRef.current) {
instRef.current = factory()
}
//StrictMode/ConcurrentMode会导致组件无法正确触发UnMount,所以只能自己做垃圾回收
if (!gcRef.current) {
gcRef.current = new GarbageCollector(() => {
if (instRef.current) {
instRef.current.dispose()
}
})
gcRef.current.open(objectRetainedByReact)
}

useCompatEffect(() => {
gcRef.current.close()
return () => {
if (instRef.current) {
instRef.current.dispose()
instRef.current = null
}
}
}, [])
return instRef.current
}
59 changes: 11 additions & 48 deletions packages/reactive-react/src/hooks/useObserver.ts
Original file line number Diff line number Diff line change
@@ -1,59 +1,22 @@
import React from 'react'
import { Tracker } from '@formily/reactive'
import { GarbageCollector, immediate } from '../shared'
import { IObserverOptions } from '../types'
import { useForceUpdate } from './useForceUpdate'

class ObjectToBeRetainedByReact {}

function objectToBeRetainedByReactFactory() {
return new ObjectToBeRetainedByReact()
}
import { useCompatFactory } from './useCompatFactory'

export const useObserver = <T extends () => any>(
view: T,
options?: IObserverOptions
): ReturnType<T> => {
const forceUpdate = useForceUpdate()
const mountedRef = React.useRef(false)
const trackerRef = React.useRef<Tracker>(null)
const gcRef = React.useRef<GarbageCollector>()
const [objectRetainedByReact] = React.useState(
objectToBeRetainedByReactFactory
const tracker = useCompatFactory(
() =>
new Tracker(() => {
if (typeof options?.scheduler === 'function') {
options.scheduler(forceUpdate)
} else {
forceUpdate()
}
}, options?.displayName)
)
if (!trackerRef.current) {
trackerRef.current = new Tracker(() => {
if (typeof options?.scheduler === 'function') {
options.scheduler(forceUpdate)
} else {
forceUpdate()
}
}, options?.displayName)
}

//StrictMode/ConcurrentMode会导致组件无法正确触发UnMount,所以只能自己做垃圾回收
if (!gcRef.current) {
gcRef.current = new GarbageCollector(() => {
if (trackerRef.current) {
trackerRef.current.dispose()
}
})
gcRef.current.open(objectRetainedByReact)
}

React.useEffect(() => {
const dispose = () => {
if (trackerRef.current && !mountedRef.current) {
trackerRef.current.dispose()
trackerRef.current = null
}
}
mountedRef.current = true
gcRef.current.close()
return () => {
mountedRef.current = false
immediate(dispose)
}
}, [])
return trackerRef.current.track(view)
return tracker.track(view)
}
1 change: 1 addition & 0 deletions packages/reactive-react/src/index.ts
Original file line number Diff line number Diff line change
@@ -1,2 +1,3 @@
export * from './observer'
export * from './hooks'
export * from './types'
2 changes: 1 addition & 1 deletion packages/reactive-react/src/observer.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import React, { forwardRef, memo, Fragment } from 'react'
import hoistNonReactStatics from 'hoist-non-react-statics'
import { useObserver } from './hooks'
import { useObserver } from './hooks/useObserver'
import { IObserverOptions, IObserverProps, ReactFC } from './types'

export function observer<
Expand Down

0 comments on commit 9df806b

Please sign in to comment.