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

feat(ngrx-toolkit): Creates provideDevtoolsConfig #151

Merged
Changes from all commits
Commits
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
33 changes: 33 additions & 0 deletions docs/docs/with-redux.md
Original file line number Diff line number Diff line change
@@ -107,3 +107,36 @@ signalStore(
})
);
```

### Configuring the Redux Devtools Extension

The `provideDevtoolsConfig` function allows you to configure the Redux DevTools integration for your NgRx SignalStore. This function is essential for setting up the DevTools with custom options. The function only needs to be called once in your appConfig or AppModule.

To use `provideDevtoolsConfig`, you need to import it and call it in your providers array.

Here is an example of how to use it with the standalone api:

```typescript
// app.config.ts
import { ApplicationConfig } from '@angular/core';
import { provideDevtoolsConfig } from '@angular-architects/ngrx-toolkit';

export const appConfig: ApplicationConfig = {
providers: [
provideDevtoolsConfig({
name: 'MyApp',
}),
],
};

// main.ts
import { bootstrapApplication } from '@angular/platform-browser';
import { AppComponent } from './app/app.component';
import { appConfig } from './app/app.config.ts';

await bootstrapApplication(AppComponent, appConfig);
```

### Additional Information

For more details on the available options and their usage, refer to the [Redux DevTools Extension documentation](https://github.com/reduxjs/redux-devtools).
4 changes: 4 additions & 0 deletions libs/ngrx-toolkit/src/index.ts
Original file line number Diff line number Diff line change
@@ -5,6 +5,10 @@ export { withMapper } from './lib/devtools/features/with-mapper';
export { withGlitchTracking } from './lib/devtools/features/with-glitch-tracking';
export { patchState, updateState } from './lib/devtools/update-state';
export { renameDevtoolsName } from './lib/devtools/rename-devtools-name';
export {
provideDevtoolsConfig,
ReduxDevtoolsConfig,
} from './lib/devtools/provide-devtools-config';

export {
withRedux,
Original file line number Diff line number Diff line change
@@ -5,6 +5,7 @@ import { StateSource } from '@ngrx/signals';
import { DevtoolsInnerOptions } from './devtools-feature';
import { throwIfNull } from '../../shared/throw-if-null';
import { Connection, StoreRegistry, Tracker } from './models';
import { REDUX_DEVTOOLS_CONFIG } from '../provide-devtools-config';

const dummyConnection: Connection = {
send: () => void true,
@@ -30,6 +31,10 @@ export class DevtoolsSyncer implements OnDestroy {
#stores: StoreRegistry = {};
readonly #isBrowser = isPlatformBrowser(inject(PLATFORM_ID));
readonly #trackers = [] as Tracker[];
readonly #devtoolsConfig = {
name: 'NgRx SignalStore',
...inject(REDUX_DEVTOOLS_CONFIG, { optional: true }),
};

/**
* Maintains the current states of all stores to avoid conflicts
@@ -50,9 +55,7 @@ export class DevtoolsSyncer implements OnDestroy {

readonly #connection: Connection = this.#isBrowser
? window.__REDUX_DEVTOOLS_EXTENSION__
? window.__REDUX_DEVTOOLS_EXTENSION__.connect({
name: 'NgRx SignalStore',
})
? window.__REDUX_DEVTOOLS_EXTENSION__.connect(this.#devtoolsConfig)
: dummyConnection
: dummyConnection;

3 changes: 2 additions & 1 deletion libs/ngrx-toolkit/src/lib/devtools/internal/models.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,13 @@
import { StateSource } from '@ngrx/signals';
import { DevtoolsInnerOptions } from './devtools-feature';
import { ReduxDevtoolsConfig } from '../provide-devtools-config';

export type Action = { type: string };
export type Connection = {
send: (action: Action, state: Record<string, unknown>) => void;
};
export type ReduxDevtoolsExtension = {
connect: (options: { name: string }) => Connection;
connect: (options: ReduxDevtoolsConfig) => Connection;
};

export type StoreRegistry = Record<
32 changes: 32 additions & 0 deletions libs/ngrx-toolkit/src/lib/devtools/provide-devtools-config.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
import { InjectionToken, ValueProvider } from '@angular/core';

/**
* Provides the configuration options for connecting to the Redux DevTools Extension.
*/
export function provideDevtoolsConfig(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should only provide those options which we are really supporting. So at the moment that would only be the name.

Copy link
Contributor Author

@mikerentmeister mikerentmeister Feb 20, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we have to manually handle maxAge? I could see that as a useful one as well.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

At the moment, we can't really support anything of the options of the original. It would be great if we get them, but not in this PR.

config: ReduxDevtoolsConfig
): ValueProvider {
return {
provide: REDUX_DEVTOOLS_CONFIG,
useValue: config,
};
}

/**
* Injection token for the configuration options for connecting to the Redux DevTools Extension.
*/
export const REDUX_DEVTOOLS_CONFIG = new InjectionToken<ReduxDevtoolsConfig>(
'ReduxDevtoolsConfig'
);

/**
* Options for connecting to the Redux DevTools Extension.
* @example
* const devToolsOptions: ReduxDevtoolsConfig = {
* name: 'My App',
* };
*/
export type ReduxDevtoolsConfig = {
/** Optional name for the devtools instance. If empty, "NgRx SignalStore" will be used. */
name?: string;
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
import { TestBed } from '@angular/core/testing';
import { provideDevtoolsConfig } from '../provide-devtools-config';
import { DevtoolsSyncer } from '../internal/devtools-syncer.service';
import { setupExtensions } from './helpers.spec';

describe('provideDevtoolsConfig', () => {
it('DevtoolsSyncer should use the default configuration if none is provided', () => {
const { connectSpy } = setupExtensions();
TestBed.inject(DevtoolsSyncer);
expect(connectSpy).toHaveBeenCalledWith({
name: 'NgRx SignalStore',
});
});

it('DevtoolsSyncer should use the configuration provided', () => {
const { connectSpy } = setupExtensions();
TestBed.configureTestingModule({
providers: [provideDevtoolsConfig({ name: 'test' })],
});
TestBed.inject(DevtoolsSyncer);
expect(connectSpy).toHaveBeenCalledWith({
name: 'test',
});
});
});