@amaster.ai/vite-plugins 1.1.0-beta.13 → 1.1.0-beta.13.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs +54 -28
- package/dist/index.cjs.map +1 -1
- package/dist/index.js +54 -28
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.cjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
'use strict';Object.defineProperty(exports,'__esModule',{value:true});var crypto=require('crypto'),m=require('fs'),
|
|
1
|
+
'use strict';Object.defineProperty(exports,'__esModule',{value:true});var crypto=require('crypto'),m=require('fs'),b=require('path'),url=require('url'),x=require('process');var _documentCurrentScript=typeof document!=='undefined'?document.currentScript:null;function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var m__default=/*#__PURE__*/_interopDefault(m);var b__default=/*#__PURE__*/_interopDefault(b);var x__default=/*#__PURE__*/_interopDefault(x);function k(n,t){let e=`${n}:${t}`;return crypto.createHash("md5").update(e).digest("hex").substring(0,12)}var E=new Set(["a","abbr","address","area","article","aside","audio","b","base","bdi","bdo","blockquote","body","br","button","canvas","caption","cite","code","col","colgroup","data","datalist","dd","del","details","dfn","dialog","div","dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","iframe","img","input","ins","kbd","label","legend","li","link","main","map","mark","meta","meter","nav","noscript","object","ol","optgroup","option","output","p","param","picture","pre","progress","q","rp","rt","ruby","s","samp","script","section","select","small","source","span","strong","style","sub","summary","sup","svg","table","tbody","td","template","textarea","tfoot","th","thead","time","title","tr","track","u","ul","var","video","wbr"]);function y(){let n=false;return {name:"vite-plugin-component-id",enforce:"pre",configResolved(t){n=t.command==="serve";},transform(t,e){if(!n||!/\.(tsx|jsx)$/.test(e)||e.includes("node_modules")||!/<[a-z]/.test(t))return null;try{let r=t,o=/<([a-z][\da-z]*)(?=[\s/>])/g,i,s=[];for(;(i=o.exec(t))!==null;){let a=i[1];if(!a)continue;let u=i.index,d=i.index+i[0].length;if(!E.has(a))continue;let c=d,p=!1,f=!1;for(;c<t.length&&!p;)t[c]===">"&&(p=!0,t.substring(u,c).includes("data-node-component-id")&&(f=!0)),c++;f||s.push({index:u,tag:a,tagEndIndex:d});}let l=t;for(let a=s.length-1;a>=0;a--){let u=s[a];if(!u)continue;let{index:d,tagEndIndex:c}=u,p=k(e,d),f=l.substring(0,c),v=l.substring(c);r=`${f} data-node-component-id="${p}"${v}`,l=r;}return {code:r,map:null}}catch{return null}}}}var L=`<script>
|
|
2
2
|
document.addEventListener("click", (e) => {
|
|
3
3
|
const element = e.target;
|
|
4
4
|
const noJumpOut = document.body.classList.contains("forbid-jump-out")
|
|
@@ -16,9 +16,9 @@ document.addEventListener("click", (e) => {
|
|
|
16
16
|
e.preventDefault();
|
|
17
17
|
}
|
|
18
18
|
});
|
|
19
|
-
</script>`;function
|
|
19
|
+
</script>`;function W(){try{let n=url.fileURLToPath((typeof document === 'undefined' ? require('u' + 'rl').pathToFileURL(__filename).href : (_documentCurrentScript && _documentCurrentScript.tagName.toUpperCase() === 'SCRIPT' && _documentCurrentScript.src || new URL('index.cjs', document.baseURI).href))),t=b.dirname(n),e=b.resolve(t,"../dist/bridge.bridge.js");return `<script>
|
|
20
20
|
${m.readFileSync(e,"utf-8")}
|
|
21
|
-
</script>`}catch(n){return console.warn("Failed to read bridge script:",n),""}}function S(){let n=false,t="";return {name:"vite-plugin-editor-bridge",configResolved(e){n=e.command==="serve",n&&(t=
|
|
21
|
+
</script>`}catch(n){return console.warn("Failed to read bridge script:",n),""}}function S(){let n=false,t="";return {name:"vite-plugin-editor-bridge",configResolved(e){n=e.command==="serve",n&&(t=W());},transformIndexHtml(e){let o=`${L}${n?t:""}</body>`;return e.replace("</body>",o)}}}function w(n){let t=false,e=n?.routesFilePath||"src/routes.tsx";return {name:"vite-plugin-routes-expose",enforce:"post",configResolved(r){t=r.command==="serve";},transform(r,o){if(!t||!o.endsWith(e))return null;try{return r.includes("window.__APP_ROUTES__")?null:{code:`${r}
|
|
22
22
|
|
|
23
23
|
// Development mode: Expose routes to window.__APP_ROUTES__
|
|
24
24
|
if (typeof window !== 'undefined') {
|
|
@@ -705,7 +705,7 @@ if (typeof window !== 'undefined') {
|
|
|
705
705
|
|
|
706
706
|
originalConsole.log('[BrowserLogs] Log collection started');
|
|
707
707
|
})();
|
|
708
|
-
</script>`;return {name:"vite-plugin-browser-logs",configResolved(e){let r=e.root||
|
|
708
|
+
</script>`;return {name:"vite-plugin-browser-logs",configResolved(e){let r=e.root||x__default.default.cwd();n=b__default.default.join(r,"browser.log");},configureServer(e){e.middlewares.use((r,o,i)=>{if(r.url==="/__browser__"&&r.method==="POST"){let s=r.headers.origin||"*",l="";r.on("data",a=>{l+=a.toString();}),r.on("end",()=>{try{let a=b__default.default.dirname(n);m__default.default.existsSync(a)||m__default.default.mkdirSync(a,{recursive:!0}),m__default.default.appendFileSync(n,`${l}
|
|
709
709
|
`,"utf-8"),o.writeHead(200,{"Content-Type":"application/json","Access-Control-Allow-Origin":s,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),o.end(JSON.stringify({success:!0}));}catch(a){console.error("[BrowserLogs] Write error:",a),o.writeHead(500,{"Content-Type":"application/json","Access-Control-Allow-Origin":s,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),o.end(JSON.stringify({success:false,error:String(a)}));}});}else if(r.url==="/__browser__"&&r.method==="OPTIONS"){let s=r.headers.origin||"*";o.writeHead(204,{"Access-Control-Allow-Origin":s,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type","Access-Control-Max-Age":"86400"}),o.end();}else if(r.url==="/__browser__"){let s=r.headers.origin||"*";o.writeHead(405,{"Content-Type":"application/json","Access-Control-Allow-Origin":s}),o.end(JSON.stringify({error:"Method not allowed"}));}else i();}),console.log("[BrowserLogs] Logs will be written to:",n);},transformIndexHtml(e){return e.replace(/<head([^>]*)>/i,`<head$1>${t}`)}}}function T(){let t=`
|
|
710
710
|
<script>
|
|
711
711
|
(function() {
|
|
@@ -971,35 +971,61 @@ if (typeof window !== 'undefined') {
|
|
|
971
971
|
})();
|
|
972
972
|
</script>`.replace(/^\s*<script>\s*/i,"").replace(/\s*<\/script>\s*$/i,"");return {name:"vite-plugin-smart-reload",apply:"serve",transformIndexHtml:{order:"pre",handler(){return [{tag:"script",injectTo:"head-prepend",children:t}]}},transform(e,r){if(!r.includes("vite/dist/client/client.mjs"))return null;let o=e.replace(/await waitForSuccessfulPing\(protocol,\s*hostAndPath\);\s*\n\s*location\.reload\(\)/,`await waitForSuccessfulPing(protocol, hostAndPath);
|
|
973
973
|
if (window.__smartReload__ && window.__smartReload__()) {
|
|
974
|
-
console.log('[SmartReload] Vite client: server
|
|
975
|
-
//
|
|
976
|
-
// \
|
|
977
|
-
|
|
978
|
-
|
|
979
|
-
|
|
980
|
-
|
|
981
|
-
|
|
982
|
-
|
|
983
|
-
|
|
984
|
-
|
|
985
|
-
|
|
986
|
-
|
|
974
|
+
console.log('[SmartReload] Vite client: server is back, fetching new WebSocket token...');
|
|
975
|
+
// PM2 \u91CD\u542F\u540E\u65B0\u670D\u52A1\u5668\u4F1A\u751F\u6210\u65B0\u7684 wsToken\uFF0C\u65E7 token \u65E0\u6CD5\u8FDE\u63A5
|
|
976
|
+
// \u901A\u8FC7 fetch /@vite/client \u83B7\u53D6\u65B0 token
|
|
977
|
+
try {
|
|
978
|
+
var clientResp = await fetch('/@vite/client');
|
|
979
|
+
var clientCode = await clientResp.text();
|
|
980
|
+
// \u4ECE\u65B0\u7684\u5BA2\u6237\u7AEF\u4EE3\u7801\u4E2D\u63D0\u53D6 wsToken
|
|
981
|
+
// Vite \u7F16\u8BD1\u540E\u7684\u4EE3\u7801\u4E2D wsToken \u662F\u4E00\u4E2A\u5B57\u7B26\u4E32\u5E38\u91CF\uFF0C\u5982: const wsToken = "xxxxxxxxxxxx"
|
|
982
|
+
var tokenMatch = clientCode.match(/const\\s+wsToken\\s*=\\s*"([^"]+)"/);
|
|
983
|
+
if (tokenMatch && tokenMatch[1]) {
|
|
984
|
+
console.log('[SmartReload] Got new wsToken from server');
|
|
985
|
+
// \u66F4\u65B0\u95ED\u5305\u4E2D\u7684 wsToken \u53D8\u91CF\uFF08Vite \u5BA2\u6237\u7AEF\u4EE3\u7801\u4E2D\u7684 const wsToken\uFF09
|
|
986
|
+
// \u6CE8\u610F\uFF1AwsToken \u662F const\uFF0C\u65E0\u6CD5\u76F4\u63A5\u4FEE\u6539\u3002\u6211\u4EEC\u9700\u8981\u4FEE\u6539 setupWebSocket \u4F7F\u7528\u7684 URL
|
|
987
|
+
// \u65B9\u6848\uFF1A\u76F4\u63A5\u521B\u5EFA WebSocket \u8FDE\u63A5\uFF0C\u7ED5\u8FC7 setupWebSocket
|
|
988
|
+
var newToken = tokenMatch[1];
|
|
989
|
+
var wsUrl = protocol + '://' + hostAndPath + '?token=' + newToken;
|
|
990
|
+
console.log('[SmartReload] Reconnecting with new token to: ' + wsUrl);
|
|
991
|
+
var newSocket = new WebSocket(wsUrl, 'vite-hmr');
|
|
992
|
+
var newIsOpened = false;
|
|
993
|
+
newSocket.addEventListener('open', function() {
|
|
994
|
+
newIsOpened = true;
|
|
995
|
+
console.log('[SmartReload] WebSocket reconnected successfully!');
|
|
996
|
+
notifyListeners('vite:ws:connect', { webSocket: newSocket });
|
|
997
|
+
}, { once: true });
|
|
998
|
+
newSocket.addEventListener('message', async function(ev) {
|
|
999
|
+
handleMessage(JSON.parse(ev.data));
|
|
1000
|
+
});
|
|
1001
|
+
newSocket.addEventListener('close', async function(ev) {
|
|
1002
|
+
if (ev.wasClean) return;
|
|
1003
|
+
if (!newIsOpened && onCloseWithoutOpen) {
|
|
1004
|
+
onCloseWithoutOpen();
|
|
1005
|
+
return;
|
|
1006
|
+
}
|
|
1007
|
+
notifyListeners('vite:ws:disconnect', { webSocket: newSocket });
|
|
1008
|
+
if (hasDocument) {
|
|
1009
|
+
console.log('[vite] server connection lost. Polling for restart...');
|
|
1010
|
+
await waitForSuccessfulPing(protocol, hostAndPath);
|
|
1011
|
+
if (window.__smartReload__ && window.__smartReload__()) {
|
|
1012
|
+
console.log('[SmartReload] Nested reconnect - reloading page');
|
|
1013
|
+
}
|
|
1014
|
+
location.reload();
|
|
1015
|
+
}
|
|
987
1016
|
});
|
|
988
|
-
|
|
989
|
-
}
|
|
990
|
-
console.log('[SmartReload]
|
|
1017
|
+
socket = newSocket;
|
|
1018
|
+
} else {
|
|
1019
|
+
console.log('[SmartReload] Could not extract new wsToken, reloading page');
|
|
1020
|
+
location.reload();
|
|
991
1021
|
}
|
|
992
|
-
}
|
|
993
|
-
|
|
994
|
-
console.log('[SmartReload] Vite client: WebSocket is ready, reconnecting now');
|
|
995
|
-
socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);
|
|
996
|
-
} else {
|
|
997
|
-
console.log('[SmartReload] Vite client: WebSocket still not ready after retries, reloading page');
|
|
1022
|
+
} catch(e) {
|
|
1023
|
+
console.log('[SmartReload] Failed to fetch new client code, reloading page', e);
|
|
998
1024
|
location.reload();
|
|
999
1025
|
}
|
|
1000
1026
|
} else {
|
|
1001
1027
|
location.reload();
|
|
1002
|
-
}`);return o!==e?{code:o,map:null}:null}}}function
|
|
1028
|
+
}`);return o!==e?{code:o,map:null}:null}}}function h(n={}){let{designWidth:t=375,maxWidth:e=750,baseFontSize:r=12,minRootSize:o=12,maxRootSize:i=24}=n;return {name:"vite-plugin-taro-style-adapter",apply:"serve",transformIndexHtml(s){let l=`
|
|
1003
1029
|
<script data-taro-flexible="true">
|
|
1004
1030
|
(function() {
|
|
1005
1031
|
var designWidth = ${t};
|
|
@@ -1086,5 +1112,5 @@ if (typeof window !== 'undefined') {
|
|
|
1086
1112
|
object-fit: contain;
|
|
1087
1113
|
}
|
|
1088
1114
|
</style>
|
|
1089
|
-
</head>`)}}}function g(n={}){let{ratio:t=2}=n;return {postcssPlugin:"postcss-rpx2px",Declaration(e){e.value.includes("rpx")&&(e.value=e.value.replace(/(-?\d*\.?\d+)rpx/gi,(r,o)=>{let i=parseFloat(o)/t;return i===0?"0":`${i}px`}));}}}g.postcss=true;function H(n={}){let{additional:t=[],autoInjectTaroApp:e=true,autoInjectVite:r=true}=n,o={},i=new Set(t);return e&&Object.keys(process.env).forEach(s=>{s.startsWith("TARO_APP_")&&i.add(s);}),r&&Object.keys(process.env).forEach(s=>{s.startsWith("VITE_")&&i.add(s);}),i.forEach(s=>{let l=process.env[s]||"";o[`process.env.${s}`]=JSON.stringify(l);}),o}function q(){return {"process.env.TARO_APP_API_BASE_URL":JSON.stringify(process.env.TARO_APP_API_BASE_URL||""),"process.env.VITE_API_BASE_URL":JSON.stringify(process.env.VITE_API_BASE_URL||"")}}function D(n={}){let t=[
|
|
1115
|
+
</head>`)}}}function g(n={}){let{ratio:t=2}=n;return {postcssPlugin:"postcss-rpx2px",Declaration(e){e.value.includes("rpx")&&(e.value=e.value.replace(/(-?\d*\.?\d+)rpx/gi,(r,o)=>{let i=parseFloat(o)/t;return i===0?"0":`${i}px`}));}}}g.postcss=true;function H(n={}){let{additional:t=[],autoInjectTaroApp:e=true,autoInjectVite:r=true}=n,o={},i=new Set(t);return e&&Object.keys(process.env).forEach(s=>{s.startsWith("TARO_APP_")&&i.add(s);}),r&&Object.keys(process.env).forEach(s=>{s.startsWith("VITE_")&&i.add(s);}),i.forEach(s=>{let l=process.env[s]||"";o[`process.env.${s}`]=JSON.stringify(l);}),o}function q(){return {"process.env.TARO_APP_API_BASE_URL":JSON.stringify(process.env.TARO_APP_API_BASE_URL||""),"process.env.VITE_API_BASE_URL":JSON.stringify(process.env.VITE_API_BASE_URL||"")}}function D(n={}){let t=[h(n.styleAdapter),{name:"dev-postcss-rpx2px-plugin",apply:"serve",config(e){x__default.default.env.TARO_ENV==="h5"&&typeof e.css?.postcss=="object"&&e.css?.postcss.plugins?.unshift(g());}},y(),S(),w(),T()];return x__default.default.env.WORKSPACE_GIT_REPO&&t.push(_()),t}exports.componentIdPlugin=y;exports.default=D;exports.editorBridgePlugin=S;exports.injectAmasterEnv=q;exports.injectTaroEnv=H;exports.routesExposePlugin=w;exports.rpx2pxPlugin=g;exports.smartReloadPlugin=T;exports.taroStyleAdapterPlugin=h;//# sourceMappingURL=index.cjs.map
|
|
1090
1116
|
//# sourceMappingURL=index.cjs.map
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts","../src/browser-logs.ts","../src/smart-reload.ts","../src/taro-style-adapter.ts","../src/postcss-rpx2px.ts","../src/taro-env-inject.ts","../src/index.ts"],"names":["generateUniqueId","filePath","position","key","createHash","HTML_TAGS","componentIdPlugin","isDev","config","code","id","transformedCode","jsxOpenTagRegex","match","matches","tag","tagStartIndex","tagEndIndex","checkIndex","foundClosing","hasComponentId","currentCode","i","item","index","uniqueId","before","after","clickHandlerScript","getBridgeScriptContent","__filename","fileURLToPath","__dirname","dirname","bridgePath","resolve","readFileSync","error","editorBridgePlugin","bridgeScript","html","scriptsToInject","routesExposePlugin","options","routesPath","browserLogsPlugin","logFilePath","injectedScript","root","process","path","devServer","req","res","next","origin","body","chunk","logDir","fs","smartReloadPlugin","pureScript","transformed","taroStyleAdapterPlugin","designWidth","maxWidth","baseFontSize","minRootSize","maxRootSize","flexibleScript","rpx2pxPlugin","ratio","decl","_match","num","pxValue","injectTaroEnv","additional","autoInjectTaroApp","autoInjectVite","constants","varsToInject","varName","value","injectAmasterEnv","devTools","plugins"],"mappings":"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AA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AA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","file":"index.cjs","sourcesContent":["import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { readFileSync } from \"fs\";\nimport { resolve, dirname } from \"path\";\nimport { fileURLToPath } from \"url\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * 读取构建后的 bridge 脚本内容\n */\nfunction getBridgeScriptContent(): string {\n try {\n // 获取当前模块的目录\n const __filename = fileURLToPath(import.meta.url);\n const __dirname = dirname(__filename);\n \n // 读取构建后的 bridge.bridge.js 文件\n const bridgePath = resolve(__dirname, \"../dist/bridge.bridge.js\");\n const bridgeContent = readFileSync(bridgePath, \"utf-8\");\n \n return `<script>\\n${bridgeContent}\\n</script>`;\n } catch (error) {\n console.warn(\"Failed to read bridge script:\", error);\n return \"\";\n }\n}\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode by reading the built bridge script\n */\nexport function editorBridgePlugin(): Plugin {\n let isDev = false;\n let bridgeScript = \"\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n \n // 在开发模式下读取 bridge 脚本内容\n if (isDev) {\n bridgeScript = getBridgeScriptContent();\n }\n },\n\n transformIndexHtml(html) {\n // 在开发模式下注入 bridge 脚本\n const devScript = isDev ? bridgeScript : \"\";\n\n const scriptsToInject = `${clickHandlerScript}${devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { Buffer } from \"node:buffer\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport process from \"node:process\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n // ============================================\n // Vite deps 504 error capture using PerformanceObserver\n // Only captures 504 errors for node_modules/.vite resources (pre-bundled dependencies)\n // ============================================\n var reportedUrls = {}; // Track reported URLs to avoid duplicates\n \n if (typeof PerformanceObserver !== 'undefined') {\n var perfObserver = new PerformanceObserver(function(list) {\n var entries = list.getEntries();\n for (var i = 0; i < entries.length; i++) {\n var perfEntry = entries[i];\n // Only check script resources\n if (perfEntry.initiatorType !== 'script') {\n continue;\n }\n \n var resourceUrl = perfEntry.name || '';\n // Only report errors for node_modules/.vite resources\n if (resourceUrl.indexOf('node_modules/.vite') === -1 && resourceUrl.indexOf('/node_modules/.vite') === -1) {\n continue;\n }\n \n // Check if response status is 504 (Gateway Timeout)\n // responseStatus is available in Resource Timing Level 2\n var status = perfEntry.responseStatus;\n if (status === 504) {\n // Avoid duplicate reports\n if (reportedUrls[resourceUrl]) {\n continue;\n }\n reportedUrls[resourceUrl] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: resourceUrl,\n status: 504,\n message: 'Vite pre-bundled dependency returned 504 Gateway Timeout: ' + resourceUrl,\n duration: perfEntry.duration,\n transferSize: perfEntry.transferSize\n };\n addLog(entry);\n }\n }\n });\n \n try {\n perfObserver.observe({ type: 'resource', buffered: true });\n } catch (e) {\n // Fallback for browsers that don't support the options\n try {\n perfObserver.observe({ entryTypes: ['resource'] });\n } catch (e2) {\n originalConsole.warn('[BrowserLogs] PerformanceObserver not supported:', e2.message);\n }\n }\n }\n \n // ============================================\n // Script load error capture (for 504 and other network errors)\n // This captures errors that PerformanceObserver might miss\n // ============================================\n \n // Use MutationObserver to watch for dynamically added script tags\n function attachScriptErrorHandler(script) {\n if (script.__errorHandlerAttached__) return;\n script.__errorHandlerAttached__ = true;\n \n script.addEventListener('error', function(event) {\n var src = script.src || '';\n \n // Only report errors for node_modules/.vite resources\n if (src.indexOf('node_modules/.vite') === -1 && src.indexOf('/node_modules/.vite') === -1) {\n return;\n }\n \n // Avoid duplicate reports\n if (reportedUrls[src]) {\n return;\n }\n reportedUrls[src] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: src,\n status: 'load-error',\n message: 'Vite pre-bundled dependency failed to load (possibly 504 Gateway Timeout): ' + src,\n errorType: event.type\n };\n addLog(entry);\n });\n }\n \n // Attach error handlers to existing scripts\n var existingScripts = document.querySelectorAll('script[src]');\n for (var i = 0; i < existingScripts.length; i++) {\n attachScriptErrorHandler(existingScripts[i]);\n }\n \n // Watch for dynamically added scripts\n if (typeof MutationObserver !== 'undefined') {\n var scriptObserver = new MutationObserver(function(mutations) {\n for (var i = 0; i < mutations.length; i++) {\n var mutation = mutations[i];\n for (var j = 0; j < mutation.addedNodes.length; j++) {\n var node = mutation.addedNodes[j];\n if (node.nodeName === 'SCRIPT' && node.src) {\n attachScriptErrorHandler(node);\n }\n // Also check child nodes\n if (node.querySelectorAll) {\n var scripts = node.querySelectorAll('script[src]');\n for (var k = 0; k < scripts.length; k++) {\n attachScriptErrorHandler(scripts[k]);\n }\n }\n }\n }\n });\n \n scriptObserver.observe(document.documentElement, {\n childList: true,\n subtree: true\n });\n }\n \n // Also intercept document.createElement to catch scripts before they're added to DOM\n var originalCreateElement = document.createElement.bind(document);\n document.createElement = function(tagName) {\n var element = originalCreateElement(tagName);\n if (tagName.toLowerCase() === 'script') {\n // Use a setter to catch when src is set\n var originalSrc = '';\n Object.defineProperty(element, '__originalSrc__', {\n get: function() { return originalSrc; },\n set: function(val) { originalSrc = val; }\n });\n \n // Defer attaching error handler until src is set\n var srcDescriptor = Object.getOwnPropertyDescriptor(HTMLScriptElement.prototype, 'src');\n if (srcDescriptor && srcDescriptor.set) {\n var originalSrcSetter = srcDescriptor.set;\n Object.defineProperty(element, 'src', {\n get: function() {\n return srcDescriptor.get ? srcDescriptor.get.call(this) : this.getAttribute('src');\n },\n set: function(value) {\n if (originalSrcSetter) {\n originalSrcSetter.call(this, value);\n } else {\n this.setAttribute('src', value);\n }\n attachScriptErrorHandler(this);\n },\n configurable: true\n });\n }\n }\n return element;\n };\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Smart Reload 插件\n *\n * 在 Vite 服务器重启期间(如 PM2 重启 web-server),拦截浏览器端的刷新行为,\n * 让浏览器静默等待服务器恢复,期间页面完全不刷新。\n * 服务器恢复后自动恢复 HMR 连接,无需手动刷新。\n *\n * 场景说明:\n * - PM2 杀掉 Vite 进程 → WebSocket 断开 → PM2 重启 → 新 Vite 启动\n * - Vite 客户端检测到 WebSocket 断开后会通过 HTTP ping 轮询等待服务器恢复\n * - 服务器恢复后 Vite 客户端会调用 location.reload() 刷新页面\n * - 本插件拦截这个刷新行为,让页面保持当前状态\n *\n * 实现原理(双层拦截):\n * 1. 服务端层:通过 transform 钩子修改 /@vite/client 源码,\n * 将 location.reload() 替换为 window.__smartReload__(),\n * 使我们能控制是否真正刷新页面\n * 2. 客户端层:注入脚本拦截 WebSocket 消息,\n * 检测服务器重启场景,在重启期间阻止刷新\n *\n * 区分两种 full-reload 场景:\n * 1. 服务器重启 → WebSocket 断开 → 静默等待,不刷新,自动恢复 HMR\n * 2. 模块变更需要 full-reload → WebSocket 保持连接 → 正常刷新\n */\nexport function smartReloadPlugin(): Plugin {\n // 注入到浏览器端的脚本\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n\n var PLUGIN_TAG = '[SmartReload]';\n // full-reload 后等待 WebSocket 断开的窗口期(ms)\n var DISCONNECT_DETECT_WINDOW = 300;\n\n // ============================================\n // 状态管理\n // ============================================\n var state = {\n // 是否已经拦截了 Vite HMR WebSocket\n wsIntercepted: false,\n // 是否正在等待判断 full-reload 类型(300ms 窗口期)\n detectingReloadType: false,\n // 延迟执行 full-reload 的定时器\n reloadTimer: null,\n // 是否是服务器重启场景\n isServerRestart: false\n };\n\n // ============================================\n // 日志工具\n // ============================================\n function log() {\n var args = [PLUGIN_TAG];\n for (var i = 0; i < arguments.length; i++) {\n args.push(arguments[i]);\n }\n console.log.apply(console, args);\n }\n\n // ============================================\n // 重置状态\n // ============================================\n function resetState() {\n state.detectingReloadType = false;\n state.isServerRestart = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n }\n\n // ============================================\n // 执行真正的 full-reload\n // ============================================\n function executeFullReload() {\n log('Executing full-reload');\n resetState();\n location.reload();\n }\n\n // ============================================\n // __smartReload__:检查是否应该阻止 reload\n //\n // 返回 true 表示是服务器重启场景,不应该刷新页面\n // 返回 false 表示应该正常刷新\n //\n // Vite 客户端代码中的 WebSocket close handler 已被修改为:\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(...); // 重新创建 WebSocket 恢复 HMR\n // } else {\n // location.reload();\n // }\n // ============================================\n window.__smartReload__ = function() {\n if (state.isServerRestart) {\n log('Server restart detected, suppressing reload. Will reconnect WebSocket.');\n resetState();\n return true; // 阻止 reload,由 Vite 客户端重新创建 WebSocket\n }\n log('Non-restart scenario, allowing reload');\n return false; // 允许正常 reload\n };\n\n // ============================================\n // 拦截 WebSocket 构造函数\n // ============================================\n var OriginalWebSocket = window.WebSocket;\n\n function isViteHmrWebSocket(url) {\n if (typeof url !== 'string') return false;\n return url.includes('/__vite_hmr') || url.includes('?token=');\n }\n\n function createWsProxy(ws) {\n var originalAddEventListener = ws.addEventListener.bind(ws);\n var originalRemoveEventListener = ws.removeEventListener.bind(ws);\n\n ws.addEventListener = function(type, listener, options) {\n if (type === 'message') {\n var wrappedListener = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted) {\n listener.call(ws, event);\n }\n };\n listener.__smartReloadWrapped__ = wrappedListener;\n return originalAddEventListener(type, wrappedListener, options);\n }\n return originalAddEventListener(type, listener, options);\n };\n\n ws.removeEventListener = function(type, listener, options) {\n if (type === 'message' && listener.__smartReloadWrapped__) {\n return originalRemoveEventListener(type, listener.__smartReloadWrapped__, options);\n }\n return originalRemoveEventListener(type, listener, options);\n };\n\n var originalOnMessage = null;\n var onMessageDescriptor = Object.getOwnPropertyDescriptor(WebSocket.prototype, 'onmessage') ||\n Object.getOwnPropertyDescriptor(ws, 'onmessage');\n\n if (onMessageDescriptor) {\n Object.defineProperty(ws, 'onmessage', {\n get: function() {\n return originalOnMessage;\n },\n set: function(handler) {\n originalOnMessage = handler;\n var wrappedHandler = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted && handler) {\n handler.call(ws, event);\n }\n };\n if (onMessageDescriptor.set) {\n onMessageDescriptor.set.call(ws, wrappedHandler);\n }\n },\n configurable: true\n });\n }\n\n originalAddEventListener('close', function(event) {\n handleWsClose(event);\n });\n\n originalAddEventListener('open', function() {\n handleWsOpen();\n });\n\n state.wsIntercepted = true;\n log('Vite HMR WebSocket intercepted');\n\n return ws;\n }\n\n window.WebSocket = function SmartReloadWebSocket(url, protocols) {\n var ws;\n if (protocols !== undefined) {\n ws = new OriginalWebSocket(url, protocols);\n } else {\n ws = new OriginalWebSocket(url);\n }\n\n if (isViteHmrWebSocket(url)) {\n return createWsProxy(ws);\n }\n\n return ws;\n };\n\n window.WebSocket.prototype = OriginalWebSocket.prototype;\n window.WebSocket.CONNECTING = OriginalWebSocket.CONNECTING;\n window.WebSocket.OPEN = OriginalWebSocket.OPEN;\n window.WebSocket.CLOSING = OriginalWebSocket.CLOSING;\n window.WebSocket.CLOSED = OriginalWebSocket.CLOSED;\n\n // ============================================\n // WebSocket 消息处理\n // ============================================\n function handleWsMessage(event) {\n var data;\n try {\n data = JSON.parse(event.data);\n } catch (e) {\n return false;\n }\n\n // 处理 full-reload 消息\n if (data.type === 'full-reload') {\n // 如果在服务器重启模式,拦截所有 full-reload\n if (state.isServerRestart) {\n log('Suppressed full-reload message during server restart');\n return true;\n }\n\n log('Intercepted full-reload, path:', data.path || '(none)');\n\n // 进入检测模式:等待看 WebSocket 是否断开\n state.detectingReloadType = true;\n\n // 如果窗口期内 WebSocket 没有断开,说明是真正的模块变更\n state.reloadTimer = setTimeout(function() {\n if (state.detectingReloadType && !state.isServerRestart) {\n log('WebSocket still connected after ' + DISCONNECT_DETECT_WINDOW + 'ms - real module change, executing reload');\n state.detectingReloadType = false;\n executeFullReload();\n }\n }, DISCONNECT_DETECT_WINDOW);\n\n return true;\n }\n\n // 处理 connected 消息\n if (data.type === 'connected') {\n if (state.isServerRestart) {\n log('HMR reconnected after server restart! Page was NOT reloaded.');\n resetState();\n }\n return false;\n }\n\n return false;\n }\n\n // ============================================\n // WebSocket 连接事件处理\n // ============================================\n function handleWsClose(event) {\n log('WebSocket closed, code:', event.code, 'wasClean:', event.wasClean);\n\n // 场景1:收到 full-reload 后 WebSocket 断开 → 服务器重启\n if (state.detectingReloadType) {\n log('Entering server restart mode: full-reload followed by disconnect');\n state.isServerRestart = true;\n state.detectingReloadType = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n return;\n }\n\n // 场景2:WebSocket 直接断开(非正常关闭)\n if (!state.isServerRestart && !event.wasClean) {\n log('Entering server restart mode: unexpected disconnect');\n state.isServerRestart = true;\n }\n }\n\n function handleWsOpen() {\n log('WebSocket opened');\n }\n\n // ============================================\n // 暴露 API 供调试使用\n // ============================================\n window.__smartReloadState__ = state;\n window.__OriginalWebSocket__ = OriginalWebSocket;\n window.__smartReloadForceReload__ = function() {\n log('Force reload triggered by user');\n resetState();\n location.reload();\n };\n\n log('Smart Reload initialized');\n})();\n</script>`;\n\n // 提取纯 JS 代码(去掉 <script> 标签)\n const pureScript = injectedScript\n .replace(/^\\s*<script>\\s*/i, \"\")\n .replace(/\\s*<\\/script>\\s*$/i, \"\");\n\n return {\n name: \"vite-plugin-smart-reload\",\n\n // 只在开发模式下注入脚本\n apply: \"serve\",\n\n transformIndexHtml: {\n order: \"pre\",\n handler() {\n return [\n {\n tag: \"script\",\n // 注入到 head 的最前面,确保在 Vite 客户端脚本 /@vite/client 之前执行\n injectTo: \"head-prepend\",\n children: pureScript,\n },\n ];\n },\n },\n\n // 修改 Vite 客户端代码:\n // 精确替换 WebSocket close handler 中的 reload 逻辑,\n // 在服务器重启场景下重新创建 WebSocket 恢复 HMR,而不是刷新页面\n //\n // Vite 对 /@vite/client 的处理流程:\n // 1. 浏览器请求 /@vite/client\n // 2. Vite 的 transformMiddleware 拦截请求\n // 3. 调用 server.transformRequest('/@vite/client')\n // 4. 内部解析为文件系统路径 node_modules/vite/dist/client/client.mjs\n // 5. 调用插件的 transform(code, resolvedId) — id 是文件系统路径\n transform(code, id) {\n // 匹配 Vite 客户端模块(文件系统路径)\n if (!id.includes(\"vite/dist/client/client.mjs\")) {\n return null;\n }\n\n // 精确替换 WebSocket close handler 中的 reload 逻辑:\n // 原始代码:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // location.reload();\n // 替换为:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);\n // } else {\n // location.reload();\n // }\n const transformed = code.replace(\n /await waitForSuccessfulPing\\(protocol,\\s*hostAndPath\\);\\s*\\n\\s*location\\.reload\\(\\)/,\n `await waitForSuccessfulPing(protocol, hostAndPath);\n if (window.__smartReload__ && window.__smartReload__()) {\n console.log('[SmartReload] Vite client: server HTTP is back, waiting for WebSocket to be ready...');\n // HTTP ping 成功不代表 WebSocket 已就绪,需要等待 WebSocket 可连接\n // 使用原始 WebSocket 构造函数避免被 SmartReload 代理拦截\n var RawWS = window.__OriginalWebSocket__ || WebSocket;\n var wsReady = false;\n for (var attempt = 0; attempt < 10 && !wsReady; attempt++) {\n await new Promise(function(r) { setTimeout(r, 1000); });\n try {\n await new Promise(function(resolve, reject) {\n var testWs = new RawWS(protocol + '://' + hostAndPath + '?token=' + __WS_TOKEN__, 'vite-hmr');\n testWs.addEventListener('open', function() { testWs.close(); resolve(); });\n testWs.addEventListener('error', function() { reject(); });\n setTimeout(function() { try { testWs.close(); } catch(e) {} reject(); }, 3000);\n });\n wsReady = true;\n } catch(e) {\n console.log('[SmartReload] WebSocket not ready yet, attempt ' + (attempt + 1) + '/10');\n }\n }\n if (wsReady) {\n console.log('[SmartReload] Vite client: WebSocket is ready, reconnecting now');\n socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);\n } else {\n console.log('[SmartReload] Vite client: WebSocket still not ready after retries, reloading page');\n location.reload();\n }\n } else {\n location.reload();\n }`,\n );\n\n if (transformed !== code) {\n return {\n code: transformed,\n map: null,\n };\n }\n\n return null;\n },\n };\n}\n","import type { Plugin } from 'vite';\n\nexport interface TaroStyleAdapterOptions {\n /**\n * Design width for responsive scaling (default: 375)\n * Should match Taro config's designWidth\n */\n designWidth?: number;\n \n /**\n * Maximum viewport width to scale (default: 750)\n * Beyond this width, font-size stays fixed\n */\n maxWidth?: number;\n\n /**\n * Base font size in pixels (default: 12)\n * Should match Taro config's baseFontSize\n */\n baseFontSize?: number;\n\n /**\n * Minimum root font size in pixels (default: 12)\n * Should match Taro config's minRootSize\n */\n minRootSize?: number;\n\n /**\n * Maximum root font size in pixels (default: 24)\n * Should match Taro config's maxRootSize\n */\n maxRootSize?: number;\n}\n\n/**\n * Taro H5 样式适配插件\n * \n * 功能:\n * 1. 动态计算根字号,使 H5 的 rem 与小程序的 rpx 行为对齐\n * 2. 隐藏 Taro 页面容器的滚动条,用于编辑器 iframe 预览\n * \n * 原理:\n * - 小程序 rpx: 1rpx = 屏幕宽度 / 750\n * - H5 rem: Taro 把 rpx 转成 rem,基于 baseFontSize\n * - 本插件动态设置 html font-size,使 rem 的实际像素值与 rpx 对齐\n * \n * 注意:此插件会覆盖 Taro 默认注入的根字号脚本,配置需与 Taro pxtransform 保持一致\n * \n * @example\n * ```ts\n * // In Taro config/dev.ts\n * import { taroStyleAdapterPlugin } from '@amaster.ai/vite-plugins'\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [\n * taroStyleAdapterPlugin({ \n * designWidth: 375,\n * baseFontSize: 12,\n * minRootSize: 12,\n * maxRootSize: 24\n * })\n * ]\n * }\n * }\n * ```\n */\nexport function taroStyleAdapterPlugin(options: TaroStyleAdapterOptions = {}): Plugin {\n const { \n designWidth = 375, \n maxWidth = 750,\n baseFontSize = 12,\n minRootSize = 12,\n maxRootSize = 24\n } = options;\n \n return {\n name: 'vite-plugin-taro-style-adapter',\n apply: 'serve', // 仅在开发模式下生效\n transformIndexHtml(html) {\n // 动态根字号脚本\n // 公式:fontSize = (clientWidth / designWidth) * baseFontSize\n // 当 designWidth=375, baseFontSize=12 时:\n // - 375px 屏幕 → 12px (1rem = 12px)\n // - 750px 屏幕 → 24px (1rem = 24px)\n // 这样 rem 值在不同屏幕宽度下的视觉比例与小程序 rpx 一致\n const flexibleScript = `\n<script data-taro-flexible=\"true\">\n(function() {\n var designWidth = ${designWidth};\n var maxWidth = ${maxWidth};\n var baseFontSize = ${baseFontSize};\n var minRootSize = ${minRootSize};\n var maxRootSize = ${maxRootSize};\n \n function setRootFontSize() {\n var docEl = document.documentElement;\n var clientWidth = docEl.clientWidth;\n \n // 限制最大宽度\n if (clientWidth > maxWidth) {\n clientWidth = maxWidth;\n }\n \n // 计算根字号: (屏幕宽度 / 设计稿宽度) * 基准字号\n var fontSize = (clientWidth / designWidth) * baseFontSize;\n \n // 应用最小/最大限制\n if (fontSize < minRootSize) {\n fontSize = minRootSize;\n } else if (fontSize > maxRootSize) {\n fontSize = maxRootSize;\n }\n \n docEl.style.fontSize = fontSize + 'px';\n }\n \n setRootFontSize();\n \n // 监听窗口变化\n window.addEventListener('resize', setRootFontSize);\n window.addEventListener('orientationchange', setRootFontSize);\n \n // 页面显示时重新计算(解决某些浏览器的 bug)\n document.addEventListener('DOMContentLoaded', setRootFontSize);\n})();\n</script>\n`;\n\n // 滚动条隐藏样式 + Taro Image 组件修复\n const styles = `\n<style data-taro-adapter=\"true\">\n /* 仅 H5 生效:隐藏 Taro 页面容器滚动条,但仍可滚动 */\n .taro_page {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro_page::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /* 隐藏 Taro ScrollView 组件的滚动条 */\n .taro-scroll-view__scroll-y,\n .taro-scroll-view__scroll-x {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro-scroll-view__scroll-y::-webkit-scrollbar,\n .taro-scroll-view__scroll-x::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /*\n * 修复 Taro Image 组件在 H5 下的尺寸问题\n * \n * 问题:Taro Image 组件使用 aspectFit 模式时,内部 img 使用 max-width/max-height: 100%\n * 导致小尺寸图片(如 SVG 图标)无法按指定尺寸显示,只显示原始尺寸\n * \n * 解决:让 img 填满 taro-image-core 容器,由容器控制尺寸\n */\n taro-image-core img.taro-img__mode-aspectfit {\n width: 100%;\n height: 100%;\n max-width: none;\n max-height: none;\n position: static;\n transform: none;\n object-fit: contain;\n }\n</style>\n`;\n \n // 在 </head> 之前插入脚本和样式\n return html.replace('</head>', `${flexibleScript}${styles}</head>`);\n }\n };\n}\n","/**\n * PostCSS 插件:H5 模式下将 rpx 预转换为 px\n *\n * 问题背景:\n * - Taro 的 pxtransform 在 H5 模式下会处理 rpx,但它把 rpx 值直接除以 rootValue\n * - 导致 96rpx 被转成 8rem(96/12),实际显示为 96px\n * - 而正确应该是:96rpx = 48px = 4rem = 48px\n *\n * 解决方案:\n * - 在 pxtransform 之前,先把 rpx 转成 px(rpx / 2)\n * - 然后 pxtransform 会正确地把 px 转成 rem(px / 12)\n *\n * 转换链路:96rpx → 48px → 4rem → 48px(正确)\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { rpx2pxPlugin } from '@amaster.ai/vite-plugins'\n * \n * // 在 postcss-config-loader-plugin 中使用\n * {\n * name: 'postcss-config-loader-plugin',\n * config(config) {\n * if (typeof config.css?.postcss === 'object') {\n * config.css?.postcss.plugins?.unshift(tailwindcss())\n * // H5 模式下添加 rpx2px 插件\n * if (process.env.TARO_ENV === 'h5') {\n * config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n * }\n * }\n * }\n * }\n * ```\n */\n\nexport interface Rpx2pxPluginOptions {\n /**\n * rpx 到 px 的转换比例(默认 2)\n * 即 1rpx = 1/ratio px\n * 默认值 2 表示 2rpx = 1px(基于 375 设计稿)\n */\n ratio?: number;\n}\n\ninterface PostCSSDeclaration {\n value: string;\n}\n\ninterface PostCSSPlugin {\n postcssPlugin: string;\n Declaration: (decl: PostCSSDeclaration) => void;\n}\n\ntype PostCSSPluginCreator = {\n (): PostCSSPlugin;\n postcss: boolean;\n};\n\nexport function rpx2pxPlugin(options: Rpx2pxPluginOptions = {}): PostCSSPlugin {\n const { ratio = 2 } = options;\n \n return {\n postcssPlugin: 'postcss-rpx2px',\n Declaration(decl: PostCSSDeclaration) {\n if (decl.value.includes('rpx')) {\n decl.value = decl.value.replace(/(-?\\d*\\.?\\d+)rpx/gi, (_match: string, num: string) => {\n const pxValue = parseFloat(num) / ratio;\n return pxValue === 0 ? '0' : `${pxValue}px`;\n });\n }\n }\n };\n}\n\n// PostCSS 插件标识\n(rpx2pxPlugin as PostCSSPluginCreator).postcss = true;\n","/**\n * Taro Environment Variables Injection Helper\n * \n * Automatically injects environment variables into Taro's defineConstants\n * so that they will be replaced at build time.\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { injectTaroEnv } from '@amaster.ai/vite-plugins/taro-env-inject'\n * \n * export default defineConfig({\n * defineConstants: {\n * ...injectTaroEnv()\n * }\n * })\n * ```\n */\n\nexport interface TaroEnvInjectOptions {\n /**\n * Additional environment variable names to inject\n * @default []\n */\n additional?: string[];\n \n /**\n * Whether to inject all TARO_APP_* prefixed variables\n * @default true\n */\n autoInjectTaroApp?: boolean;\n \n /**\n * Whether to inject all VITE_* prefixed variables\n * @default true\n */\n autoInjectVite?: boolean;\n}\n\n/**\n * Inject environment variables into Taro's defineConstants\n * \n * This function reads environment variables and formats them for Taro's defineConstants.\n * Taro will replace these at build time, converting `process.env.XXX` to actual values.\n */\nexport function injectTaroEnv(options: TaroEnvInjectOptions = {}): Record<string, string> {\n const {\n additional = [],\n autoInjectTaroApp = true,\n autoInjectVite = true,\n } = options;\n\n const constants: Record<string, string> = {};\n\n // Collect variable names to inject\n const varsToInject = new Set<string>(additional);\n\n // Auto-detect TARO_APP_* variables\n if (autoInjectTaroApp) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('TARO_APP_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Auto-detect VITE_* variables\n if (autoInjectVite) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('VITE_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Inject variables into defineConstants format\n varsToInject.forEach(varName => {\n const value = process.env[varName] || '';\n constants[`process.env.${varName}`] = JSON.stringify(value);\n });\n\n return constants;\n}\n\n/**\n * Inject specific environment variables for amaster.ai mini-program builds\n * \n * This is a convenience function that injects the standard environment variables\n * used by @amaster.ai/http-client for API base URL configuration.\n */\nexport function injectAmasterEnv(): Record<string, string> {\n return {\n 'process.env.TARO_APP_API_BASE_URL': JSON.stringify(process.env.TARO_APP_API_BASE_URL || ''),\n 'process.env.VITE_API_BASE_URL': JSON.stringify(process.env.VITE_API_BASE_URL || ''),\n };\n}\n","import { componentIdPlugin } from \"./component-id\";\nimport { editorBridgePlugin } from \"./editor-bridge\";\nimport { routesExposePlugin } from \"./routes-expose\";\nimport { browserLogsPlugin } from \"./browser-logs\";\nimport { smartReloadPlugin } from \"./smart-reload\";\nimport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\nimport { rpx2pxPlugin } from \"./postcss-rpx2px\";\nimport process from \"node:process\";\nimport type { Plugin } from \"vite\";\n\nexport interface DevToolsOptions {\n /**\n * Taro style adapter options\n * Controls H5 responsive scaling to match mini-program rpx behavior\n */\n styleAdapter?: TaroStyleAdapterOptions;\n}\n\n/**\n * 开发工具插件集合,简化调用,vite.config.ts 直接 devTools() 即可\n * @param options 配置项\n * @param options.styleAdapter Taro H5 样式适配配置,designWidth 应与 Taro config 一致\n * @returns Plugin[] Vite 插件数组\n * \n * @example\n * ```ts\n * // config/dev.ts\n * import devTools from \"@amaster.ai/vite-plugins\";\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [devTools()]\n * }\n * }\n * ```\n */\nexport default function devTools(options: DevToolsOptions = {}): Plugin[] {\n const plugins: Plugin[] = [\n taroStyleAdapterPlugin(options.styleAdapter), // Taro H5 样式适配(运行时)\n {\n // H5 开发模式下添加 rpx2px PostCSS 插件(编译时)\n name: 'dev-postcss-rpx2px-plugin',\n apply: 'serve',\n config(config) {\n if (process.env.TARO_ENV === 'h5' && typeof config.css?.postcss === 'object') {\n // 在最前面插入,确保在 pxtransform 之前执行\n config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n }\n }\n },\n componentIdPlugin(),\n editorBridgePlugin(),\n routesExposePlugin(),\n smartReloadPlugin() // 智能重载:拦截 full-reload,静默等待重连\n ]\n\n // process.env.WORKSPACE_GIT_REPO 有这个表示是在 sandbox 里面运行,启用浏览器日志插件\n if (process.env.WORKSPACE_GIT_REPO) {\n plugins.push(browserLogsPlugin());\n }\n\n return plugins;\n}\n\nexport {\n componentIdPlugin,\n editorBridgePlugin,\n routesExposePlugin,\n smartReloadPlugin\n}\n\n// Export Taro environment injection helpers\nexport { injectTaroEnv, injectAmasterEnv } from \"./taro-env-inject\";\n\n// Export Taro style adapter plugin and its types\nexport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\n\n// Export PostCSS rpx2px plugin for H5 mode\nexport { rpx2pxPlugin, type Rpx2pxPluginOptions } from \"./postcss-rpx2px\";"]}
|
|
1
|
+
{"version":3,"sources":["../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts","../src/browser-logs.ts","../src/smart-reload.ts","../src/taro-style-adapter.ts","../src/postcss-rpx2px.ts","../src/taro-env-inject.ts","../src/index.ts"],"names":["generateUniqueId","filePath","position","key","createHash","HTML_TAGS","componentIdPlugin","isDev","config","code","id","transformedCode","jsxOpenTagRegex","match","matches","tag","tagStartIndex","tagEndIndex","checkIndex","foundClosing","hasComponentId","currentCode","i","item","index","uniqueId","before","after","clickHandlerScript","getBridgeScriptContent","__filename","fileURLToPath","__dirname","dirname","bridgePath","resolve","readFileSync","error","editorBridgePlugin","bridgeScript","html","scriptsToInject","routesExposePlugin","options","routesPath","browserLogsPlugin","logFilePath","injectedScript","root","process","path","devServer","req","res","next","origin","body","chunk","logDir","fs","smartReloadPlugin","pureScript","transformed","taroStyleAdapterPlugin","designWidth","maxWidth","baseFontSize","minRootSize","maxRootSize","flexibleScript","rpx2pxPlugin","ratio","decl","_match","num","pxValue","injectTaroEnv","additional","autoInjectTaroApp","autoInjectVite","constants","varsToInject","varName","value","injectAmasterEnv","devTools","plugins"],"mappings":"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AA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AA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","file":"index.cjs","sourcesContent":["import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { readFileSync } from \"fs\";\nimport { resolve, dirname } from \"path\";\nimport { fileURLToPath } from \"url\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * 读取构建后的 bridge 脚本内容\n */\nfunction getBridgeScriptContent(): string {\n try {\n // 获取当前模块的目录\n const __filename = fileURLToPath(import.meta.url);\n const __dirname = dirname(__filename);\n \n // 读取构建后的 bridge.bridge.js 文件\n const bridgePath = resolve(__dirname, \"../dist/bridge.bridge.js\");\n const bridgeContent = readFileSync(bridgePath, \"utf-8\");\n \n return `<script>\\n${bridgeContent}\\n</script>`;\n } catch (error) {\n console.warn(\"Failed to read bridge script:\", error);\n return \"\";\n }\n}\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode by reading the built bridge script\n */\nexport function editorBridgePlugin(): Plugin {\n let isDev = false;\n let bridgeScript = \"\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n \n // 在开发模式下读取 bridge 脚本内容\n if (isDev) {\n bridgeScript = getBridgeScriptContent();\n }\n },\n\n transformIndexHtml(html) {\n // 在开发模式下注入 bridge 脚本\n const devScript = isDev ? bridgeScript : \"\";\n\n const scriptsToInject = `${clickHandlerScript}${devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { Buffer } from \"node:buffer\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport process from \"node:process\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n // ============================================\n // Vite deps 504 error capture using PerformanceObserver\n // Only captures 504 errors for node_modules/.vite resources (pre-bundled dependencies)\n // ============================================\n var reportedUrls = {}; // Track reported URLs to avoid duplicates\n \n if (typeof PerformanceObserver !== 'undefined') {\n var perfObserver = new PerformanceObserver(function(list) {\n var entries = list.getEntries();\n for (var i = 0; i < entries.length; i++) {\n var perfEntry = entries[i];\n // Only check script resources\n if (perfEntry.initiatorType !== 'script') {\n continue;\n }\n \n var resourceUrl = perfEntry.name || '';\n // Only report errors for node_modules/.vite resources\n if (resourceUrl.indexOf('node_modules/.vite') === -1 && resourceUrl.indexOf('/node_modules/.vite') === -1) {\n continue;\n }\n \n // Check if response status is 504 (Gateway Timeout)\n // responseStatus is available in Resource Timing Level 2\n var status = perfEntry.responseStatus;\n if (status === 504) {\n // Avoid duplicate reports\n if (reportedUrls[resourceUrl]) {\n continue;\n }\n reportedUrls[resourceUrl] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: resourceUrl,\n status: 504,\n message: 'Vite pre-bundled dependency returned 504 Gateway Timeout: ' + resourceUrl,\n duration: perfEntry.duration,\n transferSize: perfEntry.transferSize\n };\n addLog(entry);\n }\n }\n });\n \n try {\n perfObserver.observe({ type: 'resource', buffered: true });\n } catch (e) {\n // Fallback for browsers that don't support the options\n try {\n perfObserver.observe({ entryTypes: ['resource'] });\n } catch (e2) {\n originalConsole.warn('[BrowserLogs] PerformanceObserver not supported:', e2.message);\n }\n }\n }\n \n // ============================================\n // Script load error capture (for 504 and other network errors)\n // This captures errors that PerformanceObserver might miss\n // ============================================\n \n // Use MutationObserver to watch for dynamically added script tags\n function attachScriptErrorHandler(script) {\n if (script.__errorHandlerAttached__) return;\n script.__errorHandlerAttached__ = true;\n \n script.addEventListener('error', function(event) {\n var src = script.src || '';\n \n // Only report errors for node_modules/.vite resources\n if (src.indexOf('node_modules/.vite') === -1 && src.indexOf('/node_modules/.vite') === -1) {\n return;\n }\n \n // Avoid duplicate reports\n if (reportedUrls[src]) {\n return;\n }\n reportedUrls[src] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: src,\n status: 'load-error',\n message: 'Vite pre-bundled dependency failed to load (possibly 504 Gateway Timeout): ' + src,\n errorType: event.type\n };\n addLog(entry);\n });\n }\n \n // Attach error handlers to existing scripts\n var existingScripts = document.querySelectorAll('script[src]');\n for (var i = 0; i < existingScripts.length; i++) {\n attachScriptErrorHandler(existingScripts[i]);\n }\n \n // Watch for dynamically added scripts\n if (typeof MutationObserver !== 'undefined') {\n var scriptObserver = new MutationObserver(function(mutations) {\n for (var i = 0; i < mutations.length; i++) {\n var mutation = mutations[i];\n for (var j = 0; j < mutation.addedNodes.length; j++) {\n var node = mutation.addedNodes[j];\n if (node.nodeName === 'SCRIPT' && node.src) {\n attachScriptErrorHandler(node);\n }\n // Also check child nodes\n if (node.querySelectorAll) {\n var scripts = node.querySelectorAll('script[src]');\n for (var k = 0; k < scripts.length; k++) {\n attachScriptErrorHandler(scripts[k]);\n }\n }\n }\n }\n });\n \n scriptObserver.observe(document.documentElement, {\n childList: true,\n subtree: true\n });\n }\n \n // Also intercept document.createElement to catch scripts before they're added to DOM\n var originalCreateElement = document.createElement.bind(document);\n document.createElement = function(tagName) {\n var element = originalCreateElement(tagName);\n if (tagName.toLowerCase() === 'script') {\n // Use a setter to catch when src is set\n var originalSrc = '';\n Object.defineProperty(element, '__originalSrc__', {\n get: function() { return originalSrc; },\n set: function(val) { originalSrc = val; }\n });\n \n // Defer attaching error handler until src is set\n var srcDescriptor = Object.getOwnPropertyDescriptor(HTMLScriptElement.prototype, 'src');\n if (srcDescriptor && srcDescriptor.set) {\n var originalSrcSetter = srcDescriptor.set;\n Object.defineProperty(element, 'src', {\n get: function() {\n return srcDescriptor.get ? srcDescriptor.get.call(this) : this.getAttribute('src');\n },\n set: function(value) {\n if (originalSrcSetter) {\n originalSrcSetter.call(this, value);\n } else {\n this.setAttribute('src', value);\n }\n attachScriptErrorHandler(this);\n },\n configurable: true\n });\n }\n }\n return element;\n };\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Smart Reload 插件\n *\n * 在 Vite 服务器重启期间(如 PM2 重启 web-server),拦截浏览器端的刷新行为,\n * 让浏览器静默等待服务器恢复,期间页面完全不刷新。\n * 服务器恢复后自动恢复 HMR 连接,无需手动刷新。\n *\n * 场景说明:\n * - PM2 杀掉 Vite 进程 → WebSocket 断开 → PM2 重启 → 新 Vite 启动\n * - Vite 客户端检测到 WebSocket 断开后会通过 HTTP ping 轮询等待服务器恢复\n * - 服务器恢复后 Vite 客户端会调用 location.reload() 刷新页面\n * - 本插件拦截这个刷新行为,让页面保持当前状态\n *\n * 实现原理(双层拦截):\n * 1. 服务端层:通过 transform 钩子修改 /@vite/client 源码,\n * 将 location.reload() 替换为 window.__smartReload__(),\n * 使我们能控制是否真正刷新页面\n * 2. 客户端层:注入脚本拦截 WebSocket 消息,\n * 检测服务器重启场景,在重启期间阻止刷新\n *\n * 区分两种 full-reload 场景:\n * 1. 服务器重启 → WebSocket 断开 → 静默等待,不刷新,自动恢复 HMR\n * 2. 模块变更需要 full-reload → WebSocket 保持连接 → 正常刷新\n */\nexport function smartReloadPlugin(): Plugin {\n // 注入到浏览器端的脚本\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n\n var PLUGIN_TAG = '[SmartReload]';\n // full-reload 后等待 WebSocket 断开的窗口期(ms)\n var DISCONNECT_DETECT_WINDOW = 300;\n\n // ============================================\n // 状态管理\n // ============================================\n var state = {\n // 是否已经拦截了 Vite HMR WebSocket\n wsIntercepted: false,\n // 是否正在等待判断 full-reload 类型(300ms 窗口期)\n detectingReloadType: false,\n // 延迟执行 full-reload 的定时器\n reloadTimer: null,\n // 是否是服务器重启场景\n isServerRestart: false\n };\n\n // ============================================\n // 日志工具\n // ============================================\n function log() {\n var args = [PLUGIN_TAG];\n for (var i = 0; i < arguments.length; i++) {\n args.push(arguments[i]);\n }\n console.log.apply(console, args);\n }\n\n // ============================================\n // 重置状态\n // ============================================\n function resetState() {\n state.detectingReloadType = false;\n state.isServerRestart = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n }\n\n // ============================================\n // 执行真正的 full-reload\n // ============================================\n function executeFullReload() {\n log('Executing full-reload');\n resetState();\n location.reload();\n }\n\n // ============================================\n // __smartReload__:检查是否应该阻止 reload\n //\n // 返回 true 表示是服务器重启场景,不应该刷新页面\n // 返回 false 表示应该正常刷新\n //\n // Vite 客户端代码中的 WebSocket close handler 已被修改为:\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(...); // 重新创建 WebSocket 恢复 HMR\n // } else {\n // location.reload();\n // }\n // ============================================\n window.__smartReload__ = function() {\n if (state.isServerRestart) {\n log('Server restart detected, suppressing reload. Will reconnect WebSocket.');\n resetState();\n return true; // 阻止 reload,由 Vite 客户端重新创建 WebSocket\n }\n log('Non-restart scenario, allowing reload');\n return false; // 允许正常 reload\n };\n\n // ============================================\n // 拦截 WebSocket 构造函数\n // ============================================\n var OriginalWebSocket = window.WebSocket;\n\n function isViteHmrWebSocket(url) {\n if (typeof url !== 'string') return false;\n return url.includes('/__vite_hmr') || url.includes('?token=');\n }\n\n function createWsProxy(ws) {\n var originalAddEventListener = ws.addEventListener.bind(ws);\n var originalRemoveEventListener = ws.removeEventListener.bind(ws);\n\n ws.addEventListener = function(type, listener, options) {\n if (type === 'message') {\n var wrappedListener = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted) {\n listener.call(ws, event);\n }\n };\n listener.__smartReloadWrapped__ = wrappedListener;\n return originalAddEventListener(type, wrappedListener, options);\n }\n return originalAddEventListener(type, listener, options);\n };\n\n ws.removeEventListener = function(type, listener, options) {\n if (type === 'message' && listener.__smartReloadWrapped__) {\n return originalRemoveEventListener(type, listener.__smartReloadWrapped__, options);\n }\n return originalRemoveEventListener(type, listener, options);\n };\n\n var originalOnMessage = null;\n var onMessageDescriptor = Object.getOwnPropertyDescriptor(WebSocket.prototype, 'onmessage') ||\n Object.getOwnPropertyDescriptor(ws, 'onmessage');\n\n if (onMessageDescriptor) {\n Object.defineProperty(ws, 'onmessage', {\n get: function() {\n return originalOnMessage;\n },\n set: function(handler) {\n originalOnMessage = handler;\n var wrappedHandler = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted && handler) {\n handler.call(ws, event);\n }\n };\n if (onMessageDescriptor.set) {\n onMessageDescriptor.set.call(ws, wrappedHandler);\n }\n },\n configurable: true\n });\n }\n\n originalAddEventListener('close', function(event) {\n handleWsClose(event);\n });\n\n originalAddEventListener('open', function() {\n handleWsOpen();\n });\n\n state.wsIntercepted = true;\n log('Vite HMR WebSocket intercepted');\n\n return ws;\n }\n\n window.WebSocket = function SmartReloadWebSocket(url, protocols) {\n var ws;\n if (protocols !== undefined) {\n ws = new OriginalWebSocket(url, protocols);\n } else {\n ws = new OriginalWebSocket(url);\n }\n\n if (isViteHmrWebSocket(url)) {\n return createWsProxy(ws);\n }\n\n return ws;\n };\n\n window.WebSocket.prototype = OriginalWebSocket.prototype;\n window.WebSocket.CONNECTING = OriginalWebSocket.CONNECTING;\n window.WebSocket.OPEN = OriginalWebSocket.OPEN;\n window.WebSocket.CLOSING = OriginalWebSocket.CLOSING;\n window.WebSocket.CLOSED = OriginalWebSocket.CLOSED;\n\n // ============================================\n // WebSocket 消息处理\n // ============================================\n function handleWsMessage(event) {\n var data;\n try {\n data = JSON.parse(event.data);\n } catch (e) {\n return false;\n }\n\n // 处理 full-reload 消息\n if (data.type === 'full-reload') {\n // 如果在服务器重启模式,拦截所有 full-reload\n if (state.isServerRestart) {\n log('Suppressed full-reload message during server restart');\n return true;\n }\n\n log('Intercepted full-reload, path:', data.path || '(none)');\n\n // 进入检测模式:等待看 WebSocket 是否断开\n state.detectingReloadType = true;\n\n // 如果窗口期内 WebSocket 没有断开,说明是真正的模块变更\n state.reloadTimer = setTimeout(function() {\n if (state.detectingReloadType && !state.isServerRestart) {\n log('WebSocket still connected after ' + DISCONNECT_DETECT_WINDOW + 'ms - real module change, executing reload');\n state.detectingReloadType = false;\n executeFullReload();\n }\n }, DISCONNECT_DETECT_WINDOW);\n\n return true;\n }\n\n // 处理 connected 消息\n if (data.type === 'connected') {\n if (state.isServerRestart) {\n log('HMR reconnected after server restart! Page was NOT reloaded.');\n resetState();\n }\n return false;\n }\n\n return false;\n }\n\n // ============================================\n // WebSocket 连接事件处理\n // ============================================\n function handleWsClose(event) {\n log('WebSocket closed, code:', event.code, 'wasClean:', event.wasClean);\n\n // 场景1:收到 full-reload 后 WebSocket 断开 → 服务器重启\n if (state.detectingReloadType) {\n log('Entering server restart mode: full-reload followed by disconnect');\n state.isServerRestart = true;\n state.detectingReloadType = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n return;\n }\n\n // 场景2:WebSocket 直接断开(非正常关闭)\n if (!state.isServerRestart && !event.wasClean) {\n log('Entering server restart mode: unexpected disconnect');\n state.isServerRestart = true;\n }\n }\n\n function handleWsOpen() {\n log('WebSocket opened');\n }\n\n // ============================================\n // 暴露 API 供调试使用\n // ============================================\n window.__smartReloadState__ = state;\n window.__OriginalWebSocket__ = OriginalWebSocket;\n window.__smartReloadForceReload__ = function() {\n log('Force reload triggered by user');\n resetState();\n location.reload();\n };\n\n log('Smart Reload initialized');\n})();\n</script>`;\n\n // 提取纯 JS 代码(去掉 <script> 标签)\n const pureScript = injectedScript\n .replace(/^\\s*<script>\\s*/i, \"\")\n .replace(/\\s*<\\/script>\\s*$/i, \"\");\n\n return {\n name: \"vite-plugin-smart-reload\",\n\n // 只在开发模式下注入脚本\n apply: \"serve\",\n\n transformIndexHtml: {\n order: \"pre\",\n handler() {\n return [\n {\n tag: \"script\",\n // 注入到 head 的最前面,确保在 Vite 客户端脚本 /@vite/client 之前执行\n injectTo: \"head-prepend\",\n children: pureScript,\n },\n ];\n },\n },\n\n // 修改 Vite 客户端代码:\n // 精确替换 WebSocket close handler 中的 reload 逻辑,\n // 在服务器重启场景下重新创建 WebSocket 恢复 HMR,而不是刷新页面\n //\n // Vite 对 /@vite/client 的处理流程:\n // 1. 浏览器请求 /@vite/client\n // 2. Vite 的 transformMiddleware 拦截请求\n // 3. 调用 server.transformRequest('/@vite/client')\n // 4. 内部解析为文件系统路径 node_modules/vite/dist/client/client.mjs\n // 5. 调用插件的 transform(code, resolvedId) — id 是文件系统路径\n transform(code, id) {\n // 匹配 Vite 客户端模块(文件系统路径)\n if (!id.includes(\"vite/dist/client/client.mjs\")) {\n return null;\n }\n\n // 精确替换 WebSocket close handler 中的 reload 逻辑:\n // 原始代码:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // location.reload();\n // 替换为:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);\n // } else {\n // location.reload();\n // }\n const transformed = code.replace(\n /await waitForSuccessfulPing\\(protocol,\\s*hostAndPath\\);\\s*\\n\\s*location\\.reload\\(\\)/,\n `await waitForSuccessfulPing(protocol, hostAndPath);\n if (window.__smartReload__ && window.__smartReload__()) {\n console.log('[SmartReload] Vite client: server is back, fetching new WebSocket token...');\n // PM2 重启后新服务器会生成新的 wsToken,旧 token 无法连接\n // 通过 fetch /@vite/client 获取新 token\n try {\n var clientResp = await fetch('/@vite/client');\n var clientCode = await clientResp.text();\n // 从新的客户端代码中提取 wsToken\n // Vite 编译后的代码中 wsToken 是一个字符串常量,如: const wsToken = \"xxxxxxxxxxxx\"\n var tokenMatch = clientCode.match(/const\\\\s+wsToken\\\\s*=\\\\s*\"([^\"]+)\"/);\n if (tokenMatch && tokenMatch[1]) {\n console.log('[SmartReload] Got new wsToken from server');\n // 更新闭包中的 wsToken 变量(Vite 客户端代码中的 const wsToken)\n // 注意:wsToken 是 const,无法直接修改。我们需要修改 setupWebSocket 使用的 URL\n // 方案:直接创建 WebSocket 连接,绕过 setupWebSocket\n var newToken = tokenMatch[1];\n var wsUrl = protocol + '://' + hostAndPath + '?token=' + newToken;\n console.log('[SmartReload] Reconnecting with new token to: ' + wsUrl);\n var newSocket = new WebSocket(wsUrl, 'vite-hmr');\n var newIsOpened = false;\n newSocket.addEventListener('open', function() {\n newIsOpened = true;\n console.log('[SmartReload] WebSocket reconnected successfully!');\n notifyListeners('vite:ws:connect', { webSocket: newSocket });\n }, { once: true });\n newSocket.addEventListener('message', async function(ev) {\n handleMessage(JSON.parse(ev.data));\n });\n newSocket.addEventListener('close', async function(ev) {\n if (ev.wasClean) return;\n if (!newIsOpened && onCloseWithoutOpen) {\n onCloseWithoutOpen();\n return;\n }\n notifyListeners('vite:ws:disconnect', { webSocket: newSocket });\n if (hasDocument) {\n console.log('[vite] server connection lost. Polling for restart...');\n await waitForSuccessfulPing(protocol, hostAndPath);\n if (window.__smartReload__ && window.__smartReload__()) {\n console.log('[SmartReload] Nested reconnect - reloading page');\n }\n location.reload();\n }\n });\n socket = newSocket;\n } else {\n console.log('[SmartReload] Could not extract new wsToken, reloading page');\n location.reload();\n }\n } catch(e) {\n console.log('[SmartReload] Failed to fetch new client code, reloading page', e);\n location.reload();\n }\n } else {\n location.reload();\n }`,\n );\n\n if (transformed !== code) {\n return {\n code: transformed,\n map: null,\n };\n }\n\n return null;\n },\n };\n}\n","import type { Plugin } from 'vite';\n\nexport interface TaroStyleAdapterOptions {\n /**\n * Design width for responsive scaling (default: 375)\n * Should match Taro config's designWidth\n */\n designWidth?: number;\n \n /**\n * Maximum viewport width to scale (default: 750)\n * Beyond this width, font-size stays fixed\n */\n maxWidth?: number;\n\n /**\n * Base font size in pixels (default: 12)\n * Should match Taro config's baseFontSize\n */\n baseFontSize?: number;\n\n /**\n * Minimum root font size in pixels (default: 12)\n * Should match Taro config's minRootSize\n */\n minRootSize?: number;\n\n /**\n * Maximum root font size in pixels (default: 24)\n * Should match Taro config's maxRootSize\n */\n maxRootSize?: number;\n}\n\n/**\n * Taro H5 样式适配插件\n * \n * 功能:\n * 1. 动态计算根字号,使 H5 的 rem 与小程序的 rpx 行为对齐\n * 2. 隐藏 Taro 页面容器的滚动条,用于编辑器 iframe 预览\n * \n * 原理:\n * - 小程序 rpx: 1rpx = 屏幕宽度 / 750\n * - H5 rem: Taro 把 rpx 转成 rem,基于 baseFontSize\n * - 本插件动态设置 html font-size,使 rem 的实际像素值与 rpx 对齐\n * \n * 注意:此插件会覆盖 Taro 默认注入的根字号脚本,配置需与 Taro pxtransform 保持一致\n * \n * @example\n * ```ts\n * // In Taro config/dev.ts\n * import { taroStyleAdapterPlugin } from '@amaster.ai/vite-plugins'\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [\n * taroStyleAdapterPlugin({ \n * designWidth: 375,\n * baseFontSize: 12,\n * minRootSize: 12,\n * maxRootSize: 24\n * })\n * ]\n * }\n * }\n * ```\n */\nexport function taroStyleAdapterPlugin(options: TaroStyleAdapterOptions = {}): Plugin {\n const { \n designWidth = 375, \n maxWidth = 750,\n baseFontSize = 12,\n minRootSize = 12,\n maxRootSize = 24\n } = options;\n \n return {\n name: 'vite-plugin-taro-style-adapter',\n apply: 'serve', // 仅在开发模式下生效\n transformIndexHtml(html) {\n // 动态根字号脚本\n // 公式:fontSize = (clientWidth / designWidth) * baseFontSize\n // 当 designWidth=375, baseFontSize=12 时:\n // - 375px 屏幕 → 12px (1rem = 12px)\n // - 750px 屏幕 → 24px (1rem = 24px)\n // 这样 rem 值在不同屏幕宽度下的视觉比例与小程序 rpx 一致\n const flexibleScript = `\n<script data-taro-flexible=\"true\">\n(function() {\n var designWidth = ${designWidth};\n var maxWidth = ${maxWidth};\n var baseFontSize = ${baseFontSize};\n var minRootSize = ${minRootSize};\n var maxRootSize = ${maxRootSize};\n \n function setRootFontSize() {\n var docEl = document.documentElement;\n var clientWidth = docEl.clientWidth;\n \n // 限制最大宽度\n if (clientWidth > maxWidth) {\n clientWidth = maxWidth;\n }\n \n // 计算根字号: (屏幕宽度 / 设计稿宽度) * 基准字号\n var fontSize = (clientWidth / designWidth) * baseFontSize;\n \n // 应用最小/最大限制\n if (fontSize < minRootSize) {\n fontSize = minRootSize;\n } else if (fontSize > maxRootSize) {\n fontSize = maxRootSize;\n }\n \n docEl.style.fontSize = fontSize + 'px';\n }\n \n setRootFontSize();\n \n // 监听窗口变化\n window.addEventListener('resize', setRootFontSize);\n window.addEventListener('orientationchange', setRootFontSize);\n \n // 页面显示时重新计算(解决某些浏览器的 bug)\n document.addEventListener('DOMContentLoaded', setRootFontSize);\n})();\n</script>\n`;\n\n // 滚动条隐藏样式 + Taro Image 组件修复\n const styles = `\n<style data-taro-adapter=\"true\">\n /* 仅 H5 生效:隐藏 Taro 页面容器滚动条,但仍可滚动 */\n .taro_page {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro_page::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /* 隐藏 Taro ScrollView 组件的滚动条 */\n .taro-scroll-view__scroll-y,\n .taro-scroll-view__scroll-x {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro-scroll-view__scroll-y::-webkit-scrollbar,\n .taro-scroll-view__scroll-x::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /*\n * 修复 Taro Image 组件在 H5 下的尺寸问题\n * \n * 问题:Taro Image 组件使用 aspectFit 模式时,内部 img 使用 max-width/max-height: 100%\n * 导致小尺寸图片(如 SVG 图标)无法按指定尺寸显示,只显示原始尺寸\n * \n * 解决:让 img 填满 taro-image-core 容器,由容器控制尺寸\n */\n taro-image-core img.taro-img__mode-aspectfit {\n width: 100%;\n height: 100%;\n max-width: none;\n max-height: none;\n position: static;\n transform: none;\n object-fit: contain;\n }\n</style>\n`;\n \n // 在 </head> 之前插入脚本和样式\n return html.replace('</head>', `${flexibleScript}${styles}</head>`);\n }\n };\n}\n","/**\n * PostCSS 插件:H5 模式下将 rpx 预转换为 px\n *\n * 问题背景:\n * - Taro 的 pxtransform 在 H5 模式下会处理 rpx,但它把 rpx 值直接除以 rootValue\n * - 导致 96rpx 被转成 8rem(96/12),实际显示为 96px\n * - 而正确应该是:96rpx = 48px = 4rem = 48px\n *\n * 解决方案:\n * - 在 pxtransform 之前,先把 rpx 转成 px(rpx / 2)\n * - 然后 pxtransform 会正确地把 px 转成 rem(px / 12)\n *\n * 转换链路:96rpx → 48px → 4rem → 48px(正确)\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { rpx2pxPlugin } from '@amaster.ai/vite-plugins'\n * \n * // 在 postcss-config-loader-plugin 中使用\n * {\n * name: 'postcss-config-loader-plugin',\n * config(config) {\n * if (typeof config.css?.postcss === 'object') {\n * config.css?.postcss.plugins?.unshift(tailwindcss())\n * // H5 模式下添加 rpx2px 插件\n * if (process.env.TARO_ENV === 'h5') {\n * config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n * }\n * }\n * }\n * }\n * ```\n */\n\nexport interface Rpx2pxPluginOptions {\n /**\n * rpx 到 px 的转换比例(默认 2)\n * 即 1rpx = 1/ratio px\n * 默认值 2 表示 2rpx = 1px(基于 375 设计稿)\n */\n ratio?: number;\n}\n\ninterface PostCSSDeclaration {\n value: string;\n}\n\ninterface PostCSSPlugin {\n postcssPlugin: string;\n Declaration: (decl: PostCSSDeclaration) => void;\n}\n\ntype PostCSSPluginCreator = {\n (): PostCSSPlugin;\n postcss: boolean;\n};\n\nexport function rpx2pxPlugin(options: Rpx2pxPluginOptions = {}): PostCSSPlugin {\n const { ratio = 2 } = options;\n \n return {\n postcssPlugin: 'postcss-rpx2px',\n Declaration(decl: PostCSSDeclaration) {\n if (decl.value.includes('rpx')) {\n decl.value = decl.value.replace(/(-?\\d*\\.?\\d+)rpx/gi, (_match: string, num: string) => {\n const pxValue = parseFloat(num) / ratio;\n return pxValue === 0 ? '0' : `${pxValue}px`;\n });\n }\n }\n };\n}\n\n// PostCSS 插件标识\n(rpx2pxPlugin as PostCSSPluginCreator).postcss = true;\n","/**\n * Taro Environment Variables Injection Helper\n * \n * Automatically injects environment variables into Taro's defineConstants\n * so that they will be replaced at build time.\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { injectTaroEnv } from '@amaster.ai/vite-plugins/taro-env-inject'\n * \n * export default defineConfig({\n * defineConstants: {\n * ...injectTaroEnv()\n * }\n * })\n * ```\n */\n\nexport interface TaroEnvInjectOptions {\n /**\n * Additional environment variable names to inject\n * @default []\n */\n additional?: string[];\n \n /**\n * Whether to inject all TARO_APP_* prefixed variables\n * @default true\n */\n autoInjectTaroApp?: boolean;\n \n /**\n * Whether to inject all VITE_* prefixed variables\n * @default true\n */\n autoInjectVite?: boolean;\n}\n\n/**\n * Inject environment variables into Taro's defineConstants\n * \n * This function reads environment variables and formats them for Taro's defineConstants.\n * Taro will replace these at build time, converting `process.env.XXX` to actual values.\n */\nexport function injectTaroEnv(options: TaroEnvInjectOptions = {}): Record<string, string> {\n const {\n additional = [],\n autoInjectTaroApp = true,\n autoInjectVite = true,\n } = options;\n\n const constants: Record<string, string> = {};\n\n // Collect variable names to inject\n const varsToInject = new Set<string>(additional);\n\n // Auto-detect TARO_APP_* variables\n if (autoInjectTaroApp) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('TARO_APP_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Auto-detect VITE_* variables\n if (autoInjectVite) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('VITE_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Inject variables into defineConstants format\n varsToInject.forEach(varName => {\n const value = process.env[varName] || '';\n constants[`process.env.${varName}`] = JSON.stringify(value);\n });\n\n return constants;\n}\n\n/**\n * Inject specific environment variables for amaster.ai mini-program builds\n * \n * This is a convenience function that injects the standard environment variables\n * used by @amaster.ai/http-client for API base URL configuration.\n */\nexport function injectAmasterEnv(): Record<string, string> {\n return {\n 'process.env.TARO_APP_API_BASE_URL': JSON.stringify(process.env.TARO_APP_API_BASE_URL || ''),\n 'process.env.VITE_API_BASE_URL': JSON.stringify(process.env.VITE_API_BASE_URL || ''),\n };\n}\n","import { componentIdPlugin } from \"./component-id\";\nimport { editorBridgePlugin } from \"./editor-bridge\";\nimport { routesExposePlugin } from \"./routes-expose\";\nimport { browserLogsPlugin } from \"./browser-logs\";\nimport { smartReloadPlugin } from \"./smart-reload\";\nimport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\nimport { rpx2pxPlugin } from \"./postcss-rpx2px\";\nimport process from \"node:process\";\nimport type { Plugin } from \"vite\";\n\nexport interface DevToolsOptions {\n /**\n * Taro style adapter options\n * Controls H5 responsive scaling to match mini-program rpx behavior\n */\n styleAdapter?: TaroStyleAdapterOptions;\n}\n\n/**\n * 开发工具插件集合,简化调用,vite.config.ts 直接 devTools() 即可\n * @param options 配置项\n * @param options.styleAdapter Taro H5 样式适配配置,designWidth 应与 Taro config 一致\n * @returns Plugin[] Vite 插件数组\n * \n * @example\n * ```ts\n * // config/dev.ts\n * import devTools from \"@amaster.ai/vite-plugins\";\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [devTools()]\n * }\n * }\n * ```\n */\nexport default function devTools(options: DevToolsOptions = {}): Plugin[] {\n const plugins: Plugin[] = [\n taroStyleAdapterPlugin(options.styleAdapter), // Taro H5 样式适配(运行时)\n {\n // H5 开发模式下添加 rpx2px PostCSS 插件(编译时)\n name: 'dev-postcss-rpx2px-plugin',\n apply: 'serve',\n config(config) {\n if (process.env.TARO_ENV === 'h5' && typeof config.css?.postcss === 'object') {\n // 在最前面插入,确保在 pxtransform 之前执行\n config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n }\n }\n },\n componentIdPlugin(),\n editorBridgePlugin(),\n routesExposePlugin(),\n smartReloadPlugin() // 智能重载:拦截 full-reload,静默等待重连\n ]\n\n // process.env.WORKSPACE_GIT_REPO 有这个表示是在 sandbox 里面运行,启用浏览器日志插件\n if (process.env.WORKSPACE_GIT_REPO) {\n plugins.push(browserLogsPlugin());\n }\n\n return plugins;\n}\n\nexport {\n componentIdPlugin,\n editorBridgePlugin,\n routesExposePlugin,\n smartReloadPlugin\n}\n\n// Export Taro environment injection helpers\nexport { injectTaroEnv, injectAmasterEnv } from \"./taro-env-inject\";\n\n// Export Taro style adapter plugin and its types\nexport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\n\n// Export PostCSS rpx2px plugin for H5 mode\nexport { rpx2pxPlugin, type Rpx2pxPluginOptions } from \"./postcss-rpx2px\";"]}
|
package/dist/index.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {createHash}from'crypto';import m,{readFileSync}from'fs';import
|
|
1
|
+
import {createHash}from'crypto';import m,{readFileSync}from'fs';import b,{dirname,resolve}from'path';import {fileURLToPath}from'url';import x from'process';function k(n,t){let e=`${n}:${t}`;return createHash("md5").update(e).digest("hex").substring(0,12)}var E=new Set(["a","abbr","address","area","article","aside","audio","b","base","bdi","bdo","blockquote","body","br","button","canvas","caption","cite","code","col","colgroup","data","datalist","dd","del","details","dfn","dialog","div","dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","iframe","img","input","ins","kbd","label","legend","li","link","main","map","mark","meta","meter","nav","noscript","object","ol","optgroup","option","output","p","param","picture","pre","progress","q","rp","rt","ruby","s","samp","script","section","select","small","source","span","strong","style","sub","summary","sup","svg","table","tbody","td","template","textarea","tfoot","th","thead","time","title","tr","track","u","ul","var","video","wbr"]);function y(){let n=false;return {name:"vite-plugin-component-id",enforce:"pre",configResolved(t){n=t.command==="serve";},transform(t,e){if(!n||!/\.(tsx|jsx)$/.test(e)||e.includes("node_modules")||!/<[a-z]/.test(t))return null;try{let r=t,o=/<([a-z][\da-z]*)(?=[\s/>])/g,i,s=[];for(;(i=o.exec(t))!==null;){let a=i[1];if(!a)continue;let u=i.index,d=i.index+i[0].length;if(!E.has(a))continue;let c=d,p=!1,f=!1;for(;c<t.length&&!p;)t[c]===">"&&(p=!0,t.substring(u,c).includes("data-node-component-id")&&(f=!0)),c++;f||s.push({index:u,tag:a,tagEndIndex:d});}let l=t;for(let a=s.length-1;a>=0;a--){let u=s[a];if(!u)continue;let{index:d,tagEndIndex:c}=u,p=k(e,d),f=l.substring(0,c),v=l.substring(c);r=`${f} data-node-component-id="${p}"${v}`,l=r;}return {code:r,map:null}}catch{return null}}}}var L=`<script>
|
|
2
2
|
document.addEventListener("click", (e) => {
|
|
3
3
|
const element = e.target;
|
|
4
4
|
const noJumpOut = document.body.classList.contains("forbid-jump-out")
|
|
@@ -16,9 +16,9 @@ document.addEventListener("click", (e) => {
|
|
|
16
16
|
e.preventDefault();
|
|
17
17
|
}
|
|
18
18
|
});
|
|
19
|
-
</script>`;function
|
|
19
|
+
</script>`;function W(){try{let n=fileURLToPath(import.meta.url),t=dirname(n),e=resolve(t,"../dist/bridge.bridge.js");return `<script>
|
|
20
20
|
${readFileSync(e,"utf-8")}
|
|
21
|
-
</script>`}catch(n){return console.warn("Failed to read bridge script:",n),""}}function S(){let n=false,t="";return {name:"vite-plugin-editor-bridge",configResolved(e){n=e.command==="serve",n&&(t=
|
|
21
|
+
</script>`}catch(n){return console.warn("Failed to read bridge script:",n),""}}function S(){let n=false,t="";return {name:"vite-plugin-editor-bridge",configResolved(e){n=e.command==="serve",n&&(t=W());},transformIndexHtml(e){let o=`${L}${n?t:""}</body>`;return e.replace("</body>",o)}}}function w(n){let t=false,e=n?.routesFilePath||"src/routes.tsx";return {name:"vite-plugin-routes-expose",enforce:"post",configResolved(r){t=r.command==="serve";},transform(r,o){if(!t||!o.endsWith(e))return null;try{return r.includes("window.__APP_ROUTES__")?null:{code:`${r}
|
|
22
22
|
|
|
23
23
|
// Development mode: Expose routes to window.__APP_ROUTES__
|
|
24
24
|
if (typeof window !== 'undefined') {
|
|
@@ -705,7 +705,7 @@ if (typeof window !== 'undefined') {
|
|
|
705
705
|
|
|
706
706
|
originalConsole.log('[BrowserLogs] Log collection started');
|
|
707
707
|
})();
|
|
708
|
-
</script>`;return {name:"vite-plugin-browser-logs",configResolved(e){let r=e.root||
|
|
708
|
+
</script>`;return {name:"vite-plugin-browser-logs",configResolved(e){let r=e.root||x.cwd();n=b.join(r,"browser.log");},configureServer(e){e.middlewares.use((r,o,i)=>{if(r.url==="/__browser__"&&r.method==="POST"){let s=r.headers.origin||"*",l="";r.on("data",a=>{l+=a.toString();}),r.on("end",()=>{try{let a=b.dirname(n);m.existsSync(a)||m.mkdirSync(a,{recursive:!0}),m.appendFileSync(n,`${l}
|
|
709
709
|
`,"utf-8"),o.writeHead(200,{"Content-Type":"application/json","Access-Control-Allow-Origin":s,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),o.end(JSON.stringify({success:!0}));}catch(a){console.error("[BrowserLogs] Write error:",a),o.writeHead(500,{"Content-Type":"application/json","Access-Control-Allow-Origin":s,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),o.end(JSON.stringify({success:false,error:String(a)}));}});}else if(r.url==="/__browser__"&&r.method==="OPTIONS"){let s=r.headers.origin||"*";o.writeHead(204,{"Access-Control-Allow-Origin":s,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type","Access-Control-Max-Age":"86400"}),o.end();}else if(r.url==="/__browser__"){let s=r.headers.origin||"*";o.writeHead(405,{"Content-Type":"application/json","Access-Control-Allow-Origin":s}),o.end(JSON.stringify({error:"Method not allowed"}));}else i();}),console.log("[BrowserLogs] Logs will be written to:",n);},transformIndexHtml(e){return e.replace(/<head([^>]*)>/i,`<head$1>${t}`)}}}function T(){let t=`
|
|
710
710
|
<script>
|
|
711
711
|
(function() {
|
|
@@ -971,35 +971,61 @@ if (typeof window !== 'undefined') {
|
|
|
971
971
|
})();
|
|
972
972
|
</script>`.replace(/^\s*<script>\s*/i,"").replace(/\s*<\/script>\s*$/i,"");return {name:"vite-plugin-smart-reload",apply:"serve",transformIndexHtml:{order:"pre",handler(){return [{tag:"script",injectTo:"head-prepend",children:t}]}},transform(e,r){if(!r.includes("vite/dist/client/client.mjs"))return null;let o=e.replace(/await waitForSuccessfulPing\(protocol,\s*hostAndPath\);\s*\n\s*location\.reload\(\)/,`await waitForSuccessfulPing(protocol, hostAndPath);
|
|
973
973
|
if (window.__smartReload__ && window.__smartReload__()) {
|
|
974
|
-
console.log('[SmartReload] Vite client: server
|
|
975
|
-
//
|
|
976
|
-
// \
|
|
977
|
-
|
|
978
|
-
|
|
979
|
-
|
|
980
|
-
|
|
981
|
-
|
|
982
|
-
|
|
983
|
-
|
|
984
|
-
|
|
985
|
-
|
|
986
|
-
|
|
974
|
+
console.log('[SmartReload] Vite client: server is back, fetching new WebSocket token...');
|
|
975
|
+
// PM2 \u91CD\u542F\u540E\u65B0\u670D\u52A1\u5668\u4F1A\u751F\u6210\u65B0\u7684 wsToken\uFF0C\u65E7 token \u65E0\u6CD5\u8FDE\u63A5
|
|
976
|
+
// \u901A\u8FC7 fetch /@vite/client \u83B7\u53D6\u65B0 token
|
|
977
|
+
try {
|
|
978
|
+
var clientResp = await fetch('/@vite/client');
|
|
979
|
+
var clientCode = await clientResp.text();
|
|
980
|
+
// \u4ECE\u65B0\u7684\u5BA2\u6237\u7AEF\u4EE3\u7801\u4E2D\u63D0\u53D6 wsToken
|
|
981
|
+
// Vite \u7F16\u8BD1\u540E\u7684\u4EE3\u7801\u4E2D wsToken \u662F\u4E00\u4E2A\u5B57\u7B26\u4E32\u5E38\u91CF\uFF0C\u5982: const wsToken = "xxxxxxxxxxxx"
|
|
982
|
+
var tokenMatch = clientCode.match(/const\\s+wsToken\\s*=\\s*"([^"]+)"/);
|
|
983
|
+
if (tokenMatch && tokenMatch[1]) {
|
|
984
|
+
console.log('[SmartReload] Got new wsToken from server');
|
|
985
|
+
// \u66F4\u65B0\u95ED\u5305\u4E2D\u7684 wsToken \u53D8\u91CF\uFF08Vite \u5BA2\u6237\u7AEF\u4EE3\u7801\u4E2D\u7684 const wsToken\uFF09
|
|
986
|
+
// \u6CE8\u610F\uFF1AwsToken \u662F const\uFF0C\u65E0\u6CD5\u76F4\u63A5\u4FEE\u6539\u3002\u6211\u4EEC\u9700\u8981\u4FEE\u6539 setupWebSocket \u4F7F\u7528\u7684 URL
|
|
987
|
+
// \u65B9\u6848\uFF1A\u76F4\u63A5\u521B\u5EFA WebSocket \u8FDE\u63A5\uFF0C\u7ED5\u8FC7 setupWebSocket
|
|
988
|
+
var newToken = tokenMatch[1];
|
|
989
|
+
var wsUrl = protocol + '://' + hostAndPath + '?token=' + newToken;
|
|
990
|
+
console.log('[SmartReload] Reconnecting with new token to: ' + wsUrl);
|
|
991
|
+
var newSocket = new WebSocket(wsUrl, 'vite-hmr');
|
|
992
|
+
var newIsOpened = false;
|
|
993
|
+
newSocket.addEventListener('open', function() {
|
|
994
|
+
newIsOpened = true;
|
|
995
|
+
console.log('[SmartReload] WebSocket reconnected successfully!');
|
|
996
|
+
notifyListeners('vite:ws:connect', { webSocket: newSocket });
|
|
997
|
+
}, { once: true });
|
|
998
|
+
newSocket.addEventListener('message', async function(ev) {
|
|
999
|
+
handleMessage(JSON.parse(ev.data));
|
|
1000
|
+
});
|
|
1001
|
+
newSocket.addEventListener('close', async function(ev) {
|
|
1002
|
+
if (ev.wasClean) return;
|
|
1003
|
+
if (!newIsOpened && onCloseWithoutOpen) {
|
|
1004
|
+
onCloseWithoutOpen();
|
|
1005
|
+
return;
|
|
1006
|
+
}
|
|
1007
|
+
notifyListeners('vite:ws:disconnect', { webSocket: newSocket });
|
|
1008
|
+
if (hasDocument) {
|
|
1009
|
+
console.log('[vite] server connection lost. Polling for restart...');
|
|
1010
|
+
await waitForSuccessfulPing(protocol, hostAndPath);
|
|
1011
|
+
if (window.__smartReload__ && window.__smartReload__()) {
|
|
1012
|
+
console.log('[SmartReload] Nested reconnect - reloading page');
|
|
1013
|
+
}
|
|
1014
|
+
location.reload();
|
|
1015
|
+
}
|
|
987
1016
|
});
|
|
988
|
-
|
|
989
|
-
}
|
|
990
|
-
console.log('[SmartReload]
|
|
1017
|
+
socket = newSocket;
|
|
1018
|
+
} else {
|
|
1019
|
+
console.log('[SmartReload] Could not extract new wsToken, reloading page');
|
|
1020
|
+
location.reload();
|
|
991
1021
|
}
|
|
992
|
-
}
|
|
993
|
-
|
|
994
|
-
console.log('[SmartReload] Vite client: WebSocket is ready, reconnecting now');
|
|
995
|
-
socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);
|
|
996
|
-
} else {
|
|
997
|
-
console.log('[SmartReload] Vite client: WebSocket still not ready after retries, reloading page');
|
|
1022
|
+
} catch(e) {
|
|
1023
|
+
console.log('[SmartReload] Failed to fetch new client code, reloading page', e);
|
|
998
1024
|
location.reload();
|
|
999
1025
|
}
|
|
1000
1026
|
} else {
|
|
1001
1027
|
location.reload();
|
|
1002
|
-
}`);return o!==e?{code:o,map:null}:null}}}function
|
|
1028
|
+
}`);return o!==e?{code:o,map:null}:null}}}function h(n={}){let{designWidth:t=375,maxWidth:e=750,baseFontSize:r=12,minRootSize:o=12,maxRootSize:i=24}=n;return {name:"vite-plugin-taro-style-adapter",apply:"serve",transformIndexHtml(s){let l=`
|
|
1003
1029
|
<script data-taro-flexible="true">
|
|
1004
1030
|
(function() {
|
|
1005
1031
|
var designWidth = ${t};
|
|
@@ -1086,5 +1112,5 @@ if (typeof window !== 'undefined') {
|
|
|
1086
1112
|
object-fit: contain;
|
|
1087
1113
|
}
|
|
1088
1114
|
</style>
|
|
1089
|
-
</head>`)}}}function g(n={}){let{ratio:t=2}=n;return {postcssPlugin:"postcss-rpx2px",Declaration(e){e.value.includes("rpx")&&(e.value=e.value.replace(/(-?\d*\.?\d+)rpx/gi,(r,o)=>{let i=parseFloat(o)/t;return i===0?"0":`${i}px`}));}}}g.postcss=true;function H(n={}){let{additional:t=[],autoInjectTaroApp:e=true,autoInjectVite:r=true}=n,o={},i=new Set(t);return e&&Object.keys(process.env).forEach(s=>{s.startsWith("TARO_APP_")&&i.add(s);}),r&&Object.keys(process.env).forEach(s=>{s.startsWith("VITE_")&&i.add(s);}),i.forEach(s=>{let l=process.env[s]||"";o[`process.env.${s}`]=JSON.stringify(l);}),o}function q(){return {"process.env.TARO_APP_API_BASE_URL":JSON.stringify(process.env.TARO_APP_API_BASE_URL||""),"process.env.VITE_API_BASE_URL":JSON.stringify(process.env.VITE_API_BASE_URL||"")}}function D(n={}){let t=[
|
|
1115
|
+
</head>`)}}}function g(n={}){let{ratio:t=2}=n;return {postcssPlugin:"postcss-rpx2px",Declaration(e){e.value.includes("rpx")&&(e.value=e.value.replace(/(-?\d*\.?\d+)rpx/gi,(r,o)=>{let i=parseFloat(o)/t;return i===0?"0":`${i}px`}));}}}g.postcss=true;function H(n={}){let{additional:t=[],autoInjectTaroApp:e=true,autoInjectVite:r=true}=n,o={},i=new Set(t);return e&&Object.keys(process.env).forEach(s=>{s.startsWith("TARO_APP_")&&i.add(s);}),r&&Object.keys(process.env).forEach(s=>{s.startsWith("VITE_")&&i.add(s);}),i.forEach(s=>{let l=process.env[s]||"";o[`process.env.${s}`]=JSON.stringify(l);}),o}function q(){return {"process.env.TARO_APP_API_BASE_URL":JSON.stringify(process.env.TARO_APP_API_BASE_URL||""),"process.env.VITE_API_BASE_URL":JSON.stringify(process.env.VITE_API_BASE_URL||"")}}function D(n={}){let t=[h(n.styleAdapter),{name:"dev-postcss-rpx2px-plugin",apply:"serve",config(e){x.env.TARO_ENV==="h5"&&typeof e.css?.postcss=="object"&&e.css?.postcss.plugins?.unshift(g());}},y(),S(),w(),T()];return x.env.WORKSPACE_GIT_REPO&&t.push(_()),t}export{y as componentIdPlugin,D as default,S as editorBridgePlugin,q as injectAmasterEnv,H as injectTaroEnv,w as routesExposePlugin,g as rpx2pxPlugin,T as smartReloadPlugin,h as taroStyleAdapterPlugin};//# sourceMappingURL=index.js.map
|
|
1090
1116
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts","../src/browser-logs.ts","../src/smart-reload.ts","../src/taro-style-adapter.ts","../src/postcss-rpx2px.ts","../src/taro-env-inject.ts","../src/index.ts"],"names":["generateUniqueId","filePath","position","key","createHash","HTML_TAGS","componentIdPlugin","isDev","config","code","id","transformedCode","jsxOpenTagRegex","match","matches","tag","tagStartIndex","tagEndIndex","checkIndex","foundClosing","hasComponentId","currentCode","i","item","index","uniqueId","before","after","clickHandlerScript","getBridgeScriptContent","__filename","fileURLToPath","__dirname","dirname","bridgePath","resolve","readFileSync","error","editorBridgePlugin","bridgeScript","html","scriptsToInject","routesExposePlugin","options","routesPath","browserLogsPlugin","logFilePath","injectedScript","root","process","path","devServer","req","res","next","origin","body","chunk","logDir","fs","smartReloadPlugin","pureScript","transformed","taroStyleAdapterPlugin","designWidth","maxWidth","baseFontSize","minRootSize","maxRootSize","flexibleScript","rpx2pxPlugin","ratio","decl","_match","num","pxValue","injectTaroEnv","additional","autoInjectTaroApp","autoInjectVite","constants","varsToInject","varName","value","injectAmasterEnv","devTools","plugins"],"mappings":"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AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;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","file":"index.js","sourcesContent":["import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { readFileSync } from \"fs\";\nimport { resolve, dirname } from \"path\";\nimport { fileURLToPath } from \"url\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * 读取构建后的 bridge 脚本内容\n */\nfunction getBridgeScriptContent(): string {\n try {\n // 获取当前模块的目录\n const __filename = fileURLToPath(import.meta.url);\n const __dirname = dirname(__filename);\n \n // 读取构建后的 bridge.bridge.js 文件\n const bridgePath = resolve(__dirname, \"../dist/bridge.bridge.js\");\n const bridgeContent = readFileSync(bridgePath, \"utf-8\");\n \n return `<script>\\n${bridgeContent}\\n</script>`;\n } catch (error) {\n console.warn(\"Failed to read bridge script:\", error);\n return \"\";\n }\n}\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode by reading the built bridge script\n */\nexport function editorBridgePlugin(): Plugin {\n let isDev = false;\n let bridgeScript = \"\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n \n // 在开发模式下读取 bridge 脚本内容\n if (isDev) {\n bridgeScript = getBridgeScriptContent();\n }\n },\n\n transformIndexHtml(html) {\n // 在开发模式下注入 bridge 脚本\n const devScript = isDev ? bridgeScript : \"\";\n\n const scriptsToInject = `${clickHandlerScript}${devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { Buffer } from \"node:buffer\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport process from \"node:process\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n // ============================================\n // Vite deps 504 error capture using PerformanceObserver\n // Only captures 504 errors for node_modules/.vite resources (pre-bundled dependencies)\n // ============================================\n var reportedUrls = {}; // Track reported URLs to avoid duplicates\n \n if (typeof PerformanceObserver !== 'undefined') {\n var perfObserver = new PerformanceObserver(function(list) {\n var entries = list.getEntries();\n for (var i = 0; i < entries.length; i++) {\n var perfEntry = entries[i];\n // Only check script resources\n if (perfEntry.initiatorType !== 'script') {\n continue;\n }\n \n var resourceUrl = perfEntry.name || '';\n // Only report errors for node_modules/.vite resources\n if (resourceUrl.indexOf('node_modules/.vite') === -1 && resourceUrl.indexOf('/node_modules/.vite') === -1) {\n continue;\n }\n \n // Check if response status is 504 (Gateway Timeout)\n // responseStatus is available in Resource Timing Level 2\n var status = perfEntry.responseStatus;\n if (status === 504) {\n // Avoid duplicate reports\n if (reportedUrls[resourceUrl]) {\n continue;\n }\n reportedUrls[resourceUrl] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: resourceUrl,\n status: 504,\n message: 'Vite pre-bundled dependency returned 504 Gateway Timeout: ' + resourceUrl,\n duration: perfEntry.duration,\n transferSize: perfEntry.transferSize\n };\n addLog(entry);\n }\n }\n });\n \n try {\n perfObserver.observe({ type: 'resource', buffered: true });\n } catch (e) {\n // Fallback for browsers that don't support the options\n try {\n perfObserver.observe({ entryTypes: ['resource'] });\n } catch (e2) {\n originalConsole.warn('[BrowserLogs] PerformanceObserver not supported:', e2.message);\n }\n }\n }\n \n // ============================================\n // Script load error capture (for 504 and other network errors)\n // This captures errors that PerformanceObserver might miss\n // ============================================\n \n // Use MutationObserver to watch for dynamically added script tags\n function attachScriptErrorHandler(script) {\n if (script.__errorHandlerAttached__) return;\n script.__errorHandlerAttached__ = true;\n \n script.addEventListener('error', function(event) {\n var src = script.src || '';\n \n // Only report errors for node_modules/.vite resources\n if (src.indexOf('node_modules/.vite') === -1 && src.indexOf('/node_modules/.vite') === -1) {\n return;\n }\n \n // Avoid duplicate reports\n if (reportedUrls[src]) {\n return;\n }\n reportedUrls[src] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: src,\n status: 'load-error',\n message: 'Vite pre-bundled dependency failed to load (possibly 504 Gateway Timeout): ' + src,\n errorType: event.type\n };\n addLog(entry);\n });\n }\n \n // Attach error handlers to existing scripts\n var existingScripts = document.querySelectorAll('script[src]');\n for (var i = 0; i < existingScripts.length; i++) {\n attachScriptErrorHandler(existingScripts[i]);\n }\n \n // Watch for dynamically added scripts\n if (typeof MutationObserver !== 'undefined') {\n var scriptObserver = new MutationObserver(function(mutations) {\n for (var i = 0; i < mutations.length; i++) {\n var mutation = mutations[i];\n for (var j = 0; j < mutation.addedNodes.length; j++) {\n var node = mutation.addedNodes[j];\n if (node.nodeName === 'SCRIPT' && node.src) {\n attachScriptErrorHandler(node);\n }\n // Also check child nodes\n if (node.querySelectorAll) {\n var scripts = node.querySelectorAll('script[src]');\n for (var k = 0; k < scripts.length; k++) {\n attachScriptErrorHandler(scripts[k]);\n }\n }\n }\n }\n });\n \n scriptObserver.observe(document.documentElement, {\n childList: true,\n subtree: true\n });\n }\n \n // Also intercept document.createElement to catch scripts before they're added to DOM\n var originalCreateElement = document.createElement.bind(document);\n document.createElement = function(tagName) {\n var element = originalCreateElement(tagName);\n if (tagName.toLowerCase() === 'script') {\n // Use a setter to catch when src is set\n var originalSrc = '';\n Object.defineProperty(element, '__originalSrc__', {\n get: function() { return originalSrc; },\n set: function(val) { originalSrc = val; }\n });\n \n // Defer attaching error handler until src is set\n var srcDescriptor = Object.getOwnPropertyDescriptor(HTMLScriptElement.prototype, 'src');\n if (srcDescriptor && srcDescriptor.set) {\n var originalSrcSetter = srcDescriptor.set;\n Object.defineProperty(element, 'src', {\n get: function() {\n return srcDescriptor.get ? srcDescriptor.get.call(this) : this.getAttribute('src');\n },\n set: function(value) {\n if (originalSrcSetter) {\n originalSrcSetter.call(this, value);\n } else {\n this.setAttribute('src', value);\n }\n attachScriptErrorHandler(this);\n },\n configurable: true\n });\n }\n }\n return element;\n };\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Smart Reload 插件\n *\n * 在 Vite 服务器重启期间(如 PM2 重启 web-server),拦截浏览器端的刷新行为,\n * 让浏览器静默等待服务器恢复,期间页面完全不刷新。\n * 服务器恢复后自动恢复 HMR 连接,无需手动刷新。\n *\n * 场景说明:\n * - PM2 杀掉 Vite 进程 → WebSocket 断开 → PM2 重启 → 新 Vite 启动\n * - Vite 客户端检测到 WebSocket 断开后会通过 HTTP ping 轮询等待服务器恢复\n * - 服务器恢复后 Vite 客户端会调用 location.reload() 刷新页面\n * - 本插件拦截这个刷新行为,让页面保持当前状态\n *\n * 实现原理(双层拦截):\n * 1. 服务端层:通过 transform 钩子修改 /@vite/client 源码,\n * 将 location.reload() 替换为 window.__smartReload__(),\n * 使我们能控制是否真正刷新页面\n * 2. 客户端层:注入脚本拦截 WebSocket 消息,\n * 检测服务器重启场景,在重启期间阻止刷新\n *\n * 区分两种 full-reload 场景:\n * 1. 服务器重启 → WebSocket 断开 → 静默等待,不刷新,自动恢复 HMR\n * 2. 模块变更需要 full-reload → WebSocket 保持连接 → 正常刷新\n */\nexport function smartReloadPlugin(): Plugin {\n // 注入到浏览器端的脚本\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n\n var PLUGIN_TAG = '[SmartReload]';\n // full-reload 后等待 WebSocket 断开的窗口期(ms)\n var DISCONNECT_DETECT_WINDOW = 300;\n\n // ============================================\n // 状态管理\n // ============================================\n var state = {\n // 是否已经拦截了 Vite HMR WebSocket\n wsIntercepted: false,\n // 是否正在等待判断 full-reload 类型(300ms 窗口期)\n detectingReloadType: false,\n // 延迟执行 full-reload 的定时器\n reloadTimer: null,\n // 是否是服务器重启场景\n isServerRestart: false\n };\n\n // ============================================\n // 日志工具\n // ============================================\n function log() {\n var args = [PLUGIN_TAG];\n for (var i = 0; i < arguments.length; i++) {\n args.push(arguments[i]);\n }\n console.log.apply(console, args);\n }\n\n // ============================================\n // 重置状态\n // ============================================\n function resetState() {\n state.detectingReloadType = false;\n state.isServerRestart = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n }\n\n // ============================================\n // 执行真正的 full-reload\n // ============================================\n function executeFullReload() {\n log('Executing full-reload');\n resetState();\n location.reload();\n }\n\n // ============================================\n // __smartReload__:检查是否应该阻止 reload\n //\n // 返回 true 表示是服务器重启场景,不应该刷新页面\n // 返回 false 表示应该正常刷新\n //\n // Vite 客户端代码中的 WebSocket close handler 已被修改为:\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(...); // 重新创建 WebSocket 恢复 HMR\n // } else {\n // location.reload();\n // }\n // ============================================\n window.__smartReload__ = function() {\n if (state.isServerRestart) {\n log('Server restart detected, suppressing reload. Will reconnect WebSocket.');\n resetState();\n return true; // 阻止 reload,由 Vite 客户端重新创建 WebSocket\n }\n log('Non-restart scenario, allowing reload');\n return false; // 允许正常 reload\n };\n\n // ============================================\n // 拦截 WebSocket 构造函数\n // ============================================\n var OriginalWebSocket = window.WebSocket;\n\n function isViteHmrWebSocket(url) {\n if (typeof url !== 'string') return false;\n return url.includes('/__vite_hmr') || url.includes('?token=');\n }\n\n function createWsProxy(ws) {\n var originalAddEventListener = ws.addEventListener.bind(ws);\n var originalRemoveEventListener = ws.removeEventListener.bind(ws);\n\n ws.addEventListener = function(type, listener, options) {\n if (type === 'message') {\n var wrappedListener = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted) {\n listener.call(ws, event);\n }\n };\n listener.__smartReloadWrapped__ = wrappedListener;\n return originalAddEventListener(type, wrappedListener, options);\n }\n return originalAddEventListener(type, listener, options);\n };\n\n ws.removeEventListener = function(type, listener, options) {\n if (type === 'message' && listener.__smartReloadWrapped__) {\n return originalRemoveEventListener(type, listener.__smartReloadWrapped__, options);\n }\n return originalRemoveEventListener(type, listener, options);\n };\n\n var originalOnMessage = null;\n var onMessageDescriptor = Object.getOwnPropertyDescriptor(WebSocket.prototype, 'onmessage') ||\n Object.getOwnPropertyDescriptor(ws, 'onmessage');\n\n if (onMessageDescriptor) {\n Object.defineProperty(ws, 'onmessage', {\n get: function() {\n return originalOnMessage;\n },\n set: function(handler) {\n originalOnMessage = handler;\n var wrappedHandler = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted && handler) {\n handler.call(ws, event);\n }\n };\n if (onMessageDescriptor.set) {\n onMessageDescriptor.set.call(ws, wrappedHandler);\n }\n },\n configurable: true\n });\n }\n\n originalAddEventListener('close', function(event) {\n handleWsClose(event);\n });\n\n originalAddEventListener('open', function() {\n handleWsOpen();\n });\n\n state.wsIntercepted = true;\n log('Vite HMR WebSocket intercepted');\n\n return ws;\n }\n\n window.WebSocket = function SmartReloadWebSocket(url, protocols) {\n var ws;\n if (protocols !== undefined) {\n ws = new OriginalWebSocket(url, protocols);\n } else {\n ws = new OriginalWebSocket(url);\n }\n\n if (isViteHmrWebSocket(url)) {\n return createWsProxy(ws);\n }\n\n return ws;\n };\n\n window.WebSocket.prototype = OriginalWebSocket.prototype;\n window.WebSocket.CONNECTING = OriginalWebSocket.CONNECTING;\n window.WebSocket.OPEN = OriginalWebSocket.OPEN;\n window.WebSocket.CLOSING = OriginalWebSocket.CLOSING;\n window.WebSocket.CLOSED = OriginalWebSocket.CLOSED;\n\n // ============================================\n // WebSocket 消息处理\n // ============================================\n function handleWsMessage(event) {\n var data;\n try {\n data = JSON.parse(event.data);\n } catch (e) {\n return false;\n }\n\n // 处理 full-reload 消息\n if (data.type === 'full-reload') {\n // 如果在服务器重启模式,拦截所有 full-reload\n if (state.isServerRestart) {\n log('Suppressed full-reload message during server restart');\n return true;\n }\n\n log('Intercepted full-reload, path:', data.path || '(none)');\n\n // 进入检测模式:等待看 WebSocket 是否断开\n state.detectingReloadType = true;\n\n // 如果窗口期内 WebSocket 没有断开,说明是真正的模块变更\n state.reloadTimer = setTimeout(function() {\n if (state.detectingReloadType && !state.isServerRestart) {\n log('WebSocket still connected after ' + DISCONNECT_DETECT_WINDOW + 'ms - real module change, executing reload');\n state.detectingReloadType = false;\n executeFullReload();\n }\n }, DISCONNECT_DETECT_WINDOW);\n\n return true;\n }\n\n // 处理 connected 消息\n if (data.type === 'connected') {\n if (state.isServerRestart) {\n log('HMR reconnected after server restart! Page was NOT reloaded.');\n resetState();\n }\n return false;\n }\n\n return false;\n }\n\n // ============================================\n // WebSocket 连接事件处理\n // ============================================\n function handleWsClose(event) {\n log('WebSocket closed, code:', event.code, 'wasClean:', event.wasClean);\n\n // 场景1:收到 full-reload 后 WebSocket 断开 → 服务器重启\n if (state.detectingReloadType) {\n log('Entering server restart mode: full-reload followed by disconnect');\n state.isServerRestart = true;\n state.detectingReloadType = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n return;\n }\n\n // 场景2:WebSocket 直接断开(非正常关闭)\n if (!state.isServerRestart && !event.wasClean) {\n log('Entering server restart mode: unexpected disconnect');\n state.isServerRestart = true;\n }\n }\n\n function handleWsOpen() {\n log('WebSocket opened');\n }\n\n // ============================================\n // 暴露 API 供调试使用\n // ============================================\n window.__smartReloadState__ = state;\n window.__OriginalWebSocket__ = OriginalWebSocket;\n window.__smartReloadForceReload__ = function() {\n log('Force reload triggered by user');\n resetState();\n location.reload();\n };\n\n log('Smart Reload initialized');\n})();\n</script>`;\n\n // 提取纯 JS 代码(去掉 <script> 标签)\n const pureScript = injectedScript\n .replace(/^\\s*<script>\\s*/i, \"\")\n .replace(/\\s*<\\/script>\\s*$/i, \"\");\n\n return {\n name: \"vite-plugin-smart-reload\",\n\n // 只在开发模式下注入脚本\n apply: \"serve\",\n\n transformIndexHtml: {\n order: \"pre\",\n handler() {\n return [\n {\n tag: \"script\",\n // 注入到 head 的最前面,确保在 Vite 客户端脚本 /@vite/client 之前执行\n injectTo: \"head-prepend\",\n children: pureScript,\n },\n ];\n },\n },\n\n // 修改 Vite 客户端代码:\n // 精确替换 WebSocket close handler 中的 reload 逻辑,\n // 在服务器重启场景下重新创建 WebSocket 恢复 HMR,而不是刷新页面\n //\n // Vite 对 /@vite/client 的处理流程:\n // 1. 浏览器请求 /@vite/client\n // 2. Vite 的 transformMiddleware 拦截请求\n // 3. 调用 server.transformRequest('/@vite/client')\n // 4. 内部解析为文件系统路径 node_modules/vite/dist/client/client.mjs\n // 5. 调用插件的 transform(code, resolvedId) — id 是文件系统路径\n transform(code, id) {\n // 匹配 Vite 客户端模块(文件系统路径)\n if (!id.includes(\"vite/dist/client/client.mjs\")) {\n return null;\n }\n\n // 精确替换 WebSocket close handler 中的 reload 逻辑:\n // 原始代码:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // location.reload();\n // 替换为:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);\n // } else {\n // location.reload();\n // }\n const transformed = code.replace(\n /await waitForSuccessfulPing\\(protocol,\\s*hostAndPath\\);\\s*\\n\\s*location\\.reload\\(\\)/,\n `await waitForSuccessfulPing(protocol, hostAndPath);\n if (window.__smartReload__ && window.__smartReload__()) {\n console.log('[SmartReload] Vite client: server HTTP is back, waiting for WebSocket to be ready...');\n // HTTP ping 成功不代表 WebSocket 已就绪,需要等待 WebSocket 可连接\n // 使用原始 WebSocket 构造函数避免被 SmartReload 代理拦截\n var RawWS = window.__OriginalWebSocket__ || WebSocket;\n var wsReady = false;\n for (var attempt = 0; attempt < 10 && !wsReady; attempt++) {\n await new Promise(function(r) { setTimeout(r, 1000); });\n try {\n await new Promise(function(resolve, reject) {\n var testWs = new RawWS(protocol + '://' + hostAndPath + '?token=' + __WS_TOKEN__, 'vite-hmr');\n testWs.addEventListener('open', function() { testWs.close(); resolve(); });\n testWs.addEventListener('error', function() { reject(); });\n setTimeout(function() { try { testWs.close(); } catch(e) {} reject(); }, 3000);\n });\n wsReady = true;\n } catch(e) {\n console.log('[SmartReload] WebSocket not ready yet, attempt ' + (attempt + 1) + '/10');\n }\n }\n if (wsReady) {\n console.log('[SmartReload] Vite client: WebSocket is ready, reconnecting now');\n socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);\n } else {\n console.log('[SmartReload] Vite client: WebSocket still not ready after retries, reloading page');\n location.reload();\n }\n } else {\n location.reload();\n }`,\n );\n\n if (transformed !== code) {\n return {\n code: transformed,\n map: null,\n };\n }\n\n return null;\n },\n };\n}\n","import type { Plugin } from 'vite';\n\nexport interface TaroStyleAdapterOptions {\n /**\n * Design width for responsive scaling (default: 375)\n * Should match Taro config's designWidth\n */\n designWidth?: number;\n \n /**\n * Maximum viewport width to scale (default: 750)\n * Beyond this width, font-size stays fixed\n */\n maxWidth?: number;\n\n /**\n * Base font size in pixels (default: 12)\n * Should match Taro config's baseFontSize\n */\n baseFontSize?: number;\n\n /**\n * Minimum root font size in pixels (default: 12)\n * Should match Taro config's minRootSize\n */\n minRootSize?: number;\n\n /**\n * Maximum root font size in pixels (default: 24)\n * Should match Taro config's maxRootSize\n */\n maxRootSize?: number;\n}\n\n/**\n * Taro H5 样式适配插件\n * \n * 功能:\n * 1. 动态计算根字号,使 H5 的 rem 与小程序的 rpx 行为对齐\n * 2. 隐藏 Taro 页面容器的滚动条,用于编辑器 iframe 预览\n * \n * 原理:\n * - 小程序 rpx: 1rpx = 屏幕宽度 / 750\n * - H5 rem: Taro 把 rpx 转成 rem,基于 baseFontSize\n * - 本插件动态设置 html font-size,使 rem 的实际像素值与 rpx 对齐\n * \n * 注意:此插件会覆盖 Taro 默认注入的根字号脚本,配置需与 Taro pxtransform 保持一致\n * \n * @example\n * ```ts\n * // In Taro config/dev.ts\n * import { taroStyleAdapterPlugin } from '@amaster.ai/vite-plugins'\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [\n * taroStyleAdapterPlugin({ \n * designWidth: 375,\n * baseFontSize: 12,\n * minRootSize: 12,\n * maxRootSize: 24\n * })\n * ]\n * }\n * }\n * ```\n */\nexport function taroStyleAdapterPlugin(options: TaroStyleAdapterOptions = {}): Plugin {\n const { \n designWidth = 375, \n maxWidth = 750,\n baseFontSize = 12,\n minRootSize = 12,\n maxRootSize = 24\n } = options;\n \n return {\n name: 'vite-plugin-taro-style-adapter',\n apply: 'serve', // 仅在开发模式下生效\n transformIndexHtml(html) {\n // 动态根字号脚本\n // 公式:fontSize = (clientWidth / designWidth) * baseFontSize\n // 当 designWidth=375, baseFontSize=12 时:\n // - 375px 屏幕 → 12px (1rem = 12px)\n // - 750px 屏幕 → 24px (1rem = 24px)\n // 这样 rem 值在不同屏幕宽度下的视觉比例与小程序 rpx 一致\n const flexibleScript = `\n<script data-taro-flexible=\"true\">\n(function() {\n var designWidth = ${designWidth};\n var maxWidth = ${maxWidth};\n var baseFontSize = ${baseFontSize};\n var minRootSize = ${minRootSize};\n var maxRootSize = ${maxRootSize};\n \n function setRootFontSize() {\n var docEl = document.documentElement;\n var clientWidth = docEl.clientWidth;\n \n // 限制最大宽度\n if (clientWidth > maxWidth) {\n clientWidth = maxWidth;\n }\n \n // 计算根字号: (屏幕宽度 / 设计稿宽度) * 基准字号\n var fontSize = (clientWidth / designWidth) * baseFontSize;\n \n // 应用最小/最大限制\n if (fontSize < minRootSize) {\n fontSize = minRootSize;\n } else if (fontSize > maxRootSize) {\n fontSize = maxRootSize;\n }\n \n docEl.style.fontSize = fontSize + 'px';\n }\n \n setRootFontSize();\n \n // 监听窗口变化\n window.addEventListener('resize', setRootFontSize);\n window.addEventListener('orientationchange', setRootFontSize);\n \n // 页面显示时重新计算(解决某些浏览器的 bug)\n document.addEventListener('DOMContentLoaded', setRootFontSize);\n})();\n</script>\n`;\n\n // 滚动条隐藏样式 + Taro Image 组件修复\n const styles = `\n<style data-taro-adapter=\"true\">\n /* 仅 H5 生效:隐藏 Taro 页面容器滚动条,但仍可滚动 */\n .taro_page {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro_page::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /* 隐藏 Taro ScrollView 组件的滚动条 */\n .taro-scroll-view__scroll-y,\n .taro-scroll-view__scroll-x {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro-scroll-view__scroll-y::-webkit-scrollbar,\n .taro-scroll-view__scroll-x::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /*\n * 修复 Taro Image 组件在 H5 下的尺寸问题\n * \n * 问题:Taro Image 组件使用 aspectFit 模式时,内部 img 使用 max-width/max-height: 100%\n * 导致小尺寸图片(如 SVG 图标)无法按指定尺寸显示,只显示原始尺寸\n * \n * 解决:让 img 填满 taro-image-core 容器,由容器控制尺寸\n */\n taro-image-core img.taro-img__mode-aspectfit {\n width: 100%;\n height: 100%;\n max-width: none;\n max-height: none;\n position: static;\n transform: none;\n object-fit: contain;\n }\n</style>\n`;\n \n // 在 </head> 之前插入脚本和样式\n return html.replace('</head>', `${flexibleScript}${styles}</head>`);\n }\n };\n}\n","/**\n * PostCSS 插件:H5 模式下将 rpx 预转换为 px\n *\n * 问题背景:\n * - Taro 的 pxtransform 在 H5 模式下会处理 rpx,但它把 rpx 值直接除以 rootValue\n * - 导致 96rpx 被转成 8rem(96/12),实际显示为 96px\n * - 而正确应该是:96rpx = 48px = 4rem = 48px\n *\n * 解决方案:\n * - 在 pxtransform 之前,先把 rpx 转成 px(rpx / 2)\n * - 然后 pxtransform 会正确地把 px 转成 rem(px / 12)\n *\n * 转换链路:96rpx → 48px → 4rem → 48px(正确)\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { rpx2pxPlugin } from '@amaster.ai/vite-plugins'\n * \n * // 在 postcss-config-loader-plugin 中使用\n * {\n * name: 'postcss-config-loader-plugin',\n * config(config) {\n * if (typeof config.css?.postcss === 'object') {\n * config.css?.postcss.plugins?.unshift(tailwindcss())\n * // H5 模式下添加 rpx2px 插件\n * if (process.env.TARO_ENV === 'h5') {\n * config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n * }\n * }\n * }\n * }\n * ```\n */\n\nexport interface Rpx2pxPluginOptions {\n /**\n * rpx 到 px 的转换比例(默认 2)\n * 即 1rpx = 1/ratio px\n * 默认值 2 表示 2rpx = 1px(基于 375 设计稿)\n */\n ratio?: number;\n}\n\ninterface PostCSSDeclaration {\n value: string;\n}\n\ninterface PostCSSPlugin {\n postcssPlugin: string;\n Declaration: (decl: PostCSSDeclaration) => void;\n}\n\ntype PostCSSPluginCreator = {\n (): PostCSSPlugin;\n postcss: boolean;\n};\n\nexport function rpx2pxPlugin(options: Rpx2pxPluginOptions = {}): PostCSSPlugin {\n const { ratio = 2 } = options;\n \n return {\n postcssPlugin: 'postcss-rpx2px',\n Declaration(decl: PostCSSDeclaration) {\n if (decl.value.includes('rpx')) {\n decl.value = decl.value.replace(/(-?\\d*\\.?\\d+)rpx/gi, (_match: string, num: string) => {\n const pxValue = parseFloat(num) / ratio;\n return pxValue === 0 ? '0' : `${pxValue}px`;\n });\n }\n }\n };\n}\n\n// PostCSS 插件标识\n(rpx2pxPlugin as PostCSSPluginCreator).postcss = true;\n","/**\n * Taro Environment Variables Injection Helper\n * \n * Automatically injects environment variables into Taro's defineConstants\n * so that they will be replaced at build time.\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { injectTaroEnv } from '@amaster.ai/vite-plugins/taro-env-inject'\n * \n * export default defineConfig({\n * defineConstants: {\n * ...injectTaroEnv()\n * }\n * })\n * ```\n */\n\nexport interface TaroEnvInjectOptions {\n /**\n * Additional environment variable names to inject\n * @default []\n */\n additional?: string[];\n \n /**\n * Whether to inject all TARO_APP_* prefixed variables\n * @default true\n */\n autoInjectTaroApp?: boolean;\n \n /**\n * Whether to inject all VITE_* prefixed variables\n * @default true\n */\n autoInjectVite?: boolean;\n}\n\n/**\n * Inject environment variables into Taro's defineConstants\n * \n * This function reads environment variables and formats them for Taro's defineConstants.\n * Taro will replace these at build time, converting `process.env.XXX` to actual values.\n */\nexport function injectTaroEnv(options: TaroEnvInjectOptions = {}): Record<string, string> {\n const {\n additional = [],\n autoInjectTaroApp = true,\n autoInjectVite = true,\n } = options;\n\n const constants: Record<string, string> = {};\n\n // Collect variable names to inject\n const varsToInject = new Set<string>(additional);\n\n // Auto-detect TARO_APP_* variables\n if (autoInjectTaroApp) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('TARO_APP_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Auto-detect VITE_* variables\n if (autoInjectVite) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('VITE_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Inject variables into defineConstants format\n varsToInject.forEach(varName => {\n const value = process.env[varName] || '';\n constants[`process.env.${varName}`] = JSON.stringify(value);\n });\n\n return constants;\n}\n\n/**\n * Inject specific environment variables for amaster.ai mini-program builds\n * \n * This is a convenience function that injects the standard environment variables\n * used by @amaster.ai/http-client for API base URL configuration.\n */\nexport function injectAmasterEnv(): Record<string, string> {\n return {\n 'process.env.TARO_APP_API_BASE_URL': JSON.stringify(process.env.TARO_APP_API_BASE_URL || ''),\n 'process.env.VITE_API_BASE_URL': JSON.stringify(process.env.VITE_API_BASE_URL || ''),\n };\n}\n","import { componentIdPlugin } from \"./component-id\";\nimport { editorBridgePlugin } from \"./editor-bridge\";\nimport { routesExposePlugin } from \"./routes-expose\";\nimport { browserLogsPlugin } from \"./browser-logs\";\nimport { smartReloadPlugin } from \"./smart-reload\";\nimport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\nimport { rpx2pxPlugin } from \"./postcss-rpx2px\";\nimport process from \"node:process\";\nimport type { Plugin } from \"vite\";\n\nexport interface DevToolsOptions {\n /**\n * Taro style adapter options\n * Controls H5 responsive scaling to match mini-program rpx behavior\n */\n styleAdapter?: TaroStyleAdapterOptions;\n}\n\n/**\n * 开发工具插件集合,简化调用,vite.config.ts 直接 devTools() 即可\n * @param options 配置项\n * @param options.styleAdapter Taro H5 样式适配配置,designWidth 应与 Taro config 一致\n * @returns Plugin[] Vite 插件数组\n * \n * @example\n * ```ts\n * // config/dev.ts\n * import devTools from \"@amaster.ai/vite-plugins\";\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [devTools()]\n * }\n * }\n * ```\n */\nexport default function devTools(options: DevToolsOptions = {}): Plugin[] {\n const plugins: Plugin[] = [\n taroStyleAdapterPlugin(options.styleAdapter), // Taro H5 样式适配(运行时)\n {\n // H5 开发模式下添加 rpx2px PostCSS 插件(编译时)\n name: 'dev-postcss-rpx2px-plugin',\n apply: 'serve',\n config(config) {\n if (process.env.TARO_ENV === 'h5' && typeof config.css?.postcss === 'object') {\n // 在最前面插入,确保在 pxtransform 之前执行\n config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n }\n }\n },\n componentIdPlugin(),\n editorBridgePlugin(),\n routesExposePlugin(),\n smartReloadPlugin() // 智能重载:拦截 full-reload,静默等待重连\n ]\n\n // process.env.WORKSPACE_GIT_REPO 有这个表示是在 sandbox 里面运行,启用浏览器日志插件\n if (process.env.WORKSPACE_GIT_REPO) {\n plugins.push(browserLogsPlugin());\n }\n\n return plugins;\n}\n\nexport {\n componentIdPlugin,\n editorBridgePlugin,\n routesExposePlugin,\n smartReloadPlugin\n}\n\n// Export Taro environment injection helpers\nexport { injectTaroEnv, injectAmasterEnv } from \"./taro-env-inject\";\n\n// Export Taro style adapter plugin and its types\nexport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\n\n// Export PostCSS rpx2px plugin for H5 mode\nexport { rpx2pxPlugin, type Rpx2pxPluginOptions } from \"./postcss-rpx2px\";"]}
|
|
1
|
+
{"version":3,"sources":["../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts","../src/browser-logs.ts","../src/smart-reload.ts","../src/taro-style-adapter.ts","../src/postcss-rpx2px.ts","../src/taro-env-inject.ts","../src/index.ts"],"names":["generateUniqueId","filePath","position","key","createHash","HTML_TAGS","componentIdPlugin","isDev","config","code","id","transformedCode","jsxOpenTagRegex","match","matches","tag","tagStartIndex","tagEndIndex","checkIndex","foundClosing","hasComponentId","currentCode","i","item","index","uniqueId","before","after","clickHandlerScript","getBridgeScriptContent","__filename","fileURLToPath","__dirname","dirname","bridgePath","resolve","readFileSync","error","editorBridgePlugin","bridgeScript","html","scriptsToInject","routesExposePlugin","options","routesPath","browserLogsPlugin","logFilePath","injectedScript","root","process","path","devServer","req","res","next","origin","body","chunk","logDir","fs","smartReloadPlugin","pureScript","transformed","taroStyleAdapterPlugin","designWidth","maxWidth","baseFontSize","minRootSize","maxRootSize","flexibleScript","rpx2pxPlugin","ratio","decl","_match","num","pxValue","injectTaroEnv","additional","autoInjectTaroApp","autoInjectVite","constants","varsToInject","varName","value","injectAmasterEnv","devTools","plugins"],"mappings":"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AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;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","file":"index.js","sourcesContent":["import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { readFileSync } from \"fs\";\nimport { resolve, dirname } from \"path\";\nimport { fileURLToPath } from \"url\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * 读取构建后的 bridge 脚本内容\n */\nfunction getBridgeScriptContent(): string {\n try {\n // 获取当前模块的目录\n const __filename = fileURLToPath(import.meta.url);\n const __dirname = dirname(__filename);\n \n // 读取构建后的 bridge.bridge.js 文件\n const bridgePath = resolve(__dirname, \"../dist/bridge.bridge.js\");\n const bridgeContent = readFileSync(bridgePath, \"utf-8\");\n \n return `<script>\\n${bridgeContent}\\n</script>`;\n } catch (error) {\n console.warn(\"Failed to read bridge script:\", error);\n return \"\";\n }\n}\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode by reading the built bridge script\n */\nexport function editorBridgePlugin(): Plugin {\n let isDev = false;\n let bridgeScript = \"\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n \n // 在开发模式下读取 bridge 脚本内容\n if (isDev) {\n bridgeScript = getBridgeScriptContent();\n }\n },\n\n transformIndexHtml(html) {\n // 在开发模式下注入 bridge 脚本\n const devScript = isDev ? bridgeScript : \"\";\n\n const scriptsToInject = `${clickHandlerScript}${devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\nimport { Buffer } from \"node:buffer\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport process from \"node:process\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n // ============================================\n // Vite deps 504 error capture using PerformanceObserver\n // Only captures 504 errors for node_modules/.vite resources (pre-bundled dependencies)\n // ============================================\n var reportedUrls = {}; // Track reported URLs to avoid duplicates\n \n if (typeof PerformanceObserver !== 'undefined') {\n var perfObserver = new PerformanceObserver(function(list) {\n var entries = list.getEntries();\n for (var i = 0; i < entries.length; i++) {\n var perfEntry = entries[i];\n // Only check script resources\n if (perfEntry.initiatorType !== 'script') {\n continue;\n }\n \n var resourceUrl = perfEntry.name || '';\n // Only report errors for node_modules/.vite resources\n if (resourceUrl.indexOf('node_modules/.vite') === -1 && resourceUrl.indexOf('/node_modules/.vite') === -1) {\n continue;\n }\n \n // Check if response status is 504 (Gateway Timeout)\n // responseStatus is available in Resource Timing Level 2\n var status = perfEntry.responseStatus;\n if (status === 504) {\n // Avoid duplicate reports\n if (reportedUrls[resourceUrl]) {\n continue;\n }\n reportedUrls[resourceUrl] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: resourceUrl,\n status: 504,\n message: 'Vite pre-bundled dependency returned 504 Gateway Timeout: ' + resourceUrl,\n duration: perfEntry.duration,\n transferSize: perfEntry.transferSize\n };\n addLog(entry);\n }\n }\n });\n \n try {\n perfObserver.observe({ type: 'resource', buffered: true });\n } catch (e) {\n // Fallback for browsers that don't support the options\n try {\n perfObserver.observe({ entryTypes: ['resource'] });\n } catch (e2) {\n originalConsole.warn('[BrowserLogs] PerformanceObserver not supported:', e2.message);\n }\n }\n }\n \n // ============================================\n // Script load error capture (for 504 and other network errors)\n // This captures errors that PerformanceObserver might miss\n // ============================================\n \n // Use MutationObserver to watch for dynamically added script tags\n function attachScriptErrorHandler(script) {\n if (script.__errorHandlerAttached__) return;\n script.__errorHandlerAttached__ = true;\n \n script.addEventListener('error', function(event) {\n var src = script.src || '';\n \n // Only report errors for node_modules/.vite resources\n if (src.indexOf('node_modules/.vite') === -1 && src.indexOf('/node_modules/.vite') === -1) {\n return;\n }\n \n // Avoid duplicate reports\n if (reportedUrls[src]) {\n return;\n }\n reportedUrls[src] = true;\n \n var entry = {\n type: 'vite-deps-error',\n timestamp: new Date().toISOString(),\n level: 'error',\n url: src,\n status: 'load-error',\n message: 'Vite pre-bundled dependency failed to load (possibly 504 Gateway Timeout): ' + src,\n errorType: event.type\n };\n addLog(entry);\n });\n }\n \n // Attach error handlers to existing scripts\n var existingScripts = document.querySelectorAll('script[src]');\n for (var i = 0; i < existingScripts.length; i++) {\n attachScriptErrorHandler(existingScripts[i]);\n }\n \n // Watch for dynamically added scripts\n if (typeof MutationObserver !== 'undefined') {\n var scriptObserver = new MutationObserver(function(mutations) {\n for (var i = 0; i < mutations.length; i++) {\n var mutation = mutations[i];\n for (var j = 0; j < mutation.addedNodes.length; j++) {\n var node = mutation.addedNodes[j];\n if (node.nodeName === 'SCRIPT' && node.src) {\n attachScriptErrorHandler(node);\n }\n // Also check child nodes\n if (node.querySelectorAll) {\n var scripts = node.querySelectorAll('script[src]');\n for (var k = 0; k < scripts.length; k++) {\n attachScriptErrorHandler(scripts[k]);\n }\n }\n }\n }\n });\n \n scriptObserver.observe(document.documentElement, {\n childList: true,\n subtree: true\n });\n }\n \n // Also intercept document.createElement to catch scripts before they're added to DOM\n var originalCreateElement = document.createElement.bind(document);\n document.createElement = function(tagName) {\n var element = originalCreateElement(tagName);\n if (tagName.toLowerCase() === 'script') {\n // Use a setter to catch when src is set\n var originalSrc = '';\n Object.defineProperty(element, '__originalSrc__', {\n get: function() { return originalSrc; },\n set: function(val) { originalSrc = val; }\n });\n \n // Defer attaching error handler until src is set\n var srcDescriptor = Object.getOwnPropertyDescriptor(HTMLScriptElement.prototype, 'src');\n if (srcDescriptor && srcDescriptor.set) {\n var originalSrcSetter = srcDescriptor.set;\n Object.defineProperty(element, 'src', {\n get: function() {\n return srcDescriptor.get ? srcDescriptor.get.call(this) : this.getAttribute('src');\n },\n set: function(value) {\n if (originalSrcSetter) {\n originalSrcSetter.call(this, value);\n } else {\n this.setAttribute('src', value);\n }\n attachScriptErrorHandler(this);\n },\n configurable: true\n });\n }\n }\n return element;\n };\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Smart Reload 插件\n *\n * 在 Vite 服务器重启期间(如 PM2 重启 web-server),拦截浏览器端的刷新行为,\n * 让浏览器静默等待服务器恢复,期间页面完全不刷新。\n * 服务器恢复后自动恢复 HMR 连接,无需手动刷新。\n *\n * 场景说明:\n * - PM2 杀掉 Vite 进程 → WebSocket 断开 → PM2 重启 → 新 Vite 启动\n * - Vite 客户端检测到 WebSocket 断开后会通过 HTTP ping 轮询等待服务器恢复\n * - 服务器恢复后 Vite 客户端会调用 location.reload() 刷新页面\n * - 本插件拦截这个刷新行为,让页面保持当前状态\n *\n * 实现原理(双层拦截):\n * 1. 服务端层:通过 transform 钩子修改 /@vite/client 源码,\n * 将 location.reload() 替换为 window.__smartReload__(),\n * 使我们能控制是否真正刷新页面\n * 2. 客户端层:注入脚本拦截 WebSocket 消息,\n * 检测服务器重启场景,在重启期间阻止刷新\n *\n * 区分两种 full-reload 场景:\n * 1. 服务器重启 → WebSocket 断开 → 静默等待,不刷新,自动恢复 HMR\n * 2. 模块变更需要 full-reload → WebSocket 保持连接 → 正常刷新\n */\nexport function smartReloadPlugin(): Plugin {\n // 注入到浏览器端的脚本\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n\n var PLUGIN_TAG = '[SmartReload]';\n // full-reload 后等待 WebSocket 断开的窗口期(ms)\n var DISCONNECT_DETECT_WINDOW = 300;\n\n // ============================================\n // 状态管理\n // ============================================\n var state = {\n // 是否已经拦截了 Vite HMR WebSocket\n wsIntercepted: false,\n // 是否正在等待判断 full-reload 类型(300ms 窗口期)\n detectingReloadType: false,\n // 延迟执行 full-reload 的定时器\n reloadTimer: null,\n // 是否是服务器重启场景\n isServerRestart: false\n };\n\n // ============================================\n // 日志工具\n // ============================================\n function log() {\n var args = [PLUGIN_TAG];\n for (var i = 0; i < arguments.length; i++) {\n args.push(arguments[i]);\n }\n console.log.apply(console, args);\n }\n\n // ============================================\n // 重置状态\n // ============================================\n function resetState() {\n state.detectingReloadType = false;\n state.isServerRestart = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n }\n\n // ============================================\n // 执行真正的 full-reload\n // ============================================\n function executeFullReload() {\n log('Executing full-reload');\n resetState();\n location.reload();\n }\n\n // ============================================\n // __smartReload__:检查是否应该阻止 reload\n //\n // 返回 true 表示是服务器重启场景,不应该刷新页面\n // 返回 false 表示应该正常刷新\n //\n // Vite 客户端代码中的 WebSocket close handler 已被修改为:\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(...); // 重新创建 WebSocket 恢复 HMR\n // } else {\n // location.reload();\n // }\n // ============================================\n window.__smartReload__ = function() {\n if (state.isServerRestart) {\n log('Server restart detected, suppressing reload. Will reconnect WebSocket.');\n resetState();\n return true; // 阻止 reload,由 Vite 客户端重新创建 WebSocket\n }\n log('Non-restart scenario, allowing reload');\n return false; // 允许正常 reload\n };\n\n // ============================================\n // 拦截 WebSocket 构造函数\n // ============================================\n var OriginalWebSocket = window.WebSocket;\n\n function isViteHmrWebSocket(url) {\n if (typeof url !== 'string') return false;\n return url.includes('/__vite_hmr') || url.includes('?token=');\n }\n\n function createWsProxy(ws) {\n var originalAddEventListener = ws.addEventListener.bind(ws);\n var originalRemoveEventListener = ws.removeEventListener.bind(ws);\n\n ws.addEventListener = function(type, listener, options) {\n if (type === 'message') {\n var wrappedListener = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted) {\n listener.call(ws, event);\n }\n };\n listener.__smartReloadWrapped__ = wrappedListener;\n return originalAddEventListener(type, wrappedListener, options);\n }\n return originalAddEventListener(type, listener, options);\n };\n\n ws.removeEventListener = function(type, listener, options) {\n if (type === 'message' && listener.__smartReloadWrapped__) {\n return originalRemoveEventListener(type, listener.__smartReloadWrapped__, options);\n }\n return originalRemoveEventListener(type, listener, options);\n };\n\n var originalOnMessage = null;\n var onMessageDescriptor = Object.getOwnPropertyDescriptor(WebSocket.prototype, 'onmessage') ||\n Object.getOwnPropertyDescriptor(ws, 'onmessage');\n\n if (onMessageDescriptor) {\n Object.defineProperty(ws, 'onmessage', {\n get: function() {\n return originalOnMessage;\n },\n set: function(handler) {\n originalOnMessage = handler;\n var wrappedHandler = function(event) {\n var intercepted = handleWsMessage(event);\n if (!intercepted && handler) {\n handler.call(ws, event);\n }\n };\n if (onMessageDescriptor.set) {\n onMessageDescriptor.set.call(ws, wrappedHandler);\n }\n },\n configurable: true\n });\n }\n\n originalAddEventListener('close', function(event) {\n handleWsClose(event);\n });\n\n originalAddEventListener('open', function() {\n handleWsOpen();\n });\n\n state.wsIntercepted = true;\n log('Vite HMR WebSocket intercepted');\n\n return ws;\n }\n\n window.WebSocket = function SmartReloadWebSocket(url, protocols) {\n var ws;\n if (protocols !== undefined) {\n ws = new OriginalWebSocket(url, protocols);\n } else {\n ws = new OriginalWebSocket(url);\n }\n\n if (isViteHmrWebSocket(url)) {\n return createWsProxy(ws);\n }\n\n return ws;\n };\n\n window.WebSocket.prototype = OriginalWebSocket.prototype;\n window.WebSocket.CONNECTING = OriginalWebSocket.CONNECTING;\n window.WebSocket.OPEN = OriginalWebSocket.OPEN;\n window.WebSocket.CLOSING = OriginalWebSocket.CLOSING;\n window.WebSocket.CLOSED = OriginalWebSocket.CLOSED;\n\n // ============================================\n // WebSocket 消息处理\n // ============================================\n function handleWsMessage(event) {\n var data;\n try {\n data = JSON.parse(event.data);\n } catch (e) {\n return false;\n }\n\n // 处理 full-reload 消息\n if (data.type === 'full-reload') {\n // 如果在服务器重启模式,拦截所有 full-reload\n if (state.isServerRestart) {\n log('Suppressed full-reload message during server restart');\n return true;\n }\n\n log('Intercepted full-reload, path:', data.path || '(none)');\n\n // 进入检测模式:等待看 WebSocket 是否断开\n state.detectingReloadType = true;\n\n // 如果窗口期内 WebSocket 没有断开,说明是真正的模块变更\n state.reloadTimer = setTimeout(function() {\n if (state.detectingReloadType && !state.isServerRestart) {\n log('WebSocket still connected after ' + DISCONNECT_DETECT_WINDOW + 'ms - real module change, executing reload');\n state.detectingReloadType = false;\n executeFullReload();\n }\n }, DISCONNECT_DETECT_WINDOW);\n\n return true;\n }\n\n // 处理 connected 消息\n if (data.type === 'connected') {\n if (state.isServerRestart) {\n log('HMR reconnected after server restart! Page was NOT reloaded.');\n resetState();\n }\n return false;\n }\n\n return false;\n }\n\n // ============================================\n // WebSocket 连接事件处理\n // ============================================\n function handleWsClose(event) {\n log('WebSocket closed, code:', event.code, 'wasClean:', event.wasClean);\n\n // 场景1:收到 full-reload 后 WebSocket 断开 → 服务器重启\n if (state.detectingReloadType) {\n log('Entering server restart mode: full-reload followed by disconnect');\n state.isServerRestart = true;\n state.detectingReloadType = false;\n if (state.reloadTimer) {\n clearTimeout(state.reloadTimer);\n state.reloadTimer = null;\n }\n return;\n }\n\n // 场景2:WebSocket 直接断开(非正常关闭)\n if (!state.isServerRestart && !event.wasClean) {\n log('Entering server restart mode: unexpected disconnect');\n state.isServerRestart = true;\n }\n }\n\n function handleWsOpen() {\n log('WebSocket opened');\n }\n\n // ============================================\n // 暴露 API 供调试使用\n // ============================================\n window.__smartReloadState__ = state;\n window.__OriginalWebSocket__ = OriginalWebSocket;\n window.__smartReloadForceReload__ = function() {\n log('Force reload triggered by user');\n resetState();\n location.reload();\n };\n\n log('Smart Reload initialized');\n})();\n</script>`;\n\n // 提取纯 JS 代码(去掉 <script> 标签)\n const pureScript = injectedScript\n .replace(/^\\s*<script>\\s*/i, \"\")\n .replace(/\\s*<\\/script>\\s*$/i, \"\");\n\n return {\n name: \"vite-plugin-smart-reload\",\n\n // 只在开发模式下注入脚本\n apply: \"serve\",\n\n transformIndexHtml: {\n order: \"pre\",\n handler() {\n return [\n {\n tag: \"script\",\n // 注入到 head 的最前面,确保在 Vite 客户端脚本 /@vite/client 之前执行\n injectTo: \"head-prepend\",\n children: pureScript,\n },\n ];\n },\n },\n\n // 修改 Vite 客户端代码:\n // 精确替换 WebSocket close handler 中的 reload 逻辑,\n // 在服务器重启场景下重新创建 WebSocket 恢复 HMR,而不是刷新页面\n //\n // Vite 对 /@vite/client 的处理流程:\n // 1. 浏览器请求 /@vite/client\n // 2. Vite 的 transformMiddleware 拦截请求\n // 3. 调用 server.transformRequest('/@vite/client')\n // 4. 内部解析为文件系统路径 node_modules/vite/dist/client/client.mjs\n // 5. 调用插件的 transform(code, resolvedId) — id 是文件系统路径\n transform(code, id) {\n // 匹配 Vite 客户端模块(文件系统路径)\n if (!id.includes(\"vite/dist/client/client.mjs\")) {\n return null;\n }\n\n // 精确替换 WebSocket close handler 中的 reload 逻辑:\n // 原始代码:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // location.reload();\n // 替换为:\n // await waitForSuccessfulPing(protocol, hostAndPath);\n // if (window.__smartReload__ && window.__smartReload__()) {\n // socket = setupWebSocket(protocol, hostAndPath, onCloseWithoutOpen);\n // } else {\n // location.reload();\n // }\n const transformed = code.replace(\n /await waitForSuccessfulPing\\(protocol,\\s*hostAndPath\\);\\s*\\n\\s*location\\.reload\\(\\)/,\n `await waitForSuccessfulPing(protocol, hostAndPath);\n if (window.__smartReload__ && window.__smartReload__()) {\n console.log('[SmartReload] Vite client: server is back, fetching new WebSocket token...');\n // PM2 重启后新服务器会生成新的 wsToken,旧 token 无法连接\n // 通过 fetch /@vite/client 获取新 token\n try {\n var clientResp = await fetch('/@vite/client');\n var clientCode = await clientResp.text();\n // 从新的客户端代码中提取 wsToken\n // Vite 编译后的代码中 wsToken 是一个字符串常量,如: const wsToken = \"xxxxxxxxxxxx\"\n var tokenMatch = clientCode.match(/const\\\\s+wsToken\\\\s*=\\\\s*\"([^\"]+)\"/);\n if (tokenMatch && tokenMatch[1]) {\n console.log('[SmartReload] Got new wsToken from server');\n // 更新闭包中的 wsToken 变量(Vite 客户端代码中的 const wsToken)\n // 注意:wsToken 是 const,无法直接修改。我们需要修改 setupWebSocket 使用的 URL\n // 方案:直接创建 WebSocket 连接,绕过 setupWebSocket\n var newToken = tokenMatch[1];\n var wsUrl = protocol + '://' + hostAndPath + '?token=' + newToken;\n console.log('[SmartReload] Reconnecting with new token to: ' + wsUrl);\n var newSocket = new WebSocket(wsUrl, 'vite-hmr');\n var newIsOpened = false;\n newSocket.addEventListener('open', function() {\n newIsOpened = true;\n console.log('[SmartReload] WebSocket reconnected successfully!');\n notifyListeners('vite:ws:connect', { webSocket: newSocket });\n }, { once: true });\n newSocket.addEventListener('message', async function(ev) {\n handleMessage(JSON.parse(ev.data));\n });\n newSocket.addEventListener('close', async function(ev) {\n if (ev.wasClean) return;\n if (!newIsOpened && onCloseWithoutOpen) {\n onCloseWithoutOpen();\n return;\n }\n notifyListeners('vite:ws:disconnect', { webSocket: newSocket });\n if (hasDocument) {\n console.log('[vite] server connection lost. Polling for restart...');\n await waitForSuccessfulPing(protocol, hostAndPath);\n if (window.__smartReload__ && window.__smartReload__()) {\n console.log('[SmartReload] Nested reconnect - reloading page');\n }\n location.reload();\n }\n });\n socket = newSocket;\n } else {\n console.log('[SmartReload] Could not extract new wsToken, reloading page');\n location.reload();\n }\n } catch(e) {\n console.log('[SmartReload] Failed to fetch new client code, reloading page', e);\n location.reload();\n }\n } else {\n location.reload();\n }`,\n );\n\n if (transformed !== code) {\n return {\n code: transformed,\n map: null,\n };\n }\n\n return null;\n },\n };\n}\n","import type { Plugin } from 'vite';\n\nexport interface TaroStyleAdapterOptions {\n /**\n * Design width for responsive scaling (default: 375)\n * Should match Taro config's designWidth\n */\n designWidth?: number;\n \n /**\n * Maximum viewport width to scale (default: 750)\n * Beyond this width, font-size stays fixed\n */\n maxWidth?: number;\n\n /**\n * Base font size in pixels (default: 12)\n * Should match Taro config's baseFontSize\n */\n baseFontSize?: number;\n\n /**\n * Minimum root font size in pixels (default: 12)\n * Should match Taro config's minRootSize\n */\n minRootSize?: number;\n\n /**\n * Maximum root font size in pixels (default: 24)\n * Should match Taro config's maxRootSize\n */\n maxRootSize?: number;\n}\n\n/**\n * Taro H5 样式适配插件\n * \n * 功能:\n * 1. 动态计算根字号,使 H5 的 rem 与小程序的 rpx 行为对齐\n * 2. 隐藏 Taro 页面容器的滚动条,用于编辑器 iframe 预览\n * \n * 原理:\n * - 小程序 rpx: 1rpx = 屏幕宽度 / 750\n * - H5 rem: Taro 把 rpx 转成 rem,基于 baseFontSize\n * - 本插件动态设置 html font-size,使 rem 的实际像素值与 rpx 对齐\n * \n * 注意:此插件会覆盖 Taro 默认注入的根字号脚本,配置需与 Taro pxtransform 保持一致\n * \n * @example\n * ```ts\n * // In Taro config/dev.ts\n * import { taroStyleAdapterPlugin } from '@amaster.ai/vite-plugins'\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [\n * taroStyleAdapterPlugin({ \n * designWidth: 375,\n * baseFontSize: 12,\n * minRootSize: 12,\n * maxRootSize: 24\n * })\n * ]\n * }\n * }\n * ```\n */\nexport function taroStyleAdapterPlugin(options: TaroStyleAdapterOptions = {}): Plugin {\n const { \n designWidth = 375, \n maxWidth = 750,\n baseFontSize = 12,\n minRootSize = 12,\n maxRootSize = 24\n } = options;\n \n return {\n name: 'vite-plugin-taro-style-adapter',\n apply: 'serve', // 仅在开发模式下生效\n transformIndexHtml(html) {\n // 动态根字号脚本\n // 公式:fontSize = (clientWidth / designWidth) * baseFontSize\n // 当 designWidth=375, baseFontSize=12 时:\n // - 375px 屏幕 → 12px (1rem = 12px)\n // - 750px 屏幕 → 24px (1rem = 24px)\n // 这样 rem 值在不同屏幕宽度下的视觉比例与小程序 rpx 一致\n const flexibleScript = `\n<script data-taro-flexible=\"true\">\n(function() {\n var designWidth = ${designWidth};\n var maxWidth = ${maxWidth};\n var baseFontSize = ${baseFontSize};\n var minRootSize = ${minRootSize};\n var maxRootSize = ${maxRootSize};\n \n function setRootFontSize() {\n var docEl = document.documentElement;\n var clientWidth = docEl.clientWidth;\n \n // 限制最大宽度\n if (clientWidth > maxWidth) {\n clientWidth = maxWidth;\n }\n \n // 计算根字号: (屏幕宽度 / 设计稿宽度) * 基准字号\n var fontSize = (clientWidth / designWidth) * baseFontSize;\n \n // 应用最小/最大限制\n if (fontSize < minRootSize) {\n fontSize = minRootSize;\n } else if (fontSize > maxRootSize) {\n fontSize = maxRootSize;\n }\n \n docEl.style.fontSize = fontSize + 'px';\n }\n \n setRootFontSize();\n \n // 监听窗口变化\n window.addEventListener('resize', setRootFontSize);\n window.addEventListener('orientationchange', setRootFontSize);\n \n // 页面显示时重新计算(解决某些浏览器的 bug)\n document.addEventListener('DOMContentLoaded', setRootFontSize);\n})();\n</script>\n`;\n\n // 滚动条隐藏样式 + Taro Image 组件修复\n const styles = `\n<style data-taro-adapter=\"true\">\n /* 仅 H5 生效:隐藏 Taro 页面容器滚动条,但仍可滚动 */\n .taro_page {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro_page::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /* 隐藏 Taro ScrollView 组件的滚动条 */\n .taro-scroll-view__scroll-y,\n .taro-scroll-view__scroll-x {\n -ms-overflow-style: none; /* IE/Edge 老版 */\n scrollbar-width: none; /* Firefox */\n }\n\n .taro-scroll-view__scroll-y::-webkit-scrollbar,\n .taro-scroll-view__scroll-x::-webkit-scrollbar {\n width: 0;\n height: 0;\n display: none; /* Chrome/Safari */\n }\n\n /*\n * 修复 Taro Image 组件在 H5 下的尺寸问题\n * \n * 问题:Taro Image 组件使用 aspectFit 模式时,内部 img 使用 max-width/max-height: 100%\n * 导致小尺寸图片(如 SVG 图标)无法按指定尺寸显示,只显示原始尺寸\n * \n * 解决:让 img 填满 taro-image-core 容器,由容器控制尺寸\n */\n taro-image-core img.taro-img__mode-aspectfit {\n width: 100%;\n height: 100%;\n max-width: none;\n max-height: none;\n position: static;\n transform: none;\n object-fit: contain;\n }\n</style>\n`;\n \n // 在 </head> 之前插入脚本和样式\n return html.replace('</head>', `${flexibleScript}${styles}</head>`);\n }\n };\n}\n","/**\n * PostCSS 插件:H5 模式下将 rpx 预转换为 px\n *\n * 问题背景:\n * - Taro 的 pxtransform 在 H5 模式下会处理 rpx,但它把 rpx 值直接除以 rootValue\n * - 导致 96rpx 被转成 8rem(96/12),实际显示为 96px\n * - 而正确应该是:96rpx = 48px = 4rem = 48px\n *\n * 解决方案:\n * - 在 pxtransform 之前,先把 rpx 转成 px(rpx / 2)\n * - 然后 pxtransform 会正确地把 px 转成 rem(px / 12)\n *\n * 转换链路:96rpx → 48px → 4rem → 48px(正确)\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { rpx2pxPlugin } from '@amaster.ai/vite-plugins'\n * \n * // 在 postcss-config-loader-plugin 中使用\n * {\n * name: 'postcss-config-loader-plugin',\n * config(config) {\n * if (typeof config.css?.postcss === 'object') {\n * config.css?.postcss.plugins?.unshift(tailwindcss())\n * // H5 模式下添加 rpx2px 插件\n * if (process.env.TARO_ENV === 'h5') {\n * config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n * }\n * }\n * }\n * }\n * ```\n */\n\nexport interface Rpx2pxPluginOptions {\n /**\n * rpx 到 px 的转换比例(默认 2)\n * 即 1rpx = 1/ratio px\n * 默认值 2 表示 2rpx = 1px(基于 375 设计稿)\n */\n ratio?: number;\n}\n\ninterface PostCSSDeclaration {\n value: string;\n}\n\ninterface PostCSSPlugin {\n postcssPlugin: string;\n Declaration: (decl: PostCSSDeclaration) => void;\n}\n\ntype PostCSSPluginCreator = {\n (): PostCSSPlugin;\n postcss: boolean;\n};\n\nexport function rpx2pxPlugin(options: Rpx2pxPluginOptions = {}): PostCSSPlugin {\n const { ratio = 2 } = options;\n \n return {\n postcssPlugin: 'postcss-rpx2px',\n Declaration(decl: PostCSSDeclaration) {\n if (decl.value.includes('rpx')) {\n decl.value = decl.value.replace(/(-?\\d*\\.?\\d+)rpx/gi, (_match: string, num: string) => {\n const pxValue = parseFloat(num) / ratio;\n return pxValue === 0 ? '0' : `${pxValue}px`;\n });\n }\n }\n };\n}\n\n// PostCSS 插件标识\n(rpx2pxPlugin as PostCSSPluginCreator).postcss = true;\n","/**\n * Taro Environment Variables Injection Helper\n * \n * Automatically injects environment variables into Taro's defineConstants\n * so that they will be replaced at build time.\n * \n * @example\n * ```ts\n * // In Taro config/index.ts\n * import { injectTaroEnv } from '@amaster.ai/vite-plugins/taro-env-inject'\n * \n * export default defineConfig({\n * defineConstants: {\n * ...injectTaroEnv()\n * }\n * })\n * ```\n */\n\nexport interface TaroEnvInjectOptions {\n /**\n * Additional environment variable names to inject\n * @default []\n */\n additional?: string[];\n \n /**\n * Whether to inject all TARO_APP_* prefixed variables\n * @default true\n */\n autoInjectTaroApp?: boolean;\n \n /**\n * Whether to inject all VITE_* prefixed variables\n * @default true\n */\n autoInjectVite?: boolean;\n}\n\n/**\n * Inject environment variables into Taro's defineConstants\n * \n * This function reads environment variables and formats them for Taro's defineConstants.\n * Taro will replace these at build time, converting `process.env.XXX` to actual values.\n */\nexport function injectTaroEnv(options: TaroEnvInjectOptions = {}): Record<string, string> {\n const {\n additional = [],\n autoInjectTaroApp = true,\n autoInjectVite = true,\n } = options;\n\n const constants: Record<string, string> = {};\n\n // Collect variable names to inject\n const varsToInject = new Set<string>(additional);\n\n // Auto-detect TARO_APP_* variables\n if (autoInjectTaroApp) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('TARO_APP_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Auto-detect VITE_* variables\n if (autoInjectVite) {\n Object.keys(process.env).forEach(key => {\n if (key.startsWith('VITE_')) {\n varsToInject.add(key);\n }\n });\n }\n\n // Inject variables into defineConstants format\n varsToInject.forEach(varName => {\n const value = process.env[varName] || '';\n constants[`process.env.${varName}`] = JSON.stringify(value);\n });\n\n return constants;\n}\n\n/**\n * Inject specific environment variables for amaster.ai mini-program builds\n * \n * This is a convenience function that injects the standard environment variables\n * used by @amaster.ai/http-client for API base URL configuration.\n */\nexport function injectAmasterEnv(): Record<string, string> {\n return {\n 'process.env.TARO_APP_API_BASE_URL': JSON.stringify(process.env.TARO_APP_API_BASE_URL || ''),\n 'process.env.VITE_API_BASE_URL': JSON.stringify(process.env.VITE_API_BASE_URL || ''),\n };\n}\n","import { componentIdPlugin } from \"./component-id\";\nimport { editorBridgePlugin } from \"./editor-bridge\";\nimport { routesExposePlugin } from \"./routes-expose\";\nimport { browserLogsPlugin } from \"./browser-logs\";\nimport { smartReloadPlugin } from \"./smart-reload\";\nimport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\nimport { rpx2pxPlugin } from \"./postcss-rpx2px\";\nimport process from \"node:process\";\nimport type { Plugin } from \"vite\";\n\nexport interface DevToolsOptions {\n /**\n * Taro style adapter options\n * Controls H5 responsive scaling to match mini-program rpx behavior\n */\n styleAdapter?: TaroStyleAdapterOptions;\n}\n\n/**\n * 开发工具插件集合,简化调用,vite.config.ts 直接 devTools() 即可\n * @param options 配置项\n * @param options.styleAdapter Taro H5 样式适配配置,designWidth 应与 Taro config 一致\n * @returns Plugin[] Vite 插件数组\n * \n * @example\n * ```ts\n * // config/dev.ts\n * import devTools from \"@amaster.ai/vite-plugins\";\n * \n * export default {\n * compiler: {\n * type: 'vite',\n * vitePlugins: [devTools()]\n * }\n * }\n * ```\n */\nexport default function devTools(options: DevToolsOptions = {}): Plugin[] {\n const plugins: Plugin[] = [\n taroStyleAdapterPlugin(options.styleAdapter), // Taro H5 样式适配(运行时)\n {\n // H5 开发模式下添加 rpx2px PostCSS 插件(编译时)\n name: 'dev-postcss-rpx2px-plugin',\n apply: 'serve',\n config(config) {\n if (process.env.TARO_ENV === 'h5' && typeof config.css?.postcss === 'object') {\n // 在最前面插入,确保在 pxtransform 之前执行\n config.css?.postcss.plugins?.unshift(rpx2pxPlugin())\n }\n }\n },\n componentIdPlugin(),\n editorBridgePlugin(),\n routesExposePlugin(),\n smartReloadPlugin() // 智能重载:拦截 full-reload,静默等待重连\n ]\n\n // process.env.WORKSPACE_GIT_REPO 有这个表示是在 sandbox 里面运行,启用浏览器日志插件\n if (process.env.WORKSPACE_GIT_REPO) {\n plugins.push(browserLogsPlugin());\n }\n\n return plugins;\n}\n\nexport {\n componentIdPlugin,\n editorBridgePlugin,\n routesExposePlugin,\n smartReloadPlugin\n}\n\n// Export Taro environment injection helpers\nexport { injectTaroEnv, injectAmasterEnv } from \"./taro-env-inject\";\n\n// Export Taro style adapter plugin and its types\nexport { taroStyleAdapterPlugin, type TaroStyleAdapterOptions } from \"./taro-style-adapter\";\n\n// Export PostCSS rpx2px plugin for H5 mode\nexport { rpx2pxPlugin, type Rpx2pxPluginOptions } from \"./postcss-rpx2px\";"]}
|