UNPKG

one

Version:

One is a new React Framework that makes Vite serve both native and web.

85 lines (74 loc) 2.77 kB
/** * The MIT License (MIT) * * Copyright (c) 2023 Daishi Kato * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. **/ import { extname } from 'node:path' import * as swc from '@swc/core' import type { Plugin } from 'vite' // PLAN // see if we can use react server dom for the bridge between 'use dom' and native // https://github.com/dai-shi/waku/blob/82dea924457fba1c60a9dfb071cb69dc44d7bbf2/packages/waku/src/lib/plugins/vite-plugin-rsc-transform.ts#L143 export function useDOMPlugin(): Plugin { return { name: 'one-vite-dom-plugin', async transform(code, id, options) { if (!code.includes('use dom')) { return } const ext = extname(id) const mod = swc.parseSync(code, parseOpts(ext)) let hasUseDom = false for (let i = 0; i < mod.body.length; ++i) { const item = mod.body[i]! if (item.type === 'ExpressionStatement') { if (item.expression.type === 'StringLiteral') { if (item.expression.value === 'use dom') { hasUseDom = true break } } } else { // HACK we can't stop the loop here, because vite may put some import statements before the directives // break; } } if (!hasUseDom) { return } // does have use dom - lets transform }, } } const parseOpts = (ext: string) => { if (ext === '.ts' || ext === '.tsx') { return { syntax: 'typescript', tsx: ext.endsWith('x'), } as const } // We hoped to use 'typescript' for everything, but it fails in some cases. // https://github.com/dai-shi/waku/issues/677 return { syntax: 'ecmascript', jsx: ext.endsWith('x'), } as const }