Skip to content

Commit

Permalink
Merge branch 'main' into decorators2022
Browse files Browse the repository at this point in the history
  • Loading branch information
mweststrate authored Aug 29, 2023
2 parents 63acafc + 4ffc001 commit 78bee62
Show file tree
Hide file tree
Showing 31 changed files with 369 additions and 141 deletions.
5 changes: 0 additions & 5 deletions .changeset/polite-laws-provide.md

This file was deleted.

6 changes: 6 additions & 0 deletions packages/mobx-react-lite/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
# mobx-react-lite

## 4.0.4

### Patch Changes

- [`3ceeb865`](https://github.com/mobxjs/mobx/commit/3ceeb8651e328c4c7211c875696b3f5269fea834) [#3732](https://github.com/mobxjs/mobx/pull/3732) Thanks [@urugator](https://github.com/urugator)! - fix: #3734: `isolateGlobalState: true` makes observer stop to react to store changes

## 4.0.3

### Patch Changes
Expand Down
21 changes: 21 additions & 0 deletions packages/mobx-react-lite/__tests__/observer.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -1107,3 +1107,24 @@ test("StrictMode #3671", async () => {
)
expect(container).toHaveTextContent("1")
})

test("`isolateGlobalState` shouldn't break reactivity #3734", async () => {
mobx.configure({ isolateGlobalState: true })

const o = mobx.observable({ x: 0 })

const Cmp = observer(() => o.x as any)

const { container, unmount } = render(<Cmp />)

expect(container).toHaveTextContent("0")
act(
mobx.action(() => {
o.x++
})
)
expect(container).toHaveTextContent("1")
unmount()

mobx._resetGlobalState()
})
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ test("uncommitted components should not leak observations", async () => {
expect(count2IsObserved).toBeFalsy()
},
{
timeout: 10_000,
timeout: 15_000,
interval: 200
}
)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -126,19 +126,19 @@ describe("base useAsObservableSource should work", () => {
const { container } = render(<Parent />)

expect(container.querySelector("span")!.innerHTML).toBe("10")
expect(counterRender).toBe(2)
expect(counterRender).toBe(1)

act(() => {
;(container.querySelector("#inc")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("11")
expect(counterRender).toBe(3)
expect(counterRender).toBe(2)

act(() => {
;(container.querySelector("#incmultiplier")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("22")
expect(counterRender).toBe(5) // TODO: should be 3
expect(counterRender).toBe(4) // One from props, second from updating local observable (setState during render)
})
})

Expand Down Expand Up @@ -271,7 +271,12 @@ describe("combining observer with props and stores", () => {
store.x = 10
})

expect(renderedValues).toEqual([10, 10, 15, 15, 20]) // TODO: should have one 15 less
expect(renderedValues).toEqual([
10,
15, // props change
15, // local observable change (setState during render)
20
])

// TODO: re-enable this line. When debugging, the correct value is returned from render,
// which is also visible with renderedValues, however, querying the dom doesn't show the correct result
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -192,19 +192,19 @@ describe("base useAsObservableSource should work", () => {
const { container } = render(<Parent />)

expect(container.querySelector("span")!.innerHTML).toBe("10")
expect(counterRender).toBe(2)
expect(counterRender).toBe(1)

act(() => {
;(container.querySelector("#inc")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("11")
expect(counterRender).toBe(3)
expect(counterRender).toBe(2)

act(() => {
;(container.querySelector("#incmultiplier")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("22")
expect(counterRender).toBe(5) // TODO: should be 3
expect(counterRender).toBe(4) // One from props, second from updating local observable with effect
})
})

Expand Down Expand Up @@ -337,7 +337,12 @@ describe("combining observer with props and stores", () => {
store.x = 10
})

expect(renderedValues).toEqual([10, 10, 15, 15, 20]) // TODO: should have one 15 less
expect(renderedValues).toEqual([
10,
15, // props change
15, // local observable change during render (localStore.y = props.y)
20
])

expect(container.querySelector("div")!.textContent).toBe("20")
})
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ describe("is used to keep observable within component body", () => {
fireEvent.click(div)
expect(div.textContent).toBe("3-2")

expect(renderCount).toBe(4)
expect(renderCount).toBe(3)
})

it("actions can be used", () => {
Expand Down Expand Up @@ -418,19 +418,19 @@ describe("is used to keep observable within component body", () => {
const { container } = render(<Parent />)

expect(container.querySelector("span")!.innerHTML).toBe("10")
expect(counterRender).toBe(2)
expect(counterRender).toBe(1)

act(() => {
;(container.querySelector("#inc")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("11")
expect(counterRender).toBe(3)
expect(counterRender).toBe(2)

act(() => {
;(container.querySelector("#incmultiplier")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("22")
expect(counterRender).toBe(5) // TODO: should be 3
expect(counterRender).toBe(4) // One from props, second from updating local observable with effect
})
})
})
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -119,8 +119,7 @@ describe("is used to keep observable within component body", () => {
fireEvent.click(div)
expect(div.textContent).toBe("3-2")

// though render 4 times, mobx.observable only called once
expect(renderCount).toBe(4)
expect(renderCount).toBe(3)
})

it("actions can be used", () => {
Expand Down Expand Up @@ -424,19 +423,19 @@ describe("is used to keep observable within component body", () => {
const { container } = render(<Parent />)

expect(container.querySelector("span")!.innerHTML).toBe("10")
expect(counterRender).toBe(2)
expect(counterRender).toBe(1)

act(() => {
;(container.querySelector("#inc")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("11")
expect(counterRender).toBe(3)
expect(counterRender).toBe(2)

act(() => {
;(container.querySelector("#incmultiplier")! as any).click()
})
expect(container.querySelector("span")!.innerHTML).toBe("22")
expect(counterRender).toBe(5) // TODO: should be 3
expect(counterRender).toBe(4) // One from props, second from updating source (setState during render)
})
})
})
4 changes: 2 additions & 2 deletions packages/mobx-react-lite/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "mobx-react-lite",
"version": "4.0.3",
"version": "4.0.4",
"description": "Lightweight React bindings for MobX based on React 16.8+ and Hooks",
"source": "src/index.ts",
"main": "dist/index.js",
Expand Down Expand Up @@ -52,7 +52,7 @@
}
},
"devDependencies": {
"mobx": "^6.9.0",
"mobx": "^6.10.1",
"expose-gc": "^1.0.0"
},
"keywords": [
Expand Down
6 changes: 2 additions & 4 deletions packages/mobx-react-lite/src/useObserver.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,10 +24,8 @@ type ObserverAdministration = {
getSnapshot: Parameters<typeof React.useSyncExternalStore>[1]
}

const mobxGlobalState = _getGlobalState()

// BC
const globalStateVersionIsAvailable = typeof mobxGlobalState.stateVersion !== "undefined"
const globalStateVersionIsAvailable = typeof _getGlobalState().stateVersion !== "undefined"

function createReaction(adm: ObserverAdministration) {
adm.reaction = new Reaction(`observer${adm.name}`, () => {
Expand Down Expand Up @@ -84,7 +82,7 @@ export function useObserver<T>(render: () => T, baseComponentName: string = "obs
getSnapshot() {
// Do NOT access admRef here!
return globalStateVersionIsAvailable
? mobxGlobalState.stateVersion
? _getGlobalState().stateVersion
: adm.stateVersion
}
}
Expand Down
9 changes: 9 additions & 0 deletions packages/mobx-react/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,14 @@
# mobx-react

## 9.0.1

### Patch Changes

- [`d813746c`](https://github.com/mobxjs/mobx/commit/d813746cfaa18d80daddee3724562fed6b307c0a) [#3731](https://github.com/mobxjs/mobx/pull/3731) Thanks [@urugator](https://github.com/urugator)! - fix #3730: class component does not react to state changes performed before mount

- Updated dependencies [[`3ceeb865`](https://github.com/mobxjs/mobx/commit/3ceeb8651e328c4c7211c875696b3f5269fea834)]:
- mobx-react-lite@4.0.4

## 9.0.0

### Major Changes
Expand Down
13 changes: 5 additions & 8 deletions packages/mobx-react/__tests__/finalizationRegistry.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,12 +8,6 @@ import { observer } from "../src"

afterEach(cleanup)

function sleep(time: number) {
return new Promise<void>(res => {
setTimeout(res, time)
})
}

// TODO remove once https://github.com/mobxjs/mobx/pull/3620 is merged.
declare class WeakRef<T> {
constructor(object: T)
Expand Down Expand Up @@ -80,9 +74,12 @@ test("should not prevent GC of uncomitted components", async () => {
expect(aConstructorCount).toBe(2)
expect(aMountCount).toBe(1)

await Promise.resolve()
gc()
await sleep(50)
expect(firstARef!.deref()).toBeUndefined()
await waitFor(() => expect(firstARef!.deref()).toBeUndefined(), {
timeout: 10_000,
interval: 150
})

unmount()
})
5 changes: 2 additions & 3 deletions packages/mobx-react/__tests__/hooks.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -93,10 +93,9 @@ test("computed properties result in double render when using observer instead of
expect(seen).toEqual([
"parent",
0,
0,
"parent",
2,
2 // should contain "2" only once! But with hooks, one update is scheduled based the fact that props change, the other because the observable source changed.
2, // props changed
2 // observable source changed (setState during render)
])
expect(container).toHaveTextContent("2")
})
8 changes: 4 additions & 4 deletions packages/mobx-react/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "mobx-react",
"version": "9.0.0",
"version": "9.0.1",
"description": "React bindings for MobX. Create fully reactive components.",
"source": "src/index.ts",
"main": "dist/index.js",
Expand Down Expand Up @@ -36,7 +36,7 @@
},
"homepage": "https://mobx.js.org",
"dependencies": {
"mobx-react-lite": "^4.0.3"
"mobx-react-lite": "^4.0.4"
},
"peerDependencies": {
"mobx": "^6.9.0",
Expand All @@ -51,8 +51,8 @@
}
},
"devDependencies": {
"mobx": "^6.9.0",
"mobx-react-lite": "^4.0.3",
"mobx": "^6.10.1",
"mobx-react-lite": "^4.0.4",
"expose-gc": "^1.0.0"
},
"keywords": [
Expand Down
8 changes: 8 additions & 0 deletions packages/mobx/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,13 @@
# mobx

## 6.10.1

### Patch Changes

- [`3ceeb865`](https://github.com/mobxjs/mobx/commit/3ceeb8651e328c4c7211c875696b3f5269fea834) [#3732](https://github.com/mobxjs/mobx/pull/3732) Thanks [@urugator](https://github.com/urugator)! - - fix: #3728: Observable initialization updates state version.
- fix: Observable set initialization violates `enforceActions: "always"`.
- fix: Changing keys of observable object does not respect `enforceActions`.

## 6.10.0

### Minor Changes
Expand Down
16 changes: 15 additions & 1 deletion packages/mobx/__tests__/v5/base/extendObservable.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,9 @@ import {
isObservableProp,
isComputedProp,
isAction,
extendObservable
extendObservable,
observable,
reaction
} from "../../../src/mobx"

test("extendObservable should work", function () {
Expand Down Expand Up @@ -160,3 +162,15 @@ test("extendObservable should apply specified decorators", function () {
expect(isAction(box.someFunc)).toBe(true)
expect(box.someFunc()).toEqual(2)
})

test("extendObservable notifies about added keys", () => {
let reactionCalled = false
const o = observable({})
const disposeReaction = reaction(
() => Object.keys(o),
() => (reactionCalled = true)
)
extendObservable(o, { x: 0 })
expect(reactionCalled).toBe(true)
disposeReaction()
})
Loading

0 comments on commit 78bee62

Please sign in to comment.