Contentful Personalization & Analytics
    Preparing search index...

    Contentful Logo

    Contentful Personalization & Analytics

    Optimization React Web SDK

    Guides · Reference · Contributing

    Warning

    The Optimization SDK Suite is pre-release (alpha). Breaking changes can be published at any time.

    The Optimization React Web SDK provides React providers, hooks, router adapters, and entry-rendering primitives on top of the Optimization Web SDK. Use it when a React browser application must not manage the lower-level Web SDK instance, state subscriptions, entry resolution, and route tracking by hand.

    If you are integrating a React application, start with Getting Started, then use Integrating the Optimization React Web SDK in a React app for the step-by-step flow. This README keeps the package orientation and common setup options close at hand; generated reference documentation remains the source of truth for exported API signatures.

    Table of Contents

    Install using an NPM-compatible package manager, pnpm for example:

    pnpm install @contentful/optimization-react-web
    

    React and React DOM are application-owned peer dependencies. The SDK uses the React runtime already installed by your app instead of installing its own copy.

    Mount OptimizationRoot once near the root of your React application:

    import { OptimizationRoot } from '@contentful/optimization-react-web'

    function App() {
    return (
    <OptimizationRoot clientId="your-client-id" environment="main">
    <YourApp />
    </OptimizationRoot>
    )
    }

    For a single-locale app that fetches Contentful entries, pass the application locale to the SDK when Experience API responses and events need to use the same language:

    <OptimizationRoot clientId="your-client-id" environment="main" locale="en-US">
    <YourApp />
    </OptimizationRoot>

    Use @contentful/optimization-react-web for React browser applications that need provider-based SDK initialization, hooks, router page tracking, optimized entry rendering, automatic interaction tracking, and live update semantics. Use the lower-level Web SDK directly for non-React integrations or custom framework adapters.

    OptimizationRoot accepts most Web SDK configuration props directly and adds React-specific props such as liveUpdates, onStatesReady, and serverOptimizationState. The Web SDK autoTrackEntryInteraction option is exposed as the React trackEntryInteraction prop.

    Prop Required? Default Description
    clientId Yes N/A Shared API key for Experience API and Insights API requests
    environment No 'main' Contentful environment identifier
    api No Web SDK defaults Experience API and Insights API endpoint and request options
    app No undefined Application metadata attached to outgoing event context
    locale No undefined SDK Experience API and default event locale
    defaults No undefined Configuration/default state such as consent or persistence consent
    serverOptimizationState No undefined Server-returned Optimization state to apply before provider children mount
    allowedEventTypes No ['identify', 'page'] Event types allowed before consent is explicitly set
    trackEntryInteraction No { views: true, clicks: true, hovers: true } Automatic entry interaction tracking for OptimizedEntry elements
    cookie No { domain: undefined, expires: 365 } Anonymous ID cookie settings inherited from the Web SDK
    liveUpdates No false Whether OptimizedEntry components react continuously to SDK state
    onStatesReady No undefined Provider-managed app-level state subscription hook
    queuePolicy No SDK defaults Flush retry behavior and offline queue bounds
    logLevel No 'error' Minimum log level for the default console sink
    onEventBlocked No undefined Callback invoked when consent or guard logic blocks an event

    Use OptimizationProvider directly when an application or framework adapter needs direct provider control, including integrations that supply an SDK instance:

    <OptimizationProvider sdk={optimization}>
    <YourApp />
    </OptimizationProvider>

    Injected SDK instances render children immediately unless state setup such as onStatesReady or serverOptimizationState is provided. When state setup is provided, the provider waits until that setup is complete before children mount, and still leaves SDK teardown to the owner that created the instance.

    For server-to-browser state handoff, pass server-returned Optimization data through serverOptimizationState on OptimizationRoot or OptimizationProvider. Keep defaults for configuration or default state such as consent policy:

    <OptimizationRoot
    clientId="your-client-id"
    defaults={{ consent: true }}
    environment="main"
    serverOptimizationState={optimizationData}
    >
    <YourApp />
    </OptimizationRoot>

    For every Web SDK option that passes through this package, use the Web SDK README and generated reference documentation.

    Choose the application Contentful locale in your router, i18n layer, or app configuration. Pass that value directly to Contentful CDA requests, and pass the same value to the provider locale prop when Experience API responses and event context need to use the same language. See Locale handling in the Optimization SDK Suite for the full locale model.

    For SDK instances created from provider/root configuration, changing the locale prop calls sdk.setLocale() after initialization while the rest of the SDK config remains initialization-scoped. Locale updates do not fetch content or refresh profile state; trigger your app's normal page(), identify(), route loader, or CDA fetch flow when localized data needs to change.

    Consent policy remains application-owned. For default-on application policies that do not render an end-user consent UI, seed accepted consent on OptimizationRoot:

    <OptimizationRoot clientId="your-client-id" defaults={{ consent: true }}>
    <YourApp />
    </OptimizationRoot>

    When application policy depends on user choice, leave defaults.consent unset and call consent() from useOptimizationActions() in the relevant control:

    import { useOptimizationActions } from '@contentful/optimization-react-web'

    function ConsentButton() {
    const { consent } = useOptimizationActions()
    return <button onClick={() => consent(true)}>Accept</button>
    }

    Boolean consent calls control both event emission and durable profile-continuity persistence by default. Use sdk.consent({ events: true, persistence: false }) when events are allowed but continuity needs to stay session-only. For cross-SDK consent guidance, see Consent management in the Optimization SDK Suite.

    OptimizationRoot owns the Web SDK lifecycle. Provider-owned initialization runs after React commit, outside render, and renders no children while the SDK is pending. In normal browser rendering this uses a layout-effect path so ready children can mount before the first visible paint.

    Use the dedicated React SDK action hooks when components need common Optimization actions:

    import { useOptimizationActions } from '@contentful/optimization-react-web'

    function ProductCta() {
    const { track } = useOptimizationActions()

    return <button onClick={() => track({ event: 'purchase' })}>Buy now</button>
    }

    Use useOptimization() when a component needs direct access to the SDK instance itself, and prefer useOptimizationActions() when a component wants destructurable action methods such as track(), identify(), page(), screen(), flush(), reset(), or consent().

    Use dedicated state hooks such as useConsentState(), useProfileState(), and useSelectedOptimizationsState() when components need to render current SDK state. Prefer those hooks over subscribing to sdk.states.* directly from component effects.

    Use useEntryResolver() when a component needs manual entry resolution without the OptimizedEntry wrapper:

    useOptimization() returns the SDK instance itself. Keep that instance in a variable and call methods from it. Do not destructure SDK methods from the returned value because those methods rely on the instance this binding.

    import { useOptimization } from '@contentful/optimization-react-web'

    function ProductCta() {
    const optimization = useOptimization()

    return <button onClick={() => optimization.track({ event: 'purchase' })}>Buy now</button>
    }

    The direct SDK surface also exposes manual interaction calls such as trackView(), trackClick(), trackHover(), and trackFlagView().

    // Avoid destructuring SDK methods; this loses the instance binding.
    const { track } = useOptimization()
    import { useEntryResolver } from '@contentful/optimization-react-web'

    function HeroEntry({ baselineEntry }) {
    const { resolveEntry } = useEntryResolver()
    const resolvedEntry = resolveEntry(baselineEntry)

    return <HeroCard entry={resolvedEntry} />
    }

    Fetch Contentful entries in the app layer with one CDA locale. For localized apps, configure your application locale and pass it directly before passing entries to OptimizedEntry, useEntryResolver(), or useOptimizedEntry(). Do not pass all-locale CDA responses from withAllLocales or locale=*; these APIs expect direct single-locale field values. See Entry optimization and variant resolution for the entry contract and Locale handling in the Optimization SDK Suite for the broader locale model.

    Use useMergeTagResolver() when a component needs to resolve embedded merge tag entries:

    import { useMergeTagResolver } from '@contentful/optimization-react-web'

    function MergeTagText({ mergeTagEntry }) {
    const { getMergeTagValue } = useMergeTagResolver()

    return <span>{getMergeTagValue(mergeTagEntry) ?? ''}</span>
    }

    If a merge tag references localized profile fields such as location.city or location.country, its resolved value follows the localized profile values returned by the Experience API.

    Use onStatesReady when application code needs to subscribe to SDK state as part of provider initialization. This avoids coordinating with window.contentfulOptimization, which might not exist yet when application code runs or might have already emitted data by the time a later effect subscribes.

    <OptimizationRoot
    clientId="your-client-id"
    onStatesReady={(states) => {
    const subscriptions = [
    states.eventStream.subscribe((event) => {
    if (event) devToolsPanel.logEvent(event)
    }),
    states.blockedEventStream.subscribe((blocked) => {
    if (blocked) devToolsPanel.logBlockedEvent(blocked)
    }),
    ]

    return () => {
    subscriptions.forEach((subscription) => subscription.unsubscribe())
    }
    }}
    >
    <YourApp />
    </OptimizationRoot>

    The callback receives only sdk.states. It runs as soon as the provider has initialized the state surface and before children mount, so subscriptions can observe events emitted by child effects such as router page tracking. For component-local UI state, keep using hooks and React effects under the provider.

    OptimizedEntry resolves a baseline Contentful entry and renders either the selected variant or the baseline entry:

    import { OptimizedEntry } from '@contentful/optimization-react-web'

    function HeroEntry({ baselineEntry }) {
    return (
    <OptimizedEntry baselineEntry={baselineEntry}>
    {(resolvedEntry) => <HeroCard entry={resolvedEntry} />}
    </OptimizedEntry>
    )
    }

    Use loadingFallback, direct children, wrapper props, and nested composition patterns when needed. For optimized entries, the loading phase begins immediately while optimization is unresolved. If the state is still unresolved after 5 seconds, the component reveals baseline content so loading does not persist forever. Without a custom loadingFallback, the wrapper preserves layout by hiding the baseline until that timeout elapses. The React Web guide covers those variants in context.

    Entry interaction tracking

    OptimizedEntry emits the Web SDK's data-ctfl-* tracking attributes for resolved entries. The root config observes views, clicks, and hovers by default; pass false for any interaction type that your application does not want to observe:

    <OptimizationRoot clientId="your-client-id" trackEntryInteraction={{ hovers: false }}>
    <YourApp />
    </OptimizationRoot>

    Use OptimizedEntry props to configure Web SDK entry-tracking attributes without setting data-ctfl-* metadata manually:

    <OptimizedEntry
    baselineEntry={entry}
    clickable
    hoverDurationUpdateIntervalMs={1000}
    viewDurationUpdateIntervalMs={1000}
    >
    {(resolvedEntry) => <HeroCard entry={resolvedEntry} />}
    </OptimizedEntry>

    OptimizedEntry derives entry ID, baseline ID, optimization ID, optimization context ID, sticky state, variant index, and duplication scope from the resolved entry state.

    Use sdk.tracking.enableElement(...) from useOptimization() for manual element overrides.

    Router adapters emit page() events for supported client-side routers:

    Router Import path Mounting rule
    React Router @contentful/optimization-react-web/router/react-router Mount under a React Router data router and inside OptimizationRoot
    Next.js Pages @contentful/optimization-react-web/router/next-pages Mount once in pages/_app.tsx inside OptimizationRoot
    Next.js App Router @contentful/optimization-react-web/router/next-app Mount in app/layout.tsx inside OptimizationRoot
    TanStack Router @contentful/optimization-react-web/router/tanstack-router Mount under the TanStack router tree and inside OptimizationRoot

    For Next.js applications that need the server, client, and request-handler integration together, prefer the @contentful/optimization-nextjs adapter package.

    All adapters support static and dynamic page payload enrichment. See the React Web integration guide for router-specific examples.

    liveUpdates defaults to false, so optimized entries lock to the first resolved value. Set liveUpdates globally or per OptimizedEntry when entries must react to profile, flag, or preview changes:

    <OptimizationRoot clientId="your-client-id" liveUpdates={true}>
    <OptimizedEntry baselineEntry={entry} liveUpdates={false}>
    {(resolvedEntry) => <Card entry={resolvedEntry} />}
    </OptimizedEntry>
    </OptimizationRoot>

    The browser preview panel is provided by @contentful/optimization-web-preview-panel. When the panel is open, live updates are forced on for all OptimizedEntry components so authors can inspect variant changes immediately.

    The React SDK keeps rendering React components. OptimizedEntry does not render through custom elements, and this package does not import or register Web Components.

    Use the optional @contentful/optimization-web/web-components subpath only when a non-React app or a deliberate custom-element island needs vanilla custom elements. Framework wrappers around those elements must assign complex DOM properties such as baselineEntry, defaults, api, sdk, and callbacks after hydration, and listen for entry lifecycle events instead of trying to emulate React render props. See the Web SDK README for raw custom-element and UMD usage.

    The package-local development harness runs from packages/web/frameworks/react-web-sdk/dev/. Launch it from the repo root:

    pnpm --filter @contentful/optimization-react-web dev:launch
    

    Use the harness for package development. Use the reference implementation for end-to-end integration behavior.