-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[no-changelog-required]
- Loading branch information
Showing
6 changed files
with
289 additions
and
30 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,152 @@ | ||
import { ClassModel, action, cachedView, getSnapshot, register, types } from "../src"; | ||
|
||
@register | ||
class ViewExample extends ClassModel({ key: types.identifier, name: types.string }) { | ||
@cachedView() | ||
get slug() { | ||
return this.name.toLowerCase().replace(/ /g, "-"); | ||
} | ||
|
||
@action | ||
setName(name: string) { | ||
this.name = name; | ||
} | ||
} | ||
|
||
describe("class model cached views", () => { | ||
test("an observable instance saves the view value in a snapshot when changed", () => { | ||
const instance = ViewExample.create({ key: "1", name: "Test" }); | ||
expect(instance.slug).toEqual("test"); | ||
let snapshot = getSnapshot(instance); | ||
expect(snapshot).toEqual({ key: "1", name: "Test" }); // no snapshot output as the object hasn't changed yet | ||
instance.setName("New Name"); | ||
snapshot = getSnapshot(instance); | ||
expect(snapshot).toEqual({ key: "1", name: "New Name", slug: "new-name" }); | ||
}); | ||
|
||
test("an observable instance updates the saved view when the observed view value changes", () => { | ||
const instance = ViewExample.create({ key: "1", name: "Test" }); | ||
instance.setName("New Name"); | ||
expect(instance.slug).toEqual("new-name"); | ||
const snapshot = getSnapshot(instance); | ||
expect(snapshot).toEqual({ key: "1", name: "New Name", slug: "new-name" }); | ||
}); | ||
|
||
test("an observable instance ignores the input snapshot value as the logic may have changed", () => { | ||
const instance = ViewExample.create({ key: "1", name: "Test", slug: "outdated-cache" } as any); | ||
expect(instance.slug).toEqual("test"); | ||
}); | ||
|
||
test("an readonly instance returns the view value from the snapshot if present", () => { | ||
const instance = ViewExample.createReadOnly({ key: "1", name: "Test", slug: "test" } as any); | ||
expect(instance.slug).toEqual("test"); | ||
}); | ||
|
||
test("an readonly instance doesn't recompute the view value from the snapshot", () => { | ||
const instance = ViewExample.createReadOnly({ key: "1", name: "Test", slug: "whatever" } as any); | ||
expect(instance.slug).toEqual("whatever"); | ||
}); | ||
|
||
test("an readonly instance doesn't call the computed function if given a snapshot value", () => { | ||
const fn = jest.fn(); | ||
@register | ||
class Spy extends ClassModel({ name: types.string }) { | ||
@cachedView() | ||
get slug() { | ||
fn(); | ||
return this.name.toLowerCase().replace(/ /g, "-"); | ||
} | ||
} | ||
|
||
const instance = Spy.createReadOnly({ name: "Test", slug: "whatever" } as any); | ||
expect(instance.slug).toEqual("whatever"); | ||
expect(fn).not.toHaveBeenCalled(); | ||
}); | ||
|
||
test("an observable instance doesn't call the computed function until snapshotted", () => { | ||
const fn = jest.fn(); | ||
@register | ||
class Spy extends ClassModel({ name: types.string }) { | ||
@cachedView() | ||
get slug() { | ||
fn(); | ||
return this.name.toLowerCase().replace(/ /g, "-"); | ||
} | ||
@action | ||
setName(name: string) { | ||
this.name = name; | ||
} | ||
} | ||
|
||
const instance = Spy.create({ name: "Test", slug: "whatever" } as any); | ||
expect(fn).not.toHaveBeenCalled(); | ||
getSnapshot(instance); | ||
expect(fn).not.toHaveBeenCalled(); | ||
|
||
instance.setName("New Name"); | ||
expect(fn).toHaveBeenCalled(); | ||
}); | ||
|
||
test("an readonly instance doesn't require the snapshot to include the cache", () => { | ||
const instance = ViewExample.createReadOnly({ key: "1", name: "Test" }); | ||
expect(instance.slug).toEqual("test"); | ||
}); | ||
|
||
test("cached views can be passed nested within snapshots", () => { | ||
@register | ||
class Outer extends ClassModel({ examples: types.array(ViewExample) }) {} | ||
|
||
const instance = Outer.createReadOnly({ | ||
examples: [{ key: "1", name: "Test", slug: "test-foobar" } as any, { key: "2", name: "Test 2", slug: "test-qux" } as any], | ||
}); | ||
|
||
expect(instance.examples[0].slug).toEqual("test-foobar"); | ||
expect(instance.examples[1].slug).toEqual("test-qux"); | ||
}); | ||
|
||
describe("with a hydrator", () => { | ||
@register | ||
class HydrateExample extends ClassModel({ timestamp: types.string }) { | ||
@cachedView<Date>({ | ||
getSnapshot(value, snapshot, node) { | ||
expect(snapshot).toBeDefined(); | ||
expect(node).toBeDefined(); | ||
return value.toISOString(); | ||
}, | ||
createReadOnly(value, snapshot, node) { | ||
expect(snapshot).toBeDefined(); | ||
expect(node).toBeDefined(); | ||
return value ? new Date(value) : undefined; | ||
}, | ||
}) | ||
get startOfMonth() { | ||
const date = new Date(this.timestamp); | ||
date.setDate(0); | ||
return date; | ||
} | ||
|
||
@action | ||
setTimestamp(timestamp: string) { | ||
this.timestamp = timestamp; | ||
} | ||
} | ||
|
||
test("cached views with processors can be accessed on observable instances", () => { | ||
const instance = HydrateExample.create({ timestamp: "2021-01-02T00:00:00.000Z" }); | ||
expect(instance.startOfMonth).toEqual(new Date("2021-01-01T00:00:00.000Z")); | ||
}); | ||
|
||
test("cached views with processors can be accessed on readonly instances when there's no input data", () => { | ||
const instance = HydrateExample.createReadOnly({ timestamp: "2021-01-02T00:00:00.000Z" }); | ||
expect(instance.startOfMonth).toEqual(new Date("2021-01-01T00:00:00.000Z")); | ||
}); | ||
|
||
test("cached views with processors can be accessed on readonly instances when there is input data", () => { | ||
const instance = HydrateExample.createReadOnly({ | ||
timestamp: "2021-01-02T00:00:00.000Z", | ||
startOfMonth: "2021-02-01T00:00:00.000Z", | ||
} as any); | ||
expect(instance.startOfMonth).toEqual(new Date("2021-02-01T00:00:00.000Z")); | ||
}); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.