Files
BewlyBewly/vite-mv3-hmr.ts
Hakadao 9932c77762 chore: support safari (hmmm, i guess?) (#960)
* Add support of Safari

1. Workaround of Safari's bug of not sending cookies when using fetch in background service worker. This fixes https://github.com/hakadao/BewlyBewly/issues/275#issuecomment-1927278492. See: https://stackoverflow.com/questions/78239213/safari-extension-service-worker-not-sending-cookie
2. Add scripts to for Safari specific build and convert it to a Safari web extension.

* Fixed a bug in Safari that when the extension was installed, the page would turn into a white screen

* Safari may fire DOMContentLoaded before the codes get called

* Remove false fix; Ignore extension-safari

* Reduce memory and power usage with a nonpersistent background page on Safari and Firefox, see https://developer.apple.com/documentation/safariservices/safari_web_extensions/optimizing_your_web_extension_for_safari#3743241 and https://developer.mozilla.org/en-US/docs/Mozilla/Add-ons/WebExtensions/Background_scripts

* Build and release Safari extension with Github actions (#585)

* feat: add safari extension build

* fix: safari pathname

* fix: add build branch

* Merge

* Try building Safari extension using GitHub action

Correct versions

Debug

Minor bug fixed

Minor bug fixed

Add scripts to release

Minor bug fixed

Minor bug fixed

Minor bug fixed

---------

Co-authored-by: 梦念逍遥 <2589141604@qq.com>
Co-authored-by: Xiaolin Wang <exgphe@gmail.com>
Co-authored-by: Hakadao <a578457889743@gmail.com>

* fix: safari build (#665)

* chore: update

* chore: update knip.json to ignore "xcrun" binaries

---------

Co-authored-by: Xiaolin Wang <exgphe@gmail.com>
Co-authored-by: exgphe <14009450+exgphe@users.noreply.github.com>
Co-authored-by: 梦念逍遥 <2589141604@qq.com>
2024-08-14 00:24:09 +08:00

106 lines
3.4 KiB
TypeScript

import { dirname, join } from 'node:path'
import fs from 'fs-extra'
import type { HMRPayload, PluginOption } from 'vite'
import { isFirefox, isSafari, isWin, r } from './scripts/utils'
const targetDir = r(isFirefox ? 'extension-firefox' : isSafari ? 'extension-safari' : 'extension')
export function MV3Hmr(): PluginOption {
return {
name: 'vite-mv3-hmr',
apply: 'serve',
enforce: 'post',
async configureServer(server) {
const originWsSend: (payload: HMRPayload) => void = server.ws.send
server.ws.send = async function (payload: HMRPayload) {
if (payload.type === 'update') {
for (const update of payload.updates) {
await writeToDisk(update.path)
if (update.acceptedPath !== update.path)
await writeToDisk(update.acceptedPath)
}
payload.updates = payload.updates.map((update) => {
const isJsUpdate = update.type === 'js-update'
if (!isJsUpdate)
return update
return {
...update,
path: `${update.path}.js`,
acceptedPath: `${update.acceptedPath}.js`,
}
})
}
originWsSend.call(this, payload)
}
async function writeToDisk(url: string) {
const result = await server.transformRequest(url.replace(/^\/@id\//, ''))
let code = result?.code
if (!code)
return
const urlModule = await server.moduleGraph.getModuleByUrl(url)
const importedModules = urlModule?.importedModules
if (importedModules) {
for (const mod of importedModules) {
code = code.replace(mod.url, normalizeViteUrl(isWin
? mod.url.replace(/[A-Z]:\//, '').replace(/:/, '.')
: mod.url, mod.type)) // fix invalid colon in /@fs/C:, /@id/plugin-vue:export-helper
writeToDisk(mod.url)
}
}
if (urlModule?.url) {
code = code
.replace(/\/@vite\/client/g, '/dist/mv3client.mjs')
.replace(/\/@id\//g, '/')
.replace(/__uno.css/g, '~~uno.css')
.replace(/__x00__plugin-vue:export-helper/g, '~~x00__plugin-vue:export-helper.js')
.replace(/(\/\.vite\/deps\/\S+?)\?v=\w+/g, '$1')
if (isWin) {
code = code
.replace(/(from\s+["']\/@fs\/)[A-Z]:\//g, '$1')
}
const targetFile = normalizeFsUrl(isWin
? urlModule.url.replace(/[A-Z]:\//, '').replace(/:/, '.')
: urlModule.url, urlModule.type) // fix invalid colon in /@fs/C:, /@id/plugin-vue:export-helper
await fs.ensureDir(dirname(targetFile))
await fs.writeFile(targetFile, code)
}
}
Object.keys(server.config.build.rollupOptions.input!).map(entry => writeToDisk(`/${entry}/main.ts`))
},
}
}
function normalizeViteUrl(url: string, type: string) {
url = url.replace(/\?v=\w+$/, '')
if (type === 'js' && !url.endsWith('.js') && !url.endsWith('.mjs'))
url = `${url}.js`.replace(/vue\?/, 'vue.js_')
return url
}
function normalizeFsUrl(url: string, type: string) {
return join(
targetDir,
normalizeViteUrl(url, type)
.replace(/^\//, '')
// `\0plugin-vue:export-helper` EXPORT_HELPER_ID
.replace(/\0/g, '__x00__')
// filenames starting with "_" are reserved for use by the system.
.replace(/^_+/, match => '~'.repeat(match.length)),
)
}