Plugins

Haploid.js reuses webpack's tapable mode to provide scalability.

In fact, the preload, keep-alive and the ability to request a variety of formats entry in Haploid.js internal are accomplished through plugins, such as LoadFromEntryPlugin.tsopen in new window.

Hooks & Events

Container

The Container contains the following registrable events:

interface ContainerEvent {
  destroying: void;
  destroyed: void;

  appactivating: { appname: string | null };
  appactivated: { appname: string };
  appactivateerror: { appname: string; error: Error };
  noappactivated: { error: Error };

  appregisteredchange: void;

  appregistererror: { error: Error };
}

Both registration and unregistration events need to be prior to destruction:

container.on("appactivating", (appname) => {});
container.once("destroying", () => {});

container.off("appactivating", () => {});

Container has the following hooks:

type ContainerHooks = {
  afterrootready: AsyncSeriesBailHook<{ source: unknown }, void>;
};

container.hooks.afterrootready.tapPromise("any", ({ source }) => {});

Lifecycle

Lifecycle represents a description of the sub-application lifecycle functions and contains the following registrable events:

interface LifecycleEvent {
  beforebootstrap: void;
  afterbootstrap: void;
  bootstraperror: AppError;

  beforemount: void;
  aftermount: void;
  mounterror: AppError;

  beforeunmount: void;
  afterunmount: void;
  unmounterror: AppError;

  beforeupdate: void;
  afterupdate: void;
  updateerror: AppError;
}
app.lifecycle.on("beforemount", () => {});
app.lifecycle.once("beforeunmount", () => {});

app.lifecycle.off("beforemount", () => {});

Lifeycycle has the following hooks:

type LifecycleHooks<CustomProps> = {
  beforebootstrap: AsyncSeriesHook<void>;
  afterbootstrap: AsyncSeriesHook<void>;
  bootstraperror: AsyncSeriesHook<AppError>;

  beforemount: AsyncSeriesHook<void>;
  aftermount: AsyncSeriesHook<void>;
  mounterror: AsyncSeriesHook<AppError>;

  beforeunmount: AsyncSeriesHook<void>;
  afterunmount: AsyncSeriesHook<void>;
  unmounterror: AsyncSeriesHook<AppError>;

  beforeupdate: AsyncSeriesHook<{ props: CustomProps }>;
  afterupdate: AsyncSeriesHook<{ props: CustomProps }>;
  updateerror: AsyncSeriesHook<AppError & { props: CustomProps }>;

  bootstrap: AsyncSeriesBailHook<void, unknown>;
  mount: AsyncSeriesBailHook<void, unknown>;
  unmount: AsyncSeriesBailHook<void, unknown>;
  update: AsyncSeriesBailHook<{ props: CustomProps }, unknown>;
};

App

App contains the following registrable events:

interface AppEvent {
  statechange: { prevState: AppState; nextState: AppState };

  beforeload: void;
  afterload: void;
  loaderror: AppError;

  beforestart: void;
  afterstart: void;
  starterror: AppError;

  beforestop: void;
  afterstop: void;
  stoperror: AppError;

  beforeupdate: void;
  afterupdate: void;
  updateerror: AppError;

  beforeunload: void;
  afterunload: void;
}
app.on("beforestop", () => {});
container.once("beforeunload", () => {});

container.off("beforestop", () => {});

App has the following hooks:

type AppHooks = {
  encounterUnmountFailure: SyncBailHook<void, { ignore: boolean } | void>;
  encounterLoadingSourceCodeFailure: SyncBailHook<
    void,
    { retry: boolean; count: number } | void
  >;
  waitForLoadingOrBootstrappingWhenStop: SyncBailHook<
    void,
    { wait: boolean } | void
  >;

  resolveAssets: AsyncSeriesBailHook<
    void,
    { html?: string; scripts: ScriptNode[]; styles: StyleNode[] } | void
  >;

  resolveEnvVariables: AsyncSeriesWaterfallHook<
    Record<string, unknown>,
    Record<string, unknown>
  >;

  beforeload: AsyncSeriesHook<void>;
  afterload: AsyncSeriesHook<void>;
  loaderror: AsyncSeriesHook<AppError>;

  beforestart: AsyncSeriesHook<void>;
  afterstart: AsyncSeriesHook<void>;
  starterror: AsyncSeriesHook<AppError>;

  beforestop: AsyncSeriesHook<void>;
  afterstop: AsyncSeriesHook<void>;
  stoperror: AsyncSeriesHook<AppError>;

  beforeupdate: AsyncSeriesHook<void>;
  afterupdate: AsyncSeriesHook<void>;
  updateerror: AsyncSeriesHook<AppError>;

  beforeunload: AsyncSeriesHook<void>;
  afterunload: AsyncSeriesHook<void>;
};

Plugin

When registering an App in the Container, function plugins can be passed in:

function createMyPlugin<AdditionalOptions, CustomProps>(): AppPlugin<AdditionalOptions, CustomProps> {
    return ({ app, debug }) => {
        app.hooks...
        app.lifecycle.hooks...
    };
}

container.registerApp({
    name: 'foo'
}, [
    ['MyPlugin', createMyPlugin()]
]);

Plugins are executed in the order they were passed in.