成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

用 nodejs 寫一個(gè)命令行工具 :創(chuàng)建 react 組件的命令行工具

WelliJhon / 1046人閱讀

摘要:用寫一個(gè)命令行工具創(chuàng)建組件的命令行工具前言上周,同事抱怨說(shuō)怎么不能像那樣,使用命令行工具來(lái)生成一個(gè)組件。為什么不能將這個(gè)過(guò)程交給程序去做呢當(dāng)天晚上,我就仿照的,寫了一個(gè)生成組件的命令行工具。

用 nodejs 寫一個(gè)命令行工具 :創(chuàng)建 react 組件的命令行工具 前言

上周,同事抱怨說(shuō) react 怎么不能像 angular 那樣,使用命令行工具來(lái)生成一個(gè)組件。對(duì)呀,平時(shí)工作時(shí),想要?jiǎng)?chuàng)建一個(gè) react 的組件,都是直接 copy 一個(gè)組件,然后做一些修改。為什么不能將這個(gè)過(guò)程交給程序去做呢?當(dāng)天晚上,我就仿照 angular-cli 的 api,寫了一個(gè)生成 react 組件的命令行工具 rcli。在這里記錄一下實(shí)現(xiàn)的過(guò)程。

api 設(shè)計(jì) 0.1.0 版本的 rcli 參照 angular-cli 的設(shè)計(jì),有兩個(gè)功能:

使用 rcli new PROJECT-NAME 命令,創(chuàng)建一個(gè) react 項(xiàng)目,其中生成項(xiàng)目的腳手架當(dāng)然是 create-react-app 啦

使用 rcli g component MyComponent 命令, 創(chuàng)建一個(gè) MyComponent 組件, 這個(gè)組件是一個(gè)文件夾,在文件夾中包含 index.js、MyComponent.js、MyComponent.css 三個(gè)文件

后來(lái)發(fā)現(xiàn) rcli g component MyComponent 命令在 平時(shí)開(kāi)發(fā)過(guò)程中是不夠用的,因?yàn)檫@個(gè)命令只是創(chuàng)建了一個(gè)類組件,且繼承自 React.Component

在平時(shí)開(kāi)發(fā) 過(guò)程中,我們會(huì)用到這三類組件:

繼承自 React.Component 的類組件

繼承自 React.PureComponent 的類組件

函數(shù)組件(無(wú)狀態(tài)組件)

注: 將來(lái)可以使用 Hooks 來(lái)代替之前的類組件

于是就有了 0.2.0 版本的 rcli

0.2.0 版本的 rcli 用法
Usage: rcli [command] [options]

Commands:
  new 
  g 

`new` command options:
  -n, --use-npm                    Whether to use npm to download dependencies

`g` command options:
  -c, --component   The name of the component
  --no-folder                      Whether the component have not it"s own folder
  -p, --pure-component             Whether the component is a extend from PureComponent
  -s, --stateless                  Whether the component is a stateless component
使用 create-react-app 來(lái)創(chuàng)建一個(gè)應(yīng)用
rcli new PROJECT-NAME
cd PROJECT-NAME
yarn start

或者你可以使用 npm 安裝依賴

rcli new PROJECT-NAME --use-npm
cd PROJECT-NAME
npm start
生成純組件(繼承自 PureComponent,以提高性能)
rcli g -c MyNewComponent -p
生成類組件(有狀態(tài)組件)
rcli g -c MyNewComponent

等于:

rcli g -c ./MyNewComponent
生成函數(shù)組件(無(wú)狀態(tài)組件)
rcli g -c MyNewComponent -s
生成組件不在文件夾內(nèi)(也不包含 css 文件和 index.js 文件)
# 默認(rèn)生成的組件都會(huì)都包含在文件夾中的,若不想生成的組件被文件夾包含,則加上 --no-folder 選項(xiàng)
rcli g -c MyNewComponent --no-folder
實(shí)現(xiàn)過(guò)程 1. 創(chuàng)建項(xiàng)目

創(chuàng)建名為 hileix-rcli 的項(xiàng)目

在項(xiàng)目根目錄使用 npm init -y 初始化一個(gè) npm package 的基本信息(即生成 package.json 文件)

在項(xiàng)目根創(chuàng)建 index.js 文件,用來(lái)寫用戶輸入命令后的主要邏輯代碼

編輯 package.json 文件,添加 bin 字段:

{
  "name": "hileix-rcli",
  "version": "0.2.0",
  "description": "",
  "main": "index.js",
  "bin": {
    "rcli": "./index.js"
  },
  "scripts": {
    "test": "echo "Error: no test specified" && exit 1"
  },
  "repository": {
    "type": "git",
    "url": "git+https://github.com/hileix/rcli.git"
  },
  "keywords": [],
  "author": "hileix <[email protected]> (https://github.com/hileix)",
  "license": "MIT",
  "bugs": {
    "url": "https://github.com/hileix/rcli/issues"
  },
  "homepage": "https://github.com/hileix/rcli#readme",
  "dependencies": {
    "chalk": "^2.4.1",
    "commander": "^2.19.0",
    "cross-spawn": "^6.0.5",
    "fs-extra": "^7.0.1"
  }
}

在項(xiàng)目根目錄下,使用 npm link 命令,創(chuàng)建軟鏈接指向到本項(xiàng)目的 index.js 文件。這樣,就能再開(kāi)發(fā)的時(shí)候,直接使用 rcli 命令直接進(jìn)行測(cè)試 ~

2. rcli 會(huì)依賴一些包:

commander:tj 大神寫的一款專門處理命令行的工具。主要用來(lái)解析用戶輸入的命令、選項(xiàng)

cross-spawn:nodejs spawn 的跨平臺(tái)的版本。主要用來(lái)創(chuàng)建子進(jìn)程執(zhí)行一些命令

chalk:給命令行中的文字添加樣式。

path:nodejs path 模塊

fs-extra:提供對(duì)文件操作的方法

3.實(shí)現(xiàn) rcli new PROJECT-NAME
#!/usr/bin/env node

"use strict";

const program = require("commander");
const log = console.log;

// new command
program
  // 定義 new 命令,且后面跟一個(gè)必選的 projectName 參數(shù)
  .command("new ")
  // 對(duì) new 命令的描述
  .description("use create-react-app create a app")
  // 定義使用 new 命令之后可以使用的選項(xiàng) -n(使用 npm 來(lái)安裝依賴)
  // 在使用 create-react-app 中,我們可以可以添加 --use-npm 選項(xiàng),來(lái)使用 npm 安裝依賴(默認(rèn)使用 yarn 安裝依賴)
  // 所以,我將這個(gè)選項(xiàng)添加到了 rcli 中
  .option("-n, --use-npm", "Whether to use npm to download dependencies")
  // 定義執(zhí)行 new 命令后調(diào)用的回調(diào)函數(shù)
  // 第一個(gè)參數(shù)便是在定義 new 命令時(shí)的必選參數(shù) projectName
  // cmd 中包含了命令中選項(xiàng)的值,當(dāng)我們?cè)?new 命令中使用了 --use-npm 選項(xiàng)時(shí),cmd 中的 useNpm 屬性就會(huì)為 true,否則為 undefined
  .action(function(projectName, cmd) {
    const isUseNpm = cmd.useNpm ? true : false;
    // 創(chuàng)建 react app
    createReactApp(projectName, isUseNpm);
  });

program.parse(process.argv);

/**
 * 使用 create-react-app 創(chuàng)建項(xiàng)目
 * @param {string} projectName 項(xiàng)目名稱
 * @param {boolean} isUseNpm 是否使用 npm 安裝依賴
 */
function createReactApp(projectName, isUseNpm) {
  let args = ["create-react-app", projectName];
  if (isUseNpm) {
    args.push("--use-npm");
  }
  // 創(chuàng)建子進(jìn)程,執(zhí)行 npx create-react-app PROJECT-NAME [--use-npm] 命令
  spawn.sync("npx", args, { stdio: "inherit" });
}

上面的代碼邊實(shí)現(xiàn)了 rcli new PROJECT-NAME 的功能:

#!/usr/bin/env node 表示使用 node 執(zhí)行本腳本

4.實(shí)現(xiàn) rcli g [options]
#!/usr/bin/env node

"use strict";
const program = require("commander");
const spawn = require("cross-spawn");
const chalk = require("chalk");
const path = require("path");
const fs = require("fs-extra");

const log = console.log;

program
  // 定義 g 命令
  .command("g")
  // 命令 g 的描述
  .description("Generate a component")
  // 定義 -c 選項(xiàng),接受一個(gè)必選參數(shù) componentName:組件名稱
  .option("-c, --component-name ", "The name of the component")
  // 定義 --no-folder 選項(xiàng):表示當(dāng)使用該選項(xiàng)時(shí),組件不會(huì)被文件夾包裹
  .option("--no-folder", "Whether the component have not it is own folder")
  // 定義 -p 選項(xiàng):表示當(dāng)使用該選項(xiàng)時(shí),組件為繼承自 React.PureComponent 的類組件
  .option(
    "-p, --pure-component",
    "Whether the component is a extend from PureComponent"
  )
  // 定義 -s 選項(xiàng):表示當(dāng)使用該選項(xiàng)時(shí),組件為無(wú)狀態(tài)的函數(shù)組件
  .option(
    "-s, --stateless",
    "Whether the component is a extend from PureComponent"
  )
  // 定義執(zhí)行 g 命令后調(diào)用的回調(diào)函數(shù)
  .action(function(cmd) {
    // 當(dāng) -c 選項(xiàng)沒(méi)有傳參數(shù)進(jìn)來(lái)時(shí),報(bào)錯(cuò)、退出
    if (!cmd.componentName) {
      log(chalk.red("error: missing required argument `componentName`"));
      process.exit(1);
    }
    // 創(chuàng)建組件
    createComponent(
      cmd.componentName,
      cmd.folder,
      cmd.stateless,
      cmd.pureComponent
    );
  });

program.parse(process.argv);

/**
 * 創(chuàng)建組件
 * @param {string} componentName 組件名稱
 * @param {boolean} hasFolder 是否含有文件夾
 * @param {boolean} isStateless 是否是無(wú)狀態(tài)組件
 * @param {boolean} isPureComponent 是否是純組件
 */
function createComponent(
  componentName,
  hasFolder,
  isStateless = false,
  isPureComponent = false
) {
  let dirPath = path.join(process.cwd());
  // 組件在文件夾中
  if (hasFolder) {
    dirPath = path.join(dirPath, componentName);

    const result = fs.ensureDirSync(dirPath);
    // 目錄已存在
    if (!result) {
      log(chalk.red(`${dirPath} already exists`));
      process.exit(1);
    }
    const indexJs = getIndexJs(componentName);
    const css = "";
    fs.writeFileSync(path.join(dirPath, `index.js`), indexJs);
    fs.writeFileSync(path.join(dirPath, `${componentName}.css`), css);
  }
  let component;
  // 無(wú)狀態(tài)組件
  if (isStateless) {
    component = getStatelessComponent(componentName, hasFolder);
  } else {
    // 有狀態(tài)組件
    component = getClassComponent(
      componentName,
      isPureComponent ? "React.PureComponent" : "React.Component",
      hasFolder
    );
  }

  fs.writeFileSync(path.join(dirPath, `${componentName}.js`), component);
  log(
    chalk.green(`The ${componentName} component was successfully generated!`)
  );
  process.exit(1);
}

/**
 * 獲取類組件字符串
 * @param {string} componentName 組件名稱
 * @param {string} extendFrom 繼承自:"React.Component" | "React.PureComponent"
 * @param {boolean} hasFolder 組件是否在文件夾中(在文件夾中的話,就會(huì)自動(dòng)加載 css 文件)
 */
function getClassComponent(componentName, extendFrom, hasFolder) {
  return "省略...";
}

/**
 * 獲取無(wú)狀態(tài)組件字符串
 * @param {string} componentName 組件名稱
 * @param {boolean} hasFolder 組件是否在文件夾中(在文件夾中的話,就會(huì)自動(dòng)加載 css 文件)
 */
function getStatelessComponent(componentName, hasFolder) {
  return "省略...";
}

/**
 * 獲取 index.js 文件內(nèi)容
 * @param {string} componentName 組件名稱
 */
function getIndexJs(componentName) {
  return `import ${componentName} from "./${componentName}";
export default ${componentName};
`;
}

這樣就實(shí)現(xiàn)了 rcli g [options] 命令的功能

總結(jié)

api 設(shè)計(jì)是很重要的:好的 api 設(shè)計(jì)能讓使用者更加方便地使用,且變動(dòng)少

當(dāng)自己想不到該怎么設(shè)計(jì) api 時(shí),可以參考別人的 api,看看別人是怎么設(shè)計(jì)的好用的

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/100082.html

相關(guān)文章

  • GitHub 值得收藏前端項(xiàng)目[每月更新...]

    摘要:也是一款優(yōu)秀的響應(yīng)式框架站點(diǎn)所使用的一套框架為微信服務(wù)量身設(shè)計(jì)的一套框架一組很小的,響應(yīng)式的組件,你可以在網(wǎng)頁(yè)的項(xiàng)目上到處使用一個(gè)可定制的文件,使瀏覽器呈現(xiàn)的所有元素,更一致和符合現(xiàn)代標(biāo)準(zhǔn)。 GitHub 值得收藏的前端項(xiàng)目 整理與收集的一些比較優(yōu)秀github項(xiàng)目,方便自己閱讀,順便分享出來(lái),大家一起學(xué)習(xí),本篇文章會(huì)持續(xù)更新,版權(quán)歸原作者所有。歡迎github star與fork 預(yù)...

    maxmin 評(píng)論0 收藏0
  • webpack工程化集成React技術(shù)棧(一)

    項(xiàng)目開(kāi)始前,我們先聊一聊關(guān)于項(xiàng)目的一些說(shuō)明。該項(xiàng)目起始于2017年初,當(dāng)時(shí)公司主要技術(shù)棧為gulp+angular,鑒于react的火熱的生態(tài),在公司決定研發(fā)bss管理系統(tǒng)時(shí)選用react開(kāi)發(fā),目的也是為react native打下基礎(chǔ),以解決后期公司大前端技術(shù)棧的逐步成熟。(當(dāng)時(shí)沒(méi)有選擇vue開(kāi)發(fā)的主要原因是weex生態(tài)還不夠特別成熟),既然決定換新,項(xiàng)目的構(gòu)建也跟著一起換,從gulp轉(zhuǎn)向火熱的...

    tianhang 評(píng)論0 收藏0
  • react+express項(xiàng)目

    摘要:暴露所有內(nèi)建配置,項(xiàng)目下會(huì)新增或?qū)Σ糠峙渲梦募M(jìn)行修改。開(kāi)發(fā)環(huán)境開(kāi)發(fā)時(shí),前端項(xiàng)目和后端項(xiàng)目運(yùn)行時(shí)端口端口不同,存在跨域問(wèn)題。項(xiàng)目目錄結(jié)構(gòu)優(yōu)化項(xiàng)目目錄結(jié)構(gòu)優(yōu)化開(kāi)發(fā)目錄主要是目錄,因此需要修改的目錄主要是目錄。 1 開(kāi)發(fā)環(huán)境準(zhǔn)備(windows) 1.1 安裝nodejs和npm 1) 下載nodejs安裝包:http://nodejs.org/en/download/ nodejs安...

    Luosunce 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<