Skip to content

ViewModelsConfig configuration object

This configuration contains all options for the behavior of ViewModel instances.

The package provides a global object with this configuration, but you can also change it for each ViewModel and ViewModelStore separately using the vmConfig field.

ts
import { viewModelsConfig, ViewModelStoreBase } from "mobx-view-model";
import { withViewModel } from "mobx-view-model-react";

viewModelsConfig.comparePayload = false;

withViewModel(VM, {
  vmConfig: {
    comparePayload: false
  }
})()

new ViewModelStoreBase({
  vmConfig: {
    comparePayload: false
  }
})

Reference to source code

Recommendations

These are the recommended settings for the global configuration viewModelsConfig, which contain the most optimal values.

ts
import { viewModelsConfig, ViewModelStoreBase } from 'mobx-view-model';

viewModelsConfig.comparePayload = false;
viewModelsConfig.payloadComputed = 'struct';
viewModelsConfig.payloadObservable = 'ref';

viewModelsConfig.observable.viewModels.useDecorators = true; //false
viewModelsConfig.observable.viewModelStores.useDecorators = true; // false

startViewTransitions

Controls view transitions for view model lifecycle moments.
MDN Reference

Shape

startViewTransitions is an object with these flags:

  • mount - start transition when the view mounts
  • payloadChange - start transition when the payload changes
  • unmount - start transition when the view unmounts

In ViewModelsRawConfig you can also pass a boolean to toggle all flags at once.

WARNING

This feature is experimental and not all browsers support it yet.

comparePayload

Allows you to configure how payload should be compared

payloadComputed

Allows you to configure computed statement of the payload

payloadObservable

Indicates type of observable for ViewModel payload.

generateId()

Generates a unique identifier for a ViewModel.

This property has default implementation here

Example

Using crypto.randomUUID() to generate view model ids

ts
import { viewModelsConfig } from "mobx-view-model";

viewModelsConfig.generateId = () => crypto.randomUUID();

factory()

Factory function for creating ViewModel instances.

Can be helpful if you want to add some constructor arguments for your own ViewModel implementation

This property has default implementation here

Example

Passing RootStore as first constructor parameter

ts
import { viewModelsConfig } from "mobx-view-model";
import { rootStore } from "@/shared/store";

viewModelsConfig.factory = (config) => {
  const { VM } = config;
  return new VM(rootStore, config);
}

flushPendingReactions

How many times the library may run MobX’s pending-reaction drain (before attach/mount) after a view model instance is created. Default: 100. Use 0 to skip flushing (no-op).

fallbackComponent

A component that will be rendered while the view model is in a loading or processing state.
This is useful for showing loading spinners, skeletons, or placeholder content.

Example

tsx
viewModelsConfig.fallbackComponent = () => (
  <div className="loading-spinner">
    Loading...
  </div>
);

reactHook

Optional default for the reactHook option on withViewModel: used when the HOC config does not set reactHook (resolved as config.reactHook ?? viewModelsConfig.reactHook).

useReactIds

When true, useCreateViewModel forwards React’s useId() value into the view-model id pipeline as renderId. That id participates in generateId / generateVmId so instance ids can align with React’s stable per-component ids.

Default: false.

SSR and hydration

Matching server and client markup depends on consistent ids. Enabling useReactIds ties VM ids to React’s useId() output for that component instance, which is designed to match between SSR and the first client render when the component tree is the same—useful if VM ids must stay stable across hydration or line up with DOM id attributes derived from useId().

Example

ts
import { viewModelsConfig } from 'mobx-view-model';

viewModelsConfig.useReactIds = true;

Per-call override via useCreateViewModel / withViewModel vmConfig:

tsx
useCreateViewModel(YourVM, payload, {
  vmConfig: { useReactIds: true },
});

suspendUntil

Lets the tree wait until your condition is ready before it keeps rendering—SSR is a typical use case. Configure it through vmConfig the same way as other options on this page.

You can return nothing from the function (undefined / null) for a given instance or render—then no wait happens and the view continues as usual. Use that when suspension only applies in some cases (for example only on the server, or only until a flag exists).

TIP

A promise that resolves when you’re ready is enough (MobX when() is a common choice). Add Suspense if you want a loading or placeholder UI.

Example (global)

ts
import { when } from 'mobx';
import { viewModelsConfig } from 'mobx-view-model';

viewModelsConfig.suspendUntil = (vm) =>
  when(() => Boolean(vm.someReadyFlag));

Example (store)

ts
import { when } from 'mobx';
import { ViewModelStoreBase } from 'mobx-view-model';

new ViewModelStoreBase({
  vmConfig: {
    suspendUntil: (vm) => when(() => Boolean(vm.ctx)),
  },
});

onMount

A lifecycle hook that is called when a view model is mounted.
Useful for tracking component mounting, initializing external services, or setting up subscriptions.

Example

tsx
viewModelsConfig.onMount = (viewModel) => {
  console.log(`ViewModel ${viewModel.id} mounted`);
  // Setup analytics tracking
  analytics.track('component_mounted', { id: viewModel.id });
};

onUnmount

A lifecycle hook that is called when a view model is unmounted.
Useful for cleanup operations, removing subscriptions, or tracking component lifecycle.

Example

tsx
viewModelsConfig.onUnmount = (viewModel) => {
  console.log(`ViewModel ${viewModel.id} unmounted`);
  // Cleanup subscriptions
};

hooks

Internal event hooks for view model stores.

hooks.storeCreate

Called when a ViewModelStore instance is created.
Useful for wiring external listeners or diagnostics.

processViewComponent

A higher-order function that processes and transforms the view component before it is rendered.
This function enables component composition and modification at the ViewModel level, allowing for:

  • Wrapping components with additional functionality (error boundaries, providers, etc.)
  • Injecting props or context
  • Modifying component behavior
  • Adding global features like logging, analytics, or performance monitoring

Example

tsx
viewModelsConfig.processViewComponent = (Component) => {
  return (props) => {
    return (
      <ErrorBoundary>
        <Component {...props} />
      </ErrorBoundary>
    )
  }
}

It works only for withViewModel HOCs

wrapViewsInObserver

Wrap View components in observer() MobX HOC
This property is enabled by default.

You can turn off this behavior by setting wrapViewsInObserver to false.
Example:

tsx
import { viewModelsConfig } from "mobx-view-model";

viewModelsConfig.wrapViewsInObserver = false;

It works only for withViewModel HOCs

observable

This is a large configuration object for all base implementations in mobx-view-model, like ViewModelBase or ViewModelStoreBase.
You can modify the default behavior of wrapping in makeObservable() MobX functions.

Properties of the nested observable configs:

- disableWrapping

This removes makeObservable(this, annotations)/makeObservable(this) calls

- useDecorators

This changes the style of marking MobX annotations from "decorators style" to "non-decorators style".
Very helpful if you want to write code with "non decorators style".

This property has default value - true

Example:

ts
import { observable, action } from "mobx";
import {
  viewModelsConfig,
  ViewModelBase,
  ViewModelParams
} from "mobx-view-model";

viewModelsConfig.observable.viewModels.useDecorators = false;

class YourViewModel extends ViewModelBase  {
  constructor(params: ViewModelParams) {
    super(params);

    makeObservable(this, {
      fruitName: observable,
      setFruitName: action.bound,
    })
  }

  fruitName: string = '';

  setFruitName(fruitName: string) {
    this.fruitName = fruitName;
  }
}

Another example with "decorators style":

ts
import { observable, action } from "mobx";
import {
  viewModelsConfig,
  ViewModelBase,
  ViewModelParams
} from "mobx-view-model";

viewModelsConfig.observable.viewModels.useDecorators = true;

class YourViewModel extends ViewModelBase  {
  @observable
  fruitName: string = '';

  @action.bound
  setFruitName(fruitName: string) {
    this.fruitName = fruitName;
  }
}

- custom(context, annotationsArray)

Custom function for wrapping your entity

global configuration object

You can override default global config using import viewModelsConfig.

ts
import { viewModelsConfig } from "mobx-view-model";

You should do this before the app starts.

Usage

ts
import { viewModelsConfig } from "mobx-view-model";

// Configure payload update/reactivity behavior
viewModelsConfig.payloadObservable = 'ref';
viewModelsConfig.comparePayload = false;
viewModelsConfig.payloadComputed = 'struct';


// Disable view transitions
viewModelsConfig.startViewTransitions = {
  mount: false,
  payloadChange: false,
  unmount: false,
};

// Optional configurations (uncomment to use)
// viewModelsConfig.generateId = () => crypto.randomUUID();
// viewModelsConfig.factory = (config) => new config.VM(rootStore, config);
// viewModelsConfig.flushPendingReactions = 200;
// viewModelsConfig.fallbackComponent = () => <LoadingSpinner />;
// viewModelsConfig.onMount = (vm) => console.log('Mounted:', vm.id);
// viewModelsConfig.onUnmount = (vm) => console.log('Unmounted:', vm.id);

Possible causes of infinite re-renders due to payload access

The flexible configuration of the payload reactivity and update behavior can lead to infinite re-renders inside the View component.
This happens when the payload is changing every time the component is re-rendered.

The following ViewModel configurations can cause this problem:

json
{
  "wrapViewsInObserver": true,
  "payloadComputed": true,
  "comparePayload": "shallow",
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": true,
  "payloadComputed": true,
  "comparePayload": false,
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": true,
  "payloadComputed": true,
  "comparePayload": false,
  "payloadObservable": "ref"
}
json
{
  "wrapViewsInObserver": true,
  "payloadComputed": false,
  "comparePayload": "shallow",
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": true,
  "payloadComputed": false,
  "comparePayload": false,
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": true,
  "payloadComputed": false,
  "comparePayload": false,
  "payloadObservable": "ref"
}
json
{
  "wrapViewsInObserver": false,
  "payloadComputed": true,
  "comparePayload": "shallow",
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": false,
  "payloadComputed": true,
  "comparePayload": false,
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": false,
  "payloadComputed": true,
  "comparePayload": false,
  "payloadObservable": "ref"
}
json
{
  "wrapViewsInObserver": false,
  "payloadComputed": false,
  "comparePayload": "shallow",
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": false,
  "payloadComputed": false,
  "comparePayload": false,
  "payloadObservable": "deep"
}
json
{
  "wrapViewsInObserver": false,
  "payloadComputed": false,
  "comparePayload": false,
  "payloadObservable": "ref"
}

Released under the MIT License.