{"pageProps":{"routes":[{"title":"Documentation","heading":true,"routes":[{"title":"Getting Started","path":"/docs/getting-started.md"},{"title":"Basic Features","open":true,"routes":[{"title":"Pages","path":"/docs/basic-features/pages.md"},{"title":"Data fetching","path":"/docs/basic-features/data-fetching.md"},{"title":"Built-in CSS Support","path":"/docs/basic-features/built-in-css-support.md"},{"title":"Static File Serving","path":"/docs/basic-features/static-file-serving.md"},{"title":"TypeScript","path":"/docs/basic-features/typescript.md"},{"title":"Environment Variables","path":"/docs/basic-features/environment-variables.md"}]},{"title":"Routing","routes":[{"title":"Introduction","path":"/docs/routing/introduction.md"},{"title":"Dynamic Routes","path":"/docs/routing/dynamic-routes.md"},{"title":"Imperatively","path":"/docs/routing/imperatively.md"},{"title":"Shallow Routing","path":"/docs/routing/shallow-routing.md"}]},{"title":"API Routes","routes":[{"title":"Introduction","path":"/docs/api-routes/introduction.md"},{"title":"Dynamic API Routes","path":"/docs/api-routes/dynamic-api-routes.md"},{"title":"API Middlewares","path":"/docs/api-routes/api-middlewares.md"},{"title":"Response Helpers","path":"/docs/api-routes/response-helpers.md"}]},{"title":"Deployment","path":"/docs/deployment.md"},{"title":"Advanced Features","routes":[{"title":"Preview Mode","path":"/docs/advanced-features/preview-mode.md"},{"title":"Dynamic Import","path":"/docs/advanced-features/dynamic-import.md"},{"title":"Automatic Static Optimization","path":"/docs/advanced-features/automatic-static-optimization.md"},{"title":"Static HTML Export","path":"/docs/advanced-features/static-html-export.md"},{"title":"Absolute Imports and Module Path Aliases","path":"/docs/advanced-features/module-path-aliases.md"},{"title":"AMP Support","routes":[{"title":"Introduction","path":"/docs/advanced-features/amp-support/introduction.md"},{"title":"Adding AMP Components","path":"/docs/advanced-features/amp-support/adding-amp-components.md"},{"title":"AMP Validation","path":"/docs/advanced-features/amp-support/amp-validation.md"},{"title":"AMP in Static HTML export","path":"/docs/advanced-features/amp-support/amp-in-static-html-export.md"},{"title":"TypeScript","path":"/docs/advanced-features/amp-support/typescript.md"}]},{"title":"Customizing Babel Config","path":"/docs/advanced-features/customizing-babel-config.md"},{"title":"Customizing PostCSS Config","path":"/docs/advanced-features/customizing-postcss-config.md"},{"title":"Custom Server","path":"/docs/advanced-features/custom-server.md"},{"title":"Custom `App`","path":"/docs/advanced-features/custom-app.md"},{"title":"Custom `Document`","path":"/docs/advanced-features/custom-document.md"},{"title":"Custom Error Page","path":"/docs/advanced-features/custom-error-page.md"},{"title":"`src` Directory","path":"/docs/advanced-features/src-directory.md"},{"title":"Multi Zones","path":"/docs/advanced-features/multi-zones.md"},{"title":"Measuring performance","path":"/docs/advanced-features/measuring-performance.md"}]},{"title":"Upgrade Guide","path":"/docs/upgrading.md"},{"title":"FAQ","path":"/docs/faq.md"}]},{"title":"API Reference","heading":true,"routes":[{"title":"CLI","path":"/docs/api-reference/cli.md"},{"title":"next/router","path":"/docs/api-reference/next/router.md"},{"title":"next/link","path":"/docs/api-reference/next/link.md"},{"title":"next/head","path":"/docs/api-reference/next/head.md"},{"title":"next/amp","path":"/docs/api-reference/next/amp.md"},{"title":"Data Fetching","routes":[{"title":"getInitialProps","path":"/docs/api-reference/data-fetching/getInitialProps.md"}]},{"title":"next.config.js","routes":[{"title":"Introduction","path":"/docs/api-reference/next.config.js/introduction.md"},{"title":"Environment Variables","path":"/docs/api-reference/next.config.js/environment-variables.md"},{"title":"Custom Page Extensions","path":"/docs/api-reference/next.config.js/custom-page-extensions.md"},{"title":"CDN Support with Asset Prefix","path":"/docs/api-reference/next.config.js/cdn-support-with-asset-prefix.md"},{"title":"Build Target","path":"/docs/api-reference/next.config.js/build-target.md"},{"title":"Custom Webpack Config","path":"/docs/api-reference/next.config.js/custom-webpack-config.md"},{"title":"Compression","path":"/docs/api-reference/next.config.js/compression.md"},{"title":"Static Optimization Indicator","path":"/docs/api-reference/next.config.js/static-optimization-indicator.md"},{"title":"Runtime Configuration","path":"/docs/api-reference/next.config.js/runtime-configuration.md"},{"title":"Disabling x-powered-by","path":"/docs/api-reference/next.config.js/disabling-x-powered-by.md"},{"title":"Disabling ETag Generation","path":"/docs/api-reference/next.config.js/disabling-etag-generation.md"},{"title":"Setting a custom build directory","path":"/docs/api-reference/next.config.js/setting-a-custom-build-directory.md"},{"title":"Configuring the Build ID","path":"/docs/api-reference/next.config.js/configuring-the-build-id.md"},{"title":"Configuring onDemandEntries","path":"/docs/api-reference/next.config.js/configuring-onDemandEntries.md"},{"title":"Ignoring TypeScript Errors","path":"/docs/api-reference/next.config.js/ignoring-typescript-errors.md"},{"title":"exportPathMap","path":"/docs/api-reference/next.config.js/exportPathMap.md"}]}]}],"data":{"description":"Measure and track page performance using Next.js's build-in performance relayer"},"route":{"title":"Measuring performance","path":"/docs/advanced-features/measuring-performance.md"},"html":"
Next.js has a built-in relayer that allows you to analyze and measure the performance of\npages using different metrics.
\nTo measure any of the supported metrics, you will need to create a custom\nApp component and define a reportWebVitals
function:
// pages/_app.js\nexport function reportWebVitals(metric) {\n console.log(metric)\n}\n\nfunction MyApp({ Component, pageProps }) {\n return <Component {...pageProps} />\n}\n\nexport default MyApp\n
\nThis function is fired when the final values for any of the metrics have finished calculating on\nthe page. You can use to log any of the results to the console or send to a particular endpoint.
\nThe metric
object returned to the function consists of a number of properties:
id
: Unique identifier for the metric in the context of the current page loadname
: Metric namestartTime
: First recorded timestamp of the performance entry (if applicable)value
: Value, or duration, of performance entrylabel
: Type of metric (web-vital
or custom
)There are two types of metrics that are tracked:
\nWeb Vitals are a set of useful metrics that aim to capture the user\nexperience of a web page. The following web vitals are all included:
\nYou can handle all the results of these metrics using the web-vital
label:
export function reportWebVitals(metric) {\n if (metric.label === 'web-vital') {\n console.log(metric) // The metric object ({ id, name, startTime, value, label }) is logged to the console\n }\n}\n
\nThere's also the option of handling each of the metrics separately:
\nexport function reportWebVitals(metric) {\n switch (metric.name) {\n case 'FCP':\n // handle FCP results\n break\n case 'LCP':\n // handle LCP results\n break\n case 'CLS':\n // handle CLS results\n break\n case 'FID':\n // handle FID results\n break\n case 'TTFB':\n // handle TTFB results\n break\n default:\n break\n }\n}\n
\nA third-party library, web-vitals, is used to measure\nthese metrics. Browser compatibility depends on the particular metric, so refer to the Browser\nSupport section to find out which\nbrowsers are supported.
\nIn addition to the core metrics listed above, there are some additional custom metrics that\nmeasure the time it takes for the page to hydrate and render:
\nNext.js-hydration
: Length of time it takes for the page to start and finish hydrating (in ms)Next.js-route-change-to-render
: Length of time it takes for a page to start rendering after a\nroute change (in ms)Next.js-render
: Length of time it takes for a page to finish render after a route change (in ms)You can handle all the results of these metrics using the custom
label:
export function reportWebVitals(metric) {\n if (metric.label === 'custom') {\n console.log(metric) // The metric object ({ id, name, startTime, value, label }) is logged to the console\n }\n}\n
\nThere's also the option of handling each of the metrics separately:
\nexport function reportWebVitals(metric) {\n switch (metric.name) {\n case 'Next.js-hydration':\n // handle hydration results\n break\n case 'Next.js-route-change-to-render':\n // handle route-change to render results\n break\n case 'Next.js-render':\n // handle render results\n break\n default:\n break\n }\n}\n
\nThese metrics work in all browsers that support the User Timing API.
\nWith the relay function, you can send any of results to an analytics endpoint to measure and track\nreal user performance on your site. For example:
\nexport function reportWebVitals(metric) {\n const body = JSON.stringify(metric)\n const url = 'https://example.com/analytics'\n\n // Use `navigator.sendBeacon()` if available, falling back to `fetch()`.\n if (navigator.sendBeacon) {\n navigator.sendBeacon(url, body)\n } else {\n fetch(url, { body, method: 'POST', keepalive: true })\n }\n}\n
\n\n"},"__N_SSG":true}Note: If you use Google Analytics, using the\n
\nid
value can allow you to construct metric distributions manually (to calculate percentiles,\netc...).\nexport function reportWebVitals({ id, name, label, value }) {\n ga('send', 'event', {\n eventCategory:\n label === 'web-vital' ? 'Web Vitals' : 'Next.js custom metric',\n eventAction: name,\n eventValue: Math.round(name === 'CLS' ? value * 1000 : value), // values must be integers\n eventLabel: id, // id unique to current page load\n nonInteraction: true, // avoids affecting bounce rate.\n })\n}\n
Read more about sending results to Google Analytics here.
\n