2024-02-01 11:22:37 +00:00
|
|
|
|
/**
|
|
|
|
|
* @type { import('typescript') }
|
|
|
|
|
*/
|
|
|
|
|
var ts;
|
|
|
|
|
importScripts('./game/typescript.js');
|
2024-03-08 16:37:33 +00:00
|
|
|
|
/**
|
|
|
|
|
* @type { import('./game/compiler-sfc.browser.js') }
|
|
|
|
|
*/
|
|
|
|
|
var sfc;
|
|
|
|
|
importScripts('./game/compiler-sfc.browser.js');
|
2024-02-01 11:22:37 +00:00
|
|
|
|
// @ts-ignore
|
|
|
|
|
if (typeof ts != 'undefined') {
|
2024-02-16 11:08:45 +00:00
|
|
|
|
console.log(`ts loaded`);
|
2024-02-01 11:22:37 +00:00
|
|
|
|
} else {
|
|
|
|
|
console.log(`ts undefined`);
|
|
|
|
|
}
|
2024-03-08 16:37:33 +00:00
|
|
|
|
// @ts-ignore
|
|
|
|
|
if (typeof sfc != 'undefined') {
|
|
|
|
|
console.log(`sfc loaded`, sfc);
|
|
|
|
|
sfc.registerTS(() => ts);
|
|
|
|
|
} else {
|
|
|
|
|
console.log(`sfc undefined`);
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-01 11:22:37 +00:00
|
|
|
|
|
2024-03-08 16:37:33 +00:00
|
|
|
|
console.log('serviceWorker version 2.3');
|
2024-02-05 10:17:53 +00:00
|
|
|
|
|
|
|
|
|
self.addEventListener("install", (event) => {
|
|
|
|
|
// The promise that skipWaiting() returns can be safely ignored.
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
self.skipWaiting();
|
|
|
|
|
});
|
2024-02-01 11:22:37 +00:00
|
|
|
|
|
2024-02-16 10:54:49 +00:00
|
|
|
|
self.addEventListener("activate", (event) => {
|
|
|
|
|
// 当一个 service worker 被初始注册时,页面在下次加载之前不会使用它。 claim() 方法会立即控制这些页面
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
event.waitUntil(clients.claim());
|
|
|
|
|
});
|
|
|
|
|
|
2024-02-01 11:22:37 +00:00
|
|
|
|
self.addEventListener('message', event => {
|
|
|
|
|
console.log(event.data);
|
|
|
|
|
});
|
|
|
|
|
|
2024-03-08 16:37:33 +00:00
|
|
|
|
/**
|
|
|
|
|
* 将vue编译的结果放在这里,调用的时候直接返回就好了
|
|
|
|
|
*/
|
|
|
|
|
const vueFileMap = new Map();
|
|
|
|
|
|
2024-02-01 11:22:37 +00:00
|
|
|
|
self.addEventListener('fetch', event => {
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
const request = event.request;
|
|
|
|
|
if (typeof request.url != 'string') return console.log(request);
|
2024-03-08 16:37:33 +00:00
|
|
|
|
if (vueFileMap.has(request.url)) {
|
|
|
|
|
const rep = new Response(new Blob([vueFileMap.get(request.url)], { type: "text/javascript" }), {
|
|
|
|
|
status: 200,
|
|
|
|
|
statusText: "OK",
|
|
|
|
|
headers: new Headers({
|
|
|
|
|
"Content-Type": "text/javascript"
|
|
|
|
|
}),
|
|
|
|
|
});
|
|
|
|
|
event.respondWith(rep);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!['.ts', '.json', '.vue'].some(ext => request.url.endsWith(ext))) return;
|
2024-02-05 10:17:53 +00:00
|
|
|
|
if (request.url.endsWith('.d.ts')) return;
|
|
|
|
|
if (request.url.endsWith('.json')) {
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
if (!event.request.headers.get('origin')) return;
|
|
|
|
|
}
|
2024-02-01 11:22:37 +00:00
|
|
|
|
// 请求ts文件
|
|
|
|
|
const res = fetch(request.url, {
|
|
|
|
|
method: request.method,
|
|
|
|
|
mode: "no-cors",
|
|
|
|
|
headers: new Headers({
|
|
|
|
|
"Content-Type": "text/plain"
|
|
|
|
|
}),
|
|
|
|
|
});
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
event.respondWith(
|
|
|
|
|
res.then(res => {
|
2024-02-02 12:49:32 +00:00
|
|
|
|
if (res.status != 200) return res;
|
|
|
|
|
console.log('正在编译', request.url);
|
|
|
|
|
return res.text().then(text => {
|
2024-02-05 10:17:53 +00:00
|
|
|
|
let js;
|
|
|
|
|
if (request.url.endsWith('.json')) {
|
|
|
|
|
js = `export default ${text}`;
|
2024-03-08 16:37:33 +00:00
|
|
|
|
} else if (request.url.endsWith('.ts')) {
|
2024-02-05 10:17:53 +00:00
|
|
|
|
js = ts.transpile(text, {
|
|
|
|
|
module: ts.ModuleKind.ES2015,
|
|
|
|
|
target: ts.ScriptTarget.ES2019,
|
|
|
|
|
inlineSourceMap: true,
|
|
|
|
|
resolveJsonModule: true,
|
|
|
|
|
esModuleInterop: true,
|
|
|
|
|
}, request.url);
|
2024-03-08 16:37:33 +00:00
|
|
|
|
} else if (request.url.endsWith('.vue')) {
|
|
|
|
|
const id = Date.now().toString();
|
|
|
|
|
const scopeId = `data-v-${id}`;
|
|
|
|
|
const { descriptor } = sfc.parse(text);
|
|
|
|
|
const hasScoped = descriptor.styles.some((s) => s.scoped);
|
|
|
|
|
// 编译 script,因为可能有 script setup,还要进行 css 变量注入
|
|
|
|
|
const script = sfc.compileScript(descriptor, {
|
|
|
|
|
id: scopeId,
|
|
|
|
|
inlineTemplate: true,
|
|
|
|
|
templateOptions: {
|
|
|
|
|
scoped: hasScoped,
|
|
|
|
|
compilerOptions: {
|
|
|
|
|
scopeId: hasScoped ? scopeId : undefined,
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
// 用于存放代码,最后 join('\n') 合并成一份完整代码
|
|
|
|
|
const codeList = [];
|
|
|
|
|
// 重写 default
|
|
|
|
|
vueFileMap.set(
|
|
|
|
|
request.url + '?type=script',
|
|
|
|
|
sfc.rewriteDefault(script.content, "__sfc_main__")
|
|
|
|
|
.replace(`const __sfc_main__`, `export const __sfc_main__`)
|
|
|
|
|
.replaceAll(`from "vue"`, `from "/game/vue.esm-browser.js"`)
|
|
|
|
|
.replaceAll(`from 'vue'`, `from '/game/vue.esm-browser.js'`)
|
|
|
|
|
);
|
|
|
|
|
codeList.push(`import { __sfc_main__ } from '${request.url}?type=script'`);
|
|
|
|
|
codeList.push(`__sfc_main__.__scopeId='${scopeId}'`);
|
|
|
|
|
|
|
|
|
|
// 编译模板,转换成 render 函数
|
|
|
|
|
const template = sfc.compileTemplate({
|
|
|
|
|
source: descriptor.template.content,
|
|
|
|
|
filename: request.url, // 用于错误提示
|
|
|
|
|
id: scopeId,
|
|
|
|
|
scoped: hasScoped,
|
|
|
|
|
inlineTemplate: true,
|
|
|
|
|
compilerOptions: {
|
|
|
|
|
scopeId: hasScoped ? scopeId : undefined,
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
codeList.push(`import { render } from '${request.url}?type=template'`);
|
|
|
|
|
|
|
|
|
|
vueFileMap.set(request.url + '?type=template', template.code
|
|
|
|
|
.replace(`function render(_ctx, _cache) {`, str => str + 'console.log(_ctx);')
|
|
|
|
|
.replaceAll(`from "vue"`, `from "/game/vue.esm-browser.js"`)
|
|
|
|
|
.replaceAll(`from 'vue'`, `from '/game/vue.esm-browser.js'`)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
codeList.push(`__sfc_main__.render = render;`);
|
|
|
|
|
codeList.push(`export default __sfc_main__;`);
|
|
|
|
|
|
|
|
|
|
// 一个 Vue 文件,可能有多个 style 标签
|
|
|
|
|
for (const styleBlock of descriptor.styles) {
|
|
|
|
|
const styleCode = sfc.compileStyle({
|
|
|
|
|
source: styleBlock.content,
|
|
|
|
|
id,
|
|
|
|
|
filename: request.url,
|
|
|
|
|
scoped: styleBlock.scoped,
|
|
|
|
|
});
|
|
|
|
|
const styleDOM = `var el = document.createElement('style');\nel.innerHTML = \`${styleCode.code}\`;document.body.append(el);`;
|
|
|
|
|
codeList.push(styleDOM);
|
|
|
|
|
}
|
|
|
|
|
const code = codeList.join('\n');
|
|
|
|
|
js = code;
|
2024-02-05 10:17:53 +00:00
|
|
|
|
}
|
2024-02-02 12:49:32 +00:00
|
|
|
|
const rep = new Response(new Blob([js], { type: "text/javascript" }), {
|
|
|
|
|
status: 200,
|
|
|
|
|
statusText: "OK",
|
|
|
|
|
headers: new Headers({
|
|
|
|
|
"Content-Type": "text/javascript"
|
|
|
|
|
}),
|
|
|
|
|
});
|
|
|
|
|
return rep;
|
|
|
|
|
})
|
2024-02-01 11:22:37 +00:00
|
|
|
|
})
|
|
|
|
|
.catch(e => {
|
|
|
|
|
console.log(e);
|
|
|
|
|
throw e;
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
});
|