-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
13 changed files
with
4,374 additions
and
90 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export default () => <div>23411dd</div>; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
import { defineComponent } from 'vue'; | ||
import A from './A'; | ||
import { B } from './B'; | ||
|
||
const App = defineComponent({ | ||
data() { | ||
return { | ||
a: 1 | ||
} | ||
}, | ||
render() { | ||
const { a } = this; | ||
return ( | ||
<> | ||
{a} | ||
<div onClick={() => { this.a++; }}>Hello World!!</div> | ||
<A /> | ||
<B /> | ||
</> | ||
) | ||
} | ||
}); | ||
|
||
export default App; | ||
|
||
if (module.hot) { | ||
App.__hmrId = "${id}" | ||
const api = __VUE_HMR_RUNTIME__ | ||
module.hot.accept(); | ||
if (!api.createRecord('${id}', App)) { | ||
api.reload('${id}', App) | ||
} | ||
api.rerender('${id}', App.render); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,43 @@ | ||
import { defineComponent } from 'vue'; | ||
|
||
const cache = {} | ||
|
||
const B = defineComponent({ | ||
// setup() { | ||
// const a = ref(0) | ||
// return () => <div onClick={() => { a.value++; }}>{a.value}dssdddssd</div> | ||
// } | ||
data() { | ||
return { | ||
a: 1 | ||
} | ||
}, | ||
render() { | ||
const { a } = this; | ||
return ( | ||
<> | ||
<div onClick={() => { this.a++; }}>{a}d4s</div> | ||
<span>23dd</span> | ||
</> | ||
); | ||
} | ||
}); | ||
|
||
export { | ||
B | ||
}; | ||
|
||
if (module.hot) { | ||
B.__hmrId = "b" | ||
const api = __VUE_HMR_RUNTIME__ | ||
module.hot.accept(); | ||
if (!module.hot.data) { | ||
api.createRecord('b', B); | ||
} else if (cache.b === B.render) { | ||
api.rerender('b', B.render); | ||
} else { | ||
api.reload('b', B) | ||
} | ||
|
||
cache.b = JSON.stringify(B.render); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
import { createApp } from 'vue'; | ||
import App from './App'; | ||
|
||
createApp(App).mount('#app'); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
<!DOCTYPE html> | ||
<html lang="en"> | ||
<head> | ||
<meta charset="UTF-8"> | ||
<title>Demo</title> | ||
</head> | ||
<body> | ||
<div id="app"></div> | ||
<script src="/dist/app.js"></script> | ||
</body> | ||
</html> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,12 +1,133 @@ | ||
import webpack from 'webpack'; | ||
import * as parser from "@babel/parser"; | ||
import traverse from "@babel/traverse"; | ||
import * as webpack from 'webpack'; | ||
import * as hash from 'hash-sum'; | ||
import * as path from 'path'; | ||
import * as loaderUtils from 'loader-utils'; | ||
import * as t from '@babel/types'; | ||
import traverse from '@babel/traverse'; | ||
import { File } from '@babel/types' | ||
import { parse } from "@babel/parser"; | ||
import { isDefineComponentCall, parseComponentDecls } from './utils'; | ||
|
||
const hasJSX = (file: File) => { | ||
let fileHasJSX = false; | ||
traverse(file, { | ||
JSXElement(path) { | ||
fileHasJSX = true; | ||
path.stop(); | ||
}, | ||
JSXFragment(path) { | ||
fileHasJSX = true; | ||
path.stop(); | ||
}, | ||
}); | ||
|
||
return fileHasJSX; | ||
}; | ||
|
||
export default function loader( | ||
this: webpack.loader.LoaderContext, | ||
source: string | ||
): string { | ||
source: string, | ||
sourceMap: string | ||
) { | ||
const loaderContext = this; | ||
|
||
return source; | ||
loaderContext.cacheable?.(); | ||
const isDev = process.env.NODE_ENV === 'development'; | ||
|
||
if (isDev) { | ||
loaderContext.callback(null, source, sourceMap); | ||
} | ||
|
||
const file = parse(source); | ||
|
||
if (!hasJSX(file)) { | ||
loaderContext.callback(null, source, sourceMap); | ||
return; | ||
} | ||
|
||
const webpackRemainingChain = loaderUtils.getRemainingRequest(loaderContext).split('!'); | ||
const fullPath = webpackRemainingChain[webpackRemainingChain.length - 1]; | ||
const filename = path.relative(process.cwd(), fullPath); | ||
|
||
const declaredComponents: { name: string }[] = []; | ||
const hotComponents: { | ||
local: string; | ||
exported: string; | ||
id: string; | ||
}[] = []; | ||
let defaultIdentifier: t.Identifier | null = null; | ||
|
||
traverse(file, { | ||
VariableDeclaration(nodePath) { | ||
declaredComponents.push(...parseComponentDecls(nodePath.node)); | ||
}, | ||
ExportNamedDeclaration(nodePath) { | ||
const { specifiers = [], declaration } = nodePath.node; | ||
if (t.isVariableDeclaration(declaration)) { | ||
hotComponents.push(...parseComponentDecls(declaration).map(({ name }) => ({ | ||
local: name, | ||
exported: name, | ||
id: hash(`${filename}-${name}`), | ||
}))); | ||
} else if (specifiers.length) { | ||
for (const spec of specifiers) { | ||
if (t.isExportSpecifier(spec) && t.isIdentifier(spec.exported)) { | ||
if (declaredComponents.find(d => d.name === spec.local.name)) { | ||
hotComponents.push({ | ||
local: spec.local.name, | ||
exported: spec.exported.name, | ||
id: hash(`${filename}-${spec.exported.name}`) | ||
}); | ||
} | ||
} | ||
} | ||
} | ||
}, | ||
ExportDefaultDeclaration(nodePath) { | ||
const { declaration } = nodePath.node; | ||
if (t.isIdentifier(declaration)) { | ||
if (declaredComponents.find(d => d.name === declaration.name)) { | ||
hotComponents.push({ | ||
local: declaration.name, | ||
exported: 'default', | ||
id: hash(`${filename}-default`) | ||
}) | ||
} | ||
} else if (isDefineComponentCall(declaration)) { | ||
defaultIdentifier = nodePath.scope.generateUidIdentifier('default') | ||
hotComponents.push({ | ||
local: defaultIdentifier.name, | ||
exported: 'default', | ||
id: hash(`${filename}-default`) | ||
}); | ||
} | ||
} | ||
}); | ||
|
||
if (hotComponents.length) { | ||
if (defaultIdentifier) { | ||
const { name } = defaultIdentifier as t.Identifier; | ||
source.replace( | ||
/export default defineComponent/g, | ||
`const ${name} = defineComponent` | ||
) + `\nexport default ${name}` | ||
} | ||
|
||
let callbackCode = ''; | ||
for (const { local, exported, id } of hotComponents) { | ||
source += | ||
`\n${local}.__hmrId = '${id}'` + | ||
`\n__VUE_HMR_RUNTIME__.createRecord('${id}', ${local})` | ||
callbackCode += `\n__VUE_HMR_RUNTIME__.reload("${id}", __${exported})` | ||
} | ||
|
||
source += ` | ||
/* hot reload */ | ||
if (module.hot) { | ||
module.hot.accept() | ||
${callbackCode} | ||
} | ||
` | ||
} | ||
|
||
loaderContext.callback(null, source, sourceMap); | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
declare module 'hash-sum' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,19 @@ | ||
export function isFuntionalComponent( | ||
code: string | ||
): boolean { | ||
return !!code; | ||
import { Node } from '@babel/core'; | ||
import * as t from '@babel/types'; | ||
|
||
export function isDefineComponentCall(node?: Node | null) { | ||
return t.isCallExpression(node) && t.isIdentifier(node.callee) && node.callee.name === 'defineComponent'; | ||
} | ||
|
||
export function parseComponentDecls(node: t.VariableDeclaration) { | ||
const names = []; | ||
for (const decl of node.declarations) { | ||
if (t.isIdentifier(decl.id) && isDefineComponentCall(decl.init)) { | ||
names.push({ | ||
name: decl.id.name | ||
}); | ||
} | ||
} | ||
|
||
return names; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
const path = require('path'); | ||
|
||
const babelConfig = { | ||
plugins: [ | ||
'@vue/babel-plugin-jsx' | ||
], | ||
} | ||
|
||
module.exports = { | ||
mode: 'development', | ||
entry: { | ||
app: './examples/index.js', | ||
}, | ||
output: { | ||
path: path.resolve(__dirname, './dist'), | ||
publicPath: '/dist/', | ||
}, | ||
module: { | ||
rules: [ | ||
{ | ||
test: /\.(js|jsx)$/, | ||
loader: 'babel-loader', | ||
options: babelConfig, | ||
} | ||
], | ||
}, | ||
devServer: { | ||
historyApiFallback: true, | ||
hot: true, | ||
open: true | ||
}, | ||
resolve: { | ||
extensions: ['.jsx', '.js'] | ||
} | ||
}; |
Oops, something went wrong.