一步步搭建react后台系统

手摸手带你撸react后台管理系统

项目地址:

react后台系统

创建项目

使用create-react-app脚手架创建

 npm install -g create-react-app yarn
 create-react-app antd-demo
 cd antd-demo
 yarn start

依赖模块

  • react 16.5.2
  • react-router
  • react-rudex
  • rudex

先搭建目录结构

- src/api  存放请求以及相关接口的地方
- src/components  存放组件
- src/reducer   // 存放reducer
- src/router 存放路由
- src/utils  存放公共方法
- src/views  存放页面

规划页面, 划分组件

首先是登录页, 其实是首页,
而其他页面都在首页下面

页面

- 登录页面
- 首页

组件

- 首页内的组件
   - 侧边导航栏
   - 头部 信息栏
   - 面包屑
   - 内容页面

划分完毕, 按照页面建立路由

/src/router/index.js

import Login from '@/views/login/index'
import Index from '@/views/index/index'

export const main = [
    { path: '/login', name: '登录', component: Login },
    { path: '/', exact: true,  name: '首页', component: Index }
]

export const menus = [    // 菜单相关路由
]

export const routerConfig =  {
    main, menus
}

记得把编辑器设置成支撑jsx语法的

webstrom的设置方法

  • file -> settings -> languages & Frameworks -> javascript -> 选择react JSX

路由建立完毕, 这时候可以按照路由新建react的页面啦

  • src/views/login/index.js
    页面不必多复杂, 简单点就好,先把框架搭建出来
import React, { Component } from 'react';
class Login extends Component {
    render() {
        return (
            <div>
                Login
            </div>
        )
    }
}
export default Login;
  • src/views/index/index.js
import React, { Component } from 'react';
class Index extends Component {
    render() {
        return (
            <div>
                index
            </div>
        )
    }
}
export default Index;

弄完后, 修改一下webpack配置吧,

npm run eject  先把配置显示出来
打开 config/webpack.config.dev.js, 然后找到 resolve.alias 添加 `  '@': paths.appSrc` 配置吧, 作用是以后凡是src文件目录 都用 ‘@’来替代
module.exports = {
resolve: {
    alias: {
       '@': paths.appSrc
    }
}

改完后 重启一下

重启报错
internal/modules/cjs/loader.js:596
    throw err;
Error: Cannot find module 'chalk'

npm i chalk
好了 可以用了

路由建立完毕了, 就开始搭建路由框架吧

  1. 添加路由库
npm i -D react-router react-router-dom
  • app.js
    导入路由库, 导入路由配置, 导入路由工具
import { BrowserRouter as Router, Route, Link } from "react-router-dom";
import { main as mainConfig } from './router/index'
import { RenderRoutes } from './router/utils'

修改标签
添加Router标签
添加RenderRoutes组件

class App extends Component {
  render() {
    return (
        <Router>
          <div className="App">
              <RenderRoutes routes={mainConfig}></RenderRoutes>
          </div>
        </Router>
    );
  }
}
  1. utils
  • 添加 RenderRoutes组件
    这个组件的作用 就是循环渲染当前数组下的一维数组的内容, 将他以标签的形式展现出来,
  • RouteWithSubRoutes 是显示路由的组件, 这里你展示内容的一个标签即可
比如这个路由数组
const  routes= [
                   { path: '/login', name: '登录', component: Login },
                   { path: '/', exact: true,  name: '首页', component: Index,
                        routes: [
                            path: '/a', name: 'a', component: PageA,
                            path: '/b', name: 'b', component: PageB
                        ]
                   }
               ]
// 循环渲染当前路由数组中一维数组中的组件
export const RenderRoutes = ({routes}) => {return (routes.map((route, i) => <RouteWithSubRoutes key={i} {...route} />))};
也就是说, 这里只渲染这个数组的一唯数组, routes下面的routes数组是不会继续渲染的
  • 添加 RouteWithSubRoutes组件
根据路由信息 例如:
const route = {path: '/login', name: '登录', component: Login}
Route 是渲染路由的标签 exact是否严格匹配, render是自定义渲染内容, 返回要渲染的标签, 这里拿出route.component(Login)组件进行渲染。
别忘记, 如果有子路有的话, 需要把子路由也传递过去, routes={route.routes}, 方便下面的组件可以拿到这些信息进行渲染
// 渲染当前组件
export const RouteWithSubRoutes = route => (
    <Route
        path={route.path}
        exact={route.exact}
        render={props =>{
            return (
                <route.component {...props} routes={route.routes} />
            )
        }}
    />
);
  1. 可以看到页面了哦, 这时候最简单的路由功能以及实现
    输入 http://localhost:3000 到浏览器, 可以看到 Index
    输入 http://localhost:3000/login 到浏览器, 可以看到 login

改造Index页面

login页面暂且不管, 无非就是一个Login + 两个Input + 一个Button的事情, 这里先继续吧整体框架架构完成
下面图片截图来源自: https://github.com/yezihaohao/react-admin

后台系统主页面架构.png

这个页面被我分成了四块,侧边栏一块, 头部信息栏一块, 面包屑一块, 内容展示区一块。
这里先不做具体功能, 先把页面分块和组件做出来先

  1. 先把组件做出来
  • src/components/crumbs
import React, { Component } from 'react';
class Crumbs extends Component {
    render() {
        return (
            <div>
                crumbs
            </div>
        )
    }
}
export default Crumbs;
  • src/components/header
import React, { Component } from 'react';
class MyHeader extends Component {
    render() {
        return (
            <div>
                Header
            </div>
        )
    }
}
export default MyHeader;
  • src/components/main
import React, { Component } from 'react';
class MyMain extends Component {
    render() {
        return (
            <div>
                main
            </div>
        )
    }
}
export default MyMain;
  • src/components/slider
import React, { Component } from 'react';
class MySlider extends Component {
    render() {
        return (
            <div>
                slider
            </div>
        )
    }
}
export default MySlider;
  1. 做出组件后, 然后需要引入ant
    但是引入ant 又需要按需加载怎么办?

ant文档 里面可以找到, 但是这个方法仅限于还没有 yarn run eject 出配置文件之前。
当 yarn run eject 出配置文件之后, 该怎么办呢?
别慌, 下面有方法

  • 添加插件 yarn add babel-plugin-import --save-dev yarn add antd --save-dev
  • 在congif文件夹下webpack.config.dev.js第147行添加代码别找错了
options: {
   +        plugins: [
   +             [‘import‘, [{ libraryName: "antd", style: ‘css‘ }]],
   +          ],
              // This is a feature of `babel-loader` for webpack (not Babel itself).
              // It enables caching results in ./node_modules/.cache/babel-loader/
              // directory for faster rebuilds.
              cacheDirectory: true,
            },

|
|
| 具体细节
--------->

{
            test: /\.(js|jsx|mjs)$/,
            include: paths.appSrc,
            loader: require.resolve('babel-loader'),
            options: {
            plugins: [
               ['import', [{ libraryName: "antd", style: 'css' }]],
              ],
              // This is a feature of `babel-loader` for webpack (not Babel itself).
              // It enables caching results in ./node_modules/.cache/babel-loader/
              // directory for faster rebuilds.
              cacheDirectory: true,
            },
          },
  • 在config文件下webpack.config.prod.js第154行添加
    重复上一步操作
  1. 在views/index/index 页面引入 ant组件试一下是否成功
import React, { Component } from 'react';
import { Button }  from 'antd'
class Index extends Component {
    render() {
        return (
            <div>
                index
                <Button>sdfdssdf</Button>
            </div>
        )
    }
}
export default Index;

改在index页面2

代码中, 用 + 表示新增, - 表示减少, +-表示修改

  1. 改在组件

改造组件前, 先找到对应页面的组件
ant.designUI组件

然后找到layout组件, 从里面拿出代码 ![layout组件]


layout组件.png
  1. 记住, import必须在所有业务代码前面执行, 如果爆出下面的Bug, 请修改Impot顺序
Import in body of module; reorder to top import/first
import 必须在其它所有业务代码前面(eslint 暴出)
  1. 也把rudex一起引入了吧
  • react-redux 是react的一个库
  • rudex 是js库
npm i react-redux redux -D
  1. 修改app.js
import React, { Component } from 'react';
import { createStore } from 'redux'  //+
import { Provider } from 'react-redux' //+
import { BrowserRouter as Router, Route, Link } from "react-router-dom";
import './App.css';
import { main as mainConfig } from './router/index'
import { RenderRoutes } from './router/utils'
import { slidecollapsed } from '@/reducer/reduxs' //+
const store = createStore(slidecollapsed) // +
class App extends Component {
    render() {
        return (
            <Provider store={store}>  // +
                    <Router>
                        <div className="App">
                            <RenderRoutes routes={mainConfig}></RenderRoutes>
                        </div>
                    </Router>
            </Provider> // +
        );
    }
}

export default App

  1. 添加 ruducer/reduxs.js
  • 用来实现业务逻辑功能, 创建store的rudex
const SLIDECOLLAPSED = 'slidecollapsed'
export const slidecollapsed = (state = { slidecollapsed: false }, action) => {
    const slidecollapsed = state.slidecollapsed
    switch (action.type) {
        case SLIDECOLLAPSED:
            return Object.assign({}, state, {
                slidecollapsed: !slidecollapsed
            })
        default:
            return state
    }
}
  1. 添加reducer/connect.js
  • 添加页面中connect的参数(注入到页面中的属性和事件)
let action_slidecollapsed = {type: 'slidecollapsed'}
export const mapStateToProps = (state) => {
    return {slidecollapsed: state.slidecollapsed}
}
export const mapDispatchToProps = (dispatch) => {
    return {onSlidecollapsed: () => dispatch(action_slidecollapsed)}
}

  1. 修改views/index/index.js
import React, { Component } from 'react';
import { Layout, Menu, Icon } from 'antd'; // +
import { connect, Provider  } from 'react-redux' // +
import Crumbs  from '@/components/crumbs' // +
import MyHeader  from '@/components/header' // +
import MyMain  from '@/components/main' // +
import MySlider  from '@/components/slider' // +
import { mapStateToProps, mapDispatchToProps } from '@/reducer/connect' // +
const { Header, Content } = Layout; // +

class Index extends Component {
    constructor(props){ // +
        super(props) // +
        this.state = { // +
            onSlidecollapsed: this.props.onSlidecollapsed // +
        }; // +
    }

    toggle = () => {  // +-
        this.state.onSlidecollapsed()
    }

    render() {
        const { slidecollapsed } = this.props // +
        return (
            <Layout>
                <MySlider></MySlider>
                <Layout>
                    <Header style={{ background: '#fff', padding: 0 }}>
                        <Icon
                            className="trigger"
                            type={ slidecollapsed ? 'menu-unfold' : 'menu-fold'}   // +-
                            onClick={this.toggle}
                        />
                    </Header>
                    <Content style={{ margin: '24px 16px', padding: 24, background: '#fff', minHeight: 280 }}>
                        Content
                    </Content>
                </Layout>
            </Layout>
        );
    }
}

export default connect(mapStateToProps, mapDispatchToProps)(Index);  // +-

上面代码中, 先拿到redux注入到组件的事件 并修改toggle方法

      this.state = { // +
            onSlidecollapsed: this.props.onSlidecollapsed // +
        }; // +

           toggle = () => {  // +-
                this.state.onSlidecollapsed()
            }

然后拿到注入的属性

    const { slidecollapsed } = this.props // +

然后把属性放进Icon里面

 <Icon
                            className="trigger"
                            type={ slidecollapsed ? 'menu-unfold' : 'menu-fold'}   // +-
                            onClick={this.toggle}
                        />
  1. components/slider.js
import React, { Component } from 'react';
import { Menu, Icon } from 'antd'; // +
import {Layout} from "antd/lib/index"; // +
import { mapStateToProps, mapDispatchToProps } from '@/reducer/connect' // +
import {connect} from "react-redux"; // +
const { Sider } = Layout; // +
class MySlider extends Component {
    render() {
        const { slidecollapsed } = this.props // +
        return (
            <Sider
                trigger={null}
                collapsible
                collapsed={ slidecollapsed } // +-
            >
                <div className="logo" />
                <Menu theme="dark" mode="inline" defaultSelectedKeys={['1']}>
                    <Menu.Item key="1">
                        <Icon type="user" />
                        <span>nav 1</span>
                    </Menu.Item>
                    <Menu.Item key="2">
                        <Icon type="video-camera" />
                        <span>nav 2</span>
                    </Menu.Item>
                    <Menu.Item key="3">
                        <Icon type="upload" />
                        <span>nav 3</span>
                    </Menu.Item>
                </Menu>
            </Sider>

        )
    }
}
export default connect(mapStateToProps, mapDispatchToProps)(MySlider); // +-
  1. 继续把header组件分开
  • components/header.js
import React, { Component } from 'react';
import { Layout, Icon } from 'antd';
import { connect  } from 'react-redux'
import { mapStateToProps, mapDispatchToProps } from '@/reducer/connect'
const { Header } = Layout;
class MyHeader extends Component {
    constructor(props){
        super(props)
        this.state = {
            onSlidecollapsed: this.props.onSlidecollapsed
        };
    }
    toggle = () => {
        this.state.onSlidecollapsed()
    }
    render() {
        const { slidecollapsed } = this.props
        return (
                <Header style={{ background: '#fff', padding: 0 }}>
                    <Icon
                        className="trigger"
                        type={ slidecollapsed ? 'menu-unfold' : 'menu-fold'}
                        onClick={this.toggle}
                    />
                </Header>
        )
    }
}
export default connect(mapStateToProps, mapDispatchToProps)(MyHeader);
  • 修改views/index/index.js
import React, { Component } from 'react';
import { Layout, Menu, Icon } from 'antd';
import { connect  } from 'react-redux'
import Crumbs  from '@/components/crumbs'
import MyHeader  from '@/components/header'
import MyMain  from '@/components/main'
import MySlider  from '@/components/slider'
import { mapStateToProps, mapDispatchToProps } from '@/reducer/connect' // -
import './index.css'
const { Header, Content } = Layout;

class Index extends Component {
    render() {
        return (
            <Layout>
                <MySlider></MySlider>
                <Layout>
                    <MyHeader></MyHeader>  // +-
                    <Content style={{ margin: '24px 16px', padding: 24, background: '#fff', minHeight: 280 }}>
                        Content
                    </Content>
                </Layout>
            </Layout>
        );
    }
}

export default connect(mapStateToProps, mapDispatchToProps)(Index); // -
export default Index; // +
  1. 把内容面包屑组件加进去
  • 修改 components/crumbs.js
import React, { Component } from 'react';
import { Link, withRouter } from 'react-router-dom';
import { Breadcrumb } from 'antd';
import { connect } from 'react-redux'
import { crumbsMap } from "../reducer/connect";
import { filterData } from '@/utils/index.js'
const deepFlatten = arr => [].concat(...arr.map(v => Array.isArray(v) ? deepFlatten(v) : v));
let breadcrumbNameMap = []

class Crumbs extends Component {
    componentDidMount () {  //页面渲染完毕后调用
        this.onTrun()
    }
    onTrun () {}
    render() {
        let { location, getRouterConfig, routerConfig } = this.props
        routerConfig = filterData(routerConfig, 'routerConfig')
        this.onTrun = getRouterConfig  // 在页面刷新时不可调用, 需要页面渲染完毕时调用
        routerConfig = (typeof routerConfig === 'object' && Object.values(routerConfig)) || []
        breadcrumbNameMap = (Array.isArray(routerConfig) && deepFlatten(routerConfig)) || []
        var newBreadcrumbNameMap = breadcrumbNameMap.filter((item, i) => {
            if (item.path === location.pathname) {
                return item
            }
        })
        return (
            <div className="my-breadcrumb">
                <Breadcrumb>
                    {getBreadCurmbs(newBreadcrumbNameMap)}
                </Breadcrumb>
            </div>
        )
    }
}

const getBreadCurmbs  = (newBreadcrumbNameMap, arr = []) => {
    return arr = newBreadcrumbNameMap.map(item => {
        arr.push(
            <Breadcrumb.Item key={item.path}>
                <Link to={item.path}>
                    {item.name}
                </Link>
            </Breadcrumb.Item>
        )
        {
            Array.isArray(item.routes) && item.routes.length > 0 && getBreadCurmbs(item.routes, arr)
        }
        return arr
    })
}
export default connect(crumbsMap.mapStateToProps, crumbsMap.mapDispatchToProps)(withRouter(Crumbs));

这里使用了react-redux进行存储路由表, 然后切换action拿到路由信息并注入到props内, 通过props 拿到数据后进一步处理, 把几个路由表合并, 随后遍历路由内的子路由,并渲染出面包屑组件

  • 修改reducer/connect.js
import { action_slidecollapsed, routerConfig } from '@/reducer/action.js'

export const mapStateToProps = (state) => {
    return {slidecollapsed:  state.slidecollapsed}
}
export const mapDispatchToProps = (dispatch) => {
    return {onSlidecollapsed: () => dispatch(action_slidecollapsed)}
}

export const crumbsMap = {
    mapStateToProps (state) {
        return { routerConfig: state.routerConfig }
    },
    mapDispatchToProps (dispatch) {
        return {getRouterConfig: () => {
                return dispatch(routerConfig)
            }}
    }
}
  • 添加reducer/action.js
export const SLIDECOLLAPSED = 'slidecollapsed'
export const ROUTERCONFIG = 'routerconfig'
export const action_slidecollapsed = {type: SLIDECOLLAPSED}
export const routerConfig = { type: ROUTERCONFIG }
  • 修改reducer/reduxs.js
import { combineReducers } from 'redux';
import { routerConfig as myRouterConfig } from '@/router/index'
import {SLIDECOLLAPSED, ROUTERCONFIG} from '@/reducer/action.js'
const slidecollapsedFuc = (state = { slidecollapsed: false }, action) => {
    switch (action.type) {
        case SLIDECOLLAPSED:
            return Object.assign({}, state, {
                slidecollapsed: !state.slidecollapsed
            })
        default:
            return state
    }
}

const getRouterConfig = (state = { routerConfig: [] }, action) => {
    switch (action.type) {
        case ROUTERCONFIG:
            return  Object.assign({}, state, {
                routerConfig: myRouterConfig
            })
        default:
            return state
    }
}

export const allReducer = combineReducers({
    slidecollapsed: slidecollapsedFuc, routerConfig: getRouterConfig
})
  • 因为将 reducer拆分, 因此需要使用combineReducers, 而使用combineReducers会将原本的数据包装多一层, 因此需要进一步修改代码

  • 添加/utils/index.js

export const filterData = (state, stateName) => (typeof state ==='object' ? state[stateName] : state)
  • 修改components/crumbs.js
        let { location, getRouterConfig, routerConfig } = this.props
        routerConfig = filterData(routerConfig, 'routerConfig') // 这里需要将数据过滤一层, 因为reducer的名称 我都是按照下面变量来取的, 因此判断他们是否有下一层变量, 如果有,则拿下一层的, 如果没有, 则拿第一层的。
  • 修改components/header.js
        let { slidecollapsed } = this.props
        slidecollapsed = filterData(slidecollapsed, 'slidecollapsed')
  • 修改components/slider.js
        let { slidecollapsed } = this.props
        slidecollapsed =  filterData(slidecollapsed, 'slidecollapsed')
  1. 现在可以看到效果了
面包屑完成.png
  • 这里我在路由配置上添加了一个临时页面,
const Test = () => <h3>test</h3>
export const main = [
    { path: '/login', name: '登录', component: Login },
    { path: '/', exact: true,  name: '首页', component: Index,
        routes: [
            {path: '/test', name: '测试页面', component: Test }
        ]
    }
]

继续把main组件给拆分

  • components/main.js
import React, { Component } from 'react';
import {Layout} from "antd/lib/index";

const { Content } = Layout;
class MyMain extends Component {
    render() {
        return (
            <div>
                <Content style={{ margin: '24px 16px', padding: 24, background: '#fff', minHeight: 280 }}>
                    Content
                </Content>
            </div>
        )
    }
}
export default MyMain;
  • views/index/index.js
import React, { Component } from 'react';
import { Layout, Menu, Icon } from 'antd';
import { connect  } from 'react-redux'
import Crumbs  from '@/components/crumbs'
import MyHeader  from '@/components/header'
import MyMain  from '@/components/main'
import MySlider  from '@/components/slider'
import './index.css'
const { Header } = Layout;

class Index extends Component {
    render() {
        return (
            <Layout>
                <MySlider></MySlider>
                <Layout>
                    <MyHeader></MyHeader>
                    <Crumbs></Crumbs>
                    <MyMain></MyMain>
                </Layout>
            </Layout>
        );
    }
}

export default Index;

现在, react的大体框架已经搭建完毕。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容