Basic environment construction

Build mini-VUe3 basic development environment all according to vuE-next configuration for operation, simplify to achieve a simple, usable, efficient environment!

The following directories are created

Build - your own - vue3 ├ ─. Gitignore ├ ─ doc │ └ ─ init. Md ├ ─ examples │ └ ─ index. The HTML ├ ─ package. The json ├ ─ packages │ ├ ─ Reactivity │ │ ├ ─ package. Json │ │ └ ─ SRC │ │ └ ─ but ts │ ├ ─ size - check │ │ └ ─ package. The json │ └ ─ vue │ ├ ─ index. The js │ ├ ─ package. Json │ └ ─ SRC │ ├ ─ but ts │ └ ─ runtime. Ts ├ ─ README. Md ├ ─ a rollup. Config. Js ├ ─ scripts │ ├ ─ dev. Js │ └ ─ ├─ json ├─ crimpCopy the code

1. Create the basic configuration

1. Project initialization

yarn init
Copy the code
{
  "private": true.// Private property, which is ignored when packaging
  "version": "0.1.0 from"."workspaces": [
    // NPM multipackage management, using working directories under Packages
    "packages/*"]."description": "vue"."repository": "https://github.com/hefeng6500/build-your-own-vue3"."author": "hefeng6500 <[email protected]>"."license": "MIT"
}
Copy the code

2. Install dependency packages

yarn add @babel/core @babel/preset-env @rollup/plugin-babel @rollup/plugin-json chalk execa minimist rollup rollup-plugin-typescript2 -D -W
Copy the code

Configure the Babel

module.exports = {
  input: {},
  output: {},
  plugins: [
    tsPlugin,
    getBabelOutputPlugin({
      allowAllFormats: true.presets: ["@babel/preset-env"],}),]};Copy the code

Tsconfig. json is generated in the initial TS configuration root directory

yarn global add typescript

tsc --init
Copy the code

Modify the tsconfig. Json

{
  "target": "esnext"."module": "esnext"
}
Copy the code
const tsPlugin = ts({
  tsconfig: path.resolve(__dirname, "tsconfig.json"),
  cacheRoot: path.resolve(__dirname, "node_modules/.rts2_cache")});module.exports = {
  plugins: [tsPlugin],
};
Copy the code

Add the gitignore

dist
.DS_Store
node_modules
coverage
temp
explorations
TODOs.md
*.log
Copy the code

2. Build the development environment script

Create scripts/dev. Js scripts/utils. Js

// utils.js
const fs = require("fs");
const chalk = require('chalk');

// Read directories that need to be packed in packages directory
const targets = (exports.targets = fs.readdirSync('packages').filter(f= > {
  if(! fs.statSync(`packages/${f}`).isDirectory()) {
    return false
  }
  const pkg = require(`.. /packages/${f}/package.json`)
  if(pkg.private && ! pkg.buildOptions) {return false
  }
  return true
}))

// Fuzzy match the package name to be packaged
exports.fuzzyMatchTarget = (partialTargets, includeAllMatching) = > {
  const matched = []
  partialTargets.forEach(partialTarget= > {
    for (const target of targets) {
      if (target.match(partialTarget)) {
        matched.push(target)
        if(! includeAllMatching) {break}}}})if (matched.length) {
    return matched
  } else {
    console.log()
    console.error(
      `  ${chalk.bgRed.white(' ERROR ')} ${chalk.red(
        `Target ${chalk.underline(partialTargets)}not found! `
      )}`
    )
    console.log()

    process.exit(1)}}Copy the code

The targets exported is an array of package names in the packages directory, for example, vue and reactivity.

FuzzyMatchTarget Fuzzy matches the name of the target package to be packed and returns the matched package name. If no package is matched, the Chalk plugin will be used to highlight an error message

// dev.js
const fs = require("fs")
const execa = require('execa');
const { fuzzyMatchTarget } = require('./utils')

const args = require('minimist')(process.argv.slice(2))
const target = args._.length ? fuzzyMatchTarget(args._)[0] : 'reactivity'
const formats = args.formats || args.f
const sourceMap = args.sourcemap || args.s
// const commit = execa.sync('git', ['rev-parse', 'HEAD']).stdout.slice(0, 7)

execa(
  'rollup'['-wc'.'--environment'[// `COMMIT:${commit}`,
      `TARGET:${target}`.`FORMATS:${formats || 'global'}`,
      sourceMap ? `SOURCE_MAP:true` : ` `
    ]
      .filter(Boolean)
      .join(', ')] and {stdio: 'inherit'})Copy the code

Execa is a plug-in for managing node command-line execution, and Minimist is a plug-in for formatting command parameter options. Dev.js is packaged using execa management rollup, and -w listens for files in real time for hot updates

3. Configure rollup

import path from "path";
import ts from "rollup-plugin-typescript2";
import json from "@rollup/plugin-json";
import { getBabelOutputPlugin } from "@rollup/plugin-babel";

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

const outputConfigs = {
  "esm-bundler": {
    file: resolve(`dist/${name}.esm-bundler.js`),
    format: `es`,},"esm-browser": {
    file: resolve(`dist/${name}.esm-browser.js`),
    format: `es`,},cjs: {
    file: resolve(`dist/${name}.cjs.js`),
    format: `cjs`,},global: {
    file: resolve(`dist/${name}.global.js`),
    format: `iife`,},// runtime-only builds, for main "vue" package only
  "esm-bundler-runtime": {
    file: resolve(`dist/${name}.runtime.esm-bundler.js`),
    format: `es`,},"esm-browser-runtime": {
    file: resolve(`dist/${name}.runtime.esm-browser.js`),
    format: "es",},"global-runtime": {
    file: resolve(`dist/${name}.runtime.global.js`),
    format: "iife",}};// const defaultFormats = ["esm-bundler", "cjs"];
// const inlineFormats = process.env.FORMATS && process.env.FORMATS.split(",");

const packageFormats = packageOptions.formats;

const tsPlugin = ts({
  tsconfig: path.resolve(__dirname, "tsconfig.json"),
  cacheRoot: path.resolve(__dirname, "node_modules/.rts2_cache")});const packageConfigs = packageFormats.map((format) = >
  createConfig(format, outputConfigs[format])
);

export default packageConfigs;

function createConfig(format, output, plugins = []) {
  let entryFile = /runtime$/.test(format) ? `src/runtime.ts` : `src/index.ts`;

  const isGlobalBuild = /global/.test(format)
  
  if (isGlobalBuild) {
    output.name = packageOptions.name
  }

  return {
    input: resolve(entryFile),
    output,
    plugins: [
      json({
        namedExports: false,
      }),
      tsPlugin,
      getBabelOutputPlugin({
        allowAllFormats: true.presets: ["@babel/preset-env"],}),]}; }Copy the code

The outputConfigs object stores the rollup export object name and the file export module specification; For more front-end module specification basics, please visit: Front-end modularity: CommonJS,AMD,CMD,ES6

Format The following formats are available:

  • Es: es module is the ES module specification. It uses the import and export keywords to import and export modules
  • CJS: CommonJS specification that imports and exports modules using the module.exports and require keywords
  • Iife: Call the function expression immediately. Is a self-executing function that, when packaged, assigns the result to a global variable, like this:
var VueReactivity = function () {
  'use strict';

  function effect() {
    console.log(123);
  }

  returneffect; } ();Copy the code

Export Default packageConfigs; PackageFormats are file formats that a single NPM package configuration file needs to be exported, such as package.json under Package /vue:

{
  "name": "vue"."version": "0.1.0 from"."main": "index.js"."buildOptions": {
    "name": "Vue"."formats": [
      "esm-bundler"."esm-bundler-runtime"."cjs"."global"."global-runtime"."esm-browser"."esm-browser-runtime"]}}Copy the code

PackageConfigs packages various modules defined in formats and outputs multiple packaged files

Note: rollup.config.js is currently configured to package multiple export formats under a single NPM package, not multiple NPM packages