npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

dk-react-mobx-router

v3.4.5

Published

Router for React+MobX

Downloads

47

Readme

Config-based routing for React + MobX applications

coverage npm license size

[!WARNING]
It's fine if you use this library from NPM package with a static versioning in case you want it for some pet-project or to test it's capabilities.

But for production use it's strongly recommended to create a fork, because I do not write Changelogs and may break / add some functionality without notice.

Contents

Features

  • Has a lifecycle with beforeEnter and beforeLeave
  • Supports dynamically loaded components (from async chunks like () => import('/some-page'))
  • Supports dynamically loaded modular stores and actions for pages
  • Supports SSR
  • Ensures that every dynamic parameter from the URL has a validator
  • TypeScript works for every route and its dynamic parameters and search-query
  • It is a separate layer, so there is no more markup like <Route path="..." /> inside React components

Setup (SPA)

The setup consists of three parts:

  • Routes config: Describes your routes in a plain object
  • Router store: A MobX store or an object that includes current route data and transition history
  • Redirect function: May be a part of the router store or a separate function
  1. Install dk-react-mobx-router, dk-mobx-stateful-fn and mobx-react-lite / mobx-react
  2. Create a basic routes config routes.ts
import { createRouterConfig } from 'dk-react-mobx-router';

export const routes = createRouterConfig({
  home: {
    path: '/',
    loader: (() => import('./pages/home')),
  },
  static: {
    path: '/static',
    loader: (() => import('./pages/static')),
  },
  dynamic: {
    path: '/page/:foo/id/:bar',
    params: {
      foo: (value) => value.length > 2,
      bar: (value) => value.length > 0,
    },
    loader: (() => import('./pages/dynamic')),
  },
  // this page is necessary
  error404: {
    path: '/error404',
    props: { errorCode: 404 },
    loader: (() => import('./pages/error')),
  },
  // this page is necessary
  error500: {
    path: '/error500',
    props: { errorCode: 500 },
    loader: (() => import('./pages/error')),
  },
});
  1. Configure a router store
import { makeAutoObservable } from 'mobx';
import { InterfaceRouterStore } from 'dk-react-mobx-router';

import { routes } from 'routes';

type TInterfaceRouterStore = InterfaceRouterStore<typeof routes>;

export class RouterStore implements TInterfaceRouterStore {
  constructor() {
    makeAutoObservable(this);
  }

  routesHistory: TInterfaceRouterStore['routesHistory'] = [];
  // we will load/hydrate initial route before the app is rendered, so "as any" is safe
  currentRoute: TInterfaceRouterStore['currentRoute'] = {} as any;
}

export const routerStore = new RouterStore();
  1. Configure an action for redirecting. It may be an independent function
import { redirectToGenerator } from 'dk-react-mobx-router';
import { addState } from 'dk-mobx-stateful-fn';

import { routes } from 'routes';
import { routerStore } from 'routerStore';

export const redirectTo = addState(redirectToGenerator({
  routes,
  routerStore,
  routeError500: routes.error500,
}), 'redirectTo');

Or may be a part of RouterStore.

import { addState } from 'dk-mobx-stateful-fn';

export class RouterStore implements TInterfaceRouterStore {
  constructor() {
    makeAutoObservable(this, { redirectTo: false });
  }

  routesHistory: TInterfaceRouterStore['routesHistory'] = [];
  currentRoute: TInterfaceRouterStore['currentRoute'] = {} as any;
  
  redirectTo = addState(redirectToGenerator({
    routes,
    routerStore: this,
    routeError500: routes.error500,
  }), 'redirectTo');
}
  1. Now it is time to create a Router React component that will react to changes in routerStore.currentRoute and render the relevant page component.
import { Router as RouterMobx } from 'dk-react-mobx-router';
import { observer } from 'mobx-react-lite';

import { routes } from 'routes';
import { routerStore } from 'routerStore';

// observer decorator is not necessary, it just gives "memo"
export const Router = observer(() => {
  return (
    <RouterMobx
      routes={routes}
      redirectTo={routerStore.redirectTo}
      routerStore={routerStore}
    />
  );
});
  1. The last step to make it work is to find the initial route and render the React application (an example for SPA without SSR).
import { getInitialRoute } from 'dk-react-mobx-router';
import { observer } from 'mobx-react-lite';
import { createRoot } from 'react-dom/client';

import { Router } from 'components/Router';
import { routes } from 'routes';
import { routerStore } from 'routerStore';

const App = observer(() => {
  return (
    <>
      <div className="menu">
        <div 
          onClick={() => routerStore.redirectTo({ route: 'home' })}
          className={routerStore.currentRoute.name === 'home' ? 'active' : ''}
        >
          Home
        </div>
        <div 
          onClick={() => routerStore.redirectTo({ route: 'static' })}
          className={routerStore.currentRoute.name === 'static' ? 'active' : ''}
        >
          Static
        </div>
        <div 
          onClick={() => routerStore.redirectTo({ route: 'dynamic', params: { foo: 'test', bar: 'smth' } })}
          className={routerStore.currentRoute.name === 'dynamic' ? 'active' : ''}
        >
          Dynamic
        </div>
      </div>
      <Router />
    </>
  );
});

Promise.resolve()
  // be sure to load an initial route
  .then(() => routerStore.redirectTo(getInitialRoute({ 
    routes, 
    pathname: location.pathname + location.search, 
    fallback: 'error404'
  })))
  .then(() => createRoot(document.getElementById('app')!).render(<App />));

The full code for this example (SPA without SSR version) is here. It is configured with code-splitting for every page, but it's easy to disable this function in the bundler config.

Setup (SSR)

  1. SSR needs the store to be passed by React Context and an initial route to be passed from Node server
import { createContext } from 'react';
import { hydrateRoot } from 'react-dom/client';
import { restoreState } from 'dk-mobx-restore-state';
import { loadComponentToConfig } from 'dk-react-mobx-router';

import { routes } from './routes';
import { RouterStore } from './routerStore';

// Create context

const StoreContext = createContext(undefined as unknown as { routerStore: RouterStore });

const contextValue = { routerStore: new RouterStore() };

Promise.resolve()
  // restore current route from Node server, no need to call 'location'
  .then(() => restoreState({ target: contextValue, source: window.INITIAL_DATA }))
  .then(() => {
    // Load the async chunk for this page
    const preloadedRouteName = Object.keys(routes).find(
      (routeName) => contextValue.routerStore.currentRoute.name === routeName
    ) as keyof typeof routes;

    return loadComponentToConfig({ route: routes[preloadedRouteName] });
  })
  .then(() =>
    hydrateRoot(
      document.getElementById('app')!,
      <StoreContext.Provider value={contextValue}>
        <App />
      </StoreContext.Provider>
    )
  );
  1. And relevant Node server code for isomorphic rendering
const template500 = fs.readFileSync(path.resolve(__dirname, '../build/index.html'), 'utf-8');
      
const app = express()
  .use(serveStatic(path.resolve(__dirname, '../build/public')))
  .get('*', async (req, res) => {
    const contextValue = { routerStore: new RouterStore() };
    const reactApp = (
      <StoreContext.Provider value={contextValue}>
        <App />
      </StoreContext.Provider>
    );

    try {
      await contextValue.routerStore.redirectTo(
        getInitialRoute({
          routes,
          pathname: req.originalUrl,
          fallback: 'error404',
        })
      );
    } catch (error: any) {
      if (error.name === 'REDIRECT') {
        res.redirect(error.message);

        return;
      }

      console.error(error);

      res.status(500).send(template500);

      return;
    }

    const htmlMarkup = renderToString(reactApp);
    const storeJS = JSON.parse(JSON.stringify(contextValue));

    res.send(
      fs
        .readFileSync(path.resolve(__dirname, '../build/public/index.html'), 'utf-8')
        .replace(`<!-- HTML -->`, htmlMarkup)
        .replace('<!-- INITIAL_DATA -->', JSON.stringify(escapeAllStrings(storeJS)))
    );
  });

So, we only look at req.originalUrl on the Node server and do not look at location in the Client code; we just restore the state from window.INITIAL_DATA. The full example for SSR version is here.

Route configuration

path, params

path (string) required

params (Record<string, (value: string) => boolean>) required if the path is dynamic

Should have a leading slash. It may be a constant value or a dynamic value with :param syntax like

{
  path: '/static'
}

{
  path: '/dynamic/:param1/:param2',
  params: {
    param1: (value: string) => value.length > 0,
    param2: (value: string) => value === 'some',
  },
}

When dynamic path is used - params object with validators is necessary. Each validator should return a boolean value.

loader

(async import) required

{
  loader: () => import('./pages/home')
}

// @loadable/component should work, too
import loadable from '@loadable/component';

{
  loader: loadable(() => import('./pages/home'))
}

The imported file should have a default export with a React Component. It may also export pageName, store, actions - this will be explained in other sections of this documentation.

props

(Record<string, any>) optional

{
  error404: {
    path: '/error404',
    props: { errorCode: 404 },
    loader: () => import('./pages/error'),
  },
  error500: {
    path: '/error500',
    props: { errorCode: 500 },
    loader: () => import('./pages/error'),
  },
}

const ErrorPage = observer((props: { errorCode: number }) => {
  return `Error ${props.errorCode}`;
});

These props will be passed to the page component. In some cases it may be useful for Dependency Injection (DI).

query

(Record<string, (value: string) => boolean>) optional

{
  query: {
    foo: (value: string) => value.length > 0,
  }
}

These parameters are always optional. So, if you go to http://site?bar=1 then routerStore.currentRoute.query will be an empty object {}. Only query params which are described in the query config will be present.

beforeEnter

((config, ...args) => Promise<TypeRedirectToParams | void>) optional

config argument is not perfectly TS-typed to avoid a circular dependency, so use the console to check its contents

{
  beforeEnter(config) {
    /*
      nextUrl: string;
      nextRoute: any;
      nextPathname: string;
      nextQuery?: any;
      nextSearch?: string;

      currentUrl?: string;
      currentQuery?: any;
      currentRoute?: any;
      currentSearch?: string;
      currentPathname?: string;
    */
  
    console.log(config);
  
    if (config.nextRoute.name === 'query') {
      return Promise.resolve({ route: 'dynamic', params: { foo: 'bar' }, query: { foo: 'bar' } });
    }

    return Promise.resolve();
  }
}

This method has the capability to redirect to another page, as shown in the example above.

You can also pass some arguments to this function by adding lifecycleParams to the redirectTo function

redirectTo = redirectToGenerator({
  routes,
  routerStore: this,
  routeError500: routes.error500,
  lifecycleParams: [userStore]
});
  
{
  beforeEnter(config, userStore) {
    if (!userStore.isLoggedIn) {
      return Promise.resolve({ route: 'login' });
    }

    return Promise.resolve();
  }
}

beforeLeave

((config, ...args) => Promise<void> | Error<{ name: 'PREVENT_REDIRECT'} >) optional

config argument is not perfectly TS-typed to avoid a circular dependency, so use the console to check its contents

{
  beforeLeave(config) {
    /*
      nextUrl: string;
      nextRoute: any;
      nextPathname: string;
      nextQuery?: any;
      nextSearch?: string;

      currentUrl?: string;
      currentQuery?: any;
      currentRoute?: any;
      currentSearch?: string;
      currentPathname?: string;
    */
  
    console.log(config);
  
    if (config.nextRoute.name === 'query') {
      throw Object.assign(new Error(''), { name: 'PREVENT_REDIRECT' });
    }

    return Promise.resolve();
  }
}

This method has the capability to prevent redirect, as shown in the example above.

You can also pass some arguments to this function by adding lifecycleParams to the redirectTo function

redirectTo = redirectToGenerator({
  routes,
  routerStore: this,
  routeError500: routes.error500,
  lifecycleParams: [userStore]
});
  
{
  beforeLeave(config, userStore) {
    if (userStore.someFormNotFilled) {
      const allowLeavePage = confirm('You form data will be lost, sure to go to another page?');
    
      if (!allowLeavePage) {
        throw Object.assign(new Error(''), { name: 'PREVENT_REDIRECT' });
      }
    }

    return Promise.resolve();
  }
}

RedirectTo configuration

routes (required) - config returned from createRouterConfig function

routerStore (required) - InterfaceRouterStore-compliant object

routeError500 (required) - route config for 500 error

lifecycleParams (optional) - Array<any> This parameter is used to pass arguments to the beforeEnter and beforeLeave methods. For example, you can pass lifecycleParams: [userStore] to check if the user is logged in or has rights to view this page.

It can also be used for data-loading like

redirectTo = redirectToGenerator({
  routes,
  routerStore: this,
  routeError500: routes.error500,
  lifecycleParams: [apiStore]
});
  
{
  beforeEnter(config, apiStore) {
    return Promise.resolve()
      .then(() => apiStore.getUser())
      .then(() => apiStore.getDashboard())
      .then(() => apiStore.getCart())
      .then(() => undefined)
  }
}

If you use the SPA (no SSR) approach, then you can use userStore and apiStore directly without this dependency injection (DI), but for SSR websites, this is a powerful helper.

Router configuration

routes (required) - config returned from createRouterConfig function

routerStore (required) - InterfaceRouterStore-compliant object

redirectTo (required) - a function created with redirectToGenerator and addState

beforeMount (optional) - () => void - this function is called only once on component mount

beforeSetPageComponent (optional) - (route) => void - this function is called every time before the render of any page. config here is an extended route config

beforeUpdatePageComponent (optional) - () => void - this function is called on every page update before the render. Therefore, it will not be called for the first page render

<RouterMobx 
  routes={routes} 
  redirectTo={routerStore.redirectTo} 
  routerStore={routerStore} 
  beforeMount={() => {
    // component just mounted
  }}
  beforeSetPageComponent={(route) => {
    // some page will be rendered soon
    console.log(route); // shows which page will be loaded
  }}
  beforeUpdatePageComponent={() => {
    // some new page will be rendered soon
    // You may stop async actions and clear modular stores here
    
    cancelExecutingApi();
    cancelExecutingActions();
    someStore.reset();
  }}
/>

RedirectTo usage

route (required) - a string representing some route

params (required if in route config) - Record<string, string>

query (optional) - Record<string, string>

noHistoryPush (optional) - if true, this redirect will not be present in the browser's history

This function is fully TypeScript-typed, and TypeScript hints will be shown for autocomplete.

const routes = createRouterConfig({
  static: {
    path: '/static',
    loader: () => import('./pages/static'),
  },
  dynamic: {
    path: '/page/:foo',
    params: {
      foo: (value: string) => value.length > 0,
    },
    query: {
      q: (value: string) => value.length > 0,
    },
    loader: () => import('./pages/dynamic'),
  },
});

// Good
redirectTo({ route: 'static' })
redirectTo({ route: 'dynamic', params: { foo: 'bar' } })
redirectTo({ route: 'dynamic', params: { foo: 'bar' }, query: { q: 's' } })

// TS errors
redirectTo({ });
redirectTo({ route: 'nonExisting' });
redirectTo({ route: 'static', params: {} });
redirectTo({ route: 'dynamic' });
redirectTo({ route: 'dynamic', params: {} });
redirectTo({ route: 'dynamic', params: { a: 'b' } });
redirectTo({ route: 'dynamic', params: { foo: 'bar' }, query: { some: 'value' } });

Use cases

Show loader on redirect

redirectTo function has a mobx reactive state because of mobx-stateful-fn, so we can utilize all of its capabilities.

const GlobalHeader = observer(() => {
  console.log(`Last redirect took ${routerStore.redirectTo.state.executionTime}ms`)

  if (routerStore.redirectTo.state.isExecuting) {
    return <Loader />
  }
  
  return null;
});

If you have problems with state is undefined, do not forget to exclude redirectTo from makeAutoObservable: makeAutoObservable(this, { redirectTo: false });

Handle redirect errors

redirectTo is an async function, so it's a good practice to handle errors

redirectTo({ route: 'static' }).catch(error => console.error(error))

otherwise if some unexpected error happens in beforeEnter / beforeLeave, you will get an Uncaught exception.

Multiple redirects

This library fully supports unlimited redirects in SPA / SSR.

const routes = createRouterConfig({
  one: {
    path: '/1',
    loader: () => import('./pages/one'),
  },
  two: {
    path: '/2',
    loader: () => import('./pages/two'),
    beforeEnter(config, userStore) {
      if (!userStore.isLoggedIn) {
        return Promise.resolve({ route: 'one' });
      }

      return Promise.resolve();
    },
  },
  three: {
    path: '/3',
    loader: () => import('./pages/three'),
    beforeEnter(config, userStore) {
      if (!userStore.isLoggedIn) {
        return Promise.resolve({ route: 'two' });
      }

      return Promise.resolve();
    },
  },
  four: {
    path: '/4',
    loader: () => import('./pages/four'),
    beforeEnter(config, userStore) {
      if (!userStore.isLoggedIn) {
        return Promise.resolve({ route: 'three' });
      }

      return Promise.resolve();
    },
  },
});

redirectTo({ route: 'four' })

In this case if user goes to '/4' he will be redirected to '/3' then '/2' then '/1'. Browser's history and routerStore.routesHistory will only have ['/1']. Also, chunks for pages four, three, two will not be loaded.

Watch and react to params / query changes

routerStore.currentRoute is a mobx-observable, so you can use it inside autorun / reaction, or within a component wrapped in observer.

import { TypeCurrentRoute } from 'dk-react-mobx-router';
import { routes } from 'routes';

const MyComponent = observer(() => {
  const currentRoute = routerStore.currentRoute as TypeCurrentRoute<typeof routes.tabs>;

  useEffect(() => {
    const disposer = autorun(() => {
      console.log(toJS(currentRoute.params))
      console.log(toJS(currentRoute.query))
    })
    
    return () => disposer();
  }, []);
  
  if (currentRoute.params.tab === 'dashboard') {
    return <Dashboard />
  }
  
  if (currentRoute.params.tab === 'table') {
    return <Table />
  }
  
  return null;
});

routerStore.currentRoute as TypeCurrentRoute<typeof routes.tabs> here tells TS that we expect the route to be 'routes.tabs' so its params and query will be TS-typed.

@loadable/component support

This feature should work, but this library does not have tests for this scenario yet.

@loadable/component is used with SSR and allows loading page chunks in parallel with the main chunk, improving the initial page loading speed. Here, only some differences from a regular SSR configuration are highlighted.

Routes config:

import loadable from '@loadable/component';

const routes = createRouterConfig({
  home: {
    path: '/',
    loader: loadable(() => import('./pages/home')),
  }
};

SSR side:

import { ChunkExtractor, ChunkExtractorManager } from '@loadable/server';

const webExtractor = new ChunkExtractor({
  statsFile: path.resolve(paths.build, 'web-loadable-stats.json'),
  entrypoints: ['client'],
});

const app = (
  <ChunkExtractorManager extractor={webExtractor}>
    <StoreContext.Provider value={globals}>
      <App />
    </StoreContext.Provider>
  </ChunkExtractorManager>
);

const finalMarkup = template
  .replace(`<!-- HTML -->`, htmlMarkup)
  .replace(
    '<!-- LINKS -->',
    [webExtractor.getLinkTags(), webExtractor.getStyleTags()].join('\n')
  )
  .replace('<!-- SCRIPTS -->', webExtractor.getScriptTags())
  .replace('<!-- INITIAL_DATA -->', JSON.stringify(escapeAllStrings(storeJS)));

Client side:

import { loadableReady } from '@loadable/component';

Promise.resolve()
  .then(() => loadableReady())
  // Still need to call loadComponentToConfig but this page's chunk will not be loaded twice,
  // because it's already loaded
  .then(() => {
    const preloadedRouteName = Object.keys(routes).find(
      (routeName) => contextValue.routerStore.currentRoute.name === routeName
    ) as keyof typeof routes;

    return loadComponentToConfig({ route: routes[preloadedRouteName] });
  })
  .then(() =>
    hydrateRoot(
      document.getElementById('app')!,
      <StoreContext.Provider value={contextValue}>
        <App />
      </StoreContext.Provider>
    )
  );

Prevent rerendering if the page component is the same

Router will not rerender the page component if params or query have been changed. However, if redirected to another route, the page component will be rerendered.

const routes = createRouterConfig({
  home: {
    path: '/',
    loader: () => import('./pages/home'),
  },
  static: {
    path: '/static',
    loader: () => import('./pages/home'),
  }
};

redirectTo({ route: 'home' })
  .then(() => redirectTo({ route: 'static' }))

Here we have the same component './pages/home', but different routes. Sometimes it's necessary to prevent rerendering in this scenario. You may include in pages/home/index.ts this code

import { default as PageComponent } from './Home';

// This line
export const pageName = 'home';

// Or if you have __dirname access you can use the folder's name
export const pageName = __dirname.split('/').pop();

export default PageComponent;

This way the component will not be rerendered and the Router will not call beforeSetPageComponent or beforeUpdatePageComponent. This is useful for the 'Modular stores / actions' section.

Modular (page's) stores / actions

It's a good practice to split your stores / actions in MobX architectures by pages. This library has a built-in mechanism to do so.

Inside your page file, ex. pages/home/index.ts, include relevant exports

import { default as PageComponent } from './Home';

class HomeStore {}

export const pageName = 'home';

// This line
export const store = new HomeStore();

// And this
export const actions = {
  getData() {
    return api.fetchData().then(data => { store.data = data });
  }
}

export default PageComponent;

Then in the Router lifecycle, you may operate these variables like this:

<Router
  routes={routes}
  redirectTo={routerStore.redirectTo}
  routerStore={routerStore}
  beforeSetPageComponent={(route: any) => {
    if (!route.pageName) return;
  
    if (route.store) {
      globalStore.pages[route.pageName] = route.store;
    }
    
    if (route.actions) {
      globalActions.pages[route.pageName] = route.actions;
    }
  }}
  beforeUpdatePageComponent={() => {
    globalStore.pages = {};
    globalActions.pages = {};
  }}
/>

You may operate route.store and route.actions according to your architecture; this library simply provides a way to separate them inside the page's async chunks and initialize them when the page is loaded.

History manipulation

This library uses history.js, and it is exported for some custom scenarios. Be aware that in SSR (Node.js server code) history is null.

import { history } from 'dk-react-mobx-router';

if (typeof window !== 'undefined') {
  history.listen((params) => {
    // some logic
  });
  
  history.back();
}

Hash support inside currentRoute

There are no plans to implement it. Use history.listen to track the hash and synchronize it with your own mobx-store parameter (may be a part of the routerStore).

Why dk-mobx-stateful-fn dependency + how to write custom logic inside redirectTo

dk-mobx-stateful-fn is necessary for the Router. redirectTo is an async function, so we have to track when it has finished before rendering the page component.

However, this does not prevent you from using custom logic inside your redirect function. For example, if you want to scroll the page to the top on every redirect, you may use this:

import { TypeRedirectToParams } from 'dk-react-mobx-router';
import { TypeFnState } from 'dk-mobx-stateful-fn';
import { routes } from 'routes';

type TypeCustomRedirectTo = (<TRouteName extends keyof typeof routes>(
  params: TypeRedirectToParams<typeof routes, TRouteName>
) => Promise<void>) &
  TypeFnState;

export class RouterStore implements TInterfaceRouterStore {
  redirectTo: TypeCustomRedirectTo = addState((params) => {
    if (typeof window !== 'undefined') {
      window.scroll(0, 0);
    }

    return redirectToGenerator({
      routes,
      routerStore: this,
      routeError500: routes.error500,
    })(params);
  }, 'redirectTo');
}

It is a bit tricky to construct TypeCustomRedirectTo, but it's helpful in some cases.

Usually the Router's lifecycle, like beforeSetPageComponent, should be sufficient for most cases.