Skip to content

Commit 810d634

Browse files
authored
Merge pull request #641 from aryaemami59/netlify-docs
2 parents 1932383 + 0bc5987 commit 810d634

File tree

79 files changed

+17879
-77
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

79 files changed

+17879
-77
lines changed

.eslintrc

+12
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,18 @@
6767
"@typescript-eslint/no-unused-vars": "off",
6868
"@typescript-eslint/no-shadow": "off"
6969
}
70+
},
71+
{
72+
"parser": "@typescript-eslint/parser",
73+
"files": ["./docs/examples/**/*.{js,ts,jsx,tsx}"],
74+
"parserOptions": {
75+
"ecmaVersion": 2023,
76+
"sourceType": "module",
77+
"ecmaFeatures": { "jsx": true }
78+
},
79+
"rules": {
80+
"no-unused-vars": [0]
81+
}
7082
}
7183
]
7284
}

.gitignore

+9
Original file line numberDiff line numberDiff line change
@@ -25,3 +25,12 @@ typesversions
2525
!.yarn/versions
2626
.pnp.*
2727
*.tgz
28+
29+
website/translated_docs
30+
website/build/
31+
website/node_modules
32+
website/i18n/*
33+
website/.yarn/
34+
35+
docs/examples/**/*.js
36+
docs/examples/**/*.jsx

README.md

+40-15
Original file line numberDiff line numberDiff line change
@@ -321,7 +321,7 @@ generates a memoized selector function.
321321

322322
A memoized [output selector].
323323

324-
<details><summary><b>Type parameters</b></summary>
324+
<details><summary><b>Type Parameters</b></summary>
325325

326326
| Name | Description |
327327
| :---------------------------- | :----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
@@ -366,7 +366,7 @@ Accepts either a `memoize` function and `...memoizeOptions` rest parameter, or s
366366

367367
A customized [`createSelector`] function.
368368

369-
<details><summary><b>Type parameters</b></summary>
369+
<details><summary><b>Type Parameters</b></summary>
370370

371371
| Name | Description |
372372
| :-------------------- | :--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
@@ -489,7 +489,7 @@ A convenience function that simplifies returning an object made up of selector r
489489

490490
A memoized structured selector.
491491

492-
<details><summary><b>Type parameters</b></summary>
492+
<details><summary><b>Type Parameters</b></summary>
493493

494494
| Name | Description |
495495
| :--------------------- | :------------------------------------------------------------------------------------------------------------------------------------------------------------ |
@@ -587,6 +587,35 @@ const structuredSelector = createSelector(selectA, selectB, (a, b) => ({
587587
const result = structuredSelector({ a: 1, b: 2 }) // will produce { x: 1, y: 2 }
588588
```
589589

590+
`createStructuredSelector` takes an object whose properties are input selectors and returns a structured selector. The structured selector returns an object with the same keys as the `inputSelectorsObject` argument, but with the selectors replaced with their values.
591+
592+
```ts
593+
const selectA = state => state.a
594+
const selectB = state => state.b
595+
596+
const structuredSelector = createStructuredSelector({
597+
x: selectA,
598+
y: selectB
599+
})
600+
601+
const result = structuredSelector({ a: 1, b: 2 }) // will produce { x: 1, y: 2 }
602+
```
603+
604+
Structured selectors can be nested:
605+
606+
```ts
607+
const nestedSelector = createStructuredSelector({
608+
subA: createStructuredSelector({
609+
selectorA,
610+
selectorB
611+
}),
612+
subB: createStructuredSelector({
613+
selectorC,
614+
selectorD
615+
})
616+
})
617+
```
618+
590619
</details>
591620

592621
<div align="right">[ <a href="#table-of-contents">↑ Back to top ↑</a> ]</div>
@@ -647,7 +676,7 @@ interface LruMemoizeOptions {
647676

648677
A memoized function with a `.clearCache()` method attached.
649678

650-
<details><summary><b>Type parameters</b></summary>
679+
<details><summary><b>Type Parameters</b></summary>
651680

652681
| Name | Description |
653682
| :----- | :----------------------------------------- |
@@ -896,7 +925,7 @@ This solves the problem of having to know and set the cache size prior to creati
896925

897926
A memoized function with a `.clearCache()` method attached.
898927

899-
<details><summary><b>Type parameters</b></summary>
928+
<details><summary><b>Type Parameters</b></summary>
900929

901930
| Name | Description |
902931
| :----- | :----------------------------------------- |
@@ -1006,7 +1035,7 @@ Uses an "auto-tracking" approach inspired by the work of the Ember Glimmer team.
10061035

10071036
A memoized function with a `.clearCache()` method attached.
10081037

1009-
<details><summary><b>Type parameters</b></summary>
1038+
<details><summary><b>Type Parameters</b></summary>
10101039

10111040
| Name | Description |
10121041
| :----- | :----------------------------------------- |
@@ -1261,8 +1290,6 @@ These updates aim to enhance flexibility, performance, and developer experience.
12611290

12621291
- Removed `ParametricSelector` and `OutputParametricSelector` types. Their functionalities are now integrated into `Selector` and `OutputSelector` respectively, which inherently support additional parameters.
12631292

1264-
</details>
1265-
12661293
<div align="right">[ <a href="#table-of-contents">↑ Back to top ↑</a> ]</div>
12671294

12681295
---
@@ -1573,9 +1600,9 @@ test('selector unit test', () => {
15731600
selectTodoIds(state)
15741601
selectTodoIds(state)
15751602
selectTodoIds(state)
1576-
// The `Result Function` should not recalculate.
1603+
// The result function should not recalculate.
15771604
expect(selectTodoIds.recomputations()).toBe(1)
1578-
// `input selectors` should not recalculate.
1605+
// input selectors should not recalculate.
15791606
expect(selectTodoIds.dependencyRecomputations()).toBe(1)
15801607
})
15811608

@@ -1594,9 +1621,9 @@ test('selector unit test', () => {
15941621
selectTodoIds(state)
15951622
selectTodoIds(state)
15961623
selectTodoIds(state)
1597-
// The `result function` should not recalculate.
1624+
// The result function should not recalculate.
15981625
expect(selectTodoIds.recomputations()).to.equal(1)
1599-
// `input selectors` should not recalculate.
1626+
// input selectors should not recalculate.
16001627
expect(selectTodoIds.dependencyRecomputations()).to.equal(1)
16011628
})
16021629
```
@@ -1636,8 +1663,6 @@ If you prefer to use a curried form instead, you can create a curried selector w
16361663

16371664
<details><summary><b>Detailed Explanation: Creating Curried Selectors</b></summary>
16381665

1639-
You can try this pattern:
1640-
16411666
```ts
16421667
const currySelector = <
16431668
State,
@@ -1854,7 +1879,7 @@ Useful to reduce selectors recalculation when the same selector is repeatedly ca
18541879

18551880
### [reselect-debugger](https://github.com/vlanemcev/reselect-debugger-flipper)
18561881

1857-
[Flipper plugin](https://github.com/vlanemcev/flipper-plugin-reselect-debugger) and [and the connect app](https://github.com/vlanemcev/reselect-debugger-flipper) for debugging selectors in **React Native Apps**.
1882+
[Flipper plugin](https://github.com/vlanemcev/flipper-plugin-reselect-debugger) and [the connect app](https://github.com/vlanemcev/reselect-debugger-flipper) for debugging selectors in **React Native Apps**.
18581883

18591884
Inspired by Reselect Tools, so it also has all functionality from this library and more, but only for React Native and Flipper.
18601885

docs/examples/FAQ/MyComponent.tsx

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
import type { FC } from 'react'
2+
import { useSelectTodo } from './createParametricSelectorHook'
3+
4+
interface Props {
5+
id: number
6+
}
7+
8+
const MyComponent: FC<Props> = ({ id }) => {
9+
const todo = useSelectTodo(id)
10+
return <div>{todo?.title}</div>
11+
}
+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
import type { defaultMemoize, SelectorArray, UnknownMemoizer } from 'reselect'
2+
import { createSelector } from 'reselect'
3+
import { currySelector } from './currySelector'
4+
5+
export const createCurriedSelector = <
6+
InputSelectors extends SelectorArray,
7+
Result,
8+
OverrideMemoizeFunction extends UnknownMemoizer = typeof defaultMemoize,
9+
OverrideArgsMemoizeFunction extends UnknownMemoizer = typeof defaultMemoize
10+
>(
11+
...args: Parameters<
12+
typeof createSelector<
13+
InputSelectors,
14+
Result,
15+
OverrideMemoizeFunction,
16+
OverrideArgsMemoizeFunction
17+
>
18+
>
19+
) => {
20+
return currySelector(createSelector(...args))
21+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
import { useSelector } from 'react-redux'
2+
import { createSelector } from 'reselect'
3+
4+
interface RootState {
5+
todos: {
6+
id: number
7+
completed: boolean
8+
title: string
9+
description: string
10+
}[]
11+
alerts: { id: number; read: boolean }[]
12+
}
13+
14+
const state: RootState = {
15+
todos: [
16+
{
17+
id: 0,
18+
completed: false,
19+
title: 'Figure out if plants are really plotting world domination.',
20+
description: 'They may be.'
21+
},
22+
{
23+
id: 1,
24+
completed: true,
25+
title: 'Practice telekinesis for 15 minutes',
26+
description: 'Just do it'
27+
}
28+
],
29+
alerts: [
30+
{ id: 0, read: false },
31+
{ id: 1, read: true }
32+
]
33+
}
34+
35+
const selectTodoById = createSelector(
36+
[(state: RootState) => state.todos, (state: RootState, id: number) => id],
37+
(todos, id) => todos.find(todo => todo.id === id)
38+
)
39+
40+
export const createParametricSelectorHook = <
41+
Result,
42+
Params extends readonly unknown[]
43+
>(
44+
selector: (state: RootState, ...params: Params) => Result
45+
) => {
46+
return (...args: Params) => {
47+
return useSelector((state: RootState) => selector(state, ...args))
48+
}
49+
}
50+
51+
export const useSelectTodo = createParametricSelectorHook(selectTodoById)

docs/examples/FAQ/currySelector.ts

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import { createSelector } from 'reselect'
2+
import type { RootState } from './selectorRecomputing'
3+
4+
export const currySelector = <
5+
State,
6+
Result,
7+
Params extends readonly any[],
8+
AdditionalFields
9+
>(
10+
selector: ((state: State, ...args: Params) => Result) & AdditionalFields
11+
) => {
12+
const curriedSelector = (...args: Params) => {
13+
return (state: State) => {
14+
return selector(state, ...args)
15+
}
16+
}
17+
return Object.assign(curriedSelector, selector)
18+
}
19+
20+
const selectTodoByIdCurried = currySelector(
21+
createSelector(
22+
[(state: RootState) => state.todos, (state: RootState, id: number) => id],
23+
(todos, id) => todos.find(todo => todo.id === id)
24+
)
25+
)

docs/examples/FAQ/howToTest.test.ts

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
import { createSelector } from 'reselect'
2+
import { expect, test } from 'vitest'
3+
4+
interface RootState {
5+
todos: { id: number; completed: boolean }[]
6+
alerts: { id: number; read: boolean }[]
7+
}
8+
9+
const state: RootState = {
10+
todos: [
11+
{ id: 0, completed: false },
12+
{ id: 1, completed: true }
13+
],
14+
alerts: [
15+
{ id: 0, read: false },
16+
{ id: 1, read: true }
17+
]
18+
}
19+
20+
// With `Vitest` or `Jest`
21+
test('selector unit test', () => {
22+
const selectTodoIds = createSelector(
23+
[(state: RootState) => state.todos],
24+
todos => todos.map(({ id }) => id)
25+
)
26+
const firstResult = selectTodoIds(state)
27+
const secondResult = selectTodoIds(state)
28+
// Reference equality should pass.
29+
expect(firstResult).toBe(secondResult)
30+
// Deep equality should also pass.
31+
expect(firstResult).toStrictEqual(secondResult)
32+
selectTodoIds(state)
33+
selectTodoIds(state)
34+
selectTodoIds(state)
35+
// The result function should not recalculate.
36+
expect(selectTodoIds.recomputations()).toBe(1)
37+
// input selectors should not recalculate.
38+
expect(selectTodoIds.dependencyRecomputations()).toBe(1)
39+
})
40+
41+
// With `Chai`
42+
test('selector unit test', () => {
43+
const selectTodoIds = createSelector(
44+
[(state: RootState) => state.todos],
45+
todos => todos.map(({ id }) => id)
46+
)
47+
const firstResult = selectTodoIds(state)
48+
const secondResult = selectTodoIds(state)
49+
// Reference equality should pass.
50+
expect(firstResult).to.equal(secondResult)
51+
// Deep equality should also pass.
52+
expect(firstResult).to.deep.equal(secondResult)
53+
selectTodoIds(state)
54+
selectTodoIds(state)
55+
selectTodoIds(state)
56+
// The result function should not recalculate.
57+
expect(selectTodoIds.recomputations()).to.equal(1)
58+
// input selectors should not recalculate.
59+
expect(selectTodoIds.dependencyRecomputations()).to.equal(1)
60+
})

docs/examples/FAQ/identity.ts

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
import { createSelectorCreator } from 'reselect'
2+
3+
const identity = <Func extends (...args: any[]) => any>(func: Func) => func
4+
5+
const createNonMemoizedSelector = createSelectorCreator({
6+
memoize: identity,
7+
argsMemoize: identity
8+
})
+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import { createSelector } from 'reselect'
2+
3+
export interface RootState {
4+
todos: { id: number; completed: boolean }[]
5+
alerts: { id: number; read: boolean; type: string }[]
6+
}
7+
8+
const selectAlertsByType = createSelector(
9+
[
10+
(state: RootState) => state.alerts,
11+
(state: RootState, type: string) => type
12+
],
13+
(alerts, type) => alerts.filter(todo => todo.type === type),
14+
{
15+
argsMemoizeOptions: {
16+
// This will check the arguments passed to the output selector.
17+
equalityCheck: (a, b) => {
18+
if (a !== b) {
19+
console.log('Changed argument:', a, 'to', b)
20+
}
21+
return a === b
22+
}
23+
}
24+
}
25+
)

0 commit comments

Comments
 (0)