0%

Vue 3 Pre-Alpha 源码调试

资源

Vue 3 Pre-Alpha 源码调试步骤

  1. clone 源码

    1
    git clone https://github.com/vuejs/vue-next.git
  2. 下载依赖

    1
    2
    cd vue-next
    yarn
  1. 启用 sourcemap
    将根目录下的 rollup.config.js 配置文件内容替换为:
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138
    139
    140
    141
    142
    143
    144
    145
    146
    147
    148
    149
    150
    151
    152
    153
    154
    155
    156
    157
    158
    159
    160
    161
    162
    163
    164
    165
    166
    167
    168
    169
    170
    171
    172
    173
    174
    175
    176
    177
    178
    const fs = require('fs')
    const path = require('path')
    const ts = require('rollup-plugin-typescript2')
    const replace = require('rollup-plugin-replace')
    const alias = require('rollup-plugin-alias')
    const json = require('rollup-plugin-json')

    if (!process.env.TARGET) {
    throw new Error('TARGET package must be specified via --environment flag.')
    }

    const packagesDir = path.resolve(__dirname, 'packages')
    const packageDir = path.resolve(packagesDir, process.env.TARGET)
    const name = path.basename(packageDir)
    const resolve = p => path.resolve(packageDir, p)
    const pkg = require(resolve(`package.json`))
    const packageOptions = pkg.buildOptions || {}

    // build aliases dynamically
    const aliasOptions = { resolve: ['.ts'] }
    fs.readdirSync(packagesDir).forEach(dir => {
    if (dir === 'vue') {
    return
    }
    if (fs.statSync(path.resolve(packagesDir, dir)).isDirectory()) {
    aliasOptions[`@vue/${dir}`] = path.resolve(packagesDir, `${dir}/src/index`)
    }
    })
    const aliasPlugin = alias(aliasOptions)

    // ensure TS checks only once for each build
    let hasTSChecked = false

    const configs = {
    esm: {
    file: resolve(`dist/${name}.esm-bundler.js`),
    format: `es`
    },
    cjs: {
    file: resolve(`dist/${name}.cjs.js`),
    format: `cjs`
    },
    global: {
    file: resolve(`dist/${name}.global.js`),
    format: `iife`
    },
    'esm-browser': {
    file: resolve(`dist/${name}.esm-browser.js`),
    format: `es`
    }
    }

    const defaultFormats = ['esm', 'cjs']
    const inlineFormats = process.env.FORMATS && process.env.FORMATS.split(',')
    const packageFormats = inlineFormats || packageOptions.formats || defaultFormats
    const packageConfigs = process.env.PROD_ONLY
    ? []
    : packageFormats.map(format => createConfig(configs[format]))

    if (process.env.NODE_ENV === 'production') {
    packageFormats.forEach(format => {
    if (format === 'cjs') {
    packageConfigs.push(createProductionConfig(format))
    }
    if (format === 'global' || format === 'esm-browser') {
    packageConfigs.push(createMinifiedConfig(format))
    }
    })
    }

    module.exports = packageConfigs

    function createConfig(output, plugins = []) {
    // 启用 sourcemap
    output.sourcemap = true

    const isProductionBuild =
    process.env.__DEV__ === 'false' || /\.prod\.js$/.test(output.file)
    const isGlobalBuild = /\.global(\.prod)?\.js$/.test(output.file)
    const isBundlerESMBuild = /\.esm\.js$/.test(output.file)
    const isBrowserESMBuild = /esm-browser(\.prod)?\.js$/.test(output.file)

    if (isGlobalBuild) {
    output.name = packageOptions.name
    }

    const shouldEmitDeclarations =
    process.env.TYPES != null &&
    process.env.NODE_ENV === 'production' &&
    !hasTSChecked

    const tsPlugin = ts({
    check: process.env.NODE_ENV === 'production' && !hasTSChecked,
    tsconfig: path.resolve(__dirname, 'tsconfig.json'),
    cacheRoot: path.resolve(__dirname, 'node_modules/.rts2_cache'),
    tsconfigOverride: {
    compilerOptions: {
    declaration: shouldEmitDeclarations,
    declarationMap: shouldEmitDeclarations
    },
    exclude: ['**/__tests__']
    }
    })
    // we only need to check TS and generate declarations once for each build.
    // it also seems to run into weird issues when checking multiple times
    // during a single build.
    hasTSChecked = true

    const externals = Object.keys(aliasOptions).filter(p => p !== '@vue/shared')

    return {
    input: resolve(`src/index.ts`),
    // Global and Browser ESM builds inlines everything so that they can be
    // used alone.
    external: isGlobalBuild || isBrowserESMBuild ? [] : externals,
    plugins: [
    json({
    namedExports: false
    }),
    tsPlugin,
    aliasPlugin,
    createReplacePlugin(
    isProductionBuild,
    isBundlerESMBuild,
    (isGlobalBuild || isBrowserESMBuild) &&
    !packageOptions.enableNonBrowserBranches
    ),
    ...plugins
    ],
    output,
    onwarn: (msg, warn) => {
    if (!/Circular/.test(msg)) {
    warn(msg)
    }
    }
    }
    }

    function createReplacePlugin(isProduction, isBundlerESMBuild, isBrowserBuild) {
    return replace({
    __COMMIT__: `"${process.env.COMMIT}"`,
    __DEV__: isBundlerESMBuild
    ? // preserve to be handled by bundlers
    `process.env.NODE_ENV !== 'production'`
    : // hard coded dev/prod builds
    !isProduction,
    // If the build is expected to run directly in the browser (global / esm-browser builds)
    __BROWSER__: isBrowserBuild,
    // support options?
    // the lean build drops options related code with buildOptions.lean: true
    __FEATURE_OPTIONS__: !packageOptions.lean,
    __FEATURE_SUSPENSE__: true,
    // this is only used during tests
    __JSDOM__: false
    })
    }

    function createProductionConfig(format) {
    return createConfig({
    file: resolve(`dist/${name}.${format}.prod.js`),
    format: configs[format].format
    })
    }

    function createMinifiedConfig(format) {
    const { terser } = require('rollup-plugin-terser')
    return createConfig(
    {
    file: resolve(`dist/${name}.${format}.prod.js`),
    format: configs[format].format
    },
    [
    terser({
    module: /^esm/.test(format)
    })
    ]
    )
    }
    将根目录下的 tsconfig.json typescript 配置文件内容替换为:
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    {
    "compilerOptions": {
    "baseUrl": ".",
    "outDir": "dist",
    "sourceMap": true,
    "target": "esnext",
    "module": "esnext",
    "moduleResolution": "node",
    "allowJs": false,
    "noUnusedLocals": true,
    "strictNullChecks": true,
    "noImplicitAny": true,
    "noImplicitThis": true,
    "experimentalDecorators": true,
    "resolveJsonModule": true,
    "esModuleInterop": true,
    "removeComments": false,
    "jsx": "react",
    "lib": ["esnext", "dom"],
    "types": ["jest", "node"],
    "rootDir": ".",
    "paths": {
    "@vue/shared": ["packages/shared/src"],
    "@vue/runtime-core": ["packages/runtime-core/src"],
    "@vue/runtime-dom": ["packages/runtime-dom/src"],
    "@vue/runtime-test": ["packages/runtime-test/src"],
    "@vue/reactivity": ["packages/reactivity/src"],
    "@vue/compiler-core": ["packages/compiler-core/src"],
    "@vue/compiler-dom": ["packages/compiler-dom/src"],
    "@vue/server-renderer": ["packages/server-renderer/src"],
    "@vue/template-explorer": ["packages/template-explorer/src"]
    }
    },
    "include": [
    "packages/global.d.ts",
    "packages/runtime-dom/jsx.d.ts",
    "packages/*/src",
    "packages/*/__tests__"
    ]
    }
  2. 编译源码
    1
    yarn dev
  3. 调试
    vue-next/vue 目录下新建 index.html 文件, 内容为:
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    <!DOCTYPE html>
    <html lang="en">
    <head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>vue 3 test</title>
    <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bulma/0.7.5/css/bulma.min.css">
    </head>
    <body>
    <div id="app" class="content">
    </div>
    <div class="has-text-centered">
    <a id="button" class="button is-info is-rounded">Rounded</a>
    </div>
    <script src="./dist/vue.global.js">
    </script>
    <script>
    console.log(Vue)
    const { reactive, watch } = Vue

    const state = reactive({
    count: 0
    })

    const appEle = document.getElementById('app')
    watch(() => {
    const template = `<p class="has-text-centered">
    count is ${state.count}
    <p>`
    appEle.innerHTML = template
    })
    const increment = (e) => {
    state.count++
    }
    document.getElementById('button').addEventListener('click', increment)
    </script>
    </body>
    </html>

浏览器打开 index.html 即可调试.