Usage Guide
This guide provides detailed examples and patterns for using the PUI App SDK in your applications. For initial setup, see the Getting Started Guide.
Table of Contents
- Routing
- Authentication
- API Integration with RTK Query
- HTTP Client (Legacy)
- Redux Internals
- State Management
- Form Management
- Analytics
- Micro-Frontends
- Scripting Objects
- Wait Messages
- Navigation Prompts
- Error Handling
- ARIA Live Messages
- Decorators
- Advanced State Selectors
- Responsive Design
- Environment & URL Utilities
- Security Utilities
- Listener Middleware
- Session Management
- App Configuration
- Web Storage Events
- Micro-Frontend Communication
- Authentication Utilities
- Utility Hooks
- Form Components
- Storybook Integration
- Utility Components
- Testing
- Type Exports
- Common Patterns
- Troubleshooting
Routing
Set up routing with React Router and SDK components:
import { Routes, Route } from 'react-router-dom';
import { RequireAuth, useInjectSaga, Page } from '@elliemae/pui-app-sdk';
import { Layout } from './view/layout';
import { onSessionEvent } from './sideeffect/session';
const key = 'session';
export const AppRoutes = () => {
// Inject session management saga
useInjectSaga({ key, saga: onSessionEvent });
return (
<Routes>
<Route path="/" element={<Layout />}>
<Route
index
element={
<Page pageTitle="Dashboard">
<Dashboard />
</Page>
}
/>
<Route
path="details/:id"
element={
<Page pageTitle="Details">
<DetailsView />
</Page>
}
/>
</Route>
</Routes>
);
};
Lazy Loading Components
import { loadable } from '@elliemae/pui-app-sdk';
// Create loadable component
export const LoanView = loadable(() => import('./index'), {
fallback: <div>Loading...</div>,
});
// Use in routes
import { LoanView } from './view/loan-view/loadable';
const AppRoutes = () => (
<Routes>
<Route path="/loans" element={<LoanView />} />
</Routes>
);
Authentication
Handle authentication and logout with Redux Saga:
import {
auth,
globalConstants,
getRedirectUrl,
logout,
error,
} from '@elliemae/pui-app-sdk';
import { call, put, takeLatest } from 'redux-saga/effects';
function* endSessionHandler() {
try {
const cred = sessionStorage.getItem('Authorization') || '';
yield call(endSession, { token: cred });
// Trigger logout
yield put(
auth.logout({
clientId: globalConstants.CLIENT_ID,
redirectUri: getRedirectUrl(),
responseType: 'code',
scope: 'loc',
code: '1004',
}),
);
} catch (err) {
yield put(error.set({ messageText: (err as Error)?.message }));
}
}
export function* onSessionEvent() {
yield takeLatest(logout.CONFIRM, endSessionHandler);
}
API Integration with RTK Query
Recommended Approach: RTK Query is the preferred method for API integration, providing automatic caching, request deduplication, and simplified data fetching.
Setting Up RTK Query API
Create an API slice using RTK Query with the SDK's sdkBaseQuery:
import { createApi } from '@reduxjs/toolkit/query/react';
import { sdkBaseQuery } from '@elliemae/pui-app-sdk';
// Define API slice
export const resourcesApi = createApi({
reducerPath: 'resourcesApi',
baseQuery: sdkBaseQuery({ baseUrl: '/v1' }),
tagTypes: ['Resources'],
endpoints: (builder) => ({
getResources: builder.query<ResourceRecord[], void>({
query: () => ({
url: '/resources',
method: 'GET',
}),
providesTags: ['Resources'],
}),
getResourceById: builder.query<ResourceRecord, string>({
query: (id) => ({
url: `/resources/${id}`,
method: 'GET',
}),
providesTags: (result, error, id) => [{ type: 'Resources', id }],
}),
createResource: builder.mutation<ResourceRecord, CreateResourceRequest>({
query: (payload) => ({
url: '/resources',
method: 'POST',
data: payload,
}),
invalidatesTags: ['Resources'],
}),
updateResource: builder.mutation<
ResourceRecord,
{ id: string; data: UpdateResourceRequest }
>({
query: ({ id, data }) => ({
url: `/resources/${id}`,
method: 'PUT',
data,
}),
invalidatesTags: (result, error, { id }) => [{ type: 'Resources', id }],
}),
deleteResource: builder.mutation<void, string>({
query: (id) => ({
url: `/resources/${id}`,
method: 'DELETE',
}),
invalidatesTags: ['Resources'],
}),
}),
});
export const {
useGetResourcesQuery,
useGetResourceByIdQuery,
useCreateResourceMutation,
useUpdateResourceMutation,
useDeleteResourceMutation,
} = resourcesApi;
Injecting RTK Query API
Use the SDK's useInjectQuery hook to dynamically inject the API:
import { useInjectQuery } from '@elliemae/pui-app-sdk';
import { resourcesApi } from './api/resources';
export const Layout = () => {
// Inject RTK Query API
useInjectQuery({ api: resourcesApi });
return (
<div>
<Header />
<Outlet />
</div>
);
};
Using RTK Query in Components
import {
useGetResourcesQuery,
useCreateResourceMutation,
useUpdateResourceMutation,
} from './api/resources';
import { waitMessageAction, useAppDispatch } from '@elliemae/pui-app-sdk';
function ResourceList() {
const dispatch = useAppDispatch();
// Fetch resources with automatic caching and refetching
const { data: resources, isLoading, error, refetch } = useGetResourcesQuery();
// Mutations
const [createResource, { isLoading: isCreating }] =
useCreateResourceMutation();
const [updateResource, { isLoading: isUpdating }] =
useUpdateResourceMutation();
// Show wait message while loading
useEffect(() => {
if (isLoading || isCreating || isUpdating) {
dispatch(waitMessageAction.open('Loading...', true));
} else {
dispatch(waitMessageAction.close());
}
}, [isLoading, isCreating, isUpdating, dispatch]);
const handleCreate = async () => {
try {
await createResource({
name: 'New Resource',
description: 'Description',
}).unwrap();
// Success - data automatically refetched due to cache invalidation
} catch (err) {
console.error('Failed to create resource:', err);
}
};
const handleUpdate = async (id: string) => {
try {
await updateResource({
id,
data: { name: 'Updated Name' },
}).unwrap();
} catch (err) {
console.error('Failed to update resource:', err);
}
};
if (error) return <div>Error loading resources</div>;
if (isLoading) return <div>Loading...</div>;
return (
<div>
<button onClick={handleCreate}>Create Resource</button>
{resources?.map((resource) => (
<div key={resource.id}>
<h3>{resource.name}</h3>
<button onClick={() => handleUpdate(resource.id)}>Update</button>
</div>
))}
</div>
);
}
Advanced RTK Query Features
Conditional Fetching
function ResourceDetails({ resourceId }: { resourceId?: string }) {
// Skip query if no resourceId
const { data, isLoading } = useGetResourceByIdQuery(resourceId!, {
skip: !resourceId,
});
return <div>{data?.name}</div>;
}
Polling
function ResourceList() {
// Poll every 30 seconds
const { data } = useGetResourcesQuery(undefined, {
pollingInterval: 30000,
});
return <div>{/* render resources */}</div>;
}
Optimistic Updates
const [updateResource] = useUpdateResourceMutation();
const handleUpdate = async (id: string, newName: string) => {
try {
await updateResource({
id,
data: { name: newName },
}).unwrap();
} catch (err) {
// Update failed, cache automatically reverted
}
};
Why RTK Query?
- Automatic Caching: Reduces unnecessary network requests
- Request Deduplication: Multiple components can use the same query without duplicate requests
- Automatic Refetching: Data stays fresh with polling, refetch on focus, etc.
- Optimistic Updates: Update UI before server response
- Error Handling: Built-in error states
- Loading States: Automatic loading indicators
- TypeScript Support: Full type safety
- Less Boilerplate: No need for actions, reducers, or sagas for API calls
HTTP Client (Legacy)
Note: For new projects, prefer RTK Query for API integration. Use the HTTP client directly only when you need fine-grained control or are working with existing code.
getAuthHTTPClient
Make authenticated API calls with automatic token management:
import { getAuthHTTPClient } from '@elliemae/pui-app-sdk';
import { logger } from './utils/logger';
const basePath = '/v1/resources';
export const getResources = async () => {
try {
const { data } = await getAuthHTTPClient().get(basePath);
return data;
} catch (err) {
logger.error({
message: 'Failed to get resources',
exception: err as Error,
});
throw err;
}
};
getHTTPClient
Create an HTTP client without automatic authorization headers. Useful for public endpoints or when you supply your own auth:
import { getHTTPClient } from '@elliemae/pui-app-sdk';
const publicClient = getHTTPClient({
baseURL: 'https://public-api.example.com',
headers: { 'X-Api-Key': 'my-key' },
sendLogRocketSessionHeader: true,
});
const { data } = await publicClient.get('/status');
Options:
| Option | Type | Default | Description |
|---|---|---|---|
baseURL | string | App config serviceEndpoints.api | Base URL for requests |
headers | object | {} | Custom request headers |
sendLogRocketSessionHeader | boolean | false | Attach X-LogRocket-URL header |
onAuthorizationFailure
Register a callback to handle 401 responses. The SDK will retry the failed request with the new authorization header you return:
import { onAuthorizationFailure } from '@elliemae/pui-app-sdk';
onAuthorizationFailure(async (error) => {
const newToken = await refreshAuthToken();
sessionStorage.setItem('Authorization', newToken);
return newToken;
});
If no handler is registered, a 401 response triggers endSession (redirect to login).
httpClientProps
Bind session management callbacks to the HTTP layer. AppRoot sets these automatically, but you can configure them manually when bootstrapping without AppRoot:
import {
httpClientProps,
resetUserIdleTime,
endSession,
} from '@elliemae/pui-app-sdk';
httpClientProps.resetUserIdleTime = resetUserIdleTime;
httpClientProps.endSession = endSession;
## State Management
> **Recommended**: For API-related state management, use [RTK Query](#api-integration-with-rtk-query) instead of manually creating actions, reducers, and sagas.
### Creating Actions and Reducers (Legacy Pattern)
For non-API state or when working with existing code, use Redux Toolkit with the SDK's API action creator:
```typescript
import { createSlice, PayloadActionCreator } from '@reduxjs/toolkit';
import { getApiActionCreator } from '@elliemae/pui-app-sdk';
const resourcesActionCreator = getApiActionCreator('resources');
type GetResourceActions = {
get: PayloadActionCreator<void, string>;
getSuccess: PayloadActionCreator<Array<ResourceRecord>, string>;
getError: PayloadActionCreator<void, string>;
};
type CreateResourceActions = {
create: PayloadActionCreator<void, string>;
createSuccess: PayloadActionCreator<CreateResourceResponse, string>;
createError: PayloadActionCreator<void, string>;
};
export const resources = {
...(resourcesActionCreator('create') as unknown as CreateResourceActions),
...(resourcesActionCreator('get') as unknown as GetResourceActions),
};
const resourcesSlice = createSlice({
name: 'resources',
initialState: [] as Array<ResourceRecord>,
reducers: {
getSuccess: (state, { payload }: { payload: Array<ResourceRecord> }) => {
state.push(...payload);
},
},
});
export const { reducer } = resourcesSlice;
Typed Selectors
Create typed selectors for better type safety:
import { TypedUseSelectorHook, useSelector } from 'react-redux';
import { RootState } from '@elliemae/pui-app-sdk';
import { reducer } from './data/resources';
// Create typed selector
type AppState = RootState & { resources: ReturnType<typeof reducer> };
export const useAppSelector: TypedUseSelectorHook<AppState> = useSelector;
Using State in Components
import { useAppDispatch, useAppSelector } from '@elliemae/pui-app-sdk';
function ResourceList() {
const resourcesData = useAppSelector((state) => state.resources);
const dispatch = useAppDispatch();
useEffect(() => {
if (!resourcesData.length) {
dispatch(resources.get());
}
}, [dispatch, resourcesData.length]);
return <div>{/* render resources */}</div>;
}
Layout Component with Dynamic Injection
Inject reducers and sagas at the layout level:
import { Outlet } from 'react-router-dom';
import { useInjectReducer, useInjectSaga } from '@elliemae/pui-app-sdk';
import { reducer } from './data/resources';
import { resourceSagas } from './sideeffect/resources';
export const Layout = () => {
// Inject reducer and saga at layout level
useInjectReducer({ key: 'resources', reducer });
useInjectSaga({ key: 'resources', saga: resourceSagas });
return (
<div>
<Header />
<Outlet />
</div>
);
};
Redux Saga Side Effects (Legacy Pattern)
Note: RTK Query eliminates the need for sagas in most API use cases. Use sagas for complex async workflows, WebSocket connections, or custom side effects not related to API calls.
Handle async operations with Redux Saga:
import { all, fork } from 'redux-saga/effects';
import { logger } from './utils/logger';
import { onGetAllResources } from './get';
import { onCreateResource } from './create';
export function* resourceSagas() {
try {
yield all([fork(onCreateResource), fork(onGetAllResources)]);
} catch (err) {
logger.error({
message: 'Saga initialization error',
exception: err as Error,
});
}
}
Redux Internals
These lower-level exports are pre-wired by the SDK. You only need them for advanced bootstrapping or when not using AppRoot.
getStore
Access the global Redux store outside of React components (e.g. in sagas, utilities, or test setup):
import { getStore } from '@elliemae/pui-app-sdk';
const store = getStore();
store.dispatch(someAction());
const state = store.getState();
authReducer & authSaga
The SDK includes a built-in auth reducer and saga that handle auth.login() / auth.logout() actions. They are automatically injected by AppRoot, but can be manually wired if needed:
import { authReducer, authSaga } from '@elliemae/pui-app-sdk';
// Manual injection (only if not using AppRoot)
useInjectReducer({ key: 'auth', reducer: authReducer });
useInjectSaga({ key: 'auth', saga: authSaga });
The auth saga listens for:
auth.login()— callsauthorize()and dispatchesLOGIN_SUCCESSauth.logout()— callsendSession()with the provided params
errorMiddleware
Redux middleware that automatically dispatches error.set() for any RTK Query rejected action:
import { errorMiddleware } from '@elliemae/pui-app-sdk';
// Already included in configureStore() — no manual setup needed.
// Shown here for reference only.
When any RTK Query mutation or query is rejected with a value, the middleware extracts the error message and dispatches it to the global error toast.
createManager
Re-exported from redux-injectors. Create a standalone injector manager for manual reducer/saga injection outside React:
import { createManager } from '@elliemae/pui-app-sdk';
In most cases, use
useInjectReducer/useInjectSagahooks inside components instead.
Form Management
Build forms with validation using React Hook Form integration:
import { Form as SDKForm, useAppDispatch } from '@elliemae/pui-app-sdk';
import { useParams } from 'react-router-dom';
import { SubmitHandler } from 'react-hook-form';
import { resources } from './data/resources';
interface FormData {
firstName: string;
lastName: string;
email: string;
country?: { value: string; label: string };
}
export const MyForm = () => {
const dispatch = useAppDispatch();
const { id } = useParams<{ id: string }>();
const onSubmit: SubmitHandler<FormData> = (formData) => {
const { country, ...rest } = formData;
dispatch(
resources.save({
id,
...rest,
country: country?.value || country,
}),
);
};
return (
<SDKForm<FormData> onSubmit={onSubmit}>
<FormFields />
<SubmitButton />
</SDKForm>
);
};
Analytics
Track user events and page views using scripting objects:
import { CMicroApp } from '@elliemae/pui-app-sdk';
import { IAnalytics } from '@elliemae/pui-scripting-object';
import { logger } from './utils/logger';
let analyticsObj: IAnalytics | null = null;
export const getAnalytics = async () => {
if (!analyticsObj) {
try {
const app = CMicroApp.getInstance();
analyticsObj = (await app.getObject('analytics')) ?? null;
} catch (error) {
logger.error({
message: 'Error fetching analytics object',
exception: error as Error,
});
}
}
return analyticsObj;
};
export const getAppDetails = () => ({
appId: window.emui?.appId || '',
appUrl: window.location.href,
});
Timing Events
// Use in component
useEffect(() => {
(async () => {
const analytics = await getAnalytics();
if (analytics) {
await analytics.startTiming('PageLoad', getAppDetails());
// ... perform operation
await analytics.endTiming('PageLoad', getAppDetails());
}
})();
}, []);
Business Analytics Events
useEffect(() => {
if (recordId) {
(async () => {
const analytics = await getAnalytics();
if (analytics) {
await analytics.sendBAEvent({
event: 'RecordViewed',
recordId,
});
}
})();
}
}, [recordId]);
Direct Analytics Exports
The SDK also re-exports analytics utilities from @elliemae/pui-analytics-so:
import { Analytics, updateBAEventParameters } from '@elliemae/pui-app-sdk';
Analytics— the analytics singleton frompui-analytics-so. Prefer using the scripting object approach (above) instead of importingAnalyticsdirectly.updateBAEventParameters— update the shared BA event parameters (e.g.instanceId,userId) that are merged into everysendBAEventcall. The SDK calls this automatically duringauthorize()andendSession().
sendBAEvent (deprecated)
Deprecated: Use
Analytics.sendBAEvent()via the scripting object instead.
Legacy helper to push BA events to the GTM data layer:
import { sendBAEvent } from '@elliemae/pui-app-sdk';
sendBAEvent({
data: { eventName: 'button_click', eventCategory: 'user_interaction' },
self: true, // true = push to own dataLayer; false = send to host app
});
fetchUserSettings
Retrieve user settings from the API. Called automatically during authorize(), but available for manual use:
import { fetchUserSettings } from '@elliemae/pui-app-sdk';
const settings = await fetchUserSettings({
userName: 'jdoe',
isConsumerUser: false,
});
Micro-Frontends
Hosting Micro-Frontends
Use GuestMicroApp to embed child applications:
import { GuestMicroApp, history } from '@elliemae/pui-app-sdk';
// Simple guest micro-app component
const PricingMicroApp = () => <GuestMicroApp id="pricing" history={history} />;
export default PricingMicroApp;
Micro-Frontend Configuration
Define micro-apps in your app.config.json:
{
"microFrontendApps": {
"pricing": {
"name": "Pricing",
"hostUrl": "./pricing",
"development": {
"files": ["index.js"]
},
"production": {
"files": ["index.js"]
}
},
"services": {
"name": "Services",
"hostUrl": "./services",
"mode": "development",
"development": {
"files": ["index.dev.js", "index.css"]
},
"production": {
"files": ["index.js", "index.css"]
}
}
}
}
Navigation & History Modes
The GuestMicroApp component supports two history modes that control how the guest app's URL is managed. The mode is determined by the useParentHistory flag in the guest app's app.config.json.
useParentHistory: true (default) — Shared History
The guest app shares the host's browser history. Route changes in the guest are reflected in the host's URL bar and vice versa.
Host URL bar: https://host.example.com/auth-setup/loan/123
└─ basename ─┘└ guest route ┘
Host app (parent):
import { GuestMicroApp, history } from '@elliemae/pui-app-sdk';
const LoanApp = () => <GuestMicroApp id="loanapp" history={history} />;
Guest app config (app.config.json):
{
"useParentHistory": true
}
When to use: The guest app is the primary content area and its routes should be bookmarkable / shareable via the host's URL.
useParentHistory: false — Independent History
The guest app runs inside an iframe with its own browser history, independent of the host's URL. The host's URL bar does not change when the guest navigates internally.
Host URL bar: https://host.example.com/auth-setup (stays fixed)
Iframe URL: https://guest.example.com/dashboard (managed independently)
Guest app config (app.config.json):
{
"useParentHistory": false
}
When to use: The guest app should manage its own navigation without affecting the host's URL — e.g., an embedded tool, modal-like workflow, or self-contained feature.
Setting an Initial Route from the Host
The initialRoute prop lets the host app control which route the guest app lands on. This works with both history modes.
Scenario 1: Shared history with initial route
// Host navigates guest to /auth-setup/loan/123 on load
const LoanApp = () => (
<GuestMicroApp
id="loanapp"
history={history}
initialRoute="/auth-setup/loan/123"
/>
);
The guest app receives the parent's history, and the host's URL is set to the initial route.
Scenario 2: Independent history with initial route
// Host tells the guest to start at /app/dashboard
const MyApp = () => (
<GuestMicroApp id="myapp" history={history} initialRoute="/app/dashboard" />
);
The SDK defers the route push to the mount() phase (after the iframe is stable but before React renders), so the guest's React Router sees the correct URL on its first render — no flash of the default landing page.
Note: The
initialRoutevalue should be the full path including the guest app's basename (e.g.,/app/dashboard, not just/dashboard). The SDK pushes this path to the browser history directly, and React Router matches it against its configuredbasename.
Quick Reference
| Scenario | useParentHistory | initialRoute | Behavior |
|---|---|---|---|
| Guest shares host URL | true (default) | omitted | Guest routes reflected in host URL bar |
| Guest shares host URL, starts at specific page | true | "/app/details/42" | Host URL set to initial route on load |
| Guest has own URL space | false | omitted | Guest loads at its default route, host URL unchanged |
| Guest has own URL space, starts at specific page | false | "/app/dashboard" | Guest starts at /dashboard, host URL unchanged |
Guest App app.config.json Example
{
"appId": "myapp",
"useParentHistory": false,
"activeEnv": "dev",
"microFrontendApps": {}
}
Micro-Frontend Host API
The SDK also provides host-side APIs for apps that manage guest micro-apps.
CMicroAppHost
The host counterpart to CMicroApp (guest). Use it when your app is the host that loads and manages guest micro-apps:
import { CMicroAppHost } from '@elliemae/pui-app-sdk';
const host = CMicroAppHost.getInstance({
logger,
onInit,
onMount,
onUnmount,
});
enableReactAppForHostIntegration
Shorthand to create a CMicroApp (guest) instance for host-integration scenarios — useful when the host wants to render the guest inline:
import { enableReactAppForHostIntegration } from '@elliemae/pui-app-sdk';
const app = enableReactAppForHostIntegration({ onInit, onMount });
getMicroFrontEndAppConfig
Read a micro-app's resolved configuration from app.config.json. Returns the merged development/production config with absolute URLs:
import { getMicroFrontEndAppConfig } from '@elliemae/pui-app-sdk';
const config = getMicroFrontEndAppConfig({ id: 'pricing' });
// config.hostUrl, config.files, config.name, etc.
getNavigationLinks
Build navigation links from all micro-apps defined in app.config.json:
import { getNavigationLinks } from '@elliemae/pui-app-sdk';
const links = getNavigationLinks();
// [{ id: 'pricing', name: 'Pricing', path: '/pricing' }, ...]
getLogger
Returns the logger from whichever micro-app context is active (host or guest):
import { getLogger } from '@elliemae/pui-app-sdk';
const logger = getLogger();
logger.info('Application ready');
Deprecated Components
MicroApp and MicroIFrameApp are deprecated. Use GuestMicroApp instead:
// Deprecated — will be removed in a future release
import { MicroApp, MicroIFrameApp } from '@elliemae/pui-app-sdk';
// Use instead
import { GuestMicroApp } from '@elliemae/pui-app-sdk';
Scripting Objects
Adding Scripting Objects
Access and share scripting objects between parent and child micro-apps:
import { CMicroApp } from '@elliemae/pui-app-sdk';
import { ScriptingObjectTypes } from '@elliemae/pui-scripting-object';
const addObject = async (
app: CMicroApp,
objectName: keyof ScriptingObjectTypes,
) => {
try {
const so = await app.getObject(objectName);
if (so) {
app.addScriptingObject(so);
}
} catch (error) {
// Handle error
}
};
// Get scripting objects from parent and expose to child
export const addScriptingObjects = async () => {
const app = CMicroApp.getInstance();
await Promise.all([
addObject(app, 'analytics'),
addObject(app, 'application'),
addObject(app, 'auth'),
addObject(app, 'http'),
addObject(app, 'loan'),
]);
};
// Call during app initialization
export const onInit: OnInitCallback = ({ history, homeRoute }) => {
// ... other initialization
addScriptingObjects().catch(() => {
// Handle error
});
};
Using Scripting Objects in Components
import { CMicroApp } from '@elliemae/pui-app-sdk';
import { useFormContext } from 'react-hook-form';
const listenToLoanSyncEvent = async (setValue) => {
const app = CMicroApp.getInstance();
const loan = await app.getObject('loan');
if (loan) {
const subscriptionId = app.subscribe({
eventId: 'loan.sync',
callback: ({ eventParams }) => {
const { firstname, lastname } = eventParams;
setValue('firstname', firstname);
setValue('lastname', lastname);
// Unsubscribe after handling
app.unsubscribe({
eventId: 'loan.sync',
token: subscriptionId,
});
},
});
}
};
export const FormComponent = () => {
const { setValue } = useFormContext();
useEffect(() => {
listenToLoanSyncEvent(setValue).catch(() => {
// Handle error
});
}, [setValue]);
return <div>{/* form fields */}</div>;
};
Wait Messages
Display loading indicators with customizable wait messages:
Using WaitMessage Component
import { waitMessageAction, useAppDispatch } from '@elliemae/pui-app-sdk';
// Show/hide wait message from anywhere in your app
function MyComponent() {
const dispatch = useAppDispatch();
const handleLoadData = async () => {
// Show wait message
dispatch(waitMessageAction.open('Loading data...', true));
try {
await fetchData();
} finally {
// Hide wait message
dispatch(waitMessageAction.close());
}
};
return <button onClick={handleLoadData}>Load Data</button>;
}
Wait Message Props
size:'xs' | 'sm' | 'md' | 'lg' | 'xl'- Size of the spinnercolor:'light' | 'dark'- Color themeshowText:boolean- Show loading textwithTooltip:boolean- Show as tooltiptooltipStartPlacementPreference: Tooltip placement
Navigation Prompts
Prompt users before navigating away from unsaved changes:
import {
NavigationPrompt,
NavigationPromptActions,
useAppDispatch,
} from '@elliemae/pui-app-sdk';
function FormPage() {
const [hasUnsavedChanges, setHasUnsavedChanges] = useState(false);
const [showPrompt, setShowPrompt] = useState(false);
const dispatch = useAppDispatch();
// Listen for navigation prompt actions
useEffect(() => {
const handleConfirm = () => {
// Save data and proceed
saveData();
setShowPrompt(false);
};
const handleCancel = () => {
// Proceed without saving
setShowPrompt(false);
};
// Subscribe to actions
return () => {
// Cleanup
};
}, []);
// Show prompt when navigating with unsaved changes
const handleNavigation = () => {
if (hasUnsavedChanges) {
setShowPrompt(true);
}
};
return (
<>
<NavigationPrompt open={showPrompt} />
{/* Your form */}
</>
);
}
Dispatch Navigation Prompt Actions
import { NavigationPromptActions } from '@elliemae/pui-app-sdk';
// User confirms to save and continue
dispatch({ type: NavigationPromptActions.CONFIRM });
// User cancels or continues without saving
dispatch({ type: NavigationPromptActions.CANCEL });
Error Handling
Global Error Toast
Display error messages using the global error toast:
import { error, useAppDispatch } from '@elliemae/pui-app-sdk';
// Dispatch errors from anywhere in your app
function MyComponent() {
const dispatch = useAppDispatch();
const handleError = async () => {
try {
await riskyOperation();
} catch (err) {
dispatch(
error.set({
messageText: 'Operation failed',
description: (err as Error).message,
}),
);
}
};
return <button onClick={handleError}>Execute</button>;
}
Clear Errors
import { error } from '@elliemae/pui-app-sdk';
// Clear error state
dispatch(error.clear());
ARIA Live Messages
Announce messages to screen readers for accessibility:
import { ariaLive, useAppDispatch } from '@elliemae/pui-app-sdk';
function MyComponent() {
const dispatch = useAppDispatch();
const announceSuccess = () => {
dispatch(
ariaLive.announce({
message: 'Data saved successfully',
ariaLive: 'polite', // or 'assertive' for urgent messages
id: 'save-success',
}),
);
};
return <button onClick={announceSuccess}>Save</button>;
}
Decorators
The SDK provides TypeScript decorators for common patterns:
Function Decorators
import { decorators } from '@elliemae/pui-app-sdk';
const {
CacheUntilResolved,
Debounce,
Throttle,
Memoize,
MemoizeAsync,
RetryAsync,
AsyncSingleExecution,
QueueTask,
} = decorators.function;
class DataService {
// Cache async results until promise resolves
@CacheUntilResolved
async fetchData() {
return await api.getData();
}
// Debounce method calls
@Debounce(500)
onSearchInput(query: string) {
this.performSearch(query);
}
// Throttle method calls
@Throttle(1000)
onScroll() {
this.loadMoreData();
}
// Memoize synchronous results
@Memoize
calculateTotal(items: Item[]) {
return items.reduce((sum, item) => sum + item.price, 0);
}
// Memoize async results
@MemoizeAsync
async fetchUserData(userId: string) {
return await api.getUser(userId);
}
// Retry failed async operations
@RetryAsync(3, 1000) // 3 retries, 1000ms delay
async unreliableOperation() {
return await api.flakeyEndpoint();
}
// Ensure only one execution at a time
@AsyncSingleExecution
async saveData() {
return await api.save();
}
// Queue tasks for sequential execution
@QueueTask
async processItem(item: Item) {
return await api.process(item);
}
}
Class Decorators
import { decorators } from '@elliemae/pui-app-sdk';
const { Singleton, Mixins } = decorators.class;
// Ensure only one instance exists
@Singleton
class ConfigService {
private config: Config;
constructor() {
this.config = loadConfig();
}
}
// Mix multiple classes
@Mixins(LoggingMixin, CachingMixin)
class DataManager {
// Inherits methods from both mixins
}
Advanced State Selectors
Using useStateSelector
Select specific fields from state with better performance:
import {
useStateSelector,
useStateSelectorShallow,
getSelectField,
} from '@elliemae/pui-app-sdk';
// In your slice file
export const selectUserState = getSelectField('user');
// In your component
function UserProfile() {
// Select single field
const userName = useStateSelector(selectUserState, 'profile.name', {
defaultValue: 'Guest',
});
// Select multiple fields
const [email, phone, address] = useStateSelector(
selectUserState,
['profile.email', 'profile.phone', 'profile.address'],
{ defaultValue: ['', '', ''] },
);
// Select nested objects
const settings = useStateSelector(selectUserState, 'preferences.settings', {
defaultValue: {},
});
return <div>{userName}</div>;
}
// Use shallow equality for better performance
function UserSettings() {
const settings = useStateSelectorShallow(selectUserState, 'preferences', {
defaultValue: {},
});
return <div>{/* render settings */}</div>;
}
Responsive Design
Media Query Hook
Respond to media query changes:
import { useMediaQueryList } from '@elliemae/pui-app-sdk';
function ResponsiveComponent() {
const deviceType = useMediaQueryList(
[
'(max-width: 767px)',
'(min-width: 768px) and (max-width: 1023px)',
'(min-width: 1024px)',
],
['mobile', 'tablet', 'desktop'],
'desktop', // default value
);
return (
<div>
{deviceType === 'mobile' && <MobileLayout />}
{deviceType === 'tablet' && <TabletLayout />}
{deviceType === 'desktop' && <DesktopLayout />}
</div>
);
}
Environment & URL Utilities
Environment Detection
import { isProd, isCIBuild } from '@elliemae/pui-app-sdk';
if (isProd()) {
// NODE_ENV === 'production'
}
if (isCIBuild()) {
// CI === 'true' (running in CI pipeline)
}
History Instances
The SDK exports two pre-created history instances:
import { history, memoryHistory } from '@elliemae/pui-app-sdk';
// history — browser history, used by default for routing
// memoryHistory — in-memory history, useful for testing or non-browser environments
Use memoryHistory in tests to avoid browser URL side-effects:
import { memoryHistory } from '@elliemae/pui-app-sdk';
memoryHistory.push('/test-route');
removeDoubleSlash
Collapse accidental double slashes in URLs (preserving the protocol ://):
import { removeDoubleSlash } from '@elliemae/pui-app-sdk';
removeDoubleSlash('https://example.com//api//v1');
// → 'https://example.com/api/v1'
Themes
Available theme constants:
import { Themes } from '@elliemae/pui-app-sdk';
// Themes.EM → 'em'
// Themes.CORP_COOL → 'corpcool'
Security Utilities
PII Redaction
Automatically redact personally identifiable information from logs:
import { redactPii } from '@elliemae/pui-app-sdk';
const sensitiveData = {
name: 'John Doe',
email: 'john@example.com',
phone: '555-123-4567',
ssn: '123-45-6789',
creditCard: '4532-1234-5678-9010',
address: '123 Main St, Apt 4B',
zipcode: '12345-6789',
};
const redacted = redactPii(sensitiveData);
console.log(redacted);
// {
// name: 'John Doe',
// email: '****',
// phone: '****',
// ssn: '****',
// creditCard: '****',
// address: '****',
// zipcode: '****',
// }
// Use when logging errors or data
logger.error(redactPii({ message: 'Error', data: sensitiveData }));
Listener Middleware
Use Redux Toolkit listener middleware for side effects:
import {
startSideEffect,
createSideEffect,
removeSideEffect,
clearSideEffects,
} from '@elliemae/pui-app-sdk';
import { resources } from './data/resources';
// Start listening for actions
startSideEffect({
actionCreator: resources.get,
effect: async (action, listenerApi) => {
// Access state
const state = listenerApi.getState();
// Call API
const data = await fetchResources();
// Dispatch success action
listenerApi.dispatch(resources.getSuccess(data));
},
});
// Create a reusable listener
const listener = createSideEffect({
actionCreator: resources.create,
effect: async (action, listenerApi) => {
await createResource(action.payload);
listenerApi.dispatch(resources.get());
},
});
// Remove a listener
removeSideEffect(listener);
// Clear all listeners
clearSideEffects();
Session Management
Session Timeout Handling
The SDK provides comprehensive session timeout management with warning and expiry callbacks:
import {
subscribeToSessionExpiryWarning,
subscribeToSessionExpiry,
subscribeToResetSession,
resetUserIdleTime,
trackActivity,
} from '@elliemae/pui-app-sdk';
// Subscribe to session expiry warning (shown before session expires)
subscribeToSessionExpiryWarning((warningNotifiedAt) => {
console.log('Session will expire soon!', warningNotifiedAt);
// Show warning modal to user
});
// Subscribe to session expiry event
subscribeToSessionExpiry(() => {
console.log('Session has expired');
// Redirect to login or handle session expiry
});
// Subscribe to session reset events
subscribeToResetSession((resetWarningModal) => {
if (resetWarningModal) {
// Close warning modal if it's open
}
});
// Manually reset user idle time (e.g., on user activity)
resetUserIdleTime(true); // Pass true to reset warning modal
// Track user activity (automatically resets idle timer)
trackActivity();
Session UI Components
SessionTimeout
A self-contained component that monitors idle time, displays a session-expiry warning modal, and dispatches logout.confirm() when the session expires. Place it once in your app root:
import { SessionTimeout } from '@elliemae/pui-app-sdk';
const Application = () => (
<AppRoot store={store} history={history} theme={theme} manageSession>
<SessionTimeout />
<AppRoutes />
</AppRoot>
);
AppRootwithmanageSessionrendersSessionTimeoutautomatically. Add it manually only when assembling your own root layout.
WaitMessage
A full-screen loading indicator controlled by waitMessageAction Redux state:
import { WaitMessage } from '@elliemae/pui-app-sdk';
// Typically placed in AppRoot layout — shown/hidden via Redux
<WaitMessage size="xl" color="light" showText />;
The WaitMessage component reads its open/close state from Redux. Dispatch waitMessageAction.open() / waitMessageAction.close() to control it (see Wait Messages).
Initialize Session Management
import { listenStorageEvents, getAppConfigValue } from '@elliemae/pui-app-sdk';
import {
Environment,
Logger,
RuntimeLoggerOptions,
} from '@elliemae/pui-diagnostics';
// Initialize session management
listenStorageEvents();
// Configure logger with session data
export const onInit: OnInitCallback = ({ history, homeRoute }) => {
const sessionData = {
environment: getAppConfigValue<Environment>('activeEnv'),
appVersion: window.emui.version,
instanceId: sessionStorage.getItem('instanceId') || '',
userId: sessionStorage.getItem('userId') || '',
};
(logger as Logger).setOptions?.(sessionData as RuntimeLoggerOptions);
logger.info('Application initialized');
// ... rest of initialization
};
LogRocket Integration
Initialization
LogRocket is initialized by the host/loader layer (e.g. pui-app-loader, encw-loader) via @elliemae/pui-logrocket. Micro-apps do not need to call initLogRocket() themselves.
// Done by the loader — not by individual micro-apps
import { initLogRocket } from '@elliemae/pui-logrocket';
if (window.emui?.logRocketConfig?.appId) {
initLogRocket();
}
User Identification
The SDK automatically identifies users to LogRocket during the OAuth authorize() flow:
authorize() → getToken → introspectToken → sessionStorage.setItem('user', ...)
→ lrIdentify(userKey, { name, instanceId })
→ markLrIdentified(userKey) ← prevents re-identification
If LogRocket is not yet loaded when authorize() runs (common in encw-loader), the lrIdentify call is a no-op. The safety net is ensureLrIdentified inside @elliemae/pui-analytics-so — it automatically identifies the user from sessionStorage on the first lrTrackGuarded call after LogRocket becomes available.
Covered scenarios:
| Scenario | How identified |
|---|---|
| Fresh OAuth login | lrIdentify() in authorize() |
| Session restore (page refresh) | ensureLrIdentified reads user from sessionStorage on first track |
| Late LogRocket load | ensureLrIdentified fires when LR becomes available |
| Logout + new user login | resetLrIdentity() on logout clears state; new authorize() re-identifies |
| Duplicate tab | Fresh JS context; ensureLrIdentified reads from copied sessionStorage |
Tracking Events
Use lrTrack from the SDK's logrocket utility for custom LogRocket events. It delegates to lrTrackGuarded from @elliemae/pui-analytics-so which provides:
- Event buffering — events fired before LogRocket loads are queued (up to 20) and replayed once available
- Sliding-window throttle — max 3 identical event+payload calls per second
- Property-count warning — warns if more than 2 properties per event
import { lrTrack } from '../utils/logrocket';
// Custom event tracking
lrTrack('document-saved', { docId: '123' });
// Events before LR loads are queued and replayed later
lrTrack('login', { name: 'Jane', instanceId: 'inst-1' });
For business analytics events that should reach both GTM and LogRocket, use Analytics.sendBAEvent() via the scripting object — it calls lrTrackGuarded internally.
Logout
The endSession function handles LogRocket cleanup:
// Inside endSession() — already wired by the SDK
lrTrack('logout');
resetLrIdentity(); // clears identify state + pending event queue
sessionStorage.clear();
resetLrIdentity() ensures that:
- The next session starts with a fresh identify state
- Any queued events from the previous session are discarded
Service Worker
Initialize service worker for your app:
import { initServiceWorker } from '@elliemae/pui-app-sdk';
import { getBasePath } from './utils/paths';
// Initialize after creating CMicroApp instance
app = CMicroApp.getInstance({
logger,
onInit,
onMount,
onUnmount,
});
// Initialize service worker
initServiceWorker(getBasePath());
App Configuration
Loading Configuration
Load configuration from a JSON file:
import {
loadAppConfig,
setAppConfig,
getAppConfigValue,
} from '@elliemae/pui-app-sdk';
// Load from JSON file
await loadAppConfig('/path/to/app.config.json');
// Set configuration programmatically
setAppConfig({
appId: 'my-app',
activeEnv: 'production',
serviceEndpoints: {
api: 'https://api.example.com',
},
});
// Get specific config value
const apiEndpoint = getAppConfigValue('serviceEndpoints.api');
const appId = getAppConfigValue('appId');
Update Configuration Values
import { setAppConfigValue } from '@elliemae/pui-app-sdk';
// Update individual config value
setAppConfigValue('activeEnv', 'staging');
setAppConfigValue('serviceEndpoints.api', 'https://staging-api.example.com');
Web Storage Events
Listen to storage events across tabs/windows:
import {
listenStorageEvents,
removeStorageEvents,
} from '@elliemae/pui-app-sdk';
// Start listening to storage events
listenStorageEvents();
// Stop listening
removeStorageEvents();
Micro-Frontend Communication
Host App Data Exchange
Exchange data between host and guest apps:
import {
setHostAppData,
getHostAppDataByKey,
fetchHostAppData,
sendMessageToHost,
} from '@elliemae/pui-app-sdk';
// In host app: Set data for guest apps
setHostAppData({ userId: '123', theme: 'dark' });
// In guest app: Get host data by key
const userId = getHostAppDataByKey('userId');
// Fetch all host data
const hostData = await fetchHostAppData();
// Send message to host app
sendMessageToHost({ type: 'UPDATE_THEME', payload: 'light' });
Guest Unload Handlers
Handle guest app unload events:
import {
onGuestUnloadStart,
notifyGuestUnloadComplete,
} from '@elliemae/pui-app-sdk';
// Register unload start handler
onGuestUnloadStart(() => {
console.log('Guest app is unloading');
// Cleanup resources
});
// Notify when unload is complete
notifyGuestUnloadComplete();
Render with Host Data
Render component with host app data:
import { renderWithHostData } from '@elliemae/pui-app-sdk';
renderWithHostData(<App />, {
hostData: { userId: '123' },
containerId: 'root',
});
Authentication Utilities
Auth Helper Functions
import {
isUserAuthorized,
login,
authorize,
endSession,
getAuthorizationHeader,
setLoginParams,
} from '@elliemae/pui-app-sdk';
// Check if user is authorized
if (isUserAuthorized()) {
// User is logged in
}
// Get authorization header for API calls
const authHeader = getAuthorizationHeader();
// Returns: 'Bearer <token>'
// Set login parameters
setLoginParams({
clientId: 'your-client-id',
redirectUri: 'https://your-app.com/callback',
scope: 'openid profile',
});
// Trigger login
await login();
// Trigger authorization
await authorize();
// End session (logout)
await endSession({
clientId: 'your-client-id',
redirectUri: 'https://your-app.com',
});
Auth Subpath Exports
The @elliemae/pui-app-sdk/auth subpath exports additional low-level functions not available on the root package:
import {
getIDPInfoFromUrl,
navigateToLoginPage,
} from '@elliemae/pui-app-sdk/auth';
getIDPInfoFromUrl
Extracts the authorization code and error_code from the current URL (set by the IDP after redirect):
const { code, idpErrorCode, redirectUri } = getIDPInfoFromUrl();
if (code) {
await authorize({ code, redirectUri, ... });
}
navigateToLoginPage
Redirects the browser to the IDP login page with the specified OAuth parameters:
navigateToLoginPage({
clientId: 'my-app',
redirectUri: window.location.href,
idpErrorCode: '',
scope: 'openid',
responseType: 'code',
instanceId: 12345, // optional
siteId: 1, // optional
userId: 'jdoe', // optional
disableRememberMe: true, // optional
useCompactIdpPage: false, // optional
isSSO: false, // optional
});
In most cases, use
login()from the root package — it callsgetIDPInfoFromUrlandnavigateToLoginPage/authorizeinternally.
Require Auth Component
Protect routes requiring authentication:
import { RequireAuth } from '@elliemae/pui-app-sdk';
function App() {
return (
<RequireAuth>
<ProtectedContent />
</RequireAuth>
);
}
Utility Hooks
useInjectQuery
Inject RTK Query endpoints dynamically:
import { useInjectQuery } from '@elliemae/pui-app-sdk';
import { api } from './api';
function MyComponent() {
useInjectQuery({ key: 'users', endpoint: api.endpoints.getUsers });
// Use the endpoint
const { data } = api.endpoints.getUsers.useQuery();
return <div>{/* render data */}</div>;
}
useInjectSideEffect
Inject side effects dynamically:
import { useInjectSideEffect } from '@elliemae/pui-app-sdk';
function MyComponent() {
useInjectSideEffect({
key: 'analytics',
effect: (action, listenerApi) => {
// Track analytics
},
});
return <div>{/* component */}</div>;
}
useAppMiddleware
Access Redux middleware dynamically:
import { useAppMiddleware } from '@elliemae/pui-app-sdk';
function MyComponent() {
const middleware = useAppMiddleware();
// Use middleware
return <div>{/* component */}</div>;
}
Form Components
The SDK provides form field components with React Hook Form integration:
Available Form Components
import {
Form,
TextBox,
InputText,
LargeTextBox,
InputMask,
MaskedInputText,
ComboBox,
ComboBoxV3,
CheckBox,
Radio,
RadioGroup,
DateInput,
DatePicker,
DateTimePicker,
DateRangePicker,
Autocomplete,
Toggle,
FormSubmitButton,
FormItemLayout,
FormLayoutBlockItem,
ConnectForm,
} from '@elliemae/pui-app-sdk';
TextBox
<TextBox
name="firstName"
label="First Name"
placeholder="Enter first name"
rules={{ required: 'First name is required' }}
/>
InputMask
import { InputMask, MASK_TYPES, MASK_PIPES } from '@elliemae/pui-app-sdk';
<InputMask
name="phone"
label="Phone Number"
mask={MASK_TYPES.PHONE}
pipe={MASK_PIPES.PHONE}
/>;
ComboBox
<ComboBox
name="country"
label="Country"
options={[
{ value: 'us', label: 'United States' },
{ value: 'ca', label: 'Canada' },
]}
rules={{ required: 'Please select a country' }}
/>
DatePicker
<DatePicker
name="birthDate"
label="Date of Birth"
rules={{ required: 'Date of birth is required' }}
/>
DateRangePicker
<DateRangePicker
name="dateRange"
label="Select Date Range"
startLabel="Start Date"
endLabel="End Date"
/>
CheckBox
<CheckBox
name="agreeToTerms"
label="I agree to the terms and conditions"
rules={{ required: 'You must agree to continue' }}
/>
RadioGroup
<RadioGroup name="paymentMethod" label="Payment Method">
<Radio value="credit" label="Credit Card" />
<Radio value="debit" label="Debit Card" />
<Radio value="paypal" label="PayPal" />
</RadioGroup>
Toggle
<Toggle name="notifications" label="Enable Notifications" />
Autocomplete
<Autocomplete
name="city"
label="City"
options={cities}
onInputChange={(value) => fetchCities(value)}
/>
Form Layout
<Form onSubmit={handleSubmit}>
<FormItemLayout>
<TextBox name="firstName" label="First Name" />
<TextBox name="lastName" label="Last Name" />
</FormItemLayout>
<FormLayoutBlockItem>
<LargeTextBox name="comments" label="Comments" rows={4} />
</FormLayoutBlockItem>
<FormSubmitButton>Submit</FormSubmitButton>
</Form>
Storybook Integration
Decorate Storybook stories with app context:
import { withAppDecorator } from '@elliemae/pui-app-sdk';
export default {
title: 'Components/MyComponent',
component: MyComponent,
decorators: [withAppDecorator],
};
Utility Components
VisuallyHidden
Hide content visually but keep it accessible to screen readers:
import { VisuallyHidden } from '@elliemae/pui-app-sdk';
<VisuallyHidden>
<label htmlFor="search">Search</label>
</VisuallyHidden>;
Page Component
Set page title and metadata:
import { Page } from '@elliemae/pui-app-sdk';
<Page pageTitle="Dashboard">
<DashboardContent />
</Page>;
Testing
Testing with Redux
The SDK provides testing helpers for components with Redux:
import {
renderWithRouterRedux,
renderWithRedux,
renderWithRouter,
} from '@elliemae/pui-app-sdk';
import { screen } from '@testing-library/react';
describe('MyComponent', () => {
it('renders with Redux state', () => {
renderWithRedux(<MyComponent />, {
initialState: { user: { name: 'John' } },
});
expect(screen.getByText('John')).toBeInTheDocument();
});
it('renders with Router and Redux', () => {
renderWithRouterRedux(<MyComponent />, {
initialState: { resources: [] },
route: '/resources',
});
expect(screen.getByTestId('resource-list')).toBeInTheDocument();
});
});
Testing Library Re-exports
The SDK re-exports common @testing-library/react utilities so you can import everything from one place:
import {
render, // custom render wrapped with AppRoot + theme + store
screen,
waitFor,
waitForElementToBeRemoved,
within,
fireEvent,
act,
cleanup,
renderHook,
} from '@elliemae/pui-app-sdk';
The custom render wraps components in AppRoot with a default store and theme. For more control, use renderWithRedux or renderWithRouterRedux.
RenderWithStateAddOns
A component that injects reducers, sagas, and theme in one shot — useful in Storybook stories or integration tests:
import { RenderWithStateAddOns } from '@elliemae/pui-app-sdk';
<RenderWithStateAddOns
Component={MyComponent}
reducer={{ key: 'resources', reducer: resourcesReducer }}
saga={{ key: 'resources', saga: resourcesSaga }}
theme={myTheme}
someProp="value"
/>;
Accepts single or array values for reducer and saga:
<RenderWithStateAddOns
Component={Dashboard}
reducer={[
{ key: 'resources', reducer: resourcesReducer },
{ key: 'users', reducer: usersReducer },
]}
saga={[
{ key: 'resources', saga: resourcesSaga },
{ key: 'users', saga: usersSaga },
]}
/>
Common Patterns
Error Boundaries
import { ErrorBoundary, ErrorToast } from '@elliemae/pui-app-sdk';
function App() {
return (
<ErrorBoundary>
<ErrorToast />
<YourApp />
</ErrorBoundary>
);
}