打杂到一定时候,也要总结总结,4年前UI工作做得蛮多,目前开发工作接触的蛮多,其实就是为了项目的落地不求人,不知不觉学了些东西,闲暇之余做了一些东西。
今天有无数的React UI工具包和库。那面对这么多选择,面对新的项目挑战,我下一步应该做什么?是学习新框架,还是自己造轮子,还是去挖掘一些工具,总之,我需要一个思考,以便更好地面对未来更多的技术需求和革新。框架毕竟只是框架,抛开场景谈用途都不对,我应该思考如何从个人成长和项目需求,语言基本功的训练等方面来规划未来,规划这个开源工具的工作。这就是为什么在一年中这个繁忙的时间里,我仍然抽时间学习一些前端或后端的技术,它们如此重要。不过,首先我会考虑某些框架是否适合去采用。另外我特别关注2021年至2022年新冠病毒的大流行对工作对行业发展的各种影响。
下面是这个小玩意儿的预览:
Github资源:https://github.com/xizon/poemkit
PoemKit可以直接通过命令工作和部署到云服务器, 但仍然需要在其背后编写真正的代码,这些工具包作为UI组件和逻辑模块,可以组合成一个真正的应用程序原型。它仍然需要编程功能才能使它们在实际需求中工作。
这个项目的一个有趣的方面,其实是对webpack很强的自定义特性,如果你想快速用于产出,也可以直接导入蚂蚁组件等第三方组件库,为了进一步完善模型,PoemKit提供了50多个内置UI组件,用于构建现代Web应用程序。
先说说我的想法吧。
为了适应一些大型的网站平台和软件项目,我考虑了一些React框架,比如Nextjs,但是最终我决定暂时放弃它(只是暂时性的,对于未来的某些项目也可能直接采用),我想让网站的架构能够自由的操控,比如webpack脚手架的功能性,扩展性。一方面从零开始搭建webpack脚手架,也能够让我深入学习各方面的知识,让我系统的了解React网站的架构原理和实现方法,同时将它沉淀为开源的工具箱,能够在未来给我的项目提供必要的支持和参考。
如果进行数据传输,那么我将优先使用Redux来做跨组件状态管理器,未来根据项目需要也会使用一些便携的工具包。我将所有的UI组件都使用TypeScript来编写,一方面组件库系统化利于管理,一方面它们和整个React网站的逻辑区块不会冲突。它还要能够直接导入其它UI组件库,比如Ant Design. 于此同时,该工具箱需要内置自定义React组件,以方便集成并改善开发人员体验。
对于制作一个前后端完全分离的网站,我还需要考虑响应式和一些酷炫的交互效果,复杂的动画我使用了GSAP插件,并且让整个网站兼容Bootstrap和常见的图标库,以提供更加友好的样式支持。它和本身内置的样式和脚本不冲突,这是我对于项目交互复杂化的考虑。
最后需要考虑SEO的问题,我也会进一步解决。比如让整个React网站支持自定义的meta标签和title等,至于性能,组件之间的通信和挂载、卸载,都严格参考相关的官方文档,做了脚本优化的处理。
当然,对于一个学习人,我的代码和脚本并不是非常高端和优秀,但是整个开源产品让我的思维和整体技术得到了非常多的学习和提升,也让我对未来的新项目多了很多选择。我希望它能给大家带来一些用途,也非常感谢大家耐心阅读我的文字。
接着,我还需要为他取一个不错的名字,因为我喜欢WordPress,WordPress有一句谚语:Code is Poetry. 说的就是代码如同诗歌一般美妙。做这个工具包,我也是为了做出更加如诗美妙的网站而努力,所以PoemKit就算这么来的。
演示
服务器端运行 http://localhost:3000
介绍
- 基础环境: TypeScript 4.x.x + Babel 7.x.x + Webpack 5.x.x
- 提供结构化布局对象、UI控件等多种预置UI组件
- 可访问的URL、SEO优化,以及使用React Helmet使您的站点对搜索引擎友好
- 集开发、调试、打包和部署为一体
- 它不是一个JavaScript框架
- 没有jQuery且不绑定任何工具库
- 支持服务器端渲染(SSR)
- 组件分离,可以导入任何第三方UI组件到项目 (比如Ant Design)
- 使用Sass/SCSS来设置React组件的样式
- 自动捆绑并生成独立的核心CSS和JS文件
- 支持通过pm2自动部署到服务器
- ✂️ 插件扩展: 默认组件演示导入了诸如Bootstrap4栅格系统, GSAP动画库, FontAwesome图标库, 3D引擎等常用的第三方插件,以便能够快速扩展您的网站 (可以按需配置需要导入的库或者插件)
- ✂️ 可配置脚手架: 独立打包dist文件(您可以自主修改
webpack.config.js
使用内存挂载)
开发者基本操作:
- 配置你电脑的Node.js环境,推荐
Node 14+
版本,测试环境为v14.16.0 - 下载完资源后,进入到
poemkit
目录下,运行相关命令进行开发或者部署。 - 当你需要配置脚手架和网站基础信息和结构的时候,可编辑
package.json
文件。 - 项目文件会打包生成到
./dist/*
和./public/*.html
.
命令使用方法:
Step 1. 下载项目
$ sudo npm install poemkit
或者克隆项目:
$ git clone git://github.com/xizon/poemkit.git
Step 2. 使用命令进入 poemkit/
目录, {your_directory}换成你的目录路径
$ cd /{your_directory}/poemkit
Step 3. 安装所需要的依赖项
$ sudo npm install
Step 4. 常用的命令:
调试应用程序, 它可以用来单独检查TypeScript类型的文件而不进行编译操作,便于提高开发效率,专注整体代码的编写。
$ npm run check
打包生成CSS,JS,HTML文件
$ npm run build
运行下面的命令进行本地测试和代码检查 (仅运行Express服务器)。 使用 Ctrl + C
退出终端
$ npm run dev
Step 5. 浏览器中访问以下网址,可快速调试:
http://localhost:3000
建议在调试之前将新代码重新编译打包。
Step 6 (可选). 用PM2启动Reactjs应用程序(仅在使用Node v13.9.0或更高版本时有效)。
6.1) 安装Node和NPM
$ curl -sL https://rpm.nodesource.com/setup_14.x | sudo bash -
$ sudo yum install nodejs
$ node --version #v14.16.1
$ npm --version #6.14.12
$ which node babel-node #check the location of node and babel-node
6.2) 全局安装PM2
$ sudo npm install pm2@latest -g
6.3) 全局安装Babel
$ sudo npm install -g babel-cli
$ sudo npm install -g @babel/core @babel/cli @babel/preset-env
6.4) 全局安装TypeScript和ts-node
$ sudo npm install -g typescript ts-node
6.5) 使用PM2安装TypeScript依赖
$ sudo pm2 install typescript
6.6) PM2常用命令:
#先进入 `"poemkit/"` 目录
$ cd /{your_directory}/poemkit
#用pm2运行应用
$ pm2 start ecosystem.config.js
#其它命令
$ pm2 restart ecosystem.config.js
$ pm2 stop ecosystem.config.js
$ pm2 delete ecosystem.config.js
$ pm2 list
$ pm2 logs
6.7) 使用域名访问您的React应用。
需要在Apache或Nginx的Web服务器上部署React App。请参考网络以获取有关设置代理的教程。
Step 7 (可选). 单元测试
$ npm run test
Step 8 (可选). 部署到服务器
$ npm run deploy
撤销现有部署:
$ npm run destroy
⚙️ 温馨提示:
a) ERROR: npm update check failed.
如果出现如上错误,表示nodejs的环境或权限问题,可以使用下面的命令解决,注意把{username}换成你自己的设备里的名字。
$ sudo chown -R $USER:$(id -gn $USER) /Users/{username}/.config
b) ERROR: Node sass version 6.x.x is not compatible with ^ 4.x.x.
如果出现如上错误,请修改node-sass的版本:
$ npm install [email protected]
c) 如果升级Node版本,请执行以下代码:
$ sudo npm install
$ sudo npm rebuild node-sass
⚙️ 配置模块和别名:
您可以通过修改 webpack.config.js
的 resolve
属性来创建 import
或 require
的别名,来确保模块引入变得更简单.
webpack.config.js
:
...
const alias = {
pathConfig : './src/config',
pathComponents : './src/client/components',
pathRouter : './src/client/router',
pathHelpers : './src/client/helpers',
pathServices : './src/client/services',
pathReducers : './src/client/reducers',
pathPages : './src/client/views/_pages',
pathActions : './src/client/actions',
pathServer : './src/server',
pathStore : './src/store'
};
...
resolve: {
extensions: ['.js', '.es6', '.vue', '.jsx', '.ts', '.tsx' ],
alias: {
// 需要同时配置 `babel.config.js` 和 `tsconfig.json` 文件
'@poemkit/config': path.resolve(__dirname, alias.pathConfig ),
'@poemkit/components': path.resolve(__dirname, alias.pathComponents ),
'@poemkit/router': path.resolve(__dirname, alias.pathRouter ),
'@poemkit/helpers': path.resolve(__dirname, alias.pathHelpers ),
'@poemkit/services': path.resolve(__dirname, alias.pathServices ),
'@poemkit/reducers': path.resolve(__dirname, alias.pathReducers ),
'@poemkit/pages': path.resolve(__dirname, alias.pathPages ),
'@poemkit/actions': path.resolve(__dirname, alias.pathActions ),
'@poemkit/server': path.resolve(__dirname, alias.pathServer ),
'@poemkit/store': path.resolve(__dirname, alias.pathStore ),
}
},
...
babel.config.js
:
...
"plugins": [
["module-resolver", {
"root": ["./src"],
"alias": {
"@poemkit/config": "./src/config",
"@poemkit/components": "./src/client/components",
"@poemkit/router": "./src/client/router",
"@poemkit/helpers": "./src/client/helpers",
"@poemkit/services": "./src/client/services",
"@poemkit/reducers": "./src/client/reducers",
"@poemkit/pages": "./src/client/views/_pages",
"@poemkit/actions": "./src/client/actions",
"@poemkit/server": "./src/server",
"@poemkit/store": "./src/store"
}
}]
]
...
tsconfig.json
:
{
"compilerOptions": {
"baseUrl": "./src",
"paths": {
"@poemkit/config/*": ["config/*"],
"@poemkit/components/*": ["client/components/*"],
"@poemkit/router/*": ["client/router/*"],
"@poemkit/helpers/*": ["client/helpers/*"],
"@poemkit/services/*": ["client/services/*"],
"@poemkit/reducers/*": ["client/reducers/*"],
"@poemkit/pages/*": ["client/views/_pages/*"],
"@poemkit/actions/*": ["client/actions/*"],
"@poemkit/server/*": ["server/*"],
"@poemkit/store/*": ["store/*"]
}
}
}
package.json
:
{
"jest": {
"testEnvironment": "jsdom",
"moduleNameMapper": {
"\\.(css|less|scss|sass)$": "identity-obj-proxy",
"^@poemkit/config/(.*)": "/src/config/$1",
"^@poemkit/components/(.*)": "/src/client/components/$1",
"^@poemkit/router/(.*)": "/src/client/router/$1",
"^@poemkit/helpers/(.*)": "/src/client/helpers/$1",
"^@poemkit/services/(.*)": "/src/client/services/$1",
"^@poemkit/reducers/(.*)": "/src/client/reducers/$1",
"^@poemkit/pages/(.*)": "/src/client/views/_pages/$1",
"^@poemkit/actions/(.*)": "/src/client/actions/$1",
"^@poemkit/server/(.*)": "/src/server/$1",
"^@poemkit/store/(.*)": "/src/store/$1"
},
"transform": {
"^.+\\.(js|jsx)$": "babel-jest",
"^.+\\.(ts|tsx)?$": "ts-jest"
}
}
}
⚙️ 库相关配置:
您可以通过修改 webpack.config.js
的 output
属性来指示 webpack 如何去输出、以及在哪里输出你的「bundle、asset 和其他你所打包或使用 webpack 载入的任何内容」
...
const globs = {
port : 8080,
examples : 'public',
build : 'src/client',
dist : 'dist'
};
...
output: {
path: path.resolve(__dirname, './' + globs.dist + '/js' ),
filename: '[name].js'
},
...
⚙️ 网站配置:
修改package.json
文件即可:
{
"author": "UIUX Lab",
"name": "poemkit",
"email": "[email protected]",
"version": "1.0.0",
"projectName": "PoemKit",
"createdInfo": "UIUX Lab (https://uiux.cc)",
"projectURL": "https://uiux.cc",
"description": "React Toolkit for Building a Full Website",
...
}
⚙️ 路由配置:
路由配置同时也兼顾了网站的主导航(不包括将在主导航的页面上配置的嵌套路由)。 访问文件 ./src/client/router/RoutesConfig.js
.
路由器的一些脚本可以在文件中修改 ./src/client/router/App.js
.
⚙️ 服务端渲染配置:
服务端渲染和标签替换的一些脚本可以在文件中修改 ./src/server/renderer.js
.
⚙️ 设置服务器代理:
为了同时运行服务器和React应用程序,我们需要在 package.json
中添加proxy
键。我们已经将服务器设置为在端口3000上运行,因此将代理指向 localhost:3000
。
{
"proxy": "http://localhost:3000",
...
}
⚙️ 使用 PropTypes
检查类型:
$ npm run dev
使用下面的JavaScript脚本可以编写调试代码,在终端显示错误和警告信息。
import PropTypes from "prop-types";
import React, { Component } from 'react';
export default class YourComponentName extends Component {
public static propTypes = {};
constructor(props) {
super(props);
}
render() { ... }
}
if ( process.env.NODE_ENV === 'development' ) {
YourComponentName.propTypes = {
displayEnable: PropTypes.bool.isRequired,
htmlString: PropTypes.oneOfType([PropTypes.string, PropTypes.object])
}
}
⚙️ 自定义导入的第三方库或插件:
应用默认加载了常用的第三方库(图标,动画,3D引擎等),您可以按需加载,或者修改导入配置。第三方插件配置文件位于 ./src/client/components/_plugins/
. 第三方插件默认和自定义组件一起使用,您也可以主动移除它们。
⚙️ 网站根目录配置:
更改网站的根目录,以便您的项目上传到另一个目录时可以使用它。修改 ./src/config/websiteConfig.js
的键 rootDirectory
。
如果该文件位于根目录中,则可以将其保留为空。 如果在另一个目录中,则可以写成 "/blog" (结尾不要加斜杠)
{
"rootDirectory": ""
}
⚙️ 网站URL配置:
网站地址(URL)通常被用于SEO优化。修改 ./src/config/websiteConfig.js
的键 siteUrl
。一个完整的URL一般被命名为 https://domain-name.com
(结尾不要加斜杠和子目录名称)
{
"siteUrl": ""
}
⚙️ 接口API配置:
修改 ./src/config/websiteConfig.js
的键 API
, 如下代码:
{
"API": {
"RECEIVE_DEMO_LIST": "https://apiurl1.com",
"RECEIVE_DEMO_LISTDETAIL": "https://apiurl2.com"
}
}
目录结构
poemkit/
├── README.md
├── CHANGELOG.md
├── CONTRIBUTING.md
├── LICENSE
├── ecosystem.config.js ------------------------- # 用于pm2部署
├── babel.config.js
├── webpack.config.js
├── package-lock.json
├── package.json
├── test/ --------------------------- # 单元测试脚本
├── public/ --------------------------- # 自动生成的首页HTML静态模板
│ ├── index.html
│ ├── manifest.json
│ └── server/ ------------------------------- # 用于服务器测试的PHP脚本
│ └── assets/
├── dist/ ------------------------------- # web 编译出的文件,用于生产环境
│ ├── css/
│ │ ├── poemkit.css
│ │ └── poemkit.min.css
│ └── js/
│ │ ├── poemkit.js
│ │ └── poemkit.min.js
├── src/
│ ├── client/
│ │ ├── client.js
│ │ ├── actions/
│ │ ├── reducers/
│ │ ├── helpers/
│ │ ├── services/
│ │ ├── router/
│ │ ├── components/ ------------------------- # 独立的UI组件
│ │ │ ├── */
│ │ │ ├── _utils/ ----------------------- # 常用的实用程序和样式脚本
│ │ │ └── _plugins/ --------------------- # 第三方插件
│ │ ├── views/ ------------------------------ # 网站页面
│ │ │ ├── _pages/
│ │ │ └── _html/
│ └── server/
│ │ ├── app.js
│ │ ├── server.js
│ │ └── renderer.js
│ └── store/
│ │ └── createStore.js
│ └── config/
│ │ ├── tmpl-manifest.json ------------------ # 自动生成的 `manifest.json` 文件模板
│ │ └── websiteConfig.js -------------------- # 网站配置文件(比如根目录)
└──
兼容性
Chrome | Firefox | Edge | IE | Safari | Opera | iOS | Android |
---|---|---|---|---|---|---|---|
>= 49 | >= 45 | >=14 | >=11 | >= 9 | >= 30 | >=10 | >=4.4 |
支持的开发环境
- 支持React 17 +
- 支持ReactTypeScript 4.x.x +
- 支持ReactBabel 7.x.x +
- 支持ReactWebpack 5.x.x
这些是我的想法。如果您想讨论某些内容或对如何改进有更多意见,请随时告诉我!感谢!