Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Entity methods creator #4223

Open
wants to merge 29 commits into
base: create-slice-creators
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
f03c2d9
Revert "there are no entity methods creators in ba sing se"
EskiMojo14 Feb 18, 2024
248892a
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Feb 18, 2024
a6b8727
Revert "reset more entity files"
EskiMojo14 Feb 18, 2024
e012149
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Feb 18, 2024
063f248
Revert "again"
EskiMojo14 Feb 18, 2024
c5f65f5
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Feb 19, 2024
28a200b
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Feb 19, 2024
a687ebf
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Feb 21, 2024
7379cc2
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Apr 6, 2024
4a95c26
update type to match
EskiMojo14 Apr 6, 2024
2fb97e5
add entity methods creator back to docs
EskiMojo14 Apr 8, 2024
17cad3c
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Apr 8, 2024
8d61a90
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Apr 8, 2024
7ab0f2e
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 May 22, 2024
9febdfe
move async thunk creator module augmentation
EskiMojo14 May 22, 2024
9d9c568
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 May 22, 2024
b1dd066
create util to cut down on repetitive code
EskiMojo14 May 22, 2024
e888b32
prevent implicit return
EskiMojo14 May 22, 2024
45de908
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Sep 1, 2024
90a7737
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Sep 3, 2024
88fa75f
avoid declare module issue
EskiMojo14 Sep 3, 2024
11b873c
fix creator issue
EskiMojo14 Sep 3, 2024
870e830
Merge branch 'master' into entity-methods-creator
EskiMojo14 Oct 25, 2024
a114a44
try using a string rather than a symbol for creator type
EskiMojo14 Oct 25, 2024
07a9acf
rename generic to avoid conflict
EskiMojo14 Oct 25, 2024
0102cc3
fix Id usage
EskiMojo14 Oct 25, 2024
6d0db5e
add entity methods creator to ReducerType enum
EskiMojo14 Oct 25, 2024
4687c27
remove export
EskiMojo14 Nov 13, 2024
6194741
Merge branch 'create-slice-creators' into entity-methods-creator
EskiMojo14 Nov 29, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion docs/api/createEntityAdapter.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -239,7 +239,7 @@ In other words, they accept a state that looks like `{ids: [], entities: {}}`, a

These CRUD methods may be used in multiple ways:

- They may be passed as case reducers directly to `createReducer` and `createSlice`.
- They may be passed as case reducers directly to `createReducer` and `createSlice`. (also see the [`create.entityMethods`](./createSlice#createentitymethods-entitymethodscreator) slice creator which can assist with this)
- They may be used as "mutating" helper methods when called manually, such as a separate hand-written call to `addOne()` inside of an existing case reducer, if the `state` argument is actually an Immer `Draft` value.
- They may be used as immutable update methods when called manually, if the `state` argument is actually a plain JS object or array.

Expand Down
101 changes: 100 additions & 1 deletion docs/usage/custom-slice-creators.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -324,6 +324,105 @@ reducers: (create) => {

:::

##### `create.entityMethods` (`entityMethodsCreator`)

Creates a set of reducers for managing a normalized entity state, based on a provided [adapter](./createEntityAdapter).

```ts
import {
createEntityAdapter,
buildCreateSlice,
entityMethodsCreator,
} from '@reduxjs/toolkit'

const createAppSlice = buildCreateSlice({
creators: { entityMethods: entityMethodsCreator },
})

interface Post {
id: string
text: string
}

const postsAdapter = createEntityAdapter<Post>()

const postsSlice = createAppSlice({
name: 'posts',
initialState: postsAdapter.getInitialState(),
reducers: (create) => ({
...create.entityMethods(postsAdapter),
}),
})

export const { setOne, upsertMany, removeAll, ...etc } = postsSlice.actions
```

:::caution

Because this creator returns an object of multiple reducer definitions, it should be spread into the final object returned by the `reducers` callback.

:::

**Parameters**

- **adapter** The [adapter](../api/createEntityAdapter) to use.
- **config** The configuration object. (optional)

The configuration object can contain some of the following options:

**`selectEntityState`**

A selector to retrieve the entity state from the slice state. Defaults to `state => state`, but should be provided if the entity state is nested.

```ts no-transpile
const postsSlice = createAppSlice({
name: 'posts',
initialState: { posts: postsAdapter.getInitialState() },
reducers: (create) => ({
...create.entityMethods(postsAdapter, {
selectEntityState: (state) => state.posts,
}),
}),
})
```

**`name`, `pluralName`**

It's often desirable to modify the reducer names to be specific to the data type being used. These options allow you to do that simply.

```ts no-transpile
const postsSlice = createAppSlice({
name: 'posts',
initialState: postsAdapter.getInitialState(),
reducers: (create) => ({
...create.entityMethods(postsAdapter, {
name: 'post',
}),
}),
})

const { addOnePost, upsertManyPosts, removeAllPosts, ...etc } =
postsSlice.actions
```

`pluralName` defaults to `name + 's'`, but can be provided if this isn't desired.

```ts no-transpile
const gooseSlice = createAppSlice({
name: 'geese',
initialState: gooseAdapter.getInitialState(),
reducers: (create) => ({
...create.entityMethods(gooseAdapter, {
name: 'goose',
pluralName: 'geese',
}),
}),
})

const { addOneGoose, upsertManyGeese, removeAllGeese, ...etc } =
gooseSlice.actions
```

## Writing your own creators

In version v2.3.0, we introduced a system for including your own creators.
Expand All @@ -338,7 +437,7 @@ For example, the `create.preparedReducer` creator uses a definition that looks l

The callback form of `reducers` should return an object of reducer definitions, by calling creators and nesting the result of each under a key.

```js no-transpile
```js
reducers: (create) => ({
addTodo: create.preparedReducer(
(todo) => ({ payload: { id: nanoid(), ...todo } }),
Expand Down
62 changes: 31 additions & 31 deletions packages/toolkit/src/asyncThunkCreator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,37 +15,6 @@ import type {
import { ReducerType } from './createSlice'
import type { Id } from './tsHelpers'

export type AsyncThunkCreatorExposes<
State,
CaseReducers extends CreatorCaseReducers<State>,
> = {
actions: {
[ReducerName in keyof CaseReducers]: CaseReducers[ReducerName] extends AsyncThunkSliceReducerDefinition<
State,
infer ThunkArg,
infer Returned,
infer ThunkApiConfig
>
? AsyncThunk<Returned, ThunkArg, ThunkApiConfig>
: never
}
caseReducers: {
[ReducerName in keyof CaseReducers]: CaseReducers[ReducerName] extends AsyncThunkSliceReducerDefinition<
State,
any,
any,
any
>
? Id<
Pick<
Required<CaseReducers[ReducerName]>,
'fulfilled' | 'rejected' | 'pending' | 'settled'
>
>
: never
}
}

export type AsyncThunkSliceReducerConfig<
State,
ThunkArg extends any,
Expand Down Expand Up @@ -145,6 +114,37 @@ export interface AsyncThunkCreator<
>
}

export type AsyncThunkCreatorExposes<
State,
CaseReducers extends CreatorCaseReducers<State>,
> = {
actions: {
[ReducerName in keyof CaseReducers]: CaseReducers[ReducerName] extends AsyncThunkSliceReducerDefinition<
State,
infer ThunkArg,
infer Returned,
infer ThunkApiConfig
>
? AsyncThunk<Returned, ThunkArg, ThunkApiConfig>
: never
}
caseReducers: {
[ReducerName in keyof CaseReducers]: CaseReducers[ReducerName] extends AsyncThunkSliceReducerDefinition<
State,
any,
any,
any
>
? Id<
Pick<
Required<CaseReducers[ReducerName]>,
'fulfilled' | 'rejected' | 'pending' | 'settled'
>
>
: never
}
}

export const asyncThunkCreator: ReducerCreator<ReducerType.asyncThunk> = {
type: ReducerType.asyncThunk,
create: /* @__PURE__ */ (() => {
Expand Down
3 changes: 3 additions & 0 deletions packages/toolkit/src/createSlice.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import type {
ReducerWithInitialState,
} from './createReducer'
import { createReducer, makeGetInitialState } from './createReducer'
import type { EntityMethodsCreator } from './entities/slice_creator'
import type { ActionReducerMapBuilder, TypedActionCreator } from './mapBuilders'
import { executeReducerBuilderCallback } from './mapBuilders'
import type {
Expand All @@ -36,6 +37,7 @@ export enum ReducerType {
reducer = 'reducer',
reducerWithPrepare = 'reducerWithPrepare',
asyncThunk = 'asyncThunk',
entityMethods = 'entityMethods',
}

export type RegisteredReducerType = keyof SliceReducerCreators<
Expand Down Expand Up @@ -146,6 +148,7 @@ export interface SliceReducerCreators<
AsyncThunkCreator<State>,
AsyncThunkCreatorExposes<State, CaseReducers>
>
[ReducerType.entityMethods]: ReducerCreatorEntry<EntityMethodsCreator<State>>
}

export type ReducerCreators<
Expand Down
8 changes: 0 additions & 8 deletions packages/toolkit/src/entities/index.ts

This file was deleted.

78 changes: 64 additions & 14 deletions packages/toolkit/src/entities/models.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
import type { UncheckedIndexedAccess } from '../uncheckedindexed'
import type { Draft } from 'immer'
import type { PayloadAction } from '../createAction'
import type { CastAny, Id } from '../tsHelpers'
import type { UncheckedIndexedAccess } from '../uncheckedindexed.js'
import type { GetSelectorsOptions } from './state_selectors'
import type { CastAny, Id } from '../tsHelpers'
import type { CaseReducerDefinition } from '../createSlice'
import type { CaseReducer } from '../createReducer'

/**
* @public
Expand Down Expand Up @@ -158,12 +160,53 @@ export interface EntityStateAdapter<T, Id extends EntityId> {
/**
* @public
*/
export interface EntitySelectors<T, V, IdType extends EntityId> {
selectIds: (state: V) => IdType[]
selectEntities: (state: V) => Record<IdType, T>
selectAll: (state: V) => T[]
selectTotal: (state: V) => number
selectById: (state: V, id: IdType) => Id<UncheckedIndexedAccess<T>>
export type EntitySelectors<
T,
V,
IdType extends EntityId,
Single extends string = '',
Plural extends string = DefaultPlural<Single>,
> = Id<
{
[K in `select${Capitalize<Single>}Ids`]: (state: V) => IdType[]
} & {
[K in `select${Capitalize<Single>}Entities`]: (
state: V,
) => Record<IdType, T>
} & {
[K in `selectAll${Capitalize<Plural>}`]: (state: V) => T[]
} & {
[K in `selectTotal${Capitalize<Plural>}`]: (state: V) => number
} & {
[K in `select${Capitalize<Single>}ById`]: (
state: V,
id: IdType,
) => Id<UncheckedIndexedAccess<T>>
}
>

export type DefaultPlural<Single extends string> = Single extends ''
? ''
: `${Single}s`

export type EntityReducers<
T,
Id extends EntityId,
State = EntityState<T, Id>,
Single extends string = '',
Plural extends string = DefaultPlural<Single>,
> = {
[K in keyof EntityStateAdapter<
T,
Id
> as `${K}${Capitalize<K extends `${string}One` ? Single : Plural>}`]: EntityStateAdapter<
T,
Id
>[K] extends (state: any) => any
? CaseReducerDefinition<State, PayloadAction>
: EntityStateAdapter<T, Id>[K] extends CaseReducer<any, infer A>
? CaseReducerDefinition<State, A>
: never
}

/**
Expand All @@ -187,12 +230,19 @@ export interface EntityAdapter<T, Id extends EntityId>
extends EntityStateAdapter<T, Id>,
EntityStateFactory<T, Id>,
Required<EntityAdapterOptions<T, Id>> {
getSelectors(
getSelectors<
Single extends string = '',
Plural extends string = DefaultPlural<Single>,
>(
selectState?: undefined,
options?: GetSelectorsOptions,
): EntitySelectors<T, EntityState<T, Id>, Id>
getSelectors<V>(
options?: GetSelectorsOptions<Single, Plural>,
): EntitySelectors<T, EntityState<T, Id>, Id, Single, Plural>
getSelectors<
V,
Single extends string = '',
Plural extends string = DefaultPlural<Single>,
>(
selectState: (state: V) => EntityState<T, Id>,
options?: GetSelectorsOptions,
): EntitySelectors<T, V, Id>
options?: GetSelectorsOptions<Single, Plural>,
): EntitySelectors<T, V, Id, Single, Plural>
}
Loading