",
+ "nodeLabel": "div.css-0 > div.css-0 > div.css-or7k2a > div.css-wp059z"
+ }
+ ]
+ },
+ "guidanceLevel": 3,
+ "replacesAudits": [
+ "largest-contentful-paint-element"
+ ]
+ },
+ "lcp-discovery-insight": {
+ "id": "lcp-discovery-insight",
+ "title": "LCP request discovery",
+ "description": "Optimize LCP by making the LCP image [discoverable](https://web.dev/articles/optimize-lcp#1_eliminate_resource_load_delay) from the HTML immediately, and [avoiding lazy-loading](https://web.dev/articles/lcp-lazy-loading)",
+ "score": 0,
+ "scoreDisplayMode": "numeric",
+ "metricSavings": {
+ "LCP": 0
+ },
+ "details": {
+ "type": "list",
+ "items": [
+ {
+ "type": "checklist",
+ "items": {
+ "priorityHinted": {
+ "label": "fetchpriority=high should be applied",
+ "value": false
+ },
+ "requestDiscoverable": {
+ "label": "Request is discoverable in initial document",
+ "value": false
+ },
+ "eagerlyLoaded": {
+ "label": "lazy load not applied",
+ "value": true
+ }
+ }
+ },
+ {
+ "type": "node",
+ "lhId": "page-0-DIV",
+ "path": "2,HTML,1,BODY,1,DIV,0,DIV,0,DIV,1,DIV,0,DIV,0,DIV,0,DIV",
+ "selector": "div.css-0 > div.css-0 > div.css-or7k2a > div.css-wp059z",
+ "boundingRect": {
+ "top": 65,
+ "bottom": 1965,
+ "left": 206,
+ "right": 412,
+ "width": 206,
+ "height": 1900
+ },
+ "snippet": "
",
+ "nodeLabel": "div.css-0 > div.css-0 > div.css-or7k2a > div.css-wp059z"
+ }
+ ]
+ },
+ "guidanceLevel": 3,
+ "replacesAudits": [
+ "prioritize-lcp-image",
+ "lcp-lazy-loaded"
+ ]
+ },
+ "legacy-javascript-insight": {
+ "id": "legacy-javascript-insight",
+ "title": "Legacy JavaScript",
+ "description": "Polyfills and transforms enable older browsers to use new JavaScript features. However, many aren't necessary for modern browsers. Consider modifying your JavaScript build process to not transpile [Baseline](https://web.dev/articles/baseline-and-polyfills) features, unless you know you must support older browsers. [Learn why most sites can deploy ES6+ code without transpiling](https://philipwalton.com/articles/the-state-of-es5-on-the-web/)",
+ "score": 1,
+ "scoreDisplayMode": "metricSavings",
+ "metricSavings": {
+ "FCP": 0,
+ "LCP": 0
+ },
+ "details": {
+ "type": "table",
+ "headings": [
+ {
+ "key": "url",
+ "valueType": "url",
+ "subItemsHeading": {
+ "key": "location",
+ "valueType": "source-location"
+ },
+ "label": "URL"
+ },
+ {
+ "key": null,
+ "valueType": "code",
+ "subItemsHeading": {
+ "key": "signal"
+ },
+ "label": ""
+ },
+ {
+ "key": "wastedBytes",
+ "valueType": "bytes",
+ "label": "Wasted bytes"
+ }
+ ],
+ "items": [],
+ "debugData": {
+ "type": "debugdata",
+ "wastedBytes": 0
+ }
+ },
+ "guidanceLevel": 2
+ },
+ "modern-http-insight": {
+ "id": "modern-http-insight",
+ "title": "Modern HTTP",
+ "description": "HTTP/2 and HTTP/3 offer many benefits over HTTP/1.1, such as multiplexing. [Learn more about using modern HTTP](https://developer.chrome.com/docs/lighthouse/best-practices/uses-http2/).",
+ "score": 0.5,
+ "scoreDisplayMode": "metricSavings",
+ "metricSavings": {
+ "FCP": 0,
+ "LCP": 0
+ },
+ "details": {
+ "type": "table",
+ "headings": [
+ {
+ "key": "url",
+ "valueType": "url",
+ "label": "URL"
+ },
+ {
+ "key": "protocol",
+ "valueType": "text",
+ "label": "Protocol"
+ }
+ ],
+ "items": [
+ {
+ "url": "https://app.valuefrontier.cn/embed.min.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/chatbot/DwN8qAKtYFQtWskM?",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/64f9f179dbdcd998.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/a01885eb9d0649e5.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/bf38d9b349c92e2b.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/9e90e05c5cca6fcc.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/054994666d6806c5.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/webpack-38776d00203f938f.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/7b1f872c-c7e4e33c66cbdc9b.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/86480-b7209753f46ad59b.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/main-app-376f7cb43c26ed4c.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/77ab3b1e-92323a26522690cf.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/26423-9886dec07285c629.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/25552-24c21834bb9ce7f8.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/37008-dd800aa6e6be46e0.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/78639-954f132e09b0bd1d.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/app/(shareLayout)/layout-c7f89e27cf4215d6.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/1ae6eb87-095b6bb2b10e3fd4.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/bda40ab4-465678c6543fde64.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/fc43f782-c8f021bc75fb0f3a.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/75146d7d-fa11a4a6b704c1e9.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/f707c8ea-09423c24a938b7e9.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/adeb31b9-c8c57fad1a5d9920.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/1471f7b3-e1e02f7c4f787f79.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/05417924-0ecf2eadee09cca3.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/4701-3e6d8f235ac58458.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/42776-f08ceab89e5c9f79.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/67178-7738e8785ac3bf1d.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/82427-2c350ac2f33216e0.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/94575-56ee7d594c07f3ac.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/3365-667dcbd31ae8d940.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/75355-3adda07b8a231ae7.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/62632-8f174dd809645249.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/94486-db318921118f62c7.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/67800-076894cf02c647d3.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/19077-e5953bb35a9231eb.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/7895-7e94e2390e12ae57.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/43853-cd3a8ce8f61ef955.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/2360-5fce6327abc41446.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/83879-ddb8796acc954a33.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/88313-976b1a7475221924.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/72247-0f896dd1b92db30f.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/80172-6a3bd1d0c5fa7c8f.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/7451-86904548ba9339ca.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/57349-e362f628f036d21a.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/12011-b60a150a91df71b2.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/26408-fbf397c3ba35f15f.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/98615-99a419845e7d310e.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/64713-66ed16203b06a50a.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/81709-f648d574ebc3712f.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/29946-4716e565c15a4b42.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/69872-0946d84d22ddfeca.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/1504-d278e1a5d4d3c34c.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/15836-9828ebd31169edd1.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/34876-f521aa67cccbe648.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/59440-5c0ebe08cb85eb15.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/70346-1dd9d11dd0937896.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/2462-613d5ea523a1e5b0.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/62579-0a9996c02bc9fd5a.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/40630-e341f8f01b43f98e.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/29412-f2e28fe1350bde1b.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/21797-e76fe60bb4637732.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/49856-077719ab20996c3f.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/52360-8e10f445240de61a.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/82723-4056891612c8cfbd.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/app/(shareLayout)/chatbot/%5Btoken%5D/page-dcc37d69b6429671.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/0b8e744a-9783aef562d7021e.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/77220-4cbfba5a4b531158.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/chunks/app/layout-fa922e5f2d3ab09d.js",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/220a772cfe3c95f4.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/2f7a6ecf4e344b75.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/2da23e89afd44708.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/c31a5eb4ac1ad018.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/b7247e8b4219ed3e.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/5bb43505df05adfe.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/_next/static/css/a031600822501d72.css",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/console/api/system-features",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/api/webapp/access-mode?appCode=DwN8qAKtYFQtWskM",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/api/passport?",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/api/parameters",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/api/site",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/api/meta",
+ "protocol": "http/1.1"
+ },
+ {
+ "url": "https://app.valuefrontier.cn/logo/logo.svg",
+ "protocol": "http/1.1"
+ }
+ ]
+ },
+ "guidanceLevel": 3
+ },
+ "network-dependency-tree-insight": {
+ "id": "network-dependency-tree-insight",
+ "title": "Network dependency tree",
+ "description": "[Avoid chaining critical requests](https://developer.chrome.com/docs/lighthouse/performance/critical-request-chains) by reducing the length of chains, reducing the download size of resources, or deferring the download of unnecessary resources to improve page load.",
+ "score": 0,
+ "scoreDisplayMode": "numeric",
+ "metricSavings": {
+ "LCP": 0
+ },
+ "details": {
+ "type": "list",
+ "items": [
+ {
+ "type": "list-section",
+ "value": {
+ "type": "network-tree",
+ "chains": {
+ "E16586C47343422178C1012FC3029684": {
+ "url": "http://localhost:3000/home",
+ "navStartToEndTime": 55,
+ "transferSize": 2536,
+ "isLongest": true,
+ "children": {
+ "96081.8": {
+ "url": "http://49.232.185.254:5001/api/auth/session",
+ "navStartToEndTime": 5848,
+ "transferSize": 585,
+ "isLongest": true,
+ "children": {}
+ },
+ "96081.9": {
+ "url": "http://49.232.185.254:5001/api/auth/session",
+ "navStartToEndTime": 4226,
+ "transferSize": 585,
+ "children": {}
+ }
+ }
+ }
+ },
+ "longestChain": {
+ "duration": 5848
+ }
+ }
+ },
+ {
+ "type": "list-section",
+ "title": "Preconnected origins",
+ "description": "[preconnect](https://developer.chrome.com/docs/lighthouse/performance/uses-rel-preconnect/) hints help the browser establish a connection earlier in the page load, saving time when the first request for that origin is made. The following are the origins that the page preconnected to.",
+ "value": {
+ "type": "text",
+ "value": "no origins were preconnected"
+ }
+ },
+ {
+ "type": "list-section",
+ "title": "Preconnect candidates",
+ "description": "Add [preconnect](https://developer.chrome.com/docs/lighthouse/performance/uses-rel-preconnect/) hints to your most important origins, but try to use no more than 4.",
+ "value": {
+ "type": "text",
+ "value": "No additional origins are good candidates for preconnecting"
+ }
+ }
+ ]
+ },
+ "guidanceLevel": 1,
+ "replacesAudits": [
+ "critical-request-chains",
+ "uses-rel-preconnect"
+ ]
+ },
+ "render-blocking-insight": {
+ "id": "render-blocking-insight",
+ "title": "Render blocking requests",
+ "description": "Requests are blocking the page's initial render, which may delay LCP. [Deferring or inlining](https://web.dev/learn/performance/understanding-the-critical-path#render-blocking_resources) can move these network requests out of the critical path.",
+ "score": 1,
+ "scoreDisplayMode": "metricSavings",
+ "metricSavings": {
+ "FCP": 0,
+ "LCP": 0
+ },
+ "details": {
+ "type": "table",
+ "headings": [
+ {
+ "key": "url",
+ "valueType": "url",
+ "label": "URL"
+ },
+ {
+ "key": "totalBytes",
+ "valueType": "bytes",
+ "label": "Transfer Size"
+ },
+ {
+ "key": "wastedMs",
+ "valueType": "timespanMs",
+ "label": "Duration"
+ }
+ ],
+ "items": []
+ },
+ "guidanceLevel": 3,
+ "replacesAudits": [
+ "render-blocking-resources"
+ ]
+ },
+ "third-parties-insight": {
+ "id": "third-parties-insight",
+ "title": "3rd parties",
+ "description": "3rd party code can significantly impact load performance. [Reduce and defer loading of 3rd party code](https://web.dev/articles/optimizing-content-efficiency-loading-third-party-javascript/) to prioritize your page's content.",
+ "score": null,
+ "scoreDisplayMode": "error",
+ "errorMessage": "Maximum call stack size exceeded",
+ "errorStack": "RangeError: Maximum call stack size exceeded\n at getRelatedEvents (file:///usr/local/lib/node_modules/lighthouse/node_modules/@paulirish/trace_engine/models/trace/insights/ThirdParties.js:35:27)\n at Module.generateInsight (file:///usr/local/lib/node_modules/lighthouse/node_modules/@paulirish/trace_engine/models/trace/insights/ThirdParties.js:59:24)\n at #computeInsightSet (file:///usr/local/lib/node_modules/lighthouse/node_modules/@paulirish/trace_engine/models/trace/Processor.js:391:33)\n at #computeInsightsForNavigation (file:///usr/local/lib/node_modules/lighthouse/node_modules/@paulirish/trace_engine/models/trace/Processor.js:533:32)\n at #computeInsights (file:///usr/local/lib/node_modules/lighthouse/node_modules/@paulirish/trace_engine/models/trace/Processor.js:467:47)\n at TraceProcessor.parse (file:///usr/local/lib/node_modules/lighthouse/node_modules/@paulirish/trace_engine/models/trace/Processor.js:122:38)\n at async TraceEngineResult.runTraceEngine (file:///usr/local/lib/node_modules/lighthouse/core/computed/trace-engine-result.js:42:5)\n at async TraceEngineResult.compute_ (file:///usr/local/lib/node_modules/lighthouse/core/computed/trace-engine-result.js:254:7)",
+ "guidanceLevel": 3,
+ "replacesAudits": [
+ "third-party-summary"
+ ]
+ },
+ "viewport-insight": {
+ "id": "viewport-insight",
+ "title": "Optimize viewport for mobile",
+ "description": "Tap interactions may be [delayed by up to 300 ms](https://developer.chrome.com/blog/300ms-tap-delay-gone-away/) if the viewport is not optimized for mobile.",
+ "score": 1,
+ "scoreDisplayMode": "numeric",
+ "metricSavings": {
+ "INP": 0
+ },
+ "details": {
+ "type": "table",
+ "headings": [
+ {
+ "key": "node",
+ "valueType": "node",
+ "label": ""
+ }
+ ],
+ "items": [
+ {
+ "node": {
+ "type": "node",
+ "lhId": "page-2-META",
+ "path": "2,HTML,0,HEAD,1,META",
+ "selector": "head > meta",
+ "boundingRect": {
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "snippet": "
",
+ "nodeLabel": "head > meta"
+ }
+ }
+ ]
+ },
+ "guidanceLevel": 3,
+ "replacesAudits": [
+ "viewport"
+ ]
+ }
+ },
+ "configSettings": {
+ "output": [
+ "json"
+ ],
+ "maxWaitForFcp": 30000,
+ "maxWaitForLoad": 45000,
+ "pauseAfterFcpMs": 1000,
+ "pauseAfterLoadMs": 1000,
+ "networkQuietThresholdMs": 1000,
+ "cpuQuietThresholdMs": 1000,
+ "formFactor": "mobile",
+ "throttling": {
+ "rttMs": 150,
+ "throughputKbps": 1638.4,
+ "requestLatencyMs": 562.5,
+ "downloadThroughputKbps": 1474.5600000000002,
+ "uploadThroughputKbps": 675,
+ "cpuSlowdownMultiplier": 4
+ },
+ "throttlingMethod": "simulate",
+ "screenEmulation": {
+ "mobile": true,
+ "width": 412,
+ "height": 823,
+ "deviceScaleFactor": 1.75,
+ "disabled": false
+ },
+ "emulatedUserAgent": "Mozilla/5.0 (Linux; Android 11; moto g power (2022)) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Mobile Safari/537.36",
+ "auditMode": false,
+ "gatherMode": false,
+ "clearStorageTypes": [
+ "file_systems",
+ "shader_cache",
+ "service_workers",
+ "cache_storage"
+ ],
+ "disableStorageReset": false,
+ "debugNavigation": false,
+ "channel": "cli",
+ "usePassiveGathering": false,
+ "disableFullPageScreenshot": false,
+ "skipAboutBlank": false,
+ "blankPage": "about:blank",
+ "ignoreStatusCode": false,
+ "locale": "en-US",
+ "blockedUrlPatterns": null,
+ "additionalTraceCategories": null,
+ "extraHeaders": null,
+ "precomputedLanternData": null,
+ "onlyAudits": null,
+ "onlyCategories": [
+ "performance"
+ ],
+ "skipAudits": null
+ },
+ "categories": {
+ "performance": {
+ "title": "Performance",
+ "supportedModes": [
+ "navigation",
+ "timespan",
+ "snapshot"
+ ],
+ "auditRefs": [
+ {
+ "id": "first-contentful-paint",
+ "weight": 10,
+ "group": "metrics",
+ "acronym": "FCP"
+ },
+ {
+ "id": "largest-contentful-paint",
+ "weight": 25,
+ "group": "metrics",
+ "acronym": "LCP"
+ },
+ {
+ "id": "total-blocking-time",
+ "weight": 30,
+ "group": "metrics",
+ "acronym": "TBT"
+ },
+ {
+ "id": "cumulative-layout-shift",
+ "weight": 25,
+ "group": "metrics",
+ "acronym": "CLS"
+ },
+ {
+ "id": "speed-index",
+ "weight": 10,
+ "group": "metrics",
+ "acronym": "SI"
+ },
+ {
+ "id": "cache-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "cls-culprits-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "document-latency-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "dom-size-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "duplicated-javascript-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "font-display-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "forced-reflow-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "image-delivery-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "inp-breakdown-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "lcp-breakdown-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "lcp-discovery-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "legacy-javascript-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "modern-http-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "network-dependency-tree-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "render-blocking-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "third-parties-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "viewport-insight",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "interactive",
+ "weight": 0,
+ "group": "hidden",
+ "acronym": "TTI"
+ },
+ {
+ "id": "max-potential-fid",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "first-meaningful-paint",
+ "weight": 0,
+ "acronym": "FMP",
+ "group": "hidden"
+ },
+ {
+ "id": "render-blocking-resources",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "uses-responsive-images",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "offscreen-images",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "unminified-css",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "unminified-javascript",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "unused-css-rules",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "unused-javascript",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "uses-optimized-images",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "modern-image-formats",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "uses-text-compression",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "uses-rel-preconnect",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "server-response-time",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "redirects",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "uses-http2",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "efficient-animated-content",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "duplicated-javascript",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "legacy-javascript",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "prioritize-lcp-image",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "total-byte-weight",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "uses-long-cache-ttl",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "dom-size",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "critical-request-chains",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "user-timings",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "bootup-time",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "mainthread-work-breakdown",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "font-display",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "third-party-summary",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "third-party-facades",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "largest-contentful-paint-element",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "lcp-lazy-loaded",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "layout-shifts",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "uses-passive-event-listeners",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "no-document-write",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "long-tasks",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "non-composited-animations",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "unsized-images",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "viewport",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "bf-cache",
+ "weight": 0,
+ "group": "diagnostics"
+ },
+ {
+ "id": "network-requests",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "network-rtt",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "network-server-latency",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "main-thread-tasks",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "diagnostics",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "metrics",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "screenshot-thumbnails",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "final-screenshot",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "script-treemap-data",
+ "weight": 0,
+ "group": "hidden"
+ },
+ {
+ "id": "resource-summary",
+ "weight": 0,
+ "group": "hidden"
+ }
+ ],
+ "id": "performance",
+ "score": 0.41
+ }
+ },
+ "categoryGroups": {
+ "metrics": {
+ "title": "Metrics"
+ },
+ "insights": {
+ "title": "Insights",
+ "description": "These insights are also available in the Chrome DevTools Performance Panel - [record a trace](https://developer.chrome.com/docs/devtools/performance/reference) to view more detailed information."
+ },
+ "diagnostics": {
+ "title": "Diagnostics",
+ "description": "More information about the performance of your application. These numbers don't [directly affect](https://developer.chrome.com/docs/lighthouse/performance/performance-scoring/) the Performance score."
+ },
+ "a11y-best-practices": {
+ "title": "Best practices",
+ "description": "These items highlight common accessibility best practices."
+ },
+ "a11y-color-contrast": {
+ "title": "Contrast",
+ "description": "These are opportunities to improve the legibility of your content."
+ },
+ "a11y-names-labels": {
+ "title": "Names and labels",
+ "description": "These are opportunities to improve the semantics of the controls in your application. This may enhance the experience for users of assistive technology, like a screen reader."
+ },
+ "a11y-navigation": {
+ "title": "Navigation",
+ "description": "These are opportunities to improve keyboard navigation in your application."
+ },
+ "a11y-aria": {
+ "title": "ARIA",
+ "description": "These are opportunities to improve the usage of ARIA in your application which may enhance the experience for users of assistive technology, like a screen reader."
+ },
+ "a11y-language": {
+ "title": "Internationalization and localization",
+ "description": "These are opportunities to improve the interpretation of your content by users in different locales."
+ },
+ "a11y-audio-video": {
+ "title": "Audio and video",
+ "description": "These are opportunities to provide alternative content for audio and video. This may improve the experience for users with hearing or vision impairments."
+ },
+ "a11y-tables-lists": {
+ "title": "Tables and lists",
+ "description": "These are opportunities to improve the experience of reading tabular or list data using assistive technology, like a screen reader."
+ },
+ "seo-mobile": {
+ "title": "Mobile Friendly",
+ "description": "Make sure your pages are mobile friendly so users don’t have to pinch or zoom in order to read the content pages. [Learn how to make pages mobile-friendly](https://developers.google.com/search/mobile-sites/)."
+ },
+ "seo-content": {
+ "title": "Content Best Practices",
+ "description": "Format your HTML in a way that enables crawlers to better understand your app’s content."
+ },
+ "seo-crawl": {
+ "title": "Crawling and Indexing",
+ "description": "To appear in search results, crawlers need access to your app."
+ },
+ "best-practices-trust-safety": {
+ "title": "Trust and Safety"
+ },
+ "best-practices-ux": {
+ "title": "User Experience"
+ },
+ "best-practices-browser-compat": {
+ "title": "Browser Compatibility"
+ },
+ "best-practices-general": {
+ "title": "General"
+ },
+ "hidden": {
+ "title": ""
+ }
+ },
+ "stackPacks": [],
+ "entities": [
+ {
+ "name": "localhost",
+ "origins": [
+ "http://localhost:3000"
+ ],
+ "isFirstParty": true,
+ "isUnrecognized": true
+ },
+ {
+ "name": "valuefrontier.cn",
+ "origins": [
+ "https://app.valuefrontier.cn"
+ ],
+ "isUnrecognized": true
+ },
+ {
+ "name": "49.232.185.254",
+ "origins": [
+ "http://49.232.185.254:5001"
+ ],
+ "isUnrecognized": true
+ }
+ ],
+ "fullPageScreenshot": {
+ "screenshot": {
+ "data": "",
+ "width": 412,
+ "height": 2124
+ },
+ "nodes": {
+ "page-0-DIV": {
+ "id": "",
+ "top": 65,
+ "bottom": 2189,
+ "left": 206,
+ "right": 412,
+ "width": 206,
+ "height": 2124
+ },
+ "page-1-DIV": {
+ "id": "dify-chatbot-bubble-button",
+ "top": 2052,
+ "bottom": 2108,
+ "left": 340,
+ "right": 396,
+ "width": 56,
+ "height": 56
+ },
+ "page-2-META": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "page-3-BODY": {
+ "id": "",
+ "top": 0,
+ "bottom": 2348,
+ "left": 0,
+ "right": 412,
+ "width": 412,
+ "height": 2348
+ },
+ "page-4-H2": {
+ "id": "",
+ "top": 513,
+ "bottom": 633,
+ "left": 136,
+ "right": 219,
+ "width": 83,
+ "height": 120
+ },
+ "page-5-DIV": {
+ "id": "",
+ "top": 1384,
+ "bottom": 1534,
+ "left": 180,
+ "right": 330,
+ "width": 150,
+ "height": 150
+ },
+ "page-6-DIV": {
+ "id": "",
+ "top": 484,
+ "bottom": 684,
+ "left": 41,
+ "right": 241,
+ "width": 200,
+ "height": 200
+ },
+ "1-0-H2": {
+ "id": "",
+ "top": 513,
+ "bottom": 633,
+ "left": 136,
+ "right": 219,
+ "width": 83,
+ "height": 120
+ },
+ "1-1-BODY": {
+ "id": "",
+ "top": 0,
+ "bottom": 2348,
+ "left": 0,
+ "right": 412,
+ "width": 412,
+ "height": 2348
+ },
+ "1-2-IMG": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "1-3-DIV": {
+ "id": "",
+ "top": 65,
+ "bottom": 2189,
+ "left": 206,
+ "right": 412,
+ "width": 206,
+ "height": 2124
+ },
+ "1-4-LINK": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "1-5-LINK": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "1-6-LINK": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "1-7-LINK": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "1-8-META": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "1-9-META": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ },
+ "1-10-META": {
+ "id": "",
+ "top": 0,
+ "bottom": 0,
+ "left": 0,
+ "right": 0,
+ "width": 0,
+ "height": 0
+ }
+ }
+ },
+ "timing": {
+ "entries": [
+ {
+ "startTime": 3477.91,
+ "name": "lh:config",
+ "duration": 1214.37,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 3480.47,
+ "name": "lh:config:resolveArtifactsToDefns",
+ "duration": 247.9,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 4692.44,
+ "name": "lh:runner:gather",
+ "duration": 48779.73,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 4894.92,
+ "name": "lh:driver:connect",
+ "duration": 6.16,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 4901.35,
+ "name": "lh:driver:navigate",
+ "duration": 105.14,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 5006.82,
+ "name": "lh:gather:getBenchmarkIndex",
+ "duration": 1022.98,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 6030.02,
+ "name": "lh:gather:getVersion",
+ "duration": 0.99,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 6031.1,
+ "name": "lh:gather:getDevicePixelRatio",
+ "duration": 0.83,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 6032.72,
+ "name": "lh:prepare:navigationMode",
+ "duration": 148.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 6062.17,
+ "name": "lh:storage:clearDataForOrigin",
+ "duration": 60.78,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 6123.11,
+ "name": "lh:storage:clearBrowserCaches",
+ "duration": 52.85,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 6178.88,
+ "name": "lh:gather:prepareThrottlingAndNetwork",
+ "duration": 2.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 6315.28,
+ "name": "lh:driver:navigate",
+ "duration": 16364.07,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45197.16,
+ "name": "lh:computed:NetworkRecords",
+ "duration": 3.52,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45201.27,
+ "name": "lh:gather:getArtifact:DevtoolsLog",
+ "duration": 0.04,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45201.33,
+ "name": "lh:gather:getArtifact:Trace",
+ "duration": 0.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45201.36,
+ "name": "lh:gather:getArtifact:ConsoleMessages",
+ "duration": 0.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45201.4,
+ "name": "lh:gather:getArtifact:CSSUsage",
+ "duration": 68.65,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45270.08,
+ "name": "lh:gather:getArtifact:DOMStats",
+ "duration": 15.06,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45285.17,
+ "name": "lh:gather:getArtifact:ImageElements",
+ "duration": 67.34,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45352.54,
+ "name": "lh:gather:getArtifact:JsUsage",
+ "duration": 0.22,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45352.88,
+ "name": "lh:gather:getArtifact:LinkElements",
+ "duration": 4.51,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45356.97,
+ "name": "lh:computed:MainResource",
+ "duration": 0.27,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45357.42,
+ "name": "lh:gather:getArtifact:MetaElements",
+ "duration": 3.63,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45361.1,
+ "name": "lh:gather:getArtifact:NetworkUserAgent",
+ "duration": 0.13,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45361.25,
+ "name": "lh:gather:getArtifact:OptimizedImages",
+ "duration": 90.74,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45452.03,
+ "name": "lh:gather:getArtifact:ResponseCompression",
+ "duration": 4.69,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45456.75,
+ "name": "lh:gather:getArtifact:Scripts",
+ "duration": 0.12,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45456.91,
+ "name": "lh:gather:getArtifact:SourceMaps",
+ "duration": 0.06,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45456.99,
+ "name": "lh:gather:getArtifact:Stacks",
+ "duration": 16.79,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45457.07,
+ "name": "lh:gather:collectStacks",
+ "duration": 16.67,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45473.79,
+ "name": "lh:gather:getArtifact:Stylesheets",
+ "duration": 92.93,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45566.77,
+ "name": "lh:gather:getArtifact:TraceElements",
+ "duration": 6068.6,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45567.01,
+ "name": "lh:computed:TraceEngineResult",
+ "duration": 5997.49,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 45567.13,
+ "name": "lh:computed:ProcessedTrace",
+ "duration": 611.23,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 46222.06,
+ "name": "lh:computed:TraceEngineResult:total",
+ "duration": 5331.7,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 46222.09,
+ "name": "lh:computed:TraceEngineResult:parse",
+ "duration": 5103.48,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 46224.11,
+ "name": "lh:computed:TraceEngineResult:parse:handleEvent",
+ "duration": 3332.67,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 49556.81,
+ "name": "lh:computed:TraceEngineResult:parse:Meta:finalize",
+ "duration": 0.32,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 49557.16,
+ "name": "lh:computed:TraceEngineResult:parse:AnimationFrames:finalize",
+ "duration": 1.22,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 49558.41,
+ "name": "lh:computed:TraceEngineResult:parse:Animations:finalize",
+ "duration": 3.95,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 49562.39,
+ "name": "lh:computed:TraceEngineResult:parse:Samples:finalize",
+ "duration": 1.23,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 49563.65,
+ "name": "lh:computed:TraceEngineResult:parse:AuctionWorklets:finalize",
+ "duration": 1.25,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 49564.92,
+ "name": "lh:computed:TraceEngineResult:parse:NetworkRequests:finalize",
+ "duration": 13.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 49577.98,
+ "name": "lh:computed:TraceEngineResult:parse:Renderer:finalize",
+ "duration": 1114.74,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50692.76,
+ "name": "lh:computed:TraceEngineResult:parse:Flows:finalize",
+ "duration": 83.8,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50776.6,
+ "name": "lh:computed:TraceEngineResult:parse:AsyncJSCalls:finalize",
+ "duration": 55.62,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50832.26,
+ "name": "lh:computed:TraceEngineResult:parse:DOMStats:finalize",
+ "duration": 1.39,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50833.69,
+ "name": "lh:computed:TraceEngineResult:parse:UserTimings:finalize",
+ "duration": 1.51,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50835.24,
+ "name": "lh:computed:TraceEngineResult:parse:ExtensionTraceData:finalize",
+ "duration": 1.98,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50837.27,
+ "name": "lh:computed:TraceEngineResult:parse:LayerTree:finalize",
+ "duration": 3.5,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50840.81,
+ "name": "lh:computed:TraceEngineResult:parse:Frames:finalize",
+ "duration": 116.12,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50956.97,
+ "name": "lh:computed:TraceEngineResult:parse:GPU:finalize",
+ "duration": 1.45,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50958.47,
+ "name": "lh:computed:TraceEngineResult:parse:ImagePainting:finalize",
+ "duration": 1.43,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50959.93,
+ "name": "lh:computed:TraceEngineResult:parse:Initiators:finalize",
+ "duration": 2.61,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50962.58,
+ "name": "lh:computed:TraceEngineResult:parse:Invalidations:finalize",
+ "duration": 1.35,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50963.97,
+ "name": "lh:computed:TraceEngineResult:parse:PageLoadMetrics:finalize",
+ "duration": 2.61,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50966.61,
+ "name": "lh:computed:TraceEngineResult:parse:LargestImagePaint:finalize",
+ "duration": 1.48,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50968.11,
+ "name": "lh:computed:TraceEngineResult:parse:LargestTextPaint:finalize",
+ "duration": 1.23,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50969.36,
+ "name": "lh:computed:TraceEngineResult:parse:Screenshots:finalize",
+ "duration": 7.71,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50977.11,
+ "name": "lh:computed:TraceEngineResult:parse:LayoutShifts:finalize",
+ "duration": 1.92,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50979.05,
+ "name": "lh:computed:TraceEngineResult:parse:Memory:finalize",
+ "duration": 1.26,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50980.33,
+ "name": "lh:computed:TraceEngineResult:parse:PageFrames:finalize",
+ "duration": 1.31,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50981.68,
+ "name": "lh:computed:TraceEngineResult:parse:Scripts:finalize",
+ "duration": 2.34,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50984.05,
+ "name": "lh:computed:TraceEngineResult:parse:SelectorStats:finalize",
+ "duration": 1.27,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50985.34,
+ "name": "lh:computed:TraceEngineResult:parse:UserInteractions:finalize",
+ "duration": 1.93,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50987.32,
+ "name": "lh:computed:TraceEngineResult:parse:Workers:finalize",
+ "duration": 1.33,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50988.69,
+ "name": "lh:computed:TraceEngineResult:parse:Warnings:finalize",
+ "duration": 1.64,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 50990.42,
+ "name": "lh:computed:TraceEngineResult:parse:clone",
+ "duration": 335.1,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51325.58,
+ "name": "lh:computed:TraceEngineResult:insights",
+ "duration": 228.17,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51326.06,
+ "name": "lh:computed:TraceEngineResult:insights:CLSCulprits",
+ "duration": 0.8,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51326.89,
+ "name": "lh:computed:TraceEngineResult:insights:Cache",
+ "duration": 0.32,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51327.22,
+ "name": "lh:computed:TraceEngineResult:insights:DOMSize",
+ "duration": 0.57,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51327.81,
+ "name": "lh:computed:TraceEngineResult:insights:DocumentLatency",
+ "duration": 0.24,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51328.07,
+ "name": "lh:computed:TraceEngineResult:insights:DuplicatedJavaScript",
+ "duration": 2.1,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51330.19,
+ "name": "lh:computed:TraceEngineResult:insights:FontDisplay",
+ "duration": 0.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51330.43,
+ "name": "lh:computed:TraceEngineResult:insights:ForcedReflow",
+ "duration": 0.3,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51330.74,
+ "name": "lh:computed:TraceEngineResult:insights:INPBreakdown",
+ "duration": 0.14,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51330.9,
+ "name": "lh:computed:TraceEngineResult:insights:ImageDelivery",
+ "duration": 0.41,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51331.33,
+ "name": "lh:computed:TraceEngineResult:insights:LCPBreakdown",
+ "duration": 0.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51331.56,
+ "name": "lh:computed:TraceEngineResult:insights:LCPDiscovery",
+ "duration": 0.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51331.79,
+ "name": "lh:computed:TraceEngineResult:insights:LegacyJavaScript",
+ "duration": 0.4,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51332.2,
+ "name": "lh:computed:TraceEngineResult:insights:ModernHTTP",
+ "duration": 0.35,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51332.58,
+ "name": "lh:computed:TraceEngineResult:insights:NetworkDependencyTree",
+ "duration": 0.16,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51332.76,
+ "name": "lh:computed:TraceEngineResult:insights:RenderBlocking",
+ "duration": 0.2,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51332.97,
+ "name": "lh:computed:TraceEngineResult:insights:SlowCSSSelector",
+ "duration": 0.26,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51333.25,
+ "name": "lh:computed:TraceEngineResult:insights:ThirdParties",
+ "duration": 3.07,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51336.35,
+ "name": "lh:computed:TraceEngineResult:insights:Viewport",
+ "duration": 0.27,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51336.92,
+ "name": "lh:computed:TraceEngineResult:insights:createLanternContext",
+ "duration": 191.31,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51528.29,
+ "name": "lh:computed:TraceEngineResult:insights:CLSCulprits",
+ "duration": 0.62,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51528.93,
+ "name": "lh:computed:TraceEngineResult:insights:Cache",
+ "duration": 2.12,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51531.07,
+ "name": "lh:computed:TraceEngineResult:insights:DOMSize",
+ "duration": 0.25,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51531.34,
+ "name": "lh:computed:TraceEngineResult:insights:DocumentLatency",
+ "duration": 0.35,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51531.7,
+ "name": "lh:computed:TraceEngineResult:insights:DuplicatedJavaScript",
+ "duration": 0.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51531.93,
+ "name": "lh:computed:TraceEngineResult:insights:FontDisplay",
+ "duration": 0.03,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51531.97,
+ "name": "lh:computed:TraceEngineResult:insights:ForcedReflow",
+ "duration": 0.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51532,
+ "name": "lh:computed:TraceEngineResult:insights:INPBreakdown",
+ "duration": 0.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51532.02,
+ "name": "lh:computed:TraceEngineResult:insights:ImageDelivery",
+ "duration": 1.09,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51533.13,
+ "name": "lh:computed:TraceEngineResult:insights:LCPBreakdown",
+ "duration": 0.29,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51533.43,
+ "name": "lh:computed:TraceEngineResult:insights:LCPDiscovery",
+ "duration": 0.06,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51533.5,
+ "name": "lh:computed:TraceEngineResult:insights:LegacyJavaScript",
+ "duration": 0.1,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51533.61,
+ "name": "lh:computed:TraceEngineResult:insights:ModernHTTP",
+ "duration": 1.89,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51535.52,
+ "name": "lh:computed:TraceEngineResult:insights:NetworkDependencyTree",
+ "duration": 1.79,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51537.32,
+ "name": "lh:computed:TraceEngineResult:insights:RenderBlocking",
+ "duration": 0.29,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51537.63,
+ "name": "lh:computed:TraceEngineResult:insights:SlowCSSSelector",
+ "duration": 0.03,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51537.67,
+ "name": "lh:computed:TraceEngineResult:insights:ThirdParties",
+ "duration": 14.75,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51552.45,
+ "name": "lh:computed:TraceEngineResult:insights:Viewport",
+ "duration": 0.17,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51570.04,
+ "name": "lh:computed:ProcessedNavigation",
+ "duration": 18.08,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51588.27,
+ "name": "lh:computed:CumulativeLayoutShift",
+ "duration": 28.11,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51617.77,
+ "name": "lh:computed:Responsiveness",
+ "duration": 0.27,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51635.4,
+ "name": "lh:gather:getArtifact:ViewportDimensions",
+ "duration": 1.48,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 51636.9,
+ "name": "lh:gather:getArtifact:FullPageScreenshot",
+ "duration": 1183.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 52820.17,
+ "name": "lh:gather:getArtifact:BFCacheFailures",
+ "duration": 624.44,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53472.67,
+ "name": "lh:runner:audit",
+ "duration": 3238.12,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53472.82,
+ "name": "lh:runner:auditing",
+ "duration": 3237.56,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53475.83,
+ "name": "lh:audit:viewport",
+ "duration": 3.81,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53476.61,
+ "name": "lh:computed:ViewportMeta",
+ "duration": 0.7,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53480.2,
+ "name": "lh:audit:first-contentful-paint",
+ "duration": 20.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53480.88,
+ "name": "lh:computed:FirstContentfulPaint",
+ "duration": 15.66,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53481.21,
+ "name": "lh:computed:LanternFirstContentfulPaint",
+ "duration": 15.32,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53481.37,
+ "name": "lh:computed:PageDependencyGraph",
+ "duration": 12.28,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53493.7,
+ "name": "lh:computed:LoadSimulator",
+ "duration": 1.1,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53493.79,
+ "name": "lh:computed:NetworkAnalysis",
+ "duration": 0.94,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53500.82,
+ "name": "lh:audit:largest-contentful-paint",
+ "duration": 4.75,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53501.35,
+ "name": "lh:computed:LargestContentfulPaint",
+ "duration": 2.88,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53501.43,
+ "name": "lh:computed:LanternLargestContentfulPaint",
+ "duration": 2.78,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53506.06,
+ "name": "lh:audit:first-meaningful-paint",
+ "duration": 1.51,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53507.97,
+ "name": "lh:audit:speed-index",
+ "duration": 481.28,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53508.42,
+ "name": "lh:computed:SpeedIndex",
+ "duration": 479.55,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53508.5,
+ "name": "lh:computed:LanternSpeedIndex",
+ "duration": 479.44,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53508.55,
+ "name": "lh:computed:Speedline",
+ "duration": 461.45,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53989.3,
+ "name": "lh:audit:screenshot-thumbnails",
+ "duration": 0.64,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53989.97,
+ "name": "lh:audit:final-screenshot",
+ "duration": 29.2,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 53990.07,
+ "name": "lh:computed:Screenshots",
+ "duration": 29.02,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54019.65,
+ "name": "lh:audit:total-blocking-time",
+ "duration": 20.71,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54020.15,
+ "name": "lh:computed:TotalBlockingTime",
+ "duration": 19.04,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54020.25,
+ "name": "lh:computed:LanternTotalBlockingTime",
+ "duration": 18.92,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54020.35,
+ "name": "lh:computed:LanternInteractive",
+ "duration": 10.49,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54040.73,
+ "name": "lh:audit:max-potential-fid",
+ "duration": 11.43,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54041.19,
+ "name": "lh:computed:MaxPotentialFID",
+ "duration": 8.88,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54041.27,
+ "name": "lh:computed:LanternMaxPotentialFID",
+ "duration": 8.78,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54052.49,
+ "name": "lh:audit:cumulative-layout-shift",
+ "duration": 2.47,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54055.41,
+ "name": "lh:audit:server-response-time",
+ "duration": 2.34,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54058.12,
+ "name": "lh:audit:interactive",
+ "duration": 1.61,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54058.6,
+ "name": "lh:computed:Interactive",
+ "duration": 0.1,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54060.05,
+ "name": "lh:audit:user-timings",
+ "duration": 6.33,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54060.44,
+ "name": "lh:computed:UserTimings",
+ "duration": 4.87,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54066.68,
+ "name": "lh:audit:critical-request-chains",
+ "duration": 2.15,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54067.08,
+ "name": "lh:computed:CriticalRequestChains",
+ "duration": 0.7,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54069.1,
+ "name": "lh:audit:redirects",
+ "duration": 7.26,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54076.8,
+ "name": "lh:audit:mainthread-work-breakdown",
+ "duration": 33.6,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54077.54,
+ "name": "lh:computed:MainThreadTasks",
+ "duration": 29.65,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54110.76,
+ "name": "lh:audit:bootup-time",
+ "duration": 17.97,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54115.1,
+ "name": "lh:computed:TBTImpactTasks",
+ "duration": 10.93,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54129.01,
+ "name": "lh:audit:uses-rel-preconnect",
+ "duration": 6.88,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54136.25,
+ "name": "lh:audit:font-display",
+ "duration": 3.61,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54139.88,
+ "name": "lh:audit:diagnostics",
+ "duration": 1.24,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54141.15,
+ "name": "lh:audit:network-requests",
+ "duration": 4.28,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54141.35,
+ "name": "lh:computed:EntityClassification",
+ "duration": 3.26,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54145.76,
+ "name": "lh:audit:network-rtt",
+ "duration": 1.45,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54147.48,
+ "name": "lh:audit:network-server-latency",
+ "duration": 1.31,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54148.81,
+ "name": "lh:audit:main-thread-tasks",
+ "duration": 0.64,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54149.47,
+ "name": "lh:audit:metrics",
+ "duration": 29.18,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54149.64,
+ "name": "lh:computed:TimingSummary",
+ "duration": 28.8,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54149.97,
+ "name": "lh:computed:FirstContentfulPaintAllFrames",
+ "duration": 0.09,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54150.11,
+ "name": "lh:computed:LargestContentfulPaintAllFrames",
+ "duration": 0.08,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54150.27,
+ "name": "lh:computed:LCPBreakdown",
+ "duration": 27.54,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54150.37,
+ "name": "lh:computed:TimeToFirstByte",
+ "duration": 0.2,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54150.58,
+ "name": "lh:computed:LCPImageRecord",
+ "duration": 27.13,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54178.66,
+ "name": "lh:audit:resource-summary",
+ "duration": 1.76,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54178.83,
+ "name": "lh:computed:ResourceSummary",
+ "duration": 0.69,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54180.89,
+ "name": "lh:audit:third-party-summary",
+ "duration": 5.57,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54186.94,
+ "name": "lh:audit:third-party-facades",
+ "duration": 3.8,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54191.05,
+ "name": "lh:audit:largest-contentful-paint-element",
+ "duration": 1.95,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54193.37,
+ "name": "lh:audit:lcp-lazy-loaded",
+ "duration": 1.76,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54195.42,
+ "name": "lh:audit:layout-shifts",
+ "duration": 1.54,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54197.25,
+ "name": "lh:audit:long-tasks",
+ "duration": 11.49,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54209.26,
+ "name": "lh:audit:non-composited-animations",
+ "duration": 3.06,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54212.79,
+ "name": "lh:audit:unsized-images",
+ "duration": 2.08,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54215.19,
+ "name": "lh:audit:prioritize-lcp-image",
+ "duration": 5.43,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54220.65,
+ "name": "lh:audit:script-treemap-data",
+ "duration": 354.63,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54221.04,
+ "name": "lh:computed:JSBundles",
+ "duration": 0.1,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54221.16,
+ "name": "lh:computed:ModuleDuplication",
+ "duration": 0.23,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54221.43,
+ "name": "lh:computed:UnusedJavascriptSummary",
+ "duration": 0.34,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54221.88,
+ "name": "lh:computed:UnusedJavascriptSummary",
+ "duration": 352.95,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54575.02,
+ "name": "lh:computed:UnusedJavascriptSummary",
+ "duration": 0.16,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54575.77,
+ "name": "lh:audit:uses-long-cache-ttl",
+ "duration": 3.64,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54579.95,
+ "name": "lh:audit:total-byte-weight",
+ "duration": 2.7,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54583.01,
+ "name": "lh:audit:offscreen-images",
+ "duration": 4.7,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54588.08,
+ "name": "lh:audit:render-blocking-resources",
+ "duration": 34.15,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54589.16,
+ "name": "lh:computed:UnusedCSS",
+ "duration": 31.3,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54620.76,
+ "name": "lh:computed:NavigationInsights",
+ "duration": 0.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54621.04,
+ "name": "lh:computed:FirstContentfulPaint",
+ "duration": 0.09,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54622.58,
+ "name": "lh:audit:unminified-css",
+ "duration": 64.18,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 54687.1,
+ "name": "lh:audit:unminified-javascript",
+ "duration": 642.84,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55330.17,
+ "name": "lh:audit:unused-css-rules",
+ "duration": 1.97,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55332.35,
+ "name": "lh:audit:unused-javascript",
+ "duration": 3.22,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55335.79,
+ "name": "lh:audit:modern-image-formats",
+ "duration": 2.47,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55338.48,
+ "name": "lh:audit:uses-optimized-images",
+ "duration": 2.01,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55340.7,
+ "name": "lh:audit:uses-text-compression",
+ "duration": 1.83,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55342.74,
+ "name": "lh:audit:uses-responsive-images",
+ "duration": 2.51,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55343.07,
+ "name": "lh:computed:ImageRecords",
+ "duration": 0.3,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55345.48,
+ "name": "lh:audit:efficient-animated-content",
+ "duration": 2.24,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55347.94,
+ "name": "lh:audit:duplicated-javascript",
+ "duration": 2.27,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 55350.5,
+ "name": "lh:audit:legacy-javascript",
+ "duration": 1317.97,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56668.75,
+ "name": "lh:audit:dom-size",
+ "duration": 2.18,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56672.22,
+ "name": "lh:audit:no-document-write",
+ "duration": 1.42,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56673.87,
+ "name": "lh:audit:uses-http2",
+ "duration": 3.25,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56677.47,
+ "name": "lh:audit:uses-passive-event-listeners",
+ "duration": 1.22,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56679.01,
+ "name": "lh:audit:bf-cache",
+ "duration": 1.5,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56680.82,
+ "name": "lh:audit:cache-insight",
+ "duration": 1.91,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56683.04,
+ "name": "lh:audit:cls-culprits-insight",
+ "duration": 1.27,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56684.62,
+ "name": "lh:audit:document-latency-insight",
+ "duration": 1.08,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56686.02,
+ "name": "lh:audit:dom-size-insight",
+ "duration": 1.47,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56687.78,
+ "name": "lh:audit:duplicated-javascript-insight",
+ "duration": 1.18,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56689.29,
+ "name": "lh:audit:font-display-insight",
+ "duration": 1.21,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56690.82,
+ "name": "lh:audit:forced-reflow-insight",
+ "duration": 1.4,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56692.53,
+ "name": "lh:audit:image-delivery-insight",
+ "duration": 1.72,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56694.55,
+ "name": "lh:audit:inp-breakdown-insight",
+ "duration": 1.31,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56696.17,
+ "name": "lh:audit:lcp-breakdown-insight",
+ "duration": 1.53,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56698.06,
+ "name": "lh:audit:lcp-discovery-insight",
+ "duration": 1.19,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56699.96,
+ "name": "lh:audit:legacy-javascript-insight",
+ "duration": 1.48,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56701.82,
+ "name": "lh:audit:modern-http-insight",
+ "duration": 1.42,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56703.6,
+ "name": "lh:audit:network-dependency-tree-insight",
+ "duration": 2.06,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56705.98,
+ "name": "lh:audit:render-blocking-insight",
+ "duration": 1.3,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56707.66,
+ "name": "lh:audit:third-parties-insight",
+ "duration": 1.36,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56709.31,
+ "name": "lh:audit:viewport-insight",
+ "duration": 1.05,
+ "entryType": "measure"
+ },
+ {
+ "startTime": 56710.39,
+ "name": "lh:runner:generate",
+ "duration": 0.39,
+ "entryType": "measure"
+ }
+ ],
+ "total": 52017.850000000006
+ },
+ "i18n": {
+ "rendererFormattedStrings": {
+ "calculatorLink": "See calculator.",
+ "collapseView": "Collapse view",
+ "crcInitialNavigation": "Initial Navigation",
+ "crcLongestDurationLabel": "Maximum critical path latency:",
+ "dropdownCopyJSON": "Copy JSON",
+ "dropdownDarkTheme": "Toggle Dark Theme",
+ "dropdownPrintExpanded": "Print Expanded",
+ "dropdownPrintSummary": "Print Summary",
+ "dropdownSaveGist": "Save as Gist",
+ "dropdownSaveHTML": "Save as HTML",
+ "dropdownSaveJSON": "Save as JSON",
+ "dropdownViewUnthrottledTrace": "View Unthrottled Trace",
+ "dropdownViewer": "Open in Viewer",
+ "errorLabel": "Error!",
+ "errorMissingAuditInfo": "Report error: no audit information",
+ "expandView": "Expand view",
+ "firstPartyChipLabel": "1st party",
+ "footerIssue": "File an issue",
+ "goBackToAudits": "Go back to audits",
+ "hide": "Hide",
+ "insightsNotice": "Later this year, insights will replace performance audits. [Learn more and provide feedback here](https://github.com/GoogleChrome/lighthouse/discussions/16462).",
+ "labDataTitle": "Lab Data",
+ "lsPerformanceCategoryDescription": "[Lighthouse](https://developers.google.com/web/tools/lighthouse/) analysis of the current page on an emulated mobile network. Values are estimated and may vary.",
+ "manualAuditsGroupTitle": "Additional items to manually check",
+ "notApplicableAuditsGroupTitle": "Not applicable",
+ "openInANewTabTooltip": "Open in a new tab",
+ "opportunityResourceColumnLabel": "Opportunity",
+ "opportunitySavingsColumnLabel": "Estimated Savings",
+ "passedAuditsGroupTitle": "Passed audits",
+ "runtimeAnalysisWindow": "Initial page load",
+ "runtimeAnalysisWindowSnapshot": "Point-in-time snapshot",
+ "runtimeAnalysisWindowTimespan": "User interactions timespan",
+ "runtimeCustom": "Custom throttling",
+ "runtimeDesktopEmulation": "Emulated Desktop",
+ "runtimeMobileEmulation": "Emulated Moto G Power",
+ "runtimeNoEmulation": "No emulation",
+ "runtimeSettingsAxeVersion": "Axe version",
+ "runtimeSettingsBenchmark": "Unthrottled CPU/Memory Power",
+ "runtimeSettingsCPUThrottling": "CPU throttling",
+ "runtimeSettingsDevice": "Device",
+ "runtimeSettingsNetworkThrottling": "Network throttling",
+ "runtimeSettingsScreenEmulation": "Screen emulation",
+ "runtimeSettingsUANetwork": "User agent (network)",
+ "runtimeSingleLoad": "Single page session",
+ "runtimeSingleLoadTooltip": "This data is taken from a single page session, as opposed to field data summarizing many sessions.",
+ "runtimeSlow4g": "Slow 4G throttling",
+ "runtimeUnknown": "Unknown",
+ "show": "Show",
+ "showRelevantAudits": "Show audits relevant to:",
+ "snippetCollapseButtonLabel": "Collapse snippet",
+ "snippetExpandButtonLabel": "Expand snippet",
+ "thirdPartyResourcesLabel": "Show 3rd-party resources",
+ "throttlingProvided": "Provided by environment",
+ "toplevelWarningsMessage": "There were issues affecting this run of Lighthouse:",
+ "tryInsights": "Try insights",
+ "unattributable": "Unattributable",
+ "varianceDisclaimer": "Values are estimated and may vary. The [performance score is calculated](https://developer.chrome.com/docs/lighthouse/performance/performance-scoring/) directly from these metrics.",
+ "viewTraceLabel": "View Trace",
+ "viewTreemapLabel": "View Treemap",
+ "warningAuditsGroupTitle": "Passed audits but with warnings",
+ "warningHeader": "Warnings: "
+ },
+ "icuMessagePaths": {
+ "core/audits/viewport.js | title": [
+ "audits.viewport.title"
+ ],
+ "core/audits/viewport.js | description": [
+ "audits.viewport.description"
+ ],
+ "core/lib/i18n/i18n.js | firstContentfulPaintMetric": [
+ "audits[first-contentful-paint].title"
+ ],
+ "core/audits/metrics/first-contentful-paint.js | description": [
+ "audits[first-contentful-paint].description"
+ ],
+ "core/lib/i18n/i18n.js | seconds": [
+ {
+ "values": {
+ "timeInMs": 650.34495
+ },
+ "path": "audits[first-contentful-paint].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 28455.6899
+ },
+ "path": "audits[largest-contentful-paint].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 5404.548441431197
+ },
+ "path": "audits[speed-index].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 51540.68990000001
+ },
+ "path": "audits.interactive.displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 9049.531999999708
+ },
+ "path": "audits[mainthread-work-breakdown].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 7880.652000000009
+ },
+ "path": "audits[bootup-time].displayValue"
+ }
+ ],
+ "core/lib/i18n/i18n.js | largestContentfulPaintMetric": [
+ "audits[largest-contentful-paint].title"
+ ],
+ "core/audits/metrics/largest-contentful-paint.js | description": [
+ "audits[largest-contentful-paint].description"
+ ],
+ "core/lib/i18n/i18n.js | firstMeaningfulPaintMetric": [
+ "audits[first-meaningful-paint].title"
+ ],
+ "core/audits/metrics/first-meaningful-paint.js | description": [
+ "audits[first-meaningful-paint].description"
+ ],
+ "core/lib/i18n/i18n.js | speedIndexMetric": [
+ "audits[speed-index].title"
+ ],
+ "core/audits/metrics/speed-index.js | description": [
+ "audits[speed-index].description"
+ ],
+ "core/lib/i18n/i18n.js | totalBlockingTimeMetric": [
+ "audits[total-blocking-time].title"
+ ],
+ "core/audits/metrics/total-blocking-time.js | description": [
+ "audits[total-blocking-time].description"
+ ],
+ "core/lib/i18n/i18n.js | ms": [
+ {
+ "values": {
+ "timeInMs": 6581.000000000004
+ },
+ "path": "audits[total-blocking-time].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 7338
+ },
+ "path": "audits[max-potential-fid].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 19.20536999999998
+ },
+ "path": "audits[network-rtt].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 671.76363
+ },
+ "path": "audits[network-server-latency].displayValue"
+ },
+ {
+ "values": {
+ "timeInMs": 28455.6899
+ },
+ "path": "audits[largest-contentful-paint-element].displayValue"
+ }
+ ],
+ "core/lib/i18n/i18n.js | maxPotentialFIDMetric": [
+ "audits[max-potential-fid].title"
+ ],
+ "core/audits/metrics/max-potential-fid.js | description": [
+ "audits[max-potential-fid].description"
+ ],
+ "core/lib/i18n/i18n.js | cumulativeLayoutShiftMetric": [
+ "audits[cumulative-layout-shift].title"
+ ],
+ "core/audits/metrics/cumulative-layout-shift.js | description": [
+ "audits[cumulative-layout-shift].description"
+ ],
+ "core/audits/server-response-time.js | title": [
+ "audits[server-response-time].title"
+ ],
+ "core/audits/server-response-time.js | description": [
+ "audits[server-response-time].description"
+ ],
+ "core/audits/server-response-time.js | displayValue": [
+ {
+ "values": {
+ "timeInMs": 17.037000000000003
+ },
+ "path": "audits[server-response-time].displayValue"
+ }
+ ],
+ "core/lib/i18n/i18n.js | columnURL": [
+ "audits[server-response-time].details.headings[0].label",
+ "audits[bootup-time].details.headings[0].label",
+ "audits[uses-rel-preconnect].details.headings[0].label",
+ "audits[font-display].details.headings[0].label",
+ "audits[network-rtt].details.headings[0].label",
+ "audits[network-server-latency].details.headings[0].label",
+ "audits[long-tasks].details.headings[0].label",
+ "audits[unsized-images].details.headings[1].label",
+ "audits[uses-long-cache-ttl].details.headings[0].label",
+ "audits[total-byte-weight].details.headings[0].label",
+ "audits[unminified-css].details.headings[0].label",
+ "audits[unused-css-rules].details.headings[0].label",
+ "audits[unused-javascript].details.headings[0].label",
+ "audits[modern-image-formats].details.headings[1].label",
+ "audits[legacy-javascript].details.headings[0].label",
+ "audits[uses-http2].details.headings[0].label",
+ "audits[font-display-insight].details.headings[0].label",
+ "audits[image-delivery-insight].details.headings[0].label",
+ "audits[legacy-javascript-insight].details.headings[0].label",
+ "audits[modern-http-insight].details.headings[0].label",
+ "audits[render-blocking-insight].details.headings[0].label"
+ ],
+ "core/lib/i18n/i18n.js | columnTimeSpent": [
+ "audits[server-response-time].details.headings[1].label",
+ "audits[mainthread-work-breakdown].details.headings[1].label",
+ "audits[network-rtt].details.headings[1].label",
+ "audits[network-server-latency].details.headings[1].label"
+ ],
+ "core/lib/i18n/i18n.js | interactiveMetric": [
+ "audits.interactive.title"
+ ],
+ "core/audits/metrics/interactive.js | description": [
+ "audits.interactive.description"
+ ],
+ "core/audits/user-timings.js | title": [
+ "audits[user-timings].title"
+ ],
+ "core/audits/user-timings.js | description": [
+ "audits[user-timings].description"
+ ],
+ "core/lib/i18n/i18n.js | columnName": [
+ "audits[user-timings].details.headings[0].label",
+ "audits[non-composited-animations].details.headings[1].label"
+ ],
+ "core/audits/user-timings.js | columnType": [
+ "audits[user-timings].details.headings[1].label"
+ ],
+ "core/lib/i18n/i18n.js | columnStartTime": [
+ "audits[user-timings].details.headings[2].label",
+ "audits[long-tasks].details.headings[1].label"
+ ],
+ "core/lib/i18n/i18n.js | columnDuration": [
+ "audits[user-timings].details.headings[3].label",
+ "audits[long-tasks].details.headings[2].label",
+ "audits[lcp-breakdown-insight].details.items[0].headings[1].label",
+ "audits[render-blocking-insight].details.headings[2].label"
+ ],
+ "core/audits/critical-request-chains.js | title": [
+ "audits[critical-request-chains].title"
+ ],
+ "core/audits/critical-request-chains.js | description": [
+ "audits[critical-request-chains].description"
+ ],
+ "core/audits/redirects.js | title": [
+ "audits.redirects.title"
+ ],
+ "core/audits/redirects.js | description": [
+ "audits.redirects.description"
+ ],
+ "core/audits/mainthread-work-breakdown.js | failureTitle": [
+ "audits[mainthread-work-breakdown].title"
+ ],
+ "core/audits/mainthread-work-breakdown.js | description": [
+ "audits[mainthread-work-breakdown].description"
+ ],
+ "core/audits/mainthread-work-breakdown.js | columnCategory": [
+ "audits[mainthread-work-breakdown].details.headings[0].label"
+ ],
+ "core/audits/bootup-time.js | failureTitle": [
+ "audits[bootup-time].title"
+ ],
+ "core/audits/bootup-time.js | description": [
+ "audits[bootup-time].description"
+ ],
+ "core/audits/bootup-time.js | columnTotal": [
+ "audits[bootup-time].details.headings[1].label"
+ ],
+ "core/audits/bootup-time.js | columnScriptEval": [
+ "audits[bootup-time].details.headings[2].label"
+ ],
+ "core/audits/bootup-time.js | columnScriptParse": [
+ "audits[bootup-time].details.headings[3].label"
+ ],
+ "core/audits/uses-rel-preconnect.js | title": [
+ "audits[uses-rel-preconnect].title"
+ ],
+ "core/audits/uses-rel-preconnect.js | description": [
+ "audits[uses-rel-preconnect].description"
+ ],
+ "core/lib/i18n/i18n.js | displayValueMsSavings": [
+ {
+ "values": {
+ "wastedMs": 169.14731999999998
+ },
+ "path": "audits[uses-rel-preconnect].displayValue"
+ }
+ ],
+ "core/lib/i18n/i18n.js | columnWastedBytes": [
+ "audits[uses-rel-preconnect].details.headings[1].label",
+ "audits[font-display].details.headings[1].label",
+ "audits[unminified-css].details.headings[2].label",
+ "audits[unused-css-rules].details.headings[2].label",
+ "audits[unused-javascript].details.headings[2].label",
+ "audits[modern-image-formats].details.headings[3].label",
+ "audits[legacy-javascript].details.headings[2].label",
+ "audits[font-display-insight].details.headings[1].label",
+ "audits[image-delivery-insight].details.headings[2].label"
+ ],
+ "core/audits/font-display.js | title": [
+ "audits[font-display].title"
+ ],
+ "core/audits/font-display.js | description": [
+ "audits[font-display].description"
+ ],
+ "core/audits/network-rtt.js | title": [
+ "audits[network-rtt].title"
+ ],
+ "core/audits/network-rtt.js | description": [
+ "audits[network-rtt].description"
+ ],
+ "core/audits/network-server-latency.js | title": [
+ "audits[network-server-latency].title"
+ ],
+ "core/audits/network-server-latency.js | description": [
+ "audits[network-server-latency].description"
+ ],
+ "core/lib/i18n/i18n.js | columnResourceType": [
+ "audits[resource-summary].details.headings[0].label"
+ ],
+ "core/lib/i18n/i18n.js | columnRequests": [
+ "audits[resource-summary].details.headings[1].label"
+ ],
+ "core/lib/i18n/i18n.js | columnTransferSize": [
+ "audits[resource-summary].details.headings[2].label",
+ "audits[third-party-summary].details.headings[1].label",
+ "audits[uses-long-cache-ttl].details.headings[2].label",
+ "audits[total-byte-weight].details.headings[1].label",
+ "audits[unminified-css].details.headings[1].label",
+ "audits[unused-css-rules].details.headings[1].label",
+ "audits[unused-javascript].details.headings[1].label",
+ "audits[cache-insight].details.headings[2].label",
+ "audits[render-blocking-insight].details.headings[1].label"
+ ],
+ "core/lib/i18n/i18n.js | total": [
+ "audits[resource-summary].details.items[0].label"
+ ],
+ "core/lib/i18n/i18n.js | scriptResourceType": [
+ "audits[resource-summary].details.items[1].label"
+ ],
+ "core/lib/i18n/i18n.js | imageResourceType": [
+ "audits[resource-summary].details.items[2].label"
+ ],
+ "core/lib/i18n/i18n.js | stylesheetResourceType": [
+ "audits[resource-summary].details.items[3].label"
+ ],
+ "core/lib/i18n/i18n.js | documentResourceType": [
+ "audits[resource-summary].details.items[4].label"
+ ],
+ "core/lib/i18n/i18n.js | otherResourceType": [
+ "audits[resource-summary].details.items[5].label"
+ ],
+ "core/lib/i18n/i18n.js | mediaResourceType": [
+ "audits[resource-summary].details.items[6].label"
+ ],
+ "core/lib/i18n/i18n.js | fontResourceType": [
+ "audits[resource-summary].details.items[7].label"
+ ],
+ "core/lib/i18n/i18n.js | thirdPartyResourceType": [
+ "audits[resource-summary].details.items[8].label"
+ ],
+ "core/audits/third-party-summary.js | title": [
+ "audits[third-party-summary].title"
+ ],
+ "core/audits/third-party-summary.js | description": [
+ "audits[third-party-summary].description"
+ ],
+ "core/audits/third-party-summary.js | displayValue": [
+ {
+ "values": {
+ "timeInMs": 0.19488088687721156
+ },
+ "path": "audits[third-party-summary].displayValue"
+ }
+ ],
+ "core/audits/third-party-summary.js | columnThirdParty": [
+ "audits[third-party-summary].details.headings[0].label"
+ ],
+ "core/lib/i18n/i18n.js | columnBlockingTime": [
+ "audits[third-party-summary].details.headings[2].label"
+ ],
+ "core/audits/third-party-facades.js | title": [
+ "audits[third-party-facades].title"
+ ],
+ "core/audits/third-party-facades.js | description": [
+ "audits[third-party-facades].description"
+ ],
+ "core/audits/largest-contentful-paint-element.js | title": [
+ "audits[largest-contentful-paint-element].title"
+ ],
+ "core/audits/largest-contentful-paint-element.js | description": [
+ "audits[largest-contentful-paint-element].description"
+ ],
+ "core/lib/i18n/i18n.js | columnElement": [
+ "audits[largest-contentful-paint-element].details.items[0].headings[0].label",
+ "audits[lcp-lazy-loaded].details.headings[0].label",
+ "audits[layout-shifts].details.headings[0].label",
+ "audits[non-composited-animations].details.headings[0].label",
+ "audits[dom-size].details.headings[1].label",
+ "audits[dom-size-insight].details.headings[1].label"
+ ],
+ "core/audits/largest-contentful-paint-element.js | columnPhase": [
+ "audits[largest-contentful-paint-element].details.items[1].headings[0].label"
+ ],
+ "core/audits/largest-contentful-paint-element.js | columnPercentOfLCP": [
+ "audits[largest-contentful-paint-element].details.items[1].headings[1].label"
+ ],
+ "core/audits/largest-contentful-paint-element.js | columnTiming": [
+ "audits[largest-contentful-paint-element].details.items[1].headings[2].label"
+ ],
+ "core/audits/largest-contentful-paint-element.js | itemTTFB": [
+ "audits[largest-contentful-paint-element].details.items[1].items[0].phase"
+ ],
+ "core/audits/largest-contentful-paint-element.js | itemLoadDelay": [
+ "audits[largest-contentful-paint-element].details.items[1].items[1].phase"
+ ],
+ "core/audits/largest-contentful-paint-element.js | itemLoadTime": [
+ "audits[largest-contentful-paint-element].details.items[1].items[2].phase"
+ ],
+ "core/audits/largest-contentful-paint-element.js | itemRenderDelay": [
+ "audits[largest-contentful-paint-element].details.items[1].items[3].phase"
+ ],
+ "core/audits/lcp-lazy-loaded.js | title": [
+ "audits[lcp-lazy-loaded].title"
+ ],
+ "core/audits/lcp-lazy-loaded.js | description": [
+ "audits[lcp-lazy-loaded].description"
+ ],
+ "core/audits/layout-shifts.js | title": [
+ "audits[layout-shifts].title"
+ ],
+ "core/audits/layout-shifts.js | description": [
+ "audits[layout-shifts].description"
+ ],
+ "core/audits/layout-shifts.js | columnScore": [
+ "audits[layout-shifts].details.headings[1].label"
+ ],
+ "core/audits/long-tasks.js | title": [
+ "audits[long-tasks].title"
+ ],
+ "core/audits/long-tasks.js | description": [
+ "audits[long-tasks].description"
+ ],
+ "core/audits/long-tasks.js | displayValue": [
+ {
+ "values": {
+ "itemCount": 6
+ },
+ "path": "audits[long-tasks].displayValue"
+ }
+ ],
+ "core/audits/non-composited-animations.js | title": [
+ "audits[non-composited-animations].title"
+ ],
+ "core/audits/non-composited-animations.js | description": [
+ "audits[non-composited-animations].description"
+ ],
+ "core/audits/non-composited-animations.js | displayValue": [
+ {
+ "values": {
+ "itemCount": 1
+ },
+ "path": "audits[non-composited-animations].displayValue"
+ }
+ ],
+ "core/audits/non-composited-animations.js | unsupportedCSSProperty": [
+ {
+ "values": {
+ "propertyCount": 1,
+ "properties": "border-left-color"
+ },
+ "path": "audits[non-composited-animations].details.items[0].subItems.items[0].failureReason"
+ },
+ {
+ "values": {
+ "propertyCount": 1,
+ "properties": "border-right-color"
+ },
+ "path": "audits[non-composited-animations].details.items[0].subItems.items[1].failureReason"
+ },
+ {
+ "values": {
+ "propertyCount": 1,
+ "properties": "color"
+ },
+ "path": "audits[non-composited-animations].details.items[0].subItems.items[2].failureReason"
+ },
+ {
+ "values": {
+ "propertyCount": 1,
+ "properties": "border-top-color"
+ },
+ "path": "audits[non-composited-animations].details.items[0].subItems.items[3].failureReason"
+ },
+ {
+ "values": {
+ "propertyCount": 1,
+ "properties": "border-bottom-color"
+ },
+ "path": "audits[non-composited-animations].details.items[0].subItems.items[4].failureReason"
+ }
+ ],
+ "core/audits/unsized-images.js | title": [
+ "audits[unsized-images].title"
+ ],
+ "core/audits/unsized-images.js | description": [
+ "audits[unsized-images].description"
+ ],
+ "core/audits/prioritize-lcp-image.js | title": [
+ "audits[prioritize-lcp-image].title"
+ ],
+ "core/audits/prioritize-lcp-image.js | description": [
+ "audits[prioritize-lcp-image].description"
+ ],
+ "core/audits/byte-efficiency/uses-long-cache-ttl.js | failureTitle": [
+ "audits[uses-long-cache-ttl].title"
+ ],
+ "core/audits/byte-efficiency/uses-long-cache-ttl.js | description": [
+ "audits[uses-long-cache-ttl].description"
+ ],
+ "core/audits/byte-efficiency/uses-long-cache-ttl.js | displayValue": [
+ {
+ "values": {
+ "itemCount": 2
+ },
+ "path": "audits[uses-long-cache-ttl].displayValue"
+ }
+ ],
+ "core/lib/i18n/i18n.js | columnCacheTTL": [
+ "audits[uses-long-cache-ttl].details.headings[1].label",
+ "audits[cache-insight].details.headings[1].label"
+ ],
+ "core/audits/byte-efficiency/total-byte-weight.js | failureTitle": [
+ "audits[total-byte-weight].title"
+ ],
+ "core/audits/byte-efficiency/total-byte-weight.js | description": [
+ "audits[total-byte-weight].description"
+ ],
+ "core/audits/byte-efficiency/total-byte-weight.js | displayValue": [
+ {
+ "values": {
+ "totalBytes": 7431629
+ },
+ "path": "audits[total-byte-weight].displayValue"
+ }
+ ],
+ "core/audits/byte-efficiency/offscreen-images.js | title": [
+ "audits[offscreen-images].title"
+ ],
+ "core/audits/byte-efficiency/offscreen-images.js | description": [
+ "audits[offscreen-images].description"
+ ],
+ "core/audits/byte-efficiency/render-blocking-resources.js | title": [
+ "audits[render-blocking-resources].title"
+ ],
+ "core/audits/byte-efficiency/render-blocking-resources.js | description": [
+ "audits[render-blocking-resources].description"
+ ],
+ "core/audits/byte-efficiency/unminified-css.js | title": [
+ "audits[unminified-css].title"
+ ],
+ "core/audits/byte-efficiency/unminified-css.js | description": [
+ "audits[unminified-css].description"
+ ],
+ "core/lib/i18n/i18n.js | displayValueByteSavings": [
+ {
+ "values": {
+ "wastedBytes": 37166
+ },
+ "path": "audits[unminified-css].displayValue"
+ },
+ {
+ "values": {
+ "wastedBytes": 45936
+ },
+ "path": "audits[unused-css-rules].displayValue"
+ },
+ {
+ "values": {
+ "wastedBytes": 21175
+ },
+ "path": "audits[unused-javascript].displayValue"
+ },
+ {
+ "values": {
+ "wastedBytes": 243692.5
+ },
+ "path": "audits[modern-image-formats].displayValue"
+ },
+ {
+ "values": {
+ "wastedBytes": 446
+ },
+ "path": "audits[legacy-javascript].displayValue"
+ },
+ {
+ "values": {
+ "wastedBytes": 4113021
+ },
+ "path": "audits[cache-insight].displayValue"
+ },
+ {
+ "values": {
+ "wastedBytes": 191341
+ },
+ "path": "audits[image-delivery-insight].displayValue"
+ }
+ ],
+ "core/audits/byte-efficiency/unminified-javascript.js | title": [
+ "audits[unminified-javascript].title"
+ ],
+ "core/audits/byte-efficiency/unminified-javascript.js | description": [
+ "audits[unminified-javascript].description"
+ ],
+ "core/audits/byte-efficiency/unused-css-rules.js | title": [
+ "audits[unused-css-rules].title"
+ ],
+ "core/audits/byte-efficiency/unused-css-rules.js | description": [
+ "audits[unused-css-rules].description"
+ ],
+ "core/audits/byte-efficiency/unused-javascript.js | title": [
+ "audits[unused-javascript].title"
+ ],
+ "core/audits/byte-efficiency/unused-javascript.js | description": [
+ "audits[unused-javascript].description"
+ ],
+ "core/audits/byte-efficiency/modern-image-formats.js | title": [
+ "audits[modern-image-formats].title"
+ ],
+ "core/audits/byte-efficiency/modern-image-formats.js | description": [
+ "audits[modern-image-formats].description"
+ ],
+ "core/lib/i18n/i18n.js | columnResourceSize": [
+ "audits[modern-image-formats].details.headings[2].label",
+ "audits[image-delivery-insight].details.headings[1].label"
+ ],
+ "core/audits/byte-efficiency/uses-optimized-images.js | title": [
+ "audits[uses-optimized-images].title"
+ ],
+ "core/audits/byte-efficiency/uses-optimized-images.js | description": [
+ "audits[uses-optimized-images].description"
+ ],
+ "core/audits/byte-efficiency/uses-text-compression.js | title": [
+ "audits[uses-text-compression].title"
+ ],
+ "core/audits/byte-efficiency/uses-text-compression.js | description": [
+ "audits[uses-text-compression].description"
+ ],
+ "core/audits/byte-efficiency/uses-responsive-images.js | title": [
+ "audits[uses-responsive-images].title"
+ ],
+ "core/audits/byte-efficiency/uses-responsive-images.js | description": [
+ "audits[uses-responsive-images].description"
+ ],
+ "core/audits/byte-efficiency/efficient-animated-content.js | title": [
+ "audits[efficient-animated-content].title"
+ ],
+ "core/audits/byte-efficiency/efficient-animated-content.js | description": [
+ "audits[efficient-animated-content].description"
+ ],
+ "core/audits/byte-efficiency/duplicated-javascript.js | title": [
+ "audits[duplicated-javascript].title"
+ ],
+ "core/audits/byte-efficiency/duplicated-javascript.js | description": [
+ "audits[duplicated-javascript].description"
+ ],
+ "core/audits/byte-efficiency/legacy-javascript.js | title": [
+ "audits[legacy-javascript].title"
+ ],
+ "core/audits/byte-efficiency/legacy-javascript.js | description": [
+ "audits[legacy-javascript].description"
+ ],
+ "core/audits/dobetterweb/dom-size.js | title": [
+ "audits[dom-size].title"
+ ],
+ "core/audits/dobetterweb/dom-size.js | description": [
+ "audits[dom-size].description"
+ ],
+ "core/audits/dobetterweb/dom-size.js | displayValue": [
+ {
+ "values": {
+ "itemCount": 125
+ },
+ "path": "audits[dom-size].displayValue"
+ }
+ ],
+ "core/audits/dobetterweb/dom-size.js | columnStatistic": [
+ "audits[dom-size].details.headings[0].label"
+ ],
+ "core/audits/dobetterweb/dom-size.js | columnValue": [
+ "audits[dom-size].details.headings[2].label"
+ ],
+ "core/audits/dobetterweb/dom-size.js | statisticDOMElements": [
+ "audits[dom-size].details.items[0].statistic"
+ ],
+ "core/audits/dobetterweb/dom-size.js | statisticDOMDepth": [
+ "audits[dom-size].details.items[1].statistic"
+ ],
+ "core/audits/dobetterweb/dom-size.js | statisticDOMWidth": [
+ "audits[dom-size].details.items[2].statistic"
+ ],
+ "core/audits/dobetterweb/no-document-write.js | title": [
+ "audits[no-document-write].title"
+ ],
+ "core/audits/dobetterweb/no-document-write.js | description": [
+ "audits[no-document-write].description"
+ ],
+ "core/lib/i18n/i18n.js | columnSource": [
+ "audits[no-document-write].details.headings[0].label",
+ "audits[uses-passive-event-listeners].details.headings[0].label",
+ "audits[forced-reflow-insight].details.items[0].headings[0].label"
+ ],
+ "core/audits/dobetterweb/uses-http2.js | title": [
+ "audits[uses-http2].title"
+ ],
+ "core/audits/dobetterweb/uses-http2.js | description": [
+ "audits[uses-http2].description"
+ ],
+ "core/audits/dobetterweb/uses-http2.js | displayValue": [
+ {
+ "values": {
+ "itemCount": 83
+ },
+ "path": "audits[uses-http2].displayValue"
+ }
+ ],
+ "core/audits/dobetterweb/uses-http2.js | columnProtocol": [
+ "audits[uses-http2].details.headings[1].label"
+ ],
+ "core/audits/dobetterweb/uses-passive-event-listeners.js | title": [
+ "audits[uses-passive-event-listeners].title"
+ ],
+ "core/audits/dobetterweb/uses-passive-event-listeners.js | description": [
+ "audits[uses-passive-event-listeners].description"
+ ],
+ "core/audits/bf-cache.js | failureTitle": [
+ "audits[bf-cache].title"
+ ],
+ "core/audits/bf-cache.js | description": [
+ "audits[bf-cache].description"
+ ],
+ "core/audits/bf-cache.js | displayValue": [
+ {
+ "values": {
+ "itemCount": 1
+ },
+ "path": "audits[bf-cache].displayValue"
+ }
+ ],
+ "core/audits/bf-cache.js | failureReasonColumn": [
+ "audits[bf-cache].details.headings[0].label"
+ ],
+ "core/audits/bf-cache.js | failureTypeColumn": [
+ "audits[bf-cache].details.headings[1].label"
+ ],
+ "node_modules/@paulirish/trace_engine/panels/application/components/BackForwardCacheStrings.js | webSocket": [
+ "audits[bf-cache].details.items[0].reason"
+ ],
+ "core/audits/bf-cache.js | supportPendingFailureType": [
+ "audits[bf-cache].details.items[0].failureType"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/Cache.js | title": [
+ "audits[cache-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/Cache.js | description": [
+ "audits[cache-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/Cache.js | requestColumn": [
+ "audits[cache-insight].details.headings[0].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/CLSCulprits.js | title": [
+ "audits[cls-culprits-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/CLSCulprits.js | description": [
+ "audits[cls-culprits-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DocumentLatency.js | title": [
+ "audits[document-latency-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DocumentLatency.js | description": [
+ "audits[document-latency-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DocumentLatency.js | passingRedirects": [
+ "audits[document-latency-insight].details.items.noRedirects.label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DocumentLatency.js | passingServerResponseTime": [
+ {
+ "values": {
+ "PH1": "17 ms"
+ },
+ "path": "audits[document-latency-insight].details.items.serverResponseIsFast.label"
+ }
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DocumentLatency.js | passingTextCompression": [
+ "audits[document-latency-insight].details.items.usesCompression.label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DOMSize.js | title": [
+ "audits[dom-size-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DOMSize.js | description": [
+ "audits[dom-size-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DOMSize.js | statistic": [
+ "audits[dom-size-insight].details.headings[0].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DOMSize.js | value": [
+ "audits[dom-size-insight].details.headings[2].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DOMSize.js | totalElements": [
+ "audits[dom-size-insight].details.items[0].statistic"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DOMSize.js | maxChildren": [
+ "audits[dom-size-insight].details.items[1].statistic"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DOMSize.js | maxDOMDepth": [
+ "audits[dom-size-insight].details.items[2].statistic"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DuplicatedJavaScript.js | title": [
+ "audits[duplicated-javascript-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DuplicatedJavaScript.js | description": [
+ "audits[duplicated-javascript-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DuplicatedJavaScript.js | columnSource": [
+ "audits[duplicated-javascript-insight].details.headings[0].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/DuplicatedJavaScript.js | columnDuplicatedBytes": [
+ "audits[duplicated-javascript-insight].details.headings[1].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/FontDisplay.js | title": [
+ "audits[font-display-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/FontDisplay.js | description": [
+ "audits[font-display-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ForcedReflow.js | title": [
+ "audits[forced-reflow-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ForcedReflow.js | description": [
+ "audits[forced-reflow-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ForcedReflow.js | totalReflowTime": [
+ "audits[forced-reflow-insight].details.items[0].headings[1].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ImageDelivery.js | title": [
+ "audits[image-delivery-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ImageDelivery.js | description": [
+ "audits[image-delivery-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ImageDelivery.js | useModernFormat": [
+ "audits[image-delivery-insight].details.items[0].subItems.items[0].reason"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/INPBreakdown.js | title": [
+ "audits[inp-breakdown-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/INPBreakdown.js | description": [
+ "audits[inp-breakdown-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPBreakdown.js | title": [
+ "audits[lcp-breakdown-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPBreakdown.js | description": [
+ "audits[lcp-breakdown-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPBreakdown.js | subpart": [
+ "audits[lcp-breakdown-insight].details.items[0].headings[0].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPBreakdown.js | timeToFirstByte": [
+ "audits[lcp-breakdown-insight].details.items[0].items[0].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPBreakdown.js | resourceLoadDelay": [
+ "audits[lcp-breakdown-insight].details.items[0].items[1].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPBreakdown.js | resourceLoadDuration": [
+ "audits[lcp-breakdown-insight].details.items[0].items[2].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPBreakdown.js | elementRenderDelay": [
+ "audits[lcp-breakdown-insight].details.items[0].items[3].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPDiscovery.js | title": [
+ "audits[lcp-discovery-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPDiscovery.js | description": [
+ "audits[lcp-discovery-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPDiscovery.js | fetchPriorityShouldBeApplied": [
+ "audits[lcp-discovery-insight].details.items[0].items.priorityHinted.label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPDiscovery.js | requestDiscoverable": [
+ "audits[lcp-discovery-insight].details.items[0].items.requestDiscoverable.label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LCPDiscovery.js | lazyLoadNotApplied": [
+ "audits[lcp-discovery-insight].details.items[0].items.eagerlyLoaded.label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LegacyJavaScript.js | title": [
+ "audits[legacy-javascript-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LegacyJavaScript.js | description": [
+ "audits[legacy-javascript-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/LegacyJavaScript.js | columnWastedBytes": [
+ "audits[legacy-javascript-insight].details.headings[2].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ModernHTTP.js | title": [
+ "audits[modern-http-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ModernHTTP.js | description": [
+ "audits[modern-http-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ModernHTTP.js | protocol": [
+ "audits[modern-http-insight].details.headings[1].label"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | title": [
+ "audits[network-dependency-tree-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | description": [
+ "audits[network-dependency-tree-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | preconnectOriginsTableTitle": [
+ "audits[network-dependency-tree-insight].details.items[1].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | preconnectOriginsTableDescription": [
+ "audits[network-dependency-tree-insight].details.items[1].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | noPreconnectOrigins": [
+ "audits[network-dependency-tree-insight].details.items[1].value.value"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | estSavingTableTitle": [
+ "audits[network-dependency-tree-insight].details.items[2].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | estSavingTableDescription": [
+ "audits[network-dependency-tree-insight].details.items[2].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/NetworkDependencyTree.js | noPreconnectCandidates": [
+ "audits[network-dependency-tree-insight].details.items[2].value.value"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/RenderBlocking.js | title": [
+ "audits[render-blocking-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/RenderBlocking.js | description": [
+ "audits[render-blocking-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ThirdParties.js | title": [
+ "audits[third-parties-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/ThirdParties.js | description": [
+ "audits[third-parties-insight].description"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/Viewport.js | title": [
+ "audits[viewport-insight].title"
+ ],
+ "node_modules/@paulirish/trace_engine/models/trace/insights/Viewport.js | description": [
+ "audits[viewport-insight].description"
+ ],
+ "core/config/default-config.js | performanceCategoryTitle": [
+ "categories.performance.title"
+ ],
+ "core/config/default-config.js | metricGroupTitle": [
+ "categoryGroups.metrics.title"
+ ],
+ "core/config/default-config.js | insightsGroupTitle": [
+ "categoryGroups.insights.title"
+ ],
+ "core/config/default-config.js | insightsGroupDescription": [
+ "categoryGroups.insights.description"
+ ],
+ "core/config/default-config.js | diagnosticsGroupTitle": [
+ "categoryGroups.diagnostics.title"
+ ],
+ "core/config/default-config.js | diagnosticsGroupDescription": [
+ "categoryGroups.diagnostics.description"
+ ],
+ "core/config/default-config.js | a11yBestPracticesGroupTitle": [
+ "categoryGroups[a11y-best-practices].title"
+ ],
+ "core/config/default-config.js | a11yBestPracticesGroupDescription": [
+ "categoryGroups[a11y-best-practices].description"
+ ],
+ "core/config/default-config.js | a11yColorContrastGroupTitle": [
+ "categoryGroups[a11y-color-contrast].title"
+ ],
+ "core/config/default-config.js | a11yColorContrastGroupDescription": [
+ "categoryGroups[a11y-color-contrast].description"
+ ],
+ "core/config/default-config.js | a11yNamesLabelsGroupTitle": [
+ "categoryGroups[a11y-names-labels].title"
+ ],
+ "core/config/default-config.js | a11yNamesLabelsGroupDescription": [
+ "categoryGroups[a11y-names-labels].description"
+ ],
+ "core/config/default-config.js | a11yNavigationGroupTitle": [
+ "categoryGroups[a11y-navigation].title"
+ ],
+ "core/config/default-config.js | a11yNavigationGroupDescription": [
+ "categoryGroups[a11y-navigation].description"
+ ],
+ "core/config/default-config.js | a11yAriaGroupTitle": [
+ "categoryGroups[a11y-aria].title"
+ ],
+ "core/config/default-config.js | a11yAriaGroupDescription": [
+ "categoryGroups[a11y-aria].description"
+ ],
+ "core/config/default-config.js | a11yLanguageGroupTitle": [
+ "categoryGroups[a11y-language].title"
+ ],
+ "core/config/default-config.js | a11yLanguageGroupDescription": [
+ "categoryGroups[a11y-language].description"
+ ],
+ "core/config/default-config.js | a11yAudioVideoGroupTitle": [
+ "categoryGroups[a11y-audio-video].title"
+ ],
+ "core/config/default-config.js | a11yAudioVideoGroupDescription": [
+ "categoryGroups[a11y-audio-video].description"
+ ],
+ "core/config/default-config.js | a11yTablesListsVideoGroupTitle": [
+ "categoryGroups[a11y-tables-lists].title"
+ ],
+ "core/config/default-config.js | a11yTablesListsVideoGroupDescription": [
+ "categoryGroups[a11y-tables-lists].description"
+ ],
+ "core/config/default-config.js | seoMobileGroupTitle": [
+ "categoryGroups[seo-mobile].title"
+ ],
+ "core/config/default-config.js | seoMobileGroupDescription": [
+ "categoryGroups[seo-mobile].description"
+ ],
+ "core/config/default-config.js | seoContentGroupTitle": [
+ "categoryGroups[seo-content].title"
+ ],
+ "core/config/default-config.js | seoContentGroupDescription": [
+ "categoryGroups[seo-content].description"
+ ],
+ "core/config/default-config.js | seoCrawlingGroupTitle": [
+ "categoryGroups[seo-crawl].title"
+ ],
+ "core/config/default-config.js | seoCrawlingGroupDescription": [
+ "categoryGroups[seo-crawl].description"
+ ],
+ "core/config/default-config.js | bestPracticesTrustSafetyGroupTitle": [
+ "categoryGroups[best-practices-trust-safety].title"
+ ],
+ "core/config/default-config.js | bestPracticesUXGroupTitle": [
+ "categoryGroups[best-practices-ux].title"
+ ],
+ "core/config/default-config.js | bestPracticesBrowserCompatGroupTitle": [
+ "categoryGroups[best-practices-browser-compat].title"
+ ],
+ "core/config/default-config.js | bestPracticesGeneralGroupTitle": [
+ "categoryGroups[best-practices-general].title"
+ ]
+ }
+ }
+}
\ No newline at end of file
diff --git a/optimize-images.js b/optimize-images.js
new file mode 100644
index 00000000..31dec1ab
--- /dev/null
+++ b/optimize-images.js
@@ -0,0 +1,129 @@
+// 图片优化脚本 - 使用sharp压缩PNG图片
+const sharp = require('sharp');
+const fs = require('fs');
+const path = require('path');
+
+// 需要优化的大图片列表 (> 500KB)
+const LARGE_IMAGES = [
+ 'CoverImage.png',
+ 'BasicImage.png',
+ 'teams-image.png',
+ 'hand-background.png',
+ 'basic-auth.png',
+ 'BgMusicCard.png',
+ 'Landing2.png',
+ 'Landing3.png',
+ 'Landing1.png',
+ 'smart-home.png',
+ 'automotive-background-card.png'
+];
+
+const IMG_DIR = path.join(__dirname, 'src/assets/img');
+const BACKUP_DIR = path.join(IMG_DIR, 'original-backup');
+
+// 确保备份目录存在
+if (!fs.existsSync(BACKUP_DIR)) {
+ fs.mkdirSync(BACKUP_DIR, { recursive: true });
+}
+
+console.log('🎨 开始优化图片...');
+console.log('================================\n');
+
+let totalBefore = 0;
+let totalAfter = 0;
+let optimizedCount = 0;
+
+async function optimizeImage(filename) {
+ const srcPath = path.join(IMG_DIR, filename);
+ const backupPath = path.join(BACKUP_DIR, filename);
+
+ if (!fs.existsSync(srcPath)) {
+ console.log(`⚠️ 跳过: ${filename} (文件不存在)`);
+ return;
+ }
+
+ try {
+ // 获取原始大小
+ const beforeStats = fs.statSync(srcPath);
+ const beforeSize = beforeStats.size;
+ totalBefore += beforeSize;
+
+ // 备份原始文件
+ if (!fs.existsSync(backupPath)) {
+ fs.copyFileSync(srcPath, backupPath);
+ }
+
+ // 读取图片元数据
+ const metadata = await sharp(srcPath).metadata();
+
+ // 优化策略:
+ // 1. 如果宽度 > 2000px,缩放到 2000px
+ // 2. 压缩质量到 85
+ // 3. 使用 pngquant 算法压缩
+ let pipeline = sharp(srcPath);
+
+ if (metadata.width > 2000) {
+ pipeline = pipeline.resize(2000, null, {
+ withoutEnlargement: true,
+ fit: 'inside'
+ });
+ }
+
+ // PNG优化
+ pipeline = pipeline.png({
+ quality: 85,
+ compressionLevel: 9,
+ adaptiveFiltering: true,
+ force: true
+ });
+
+ // 保存优化后的图片
+ await pipeline.toFile(srcPath + '.tmp');
+
+ // 替换原文件
+ fs.renameSync(srcPath + '.tmp', srcPath);
+
+ // 获取优化后的大小
+ const afterStats = fs.statSync(srcPath);
+ const afterSize = afterStats.size;
+ totalAfter += afterSize;
+
+ // 计算节省的大小
+ const saved = beforeSize - afterSize;
+ const percent = Math.round((saved / beforeSize) * 100);
+
+ if (saved > 0) {
+ optimizedCount++;
+ console.log(`✅ ${filename}`);
+ console.log(` ${Math.round(beforeSize/1024)} KB → ${Math.round(afterSize/1024)} KB`);
+ console.log(` 节省: ${Math.round(saved/1024)} KB (-${percent}%)\n`);
+ } else {
+ console.log(`ℹ️ ${filename} - 已经是最优化状态\n`);
+ }
+
+ } catch (error) {
+ console.error(`❌ ${filename} 优化失败:`, error.message);
+ }
+}
+
+async function main() {
+ // 依次优化每个图片
+ for (const img of LARGE_IMAGES) {
+ await optimizeImage(img);
+ }
+
+ console.log('================================');
+ console.log('📊 优化总结:\n');
+ console.log(` 优化前总大小: ${Math.round(totalBefore/1024/1024)} MB`);
+ console.log(` 优化后总大小: ${Math.round(totalAfter/1024/1024)} MB`);
+
+ const totalSaved = totalBefore - totalAfter;
+ const totalPercent = Math.round((totalSaved / totalBefore) * 100);
+
+ console.log(` 节省空间: ${Math.round(totalSaved/1024/1024)} MB (-${totalPercent}%)`);
+ console.log(` 成功优化: ${optimizedCount}/${LARGE_IMAGES.length} 个文件\n`);
+ console.log('✅ 图片优化完成!');
+ console.log(`📁 原始文件已备份到: ${BACKUP_DIR}\n`);
+}
+
+main().catch(console.error);
diff --git a/package.json b/package.json
index 6a85a5fc..d7633a9d 100755
--- a/package.json
+++ b/package.json
@@ -36,6 +36,7 @@
"framer-motion": "^4.1.17",
"fullcalendar": "^5.9.0",
"globalize": "^1.7.0",
+ "history": "^5.3.0",
"leaflet": "^1.9.4",
"lucide-react": "^0.540.0",
"match-sorter": "6.3.0",
@@ -60,7 +61,7 @@
"react-quill": "^2.0.0-beta.4",
"react-responsive": "^10.0.1",
"react-responsive-masonry": "^2.7.1",
- "react-router-dom": "^6.4.0",
+ "react-router-dom": "^6.30.1",
"react-scripts": "^5.0.1",
"react-scroll": "^1.8.4",
"react-scroll-into-view": "^2.1.3",
@@ -85,10 +86,14 @@
"@types/react": "18.2.0",
"@types/react-dom": "18.2.0"
},
+ "overrides": {
+ "uuid": "^9.0.1"
+ },
"scripts": {
- "start": "react-scripts --openssl-legacy-provider start",
- "build": "react-scripts build && gulp licenses",
- "test": "react-scripts test --env=jsdom",
+ "start": "NODE_OPTIONS='--openssl-legacy-provider --max_old_space_size=4096' craco start",
+ "build": "NODE_OPTIONS='--openssl-legacy-provider --max_old_space_size=4096' craco build && gulp licenses",
+ "build:analyze": "NODE_OPTIONS='--openssl-legacy-provider --max_old_space_size=4096' ANALYZE=true craco build",
+ "test": "craco test --env=jsdom",
"eject": "react-scripts eject",
"deploy": "npm run build",
"lint:check": "eslint . --ext=js,jsx; exit 0",
@@ -96,16 +101,24 @@
"install:clean": "rm -rf node_modules/ && rm -rf package-lock.json && npm install && npm start"
},
"devDependencies": {
+ "@craco/craco": "^7.1.0",
"ajv": "^8.17.1",
"autoprefixer": "^10.4.21",
"eslint-config-prettier": "8.3.0",
"eslint-plugin-prettier": "3.4.0",
"gulp": "4.0.2",
"gulp-append-prepend": "1.0.9",
+ "imagemin": "^9.0.1",
+ "imagemin-mozjpeg": "^10.0.0",
+ "imagemin-pngquant": "^10.0.0",
"postcss": "^8.5.6",
"prettier": "2.2.1",
"react-error-overlay": "6.0.9",
- "tailwindcss": "^3.4.17"
+ "sharp": "^0.34.4",
+ "tailwindcss": "^3.4.17",
+ "ts-node": "^10.9.2",
+ "webpack-bundle-analyzer": "^4.10.2",
+ "yn": "^5.1.0"
},
"browserslist": {
"production": [
diff --git a/serve.log b/serve.log
new file mode 100644
index 00000000..012db7e5
--- /dev/null
+++ b/serve.log
@@ -0,0 +1,3 @@
+ INFO Accepting connections at http://localhost:58321
+
+ INFO Gracefully shutting down. Please wait...
diff --git a/src/App.js b/src/App.js
index d070db22..722443e7 100755
--- a/src/App.js
+++ b/src/App.js
@@ -9,7 +9,7 @@
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Visionware.
*/
-import React from "react";
+import React, { Suspense, useEffect } from "react";
import { ChakraProvider } from '@chakra-ui/react';
import { Routes, Route, Navigate } from "react-router-dom";
@@ -19,34 +19,44 @@ import { Box, useColorMode } from '@chakra-ui/react';
// Core Components
import theme from "theme/theme.js";
-// Layouts
+// Loading Component
+import PageLoader from "components/Loading/PageLoader";
+
+// Layouts - 保持同步导入(需要立即加载)
import Admin from "layouts/Admin";
import Auth from "layouts/Auth";
import HomeLayout from "layouts/Home";
-// Views
-import Community from "views/Community";
-import LimitAnalyse from "views/LimitAnalyse";
-import ForecastReport from "views/Company/ForecastReport";
-import ConceptCenter from "views/Concept";
-import FinancialPanorama from "views/Company/FinancialPanorama";
-import CompanyIndex from "views/Company";
-import MarketDataView from "views/Company/MarketDataView";
-import StockOverview from "views/StockOverview";
-import EventDetail from "views/EventDetail";
-import TradingSimulation from "views/TradingSimulation";
+// ⚡ 使用 React.lazy() 实现路由懒加载
+// 首屏不需要的组件按需加载,大幅减少初始 JS 包大小
+const Community = React.lazy(() => import("views/Community"));
+const LimitAnalyse = React.lazy(() => import("views/LimitAnalyse"));
+const ForecastReport = React.lazy(() => import("views/Company/ForecastReport"));
+const ConceptCenter = React.lazy(() => import("views/Concept"));
+const FinancialPanorama = React.lazy(() => import("views/Company/FinancialPanorama"));
+const CompanyIndex = React.lazy(() => import("views/Company"));
+const MarketDataView = React.lazy(() => import("views/Company/MarketDataView"));
+const StockOverview = React.lazy(() => import("views/StockOverview"));
+const EventDetail = React.lazy(() => import("views/EventDetail"));
+const TradingSimulation = React.lazy(() => import("views/TradingSimulation"));
+
// Contexts
import { AuthProvider } from "contexts/AuthContext";
+import { AuthModalProvider } from "contexts/AuthModalContext";
// Components
import ProtectedRoute from "components/ProtectedRoute";
+import ErrorBoundary from "components/ErrorBoundary";
+import AuthModalManager from "components/Auth/AuthModalManager";
function AppContent() {
const { colorMode } = useColorMode();
return (
-
+ {/* ⚡ Suspense 边界:懒加载组件加载时显示 Loading */}
+ }>
+
{/* 首页路由 */}
} />
@@ -110,14 +120,10 @@ function AppContent() {
{/* 事件详情独立页面路由(不经 Admin 布局) */}
} />
- {/* 模拟盘交易系统路由 - 需要登录 */}
-
-
-
- }
+ {/* 模拟盘交易系统路由 - 无需登录 */}
+ }
/>
{/* 管理后台路由 - 需要登录 */}
@@ -139,16 +145,45 @@ function AppContent() {
{/* 404 页面 */}
} />
+
);
}
export default function App() {
+ // 全局错误处理:捕获未处理的 Promise rejection
+ useEffect(() => {
+ const handleUnhandledRejection = (event) => {
+ console.error('未捕获的 Promise rejection:', event.reason);
+ // 阻止默认的错误处理(防止崩溃)
+ event.preventDefault();
+ };
+
+ const handleError = (event) => {
+ console.error('全局错误:', event.error);
+ // 阻止默认的错误处理(防止崩溃)
+ event.preventDefault();
+ };
+
+ window.addEventListener('unhandledrejection', handleUnhandledRejection);
+ window.addEventListener('error', handleError);
+
+ return () => {
+ window.removeEventListener('unhandledrejection', handleUnhandledRejection);
+ window.removeEventListener('error', handleError);
+ };
+ }, []);
+
return (
-
-
-
+
+
+
+
+
+
+
+
);
}
\ No newline at end of file
diff --git a/src/assets/img/BasicImage.png b/src/assets/img/BasicImage.png
old mode 100755
new mode 100644
index 7b82e11c..7226bff7
Binary files a/src/assets/img/BasicImage.png and b/src/assets/img/BasicImage.png differ
diff --git a/src/assets/img/BgMusicCard.png b/src/assets/img/BgMusicCard.png
old mode 100755
new mode 100644
index 6488b4e8..04ba9628
Binary files a/src/assets/img/BgMusicCard.png and b/src/assets/img/BgMusicCard.png differ
diff --git a/src/assets/img/CoverImage.png b/src/assets/img/CoverImage.png
old mode 100755
new mode 100644
index cbf02623..3937ce0f
Binary files a/src/assets/img/CoverImage.png and b/src/assets/img/CoverImage.png differ
diff --git a/src/assets/img/Landing1.png b/src/assets/img/Landing1.png
old mode 100755
new mode 100644
index 987c4cbc..9877e802
Binary files a/src/assets/img/Landing1.png and b/src/assets/img/Landing1.png differ
diff --git a/src/assets/img/Landing2.png b/src/assets/img/Landing2.png
old mode 100755
new mode 100644
index 8cd7caba..27bd2d9b
Binary files a/src/assets/img/Landing2.png and b/src/assets/img/Landing2.png differ
diff --git a/src/assets/img/Landing3.png b/src/assets/img/Landing3.png
old mode 100755
new mode 100644
index 72d5e527..705dff3c
Binary files a/src/assets/img/Landing3.png and b/src/assets/img/Landing3.png differ
diff --git a/src/assets/img/automotive-background-card.png b/src/assets/img/automotive-background-card.png
old mode 100755
new mode 100644
index 669b68b6..f4cb6c29
Binary files a/src/assets/img/automotive-background-card.png and b/src/assets/img/automotive-background-card.png differ
diff --git a/src/assets/img/basic-auth.png b/src/assets/img/basic-auth.png
old mode 100755
new mode 100644
index 3df52455..4332d738
Binary files a/src/assets/img/basic-auth.png and b/src/assets/img/basic-auth.png differ
diff --git a/src/assets/img/hand-background.png b/src/assets/img/hand-background.png
old mode 100755
new mode 100644
index 8e05de33..8028b6f4
Binary files a/src/assets/img/hand-background.png and b/src/assets/img/hand-background.png differ
diff --git a/src/assets/img/original-backup/BasicImage.png b/src/assets/img/original-backup/BasicImage.png
new file mode 100755
index 00000000..7b82e11c
Binary files /dev/null and b/src/assets/img/original-backup/BasicImage.png differ
diff --git a/src/assets/img/original-backup/BgMusicCard.png b/src/assets/img/original-backup/BgMusicCard.png
new file mode 100755
index 00000000..6488b4e8
Binary files /dev/null and b/src/assets/img/original-backup/BgMusicCard.png differ
diff --git a/src/assets/img/original-backup/CoverImage.png b/src/assets/img/original-backup/CoverImage.png
new file mode 100755
index 00000000..cbf02623
Binary files /dev/null and b/src/assets/img/original-backup/CoverImage.png differ
diff --git a/src/assets/img/original-backup/Landing1.png b/src/assets/img/original-backup/Landing1.png
new file mode 100755
index 00000000..987c4cbc
Binary files /dev/null and b/src/assets/img/original-backup/Landing1.png differ
diff --git a/src/assets/img/original-backup/Landing2.png b/src/assets/img/original-backup/Landing2.png
new file mode 100755
index 00000000..8cd7caba
Binary files /dev/null and b/src/assets/img/original-backup/Landing2.png differ
diff --git a/src/assets/img/original-backup/Landing3.png b/src/assets/img/original-backup/Landing3.png
new file mode 100755
index 00000000..72d5e527
Binary files /dev/null and b/src/assets/img/original-backup/Landing3.png differ
diff --git a/src/assets/img/original-backup/automotive-background-card.png b/src/assets/img/original-backup/automotive-background-card.png
new file mode 100755
index 00000000..669b68b6
Binary files /dev/null and b/src/assets/img/original-backup/automotive-background-card.png differ
diff --git a/src/assets/img/original-backup/basic-auth.png b/src/assets/img/original-backup/basic-auth.png
new file mode 100755
index 00000000..3df52455
Binary files /dev/null and b/src/assets/img/original-backup/basic-auth.png differ
diff --git a/src/assets/img/original-backup/hand-background.png b/src/assets/img/original-backup/hand-background.png
new file mode 100755
index 00000000..8e05de33
Binary files /dev/null and b/src/assets/img/original-backup/hand-background.png differ
diff --git a/src/assets/img/original-backup/smart-home.png b/src/assets/img/original-backup/smart-home.png
new file mode 100755
index 00000000..c11f53ea
Binary files /dev/null and b/src/assets/img/original-backup/smart-home.png differ
diff --git a/src/assets/img/original-backup/teams-image.png b/src/assets/img/original-backup/teams-image.png
new file mode 100755
index 00000000..9b5b3c21
Binary files /dev/null and b/src/assets/img/original-backup/teams-image.png differ
diff --git a/src/assets/img/smart-home.png b/src/assets/img/smart-home.png
old mode 100755
new mode 100644
index c11f53ea..769e30e9
Binary files a/src/assets/img/smart-home.png and b/src/assets/img/smart-home.png differ
diff --git a/src/assets/img/teams-image.png b/src/assets/img/teams-image.png
old mode 100755
new mode 100644
index 9b5b3c21..1a6695ad
Binary files a/src/assets/img/teams-image.png and b/src/assets/img/teams-image.png differ
diff --git a/src/components/Auth/AuthBackground.js b/src/components/Auth/AuthBackground.js
new file mode 100644
index 00000000..83533e21
--- /dev/null
+++ b/src/components/Auth/AuthBackground.js
@@ -0,0 +1,55 @@
+// src/components/Auth/AuthBackground.js
+import React from "react";
+import { Box } from "@chakra-ui/react";
+
+/**
+ * 认证页面通用背景组件
+ * 用于登录和注册页面的动态渐变背景
+ */
+export default function AuthBackground() {
+ return (
+
+ );
+}
diff --git a/src/components/Auth/AuthFooter.js b/src/components/Auth/AuthFooter.js
new file mode 100644
index 00000000..099267e7
--- /dev/null
+++ b/src/components/Auth/AuthFooter.js
@@ -0,0 +1,58 @@
+import React from "react";
+import { HStack, Text, Link as ChakraLink } from "@chakra-ui/react";
+import { Link } from "react-router-dom";
+
+/**
+ * 认证页面底部组件
+ * 包含页面切换链接和登录方式切换链接
+ *
+ * 支持两种模式:
+ * 1. 页面模式:使用 linkTo 进行路由跳转
+ * 2. 弹窗模式:使用 onClick 进行弹窗切换
+ */
+export default function AuthFooter({
+ // 左侧链接配置
+ linkText, // 提示文本,如 "还没有账号," 或 "已有账号?"
+ linkLabel, // 链接文本,如 "去注册" 或 "去登录"
+ linkTo, // 链接路径,如 "/auth/sign-up" 或 "/auth/sign-in"(页面模式)
+ onClick, // 点击回调函数(弹窗模式,优先级高于 linkTo)
+
+ // 右侧切换配置
+ useVerificationCode, // 当前是否使用验证码登录
+ onSwitchMethod // 切换登录方式的回调函数
+}) {
+ return (
+
+ {/* 左侧:页面切换链接(去注册/去登录) */}
+ {onClick ? (
+ // 弹窗模式:使用 onClick
+
+ {linkText}
+ {linkLabel}
+
+ ) : (
+ // 页面模式:使用 Link 组件跳转
+
+ {linkText}
+ {linkLabel}
+
+ )}
+
+ {/* 右侧:登录方式切换链接(仅在提供了切换方法时显示) */}
+ {onSwitchMethod && (
+ {
+ e.preventDefault();
+ onSwitchMethod();
+ }}
+ >
+ {useVerificationCode ? '密码登陆' : '验证码登陆'}
+
+ )}
+
+ );
+}
diff --git a/src/components/Auth/AuthFormContent.js b/src/components/Auth/AuthFormContent.js
new file mode 100644
index 00000000..38feb673
--- /dev/null
+++ b/src/components/Auth/AuthFormContent.js
@@ -0,0 +1,460 @@
+// src/components/Auth/AuthFormContent.js
+// 统一的认证表单组件
+import React, { useState, useEffect, useRef } from "react";
+import { useNavigate } from "react-router-dom";
+import {
+ Box,
+ Button,
+ FormControl,
+ Input,
+ Heading,
+ VStack,
+ HStack,
+ Stack,
+ useToast,
+ Icon,
+ FormErrorMessage,
+ Center,
+ AlertDialog,
+ AlertDialogBody,
+ AlertDialogFooter,
+ AlertDialogHeader,
+ AlertDialogContent,
+ AlertDialogOverlay,
+ Text,
+ Link as ChakraLink,
+ useBreakpointValue,
+ Divider,
+ IconButton,
+} from "@chakra-ui/react";
+import { FaLock, FaWeixin } from "react-icons/fa";
+import { useAuth } from "../../contexts/AuthContext";
+import { useAuthModal } from "../../contexts/AuthModalContext";
+import { authService } from "../../services/authService";
+import AuthHeader from './AuthHeader';
+import VerificationCodeInput from './VerificationCodeInput';
+import WechatRegister from './WechatRegister';
+
+// 统一配置对象
+const AUTH_CONFIG = {
+ // UI文本
+ title: "价值前沿",
+ subtitle: "开启您的投资之旅",
+ formTitle: "登陆/注册",
+ buttonText: "登录/注册",
+ loadingText: "验证中...",
+ successTitle: "验证成功",
+ successDescription: "欢迎!",
+ errorTitle: "验证失败",
+
+ // API配置
+ api: {
+ endpoint: '/api/auth/register/phone',
+ purpose: 'login', // ⚡ 统一使用 'login' 模式
+ },
+
+ // 功能开关
+ features: {
+ successDelay: 1000, // 延迟1秒显示成功提示
+ }
+};
+
+export default function AuthFormContent() {
+ const toast = useToast();
+ const navigate = useNavigate();
+ const { checkSession } = useAuth();
+ const { handleLoginSuccess } = useAuthModal();
+
+ // 使用统一配置
+ const config = AUTH_CONFIG;
+
+ // 追踪组件挂载状态,防止内存泄漏
+ const isMountedRef = useRef(true);
+ const cancelRef = useRef(); // AlertDialog 需要的 ref
+
+ // 页面状态
+ const [isLoading, setIsLoading] = useState(false);
+ const [errors, setErrors] = useState({});
+
+ // 昵称设置引导对话框
+ const [showNicknamePrompt, setShowNicknamePrompt] = useState(false);
+ const [currentPhone, setCurrentPhone] = useState("");
+
+ // 响应式布局配置
+ const isMobile = useBreakpointValue({ base: true, md: false });
+ const stackDirection = useBreakpointValue({ base: "column", md: "row" });
+ const stackSpacing = useBreakpointValue({ base: 4, md: 8 });
+
+ // 表单数据
+ const [formData, setFormData] = useState({
+ phone: "",
+ verificationCode: "",
+ });
+
+ // 验证码状态
+ const [verificationCodeSent, setVerificationCodeSent] = useState(false);
+ const [sendingCode, setSendingCode] = useState(false);
+ const [countdown, setCountdown] = useState(0);
+
+ // 输入框变化处理
+ const handleInputChange = (e) => {
+ const { name, value } = e.target;
+ setFormData(prev => ({
+ ...prev,
+ [name]: value
+ }));
+ };
+
+ // 倒计时逻辑
+ useEffect(() => {
+ let timer;
+ let isMounted = true;
+
+ if (countdown > 0) {
+ timer = setInterval(() => {
+ if (isMounted) {
+ setCountdown(prev => prev - 1);
+ }
+ }, 1000);
+ } else if (countdown === 0 && isMounted) {
+ setVerificationCodeSent(false);
+ }
+
+ return () => {
+ isMounted = false;
+ if (timer) clearInterval(timer);
+ };
+ }, [countdown]);
+
+ // 发送验证码
+ const sendVerificationCode = async () => {
+ const credential = formData.phone;
+
+ if (!credential) {
+ toast({
+ title: "请先输入手机号",
+ status: "warning",
+ duration: 3000,
+ });
+ return;
+ }
+
+ if (!/^1[3-9]\d{9}$/.test(credential)) {
+ toast({
+ title: "请输入有效的手机号",
+ status: "warning",
+ duration: 3000,
+ });
+ return;
+ }
+
+ try {
+ setSendingCode(true);
+ const response = await fetch('/api/auth/send-verification-code', {
+ method: 'POST',
+ headers: {
+ 'Content-Type': 'application/json',
+ },
+ credentials: 'include', // 必须包含以支持跨域 session cookie
+ body: JSON.stringify({
+ credential,
+ type: 'phone',
+ purpose: config.api.purpose // 根据模式使用不同的purpose
+ }),
+ });
+
+ if (!response) {
+ throw new Error('网络请求失败,请检查网络连接');
+ }
+
+ const data = await response.json();
+
+ if (!isMountedRef.current) return;
+
+ if (!data) {
+ throw new Error('服务器响应为空');
+ }
+
+ if (response.ok && data.success) {
+ toast({
+ title: "验证码已发送",
+ description: "验证码已发送到您的手机号",
+ status: "success",
+ duration: 3000,
+ });
+ setVerificationCodeSent(true);
+ setCountdown(60);
+ } else {
+ throw new Error(data.error || '发送验证码失败');
+ }
+ } catch (error) {
+ if (isMountedRef.current) {
+ toast({
+ title: "发送验证码失败",
+ description: error.message || "请稍后重试",
+ status: "error",
+ duration: 3000,
+ });
+ }
+ } finally {
+ if (isMountedRef.current) {
+ setSendingCode(false);
+ }
+ }
+ };
+
+ // 提交处理(登录或注册)
+ const handleSubmit = async (e) => {
+ e.preventDefault();
+ setIsLoading(true);
+
+ try {
+ const { phone, verificationCode, nickname } = formData;
+
+ // 表单验证
+ if (!phone || !verificationCode) {
+ toast({
+ title: "请填写完整信息",
+ description: "手机号和验证码不能为空",
+ status: "warning",
+ duration: 3000,
+ });
+ return;
+ }
+
+ if (!/^1[3-9]\d{9}$/.test(phone)) {
+ toast({
+ title: "请输入有效的手机号",
+ status: "warning",
+ duration: 3000,
+ });
+ return;
+ }
+
+ // 构建请求体
+ const requestBody = {
+ credential: phone,
+ verification_code: verificationCode,
+ login_type: 'phone',
+ };
+
+ // 调用API(根据模式选择不同的endpoint
+ const response = await fetch('/api/auth/login-with-code', {
+ method: 'POST',
+ headers: {
+ 'Content-Type': 'application/json',
+ },
+ credentials: 'include', // 必须包含以支持跨域 session cookie
+ body: JSON.stringify(requestBody),
+ });
+
+ if (!response) {
+ throw new Error('网络请求失败,请检查网络连接');
+ }
+
+ const data = await response.json();
+
+ if (!isMountedRef.current) return;
+
+ if (!data) {
+ throw new Error('服务器响应为空');
+ }
+
+ if (response.ok && data.success) {
+ // 更新session
+ await checkSession();
+
+ toast({
+ title: config.successTitle,
+ description: config.successDescription,
+ status: "success",
+ duration: 2000,
+ });
+
+ // 检查是否为新注册用户
+ if (data.isNewUser) {
+ // 新注册用户,延迟后显示昵称设置引导
+ setTimeout(() => {
+ setCurrentPhone(phone);
+ setShowNicknamePrompt(true);
+ }, config.features.successDelay);
+ } else {
+ // 已有用户,直接登录成功
+ setTimeout(() => {
+ handleLoginSuccess({ phone });
+ }, config.features.successDelay);
+ }
+ } else {
+ throw new Error(data.error || `${config.errorTitle}`);
+ }
+ } catch (error) {
+ console.error('Auth error:', error);
+ if (isMountedRef.current) {
+ toast({
+ title: config.errorTitle,
+ description: error.message || "请稍后重试",
+ status: "error",
+ duration: 3000,
+ });
+ }
+ } finally {
+ if (isMountedRef.current) {
+ setIsLoading(false);
+ }
+ }
+ };
+
+ // 微信H5登录处理
+ const handleWechatH5Login = async () => {
+ try {
+ // 1. 构建回调URL
+ const redirectUrl = `${window.location.origin}/home/wechat-callback`;
+
+ // 2. 显示提示
+ toast({
+ title: "即将跳转",
+ description: "正在跳转到微信授权页面...",
+ status: "info",
+ duration: 2000,
+ isClosable: true,
+ });
+
+ // 3. 获取微信H5授权URL
+ const response = await authService.getWechatH5AuthUrl(redirectUrl);
+
+ if (!response || !response.auth_url) {
+ throw new Error('获取授权链接失败');
+ }
+
+ // 4. 延迟跳转,让用户看到提示
+ setTimeout(() => {
+ window.location.href = response.auth_url;
+ }, 500);
+ } catch (error) {
+ console.error('微信H5登录失败:', error);
+ toast({
+ title: "跳转失败",
+ description: error.message || "请稍后重试",
+ status: "error",
+ duration: 3000,
+ isClosable: true,
+ });
+ }
+ };
+
+ // 组件卸载时清理
+ useEffect(() => {
+ isMountedRef.current = true;
+
+ return () => {
+ isMountedRef.current = false;
+ };
+ }, []);
+
+ return (
+ <>
+
+
+
+
+
+
+
+ {/* 桌面端:右侧二维码扫描 */}
+ {!isMobile && (
+
+
+
+
+
+ )}
+
+
+
+ {/* 只在需要时才渲染 AlertDialog,避免创建不必要的 Portal */}
+ {showNicknamePrompt && (
+
{ setShowNicknamePrompt(false); handleLoginSuccess({ phone: currentPhone }); }} isCentered closeOnEsc={true} closeOnOverlayClick={false}>
+
+
+ 完善个人信息
+ 您已成功注册!是否前往个人中心设置昵称和其他信息?
+
+
+
+
+
+
+
+ )}
+ >
+ );
+}
diff --git a/src/components/Auth/AuthHeader.js b/src/components/Auth/AuthHeader.js
new file mode 100644
index 00000000..ba256868
--- /dev/null
+++ b/src/components/Auth/AuthHeader.js
@@ -0,0 +1,23 @@
+// src/components/Auth/AuthHeader.js
+import React from "react";
+import { Heading, Text, VStack } from "@chakra-ui/react";
+
+/**
+ * 认证页面通用头部组件
+ * 用于显示页面标题和描述
+ *
+ * @param {string} title - 主标题文字
+ * @param {string} subtitle - 副标题文字
+ */
+export default function AuthHeader({ title, subtitle }) {
+ return (
+
+
+ {title}
+
+
+ {subtitle}
+
+
+ );
+}
diff --git a/src/components/Auth/AuthModalManager.js b/src/components/Auth/AuthModalManager.js
new file mode 100644
index 00000000..40e38cd6
--- /dev/null
+++ b/src/components/Auth/AuthModalManager.js
@@ -0,0 +1,105 @@
+// src/components/Auth/AuthModalManager.js
+import React from 'react';
+import {
+ Modal,
+ ModalOverlay,
+ ModalContent,
+ ModalBody,
+ ModalCloseButton,
+ useBreakpointValue
+} from '@chakra-ui/react';
+import { useAuthModal } from '../../contexts/AuthModalContext';
+import AuthFormContent from './AuthFormContent';
+
+/**
+ * 全局认证弹窗管理器
+ * 统一的登录/注册弹窗
+ */
+export default function AuthModalManager() {
+ const {
+ isAuthModalOpen,
+ closeModal
+ } = useAuthModal();
+
+ // 响应式尺寸配置
+ const modalSize = useBreakpointValue({
+ base: "md", // 移动端:md(不占满全屏)
+ sm: "md", // 小屏:md
+ md: "lg", // 中屏:lg
+ lg: "xl" // 大屏:xl(更紧凑)
+ });
+
+ // 响应式宽度配置
+ const modalMaxW = useBreakpointValue({
+ base: "90%", // 移动端:屏幕宽度的90%
+ sm: "90%", // 小屏:90%
+ md: "700px", // 中屏:固定700px
+ lg: "700px" // 大屏:固定700px
+ });
+
+ // 响应式水平边距
+ const modalMx = useBreakpointValue({
+ base: 4, // 移动端:左右各16px边距
+ md: "auto" // 桌面端:自动居中
+ });
+
+ // 响应式垂直边距
+ const modalMy = useBreakpointValue({
+ base: 8, // 移动端:上下各32px边距
+ md: 8 // 桌面端:上下各32px边距
+ });
+
+ // 条件渲染:只在打开时才渲染 Modal,避免创建不必要的 Portal
+ if (!isAuthModalOpen) {
+ return null;
+ }
+
+ return (
+
+ {/* 半透明背景 + 模糊效果 */}
+
+
+ {/* 弹窗内容容器 */}
+
+ {/* 关闭按钮 */}
+
+
+ {/* 弹窗主体内容 */}
+
+
+
+
+
+ );
+}
diff --git a/src/components/Auth/VerificationCodeInput.js b/src/components/Auth/VerificationCodeInput.js
new file mode 100644
index 00000000..bc251055
--- /dev/null
+++ b/src/components/Auth/VerificationCodeInput.js
@@ -0,0 +1,56 @@
+import React from "react";
+import { FormControl, FormErrorMessage, HStack, Input, Button } from "@chakra-ui/react";
+
+/**
+ * 通用验证码输入组件
+ */
+export default function VerificationCodeInput({
+ value,
+ onChange,
+ onSendCode,
+ countdown,
+ isLoading,
+ isSending,
+ error,
+ placeholder = "请输入6位验证码",
+ buttonText = "获取验证码",
+ countdownText = (count) => `${count}s`,
+ colorScheme = "green",
+ isRequired = true
+}) {
+ // 包装 onSendCode,确保所有错误都被捕获,防止被 ErrorBoundary 捕获
+ const handleSendCode = async () => {
+ try {
+ if (onSendCode) {
+ await onSendCode();
+ }
+ } catch (error) {
+ // 错误已经在父组件处理,这里只需要防止未捕获的 Promise rejection
+ console.error('Send code error (caught in VerificationCodeInput):', error);
+ }
+ };
+
+ return (
+
+
+
+
+
+ {error}
+
+ );
+}
diff --git a/src/components/Auth/WechatRegister.js b/src/components/Auth/WechatRegister.js
new file mode 100644
index 00000000..c55689b1
--- /dev/null
+++ b/src/components/Auth/WechatRegister.js
@@ -0,0 +1,463 @@
+import React, { useState, useEffect, useLayoutEffect, useRef, useCallback } from "react";
+import {
+ Box,
+ Button,
+ VStack,
+ Text,
+ Icon,
+ useToast,
+ Spinner
+} from "@chakra-ui/react";
+import { FaQrcode } from "react-icons/fa";
+import { useNavigate } from "react-router-dom";
+import { authService, WECHAT_STATUS, STATUS_MESSAGES } from "../../services/authService";
+
+// 配置常量
+const POLL_INTERVAL = 2000; // 轮询间隔:2秒
+const BACKUP_POLL_INTERVAL = 3000; // 备用轮询间隔:3秒
+const QR_CODE_TIMEOUT = 300000; // 二维码超时:5分钟
+
+export default function WechatRegister() {
+ // 状态管理
+ const [wechatAuthUrl, setWechatAuthUrl] = useState("");
+ const [wechatSessionId, setWechatSessionId] = useState("");
+ const [wechatStatus, setWechatStatus] = useState(WECHAT_STATUS.NONE);
+ const [isLoading, setIsLoading] = useState(false);
+ const [scale, setScale] = useState(1); // iframe 缩放比例
+
+ // 使用 useRef 管理定时器,避免闭包问题和内存泄漏
+ const pollIntervalRef = useRef(null);
+ const backupPollIntervalRef = useRef(null); // 备用轮询定时器
+ const timeoutRef = useRef(null);
+ const isMountedRef = useRef(true); // 追踪组件挂载状态
+ const containerRef = useRef(null); // 容器DOM引用
+
+ const navigate = useNavigate();
+ const toast = useToast();
+
+ /**
+ * 显示统一的错误提示
+ */
+ const showError = useCallback((title, description) => {
+ toast({
+ title,
+ description,
+ status: "error",
+ duration: 3000,
+ isClosable: true,
+ });
+ }, [toast]);
+
+ /**
+ * 显示成功提示
+ */
+ const showSuccess = useCallback((title, description) => {
+ toast({
+ title,
+ description,
+ status: "success",
+ duration: 2000,
+ isClosable: true,
+ });
+ }, [toast]);
+
+ /**
+ * 清理所有定时器
+ */
+ const clearTimers = useCallback(() => {
+ if (pollIntervalRef.current) {
+ clearInterval(pollIntervalRef.current);
+ pollIntervalRef.current = null;
+ }
+ if (backupPollIntervalRef.current) {
+ clearInterval(backupPollIntervalRef.current);
+ backupPollIntervalRef.current = null;
+ }
+ if (timeoutRef.current) {
+ clearTimeout(timeoutRef.current);
+ timeoutRef.current = null;
+ }
+ }, []);
+
+ /**
+ * 处理登录成功
+ */
+ const handleLoginSuccess = useCallback(async (sessionId, status) => {
+ try {
+ const response = await authService.loginWithWechat(sessionId);
+
+ if (response?.success) {
+ // Session cookie 会自动管理,不需要手动存储
+ // 如果后端返回了 token,可以选择性存储(兼容旧方式)
+ if (response.token) {
+ localStorage.setItem('token', response.token);
+ }
+ if (response.user) {
+ localStorage.setItem('user', JSON.stringify(response.user));
+ }
+
+ showSuccess(
+ status === WECHAT_STATUS.LOGIN_SUCCESS ? "登录成功" : "注册成功",
+ "正在跳转..."
+ );
+
+ // 延迟跳转,让用户看到成功提示
+ setTimeout(() => {
+ navigate("/home");
+ }, 1000);
+ } else {
+ throw new Error(response?.error || '登录失败');
+ }
+ } catch (error) {
+ console.error('登录失败:', error);
+ showError("登录失败", error.message || "请重试");
+ }
+ }, [navigate, showSuccess, showError]);
+
+ /**
+ * 检查微信扫码状态
+ */
+ const checkWechatStatus = useCallback(async () => {
+ // 检查组件是否已卸载
+ if (!isMountedRef.current || !wechatSessionId) return;
+
+ try {
+ const response = await authService.checkWechatStatus(wechatSessionId);
+
+ // 安全检查:确保 response 存在且包含 status
+ if (!response || typeof response.status === 'undefined') {
+ console.warn('微信状态检查返回无效数据:', response);
+ return;
+ }
+
+ const { status } = response;
+
+ // 组件卸载后不再更新状态
+ if (!isMountedRef.current) return;
+
+ setWechatStatus(status);
+
+ // 处理成功状态
+ if (status === WECHAT_STATUS.LOGIN_SUCCESS || status === WECHAT_STATUS.REGISTER_SUCCESS) {
+ clearTimers(); // 停止轮询
+ await handleLoginSuccess(wechatSessionId, status);
+ }
+ // 处理过期状态
+ else if (status === WECHAT_STATUS.EXPIRED) {
+ clearTimers();
+ if (isMountedRef.current) {
+ toast({
+ title: "授权已过期",
+ description: "请重新获取授权",
+ status: "warning",
+ duration: 3000,
+ isClosable: true,
+ });
+ }
+ }
+ } catch (error) {
+ console.error("检查微信状态失败:", error);
+ // 轮询过程中的错误不显示给用户,避免频繁提示
+ // 但如果错误持续发生,停止轮询避免无限重试
+ if (error.message.includes('网络连接失败')) {
+ clearTimers();
+ if (isMountedRef.current) {
+ toast({
+ title: "网络连接失败",
+ description: "请检查网络后重试",
+ status: "error",
+ duration: 3000,
+ isClosable: true,
+ });
+ }
+ }
+ }
+ }, [wechatSessionId, handleLoginSuccess, clearTimers, toast]);
+
+ /**
+ * 启动轮询
+ */
+ const startPolling = useCallback(() => {
+ // 清理旧的定时器
+ clearTimers();
+
+ // 启动轮询
+ pollIntervalRef.current = setInterval(() => {
+ checkWechatStatus();
+ }, POLL_INTERVAL);
+
+ // 设置超时
+ timeoutRef.current = setTimeout(() => {
+ clearTimers();
+ setWechatStatus(WECHAT_STATUS.EXPIRED);
+ }, QR_CODE_TIMEOUT);
+ }, [checkWechatStatus, clearTimers]);
+
+ /**
+ * 获取微信二维码
+ */
+ const getWechatQRCode = useCallback(async () => {
+ try {
+ setIsLoading(true);
+
+ // 生产环境:调用真实 API
+ const response = await authService.getWechatQRCode();
+
+ // 检查组件是否已卸载
+ if (!isMountedRef.current) return;
+
+ // 安全检查:确保响应包含必要字段
+ if (!response) {
+ throw new Error('服务器无响应');
+ }
+
+ if (response.code !== 0) {
+ throw new Error(response.message || '获取二维码失败');
+ }
+
+ setWechatAuthUrl(response.data.auth_url);
+ setWechatSessionId(response.data.session_id);
+ setWechatStatus(WECHAT_STATUS.WAITING);
+
+ // 启动轮询检查扫码状态
+ startPolling();
+ } catch (error) {
+ console.error('获取微信授权失败:', error);
+ if (isMountedRef.current) {
+ showError("获取微信授权失败", error.message || "请稍后重试");
+ }
+ } finally {
+ if (isMountedRef.current) {
+ setIsLoading(false);
+ }
+ }
+ }, [startPolling, showError]);
+
+ /**
+ * 安全的按钮点击处理,确保所有错误都被捕获,防止被 ErrorBoundary 捕获
+ */
+ const handleGetQRCodeClick = useCallback(async () => {
+ try {
+ await getWechatQRCode();
+ } catch (error) {
+ // 错误已经在 getWechatQRCode 中处理,这里只需要防止未捕获的 Promise rejection
+ console.error('QR code button click error (caught in handler):', error);
+ }
+ }, [getWechatQRCode]);
+
+ /**
+ * 组件卸载时清理定时器和标记组件状态
+ */
+ useEffect(() => {
+ isMountedRef.current = true;
+
+ return () => {
+ isMountedRef.current = false;
+ clearTimers();
+ };
+ }, [clearTimers]);
+
+ /**
+ * 备用轮询机制 - 防止丢失状态
+ * 每3秒检查一次,仅在获取到二维码URL且状态为waiting时执行
+ */
+ useEffect(() => {
+ // 只在有auth_url、session_id且状态为waiting时启动备用轮询
+ if (wechatAuthUrl && wechatSessionId && wechatStatus === WECHAT_STATUS.WAITING) {
+ console.log('备用轮询:启动备用轮询机制');
+
+ backupPollIntervalRef.current = setInterval(() => {
+ try {
+ if (wechatStatus === WECHAT_STATUS.WAITING && isMountedRef.current) {
+ console.log('备用轮询:检查微信状态');
+ // 添加 .catch() 静默处理异步错误,防止被 ErrorBoundary 捕获
+ checkWechatStatus().catch(error => {
+ console.warn('备用轮询检查失败(静默处理):', error);
+ });
+ }
+ } catch (error) {
+ // 捕获所有同步错误,防止被 ErrorBoundary 捕获
+ console.warn('备用轮询执行出错(静默处理):', error);
+ }
+ }, BACKUP_POLL_INTERVAL);
+ }
+
+ // 清理备用轮询
+ return () => {
+ if (backupPollIntervalRef.current) {
+ clearInterval(backupPollIntervalRef.current);
+ backupPollIntervalRef.current = null;
+ }
+ };
+ }, [wechatAuthUrl, wechatSessionId, wechatStatus, checkWechatStatus]);
+
+ /**
+ * 测量容器尺寸并计算缩放比例
+ */
+ useLayoutEffect(() => {
+ // 微信授权页面的原始尺寸
+ const ORIGINAL_WIDTH = 600;
+ const ORIGINAL_HEIGHT = 800;
+
+ const calculateScale = () => {
+ if (containerRef.current) {
+ const { width, height } = containerRef.current.getBoundingClientRect();
+
+ // 计算宽高比例,取较小值确保完全适配
+ const scaleX = width / ORIGINAL_WIDTH;
+ const scaleY = height / ORIGINAL_HEIGHT;
+ const newScale = Math.min(scaleX, scaleY, 1.0); // 最大不超过1.0
+
+ // 设置最小缩放比例为0.3,避免过小
+ setScale(Math.max(newScale, 0.3));
+ }
+ };
+
+ // 初始计算
+ calculateScale();
+
+ // 使用 ResizeObserver 监听容器尺寸变化
+ const resizeObserver = new ResizeObserver(() => {
+ calculateScale();
+ });
+
+ if (containerRef.current) {
+ resizeObserver.observe(containerRef.current);
+ }
+
+ // 清理
+ return () => {
+ resizeObserver.disconnect();
+ };
+ }, [wechatStatus]); // 当状态变化时重新计算
+
+ /**
+ * 渲染状态提示文本
+ */
+ const renderStatusText = () => {
+ if (!wechatAuthUrl || wechatStatus === WECHAT_STATUS.NONE || wechatStatus === WECHAT_STATUS.EXPIRED) {
+ return null;
+ }
+
+ return (
+
+ {STATUS_MESSAGES[wechatStatus]}
+
+ );
+ };
+
+ return (
+
+ {wechatStatus === WECHAT_STATUS.WAITING ? (
+ <>
+
+ 微信扫码
+
+
+
+
+ {/* {renderStatusText()} */}
+ >
+ ) : (
+ <>
+
+ 微信扫码
+
+
+
+ {/* 灰色二维码底图 - 始终显示 */}
+
+
+ {/* 加载动画 */}
+ {isLoading && (
+
+
+
+ )}
+
+ {/* 显示获取/刷新二维码按钮 */}
+ {(wechatStatus === WECHAT_STATUS.NONE || wechatStatus === WECHAT_STATUS.EXPIRED) && (
+
+
+ }
+ _hover={{ bg: "green.50" }}
+ >
+ {wechatStatus === WECHAT_STATUS.EXPIRED ? "点击刷新" : "获取二维码"}
+
+ {wechatStatus === WECHAT_STATUS.EXPIRED && (
+
+ 二维码已过期
+
+ )}
+
+
+ )}
+
+
+ {/* 扫码状态提示 */}
+ {/* {renderStatusText()} */}
+ >
+ )}
+
+ );
+}
diff --git a/src/components/Citation/CitationMark.js b/src/components/Citation/CitationMark.js
new file mode 100644
index 00000000..aa15e0a4
--- /dev/null
+++ b/src/components/Citation/CitationMark.js
@@ -0,0 +1,140 @@
+// src/components/Citation/CitationMark.js
+import React, { useState } from 'react';
+import { Popover, Typography, Space, Divider } from 'antd';
+import { FileTextOutlined, UserOutlined, CalendarOutlined } from '@ant-design/icons';
+
+const { Text } = Typography;
+
+/**
+ * 引用标记组件 - 显示上标引用【1】【2】【3】
+ * 支持悬浮(桌面)和点击(移动)两种交互方式
+ *
+ * @param {Object} props
+ * @param {number} props.citationId - 引用 ID(1, 2, 3...)
+ * @param {Object} props.citation - 引用数据对象
+ * @param {string} props.citation.author - 作者
+ * @param {string} props.citation.report_title - 报告标题
+ * @param {string} props.citation.declare_date - 发布日期
+ * @param {string} props.citation.sentences - 摘要片段
+ */
+const CitationMark = ({ citationId, citation }) => {
+ const [popoverVisible, setPopoverVisible] = useState(false);
+
+ // 如果没有引用数据,不渲染
+ if (!citation) {
+ return null;
+ }
+
+ // 引用卡片内容
+ const citationContent = (
+
+ {/* 作者 */}
+
+
+
+ 作者
+
+ {citation.author}
+
+
+
+
+
+ {/* 报告标题 */}
+
+
+
+ 报告标题
+
+ {citation.report_title}
+
+
+
+
+
+ {/* 发布日期 */}
+
+
+
+ 发布日期
+
+ {citation.declare_date}
+
+
+
+
+
+ {/* 摘要片段 */}
+
+
+ 摘要片段
+
+
+ {citation.sentences}
+
+
+
+ );
+
+ // 检测是否为移动设备
+ const isMobile = () => {
+ return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(
+ navigator.userAgent
+ );
+ };
+
+ // 移动端:仅点击触发
+ // 桌面端:悬浮 + 点击都触发
+ const triggerType = isMobile() ? 'click' : ['hover', 'click'];
+
+ return (
+
+ {
+ if (!isMobile()) {
+ e.target.style.color = '#40a9ff';
+ e.target.style.textDecoration = 'underline';
+ }
+ }}
+ onMouseLeave={(e) => {
+ if (!isMobile()) {
+ e.target.style.color = '#1890ff';
+ e.target.style.textDecoration = 'none';
+ }
+ }}
+ onClick={() => {
+ setPopoverVisible(!popoverVisible);
+ }}
+ >
+ 【{citationId}】
+
+
+ );
+};
+
+export default CitationMark;
diff --git a/src/components/Citation/CitedContent.js b/src/components/Citation/CitedContent.js
new file mode 100644
index 00000000..1691e9f1
--- /dev/null
+++ b/src/components/Citation/CitedContent.js
@@ -0,0 +1,104 @@
+// src/components/Citation/CitedContent.js
+import React from 'react';
+import { Typography, Space, Tag } from 'antd';
+import { RobotOutlined, FileSearchOutlined } from '@ant-design/icons';
+import CitationMark from './CitationMark';
+import { processCitationData } from '../../utils/citationUtils';
+
+const { Text } = Typography;
+
+/**
+ * 带引用标注的内容组件
+ * 展示拼接的文本,每句话后显示上标引用【1】【2】【3】
+ * 支持鼠标悬浮和点击查看引用来源
+ *
+ * @param {Object} props
+ * @param {Object} props.data - API 返回的原始数据 { data: [...] }
+ * @param {string} props.title - 标题文本,默认 "AI 分析结果"
+ * @param {boolean} props.showAIBadge - 是否显示 AI 生成标识,默认 true
+ * @param {Object} props.containerStyle - 容器额外样式(可选)
+ *
+ * @example
+ *
+ */
+const CitedContent = ({
+ data,
+ title = 'AI 分析结果',
+ showAIBadge = true,
+ containerStyle = {}
+}) => {
+ // 处理数据
+ const processed = processCitationData(data);
+
+ // 如果数据无效,不渲染
+ if (!processed) {
+ console.warn('CitedContent: Invalid data, not rendering');
+ return null;
+ }
+
+ return (
+
+ {/* 标题栏 */}
+
+
+
+
+ {title}
+
+
+ {showAIBadge && (
+ }
+ color="purple"
+ style={{ margin: 0 }}
+ >
+ AI 生成
+
+ )}
+
+
+ {/* 带引用的文本内容 */}
+
+ {processed.segments.map((segment, index) => (
+
+ {/* 文本片段 */}
+
+ {segment.text}
+
+
+ {/* 引用标记 */}
+
+
+ {/* 在片段之间添加逗号分隔符(最后一个不加) */}
+ {index < processed.segments.length - 1 && (
+ ,
+ )}
+
+ ))}
+
+
+ );
+};
+
+export default CitedContent;
diff --git a/src/components/ErrorBoundary.js b/src/components/ErrorBoundary.js
index 0b69aef5..de2176b5 100755
--- a/src/components/ErrorBoundary.js
+++ b/src/components/ErrorBoundary.js
@@ -77,4 +77,4 @@ class ErrorBoundary extends React.Component {
}
}
-export
\ No newline at end of file
+export default ErrorBoundary;
\ No newline at end of file
diff --git a/src/components/Loading/PageLoader.js b/src/components/Loading/PageLoader.js
new file mode 100644
index 00000000..38fa4027
--- /dev/null
+++ b/src/components/Loading/PageLoader.js
@@ -0,0 +1,33 @@
+// src/components/Loading/PageLoader.js
+import React from 'react';
+import { Box, Spinner, Text, VStack } from '@chakra-ui/react';
+
+/**
+ * 页面加载组件 - 用于路由懒加载的 fallback
+ * 优雅的加载动画,提升用户体验
+ */
+export default function PageLoader({ message = '加载中...' }) {
+ return (
+
+
+
+
+ {message}
+
+
+
+ );
+}
diff --git a/src/components/Navbars/HomeNavbar.js b/src/components/Navbars/HomeNavbar.js
index 8cf5ec32..a106fb07 100644
--- a/src/components/Navbars/HomeNavbar.js
+++ b/src/components/Navbars/HomeNavbar.js
@@ -36,6 +36,7 @@ import { ChevronDownIcon, HamburgerIcon, SunIcon, MoonIcon } from '@chakra-ui/ic
import { FiStar, FiCalendar } from 'react-icons/fi';
import { useNavigate } from 'react-router-dom';
import { useAuth } from '../../contexts/AuthContext';
+import { useAuthModal } from '../../contexts/AuthModalContext';
/** 桌面端导航 - 完全按照原网站
* @TODO 添加逻辑 不展示导航case
@@ -45,7 +46,7 @@ import { useAuth } from '../../contexts/AuthContext';
const NavItems = ({ isAuthenticated, user }) => {
const navigate = useNavigate();
- if (!isAuthenticated && !user) {
+ if (isAuthenticated && user) {
return (
) : (
- // 未登录状态
+ // 未登录状态 - 单一按钮
{/* 关联描述 */}
- {stock?.relation_desc && (
+ {stock?.relation_desc?.data ? (
+ // 使用引用组件(带研报来源)
+
+ ) : stock?.relation_desc ? (
+ // 降级显示(无引用数据)
- )}
+ ) : null}
{/* 调试信息 */}
{process.env.NODE_ENV === 'development' && chartData && (
diff --git a/src/components/UserAgreementModal.js b/src/components/UserAgreementModal.js
index 98bbe7c0..34f72a25 100644
--- a/src/components/UserAgreementModal.js
+++ b/src/components/UserAgreementModal.js
@@ -18,6 +18,11 @@ const UserAgreementModal = ({ isOpen, onClose }) => {
const headingColor = useColorModeValue("gray.800", "white");
const textColor = useColorModeValue("gray.600", "gray.300");
+ // Conditional rendering: only render Modal when open
+ if (!isOpen) {
+ return null;
+ }
+
return (