react hooks umi例子

2019-12-08  本文已影响0人  龚达耶

随着react hooks的出现我们越来越多的项目已经转由hooks来写,没有看过之前关于hooks的介绍的同学们可以移步

传送门

那么我们如果需要路由或者状态之类的支持时我们就需要用到umi和dva。

= umi@2起,dva的整合可以直接通过 umi-plugin-react 来配置

我们当然可以看官方demo

传送门

我们今天结合官方文档来完成属于自己的react hooks结合umi的例子

我们这个例子是创建一个聊天机器人的前端界面

yarn create umi

具体是否需要到dva还是antd根据自己需求而定

image.png

首先我们假定这个聊天机器人有两个页面第一个为主页通过主页我们有一个弹框可以进入一个聊天界面

首先我们先配置路由

在.umirc.js中配置为chat的路径

routes: [
    {
      path: '/',
      component: '../layouts/index',
      routes: [
        { path: '/', component: '../pages/index' },
        { path: '/chat', component: '../pages/chat' } // 有一个为chat的页面
      ]
    }
  ]

接下来我们修改我们首页

//index.js
import React from 'react';
import styles from './index.css';
import handshake from '../assets/handshake.png';
import { Button } from 'antd';



export default function() {
  const handleClick = () => {
    window.open("/chat", "newwindow", "height=600, width=440, top=0, left=400, toolbar=no, menubar=no, scrollbars=no, resizable=no,location=no, status=no");
  }
  return (
    <div className={styles.container}>
      <p className={styles.slogan}>AccessHub Chatbot</p>
      <div className={styles.f1}>
        <Button className={styles.font}  onClick={handleClick} style={{background: '#20B2AA', color: '#faf0e6', fontFamily: 'IBMPlexSans,Helvetica Neue,Helvetica,Arial,sans-serif', width: '300px', height: '60px'}}>Click here to talk with me </Button>
      </div>
      <div className={styles.logo}>
        <img className={styles.image} src={handshake} />
      </div>
    </div>
  );
}

image.png

当我们点击中间button是就会到我们的/chat里

接下来我们在pages里创建chat的文件夹在里面创建对应的modles services

眼尖的同学们发现了我们用了sass 我们可以用任意的sass或者less

如果是sass的话我们需要

yarn add node-sass sass-loader 
image.png

我们先从services开始

创建chat.js

如果我们要请求一些接口我们就放在services里已机器人为例我么现在需要两个接口一个是获取进来的用户的信息一个是问机器人问题

import request from '../../../utils/request';

export function getUser() {
    return request('/user_infos');
}

export function ask(data) {
    return request('/answers', {
        method: 'POST',
        body: JSON.stringify(data),
    });
}

这是我们在src目录创建一个utils文件夹创建request.js来方便我们请求

import { fetch } from 'dva';

const domain = 'url location'; // http://0.0.0.0:8000
function parseJSON(response) {
  console.log(response)
  return response.json();
}

function checkStatus(response) {
  if (response.status >= 200 && response.status < 300) {
    return response;
  }

  const error = new Error(response.statusText);
  error.response = response;
  throw error;
}

/**
 * Requests a URL, returning a promise.
 *
 * @param  {string} url       The URL we want to request
 * @param  {object} [options] The options we want to pass to "fetch"
 * @return {object}           An object containing either "data" or "err"
 */
export default function request(url, options) {
  console.log(domain+url)
  return fetch(domain+url, options)
    .then(checkStatus)
    .then(parseJSON)
    .then(data => ({ data }))
    .catch(err => ({ err }));
}

接下来我们在models创建chat.js

在 umi 项目中,你可以使用 dva 来处理数据流,以响应一些复杂的交互操作。这些处理数据流的文件统一放在 models 文件夹下,每一个文件默认导出一个对象,里面包含数据和处理数据的方法,通常我们称之为 model

namespace

文件名即 namespace,可以省去 model 导出的 namespace key

namespace: 'chat'

state

将要用到的数据保存进去

state: {
      userInfo: {},
      answer: {},
      question: ''
    },

subscriptions

初始化时判断是否一来就要发起请求

 subscriptions: {
      setup({ dispatch, history }) {
        return history.listen(({ pathname, query }) => {
          if (pathname === '/chat') {
            dispatch({ type: 'fetch' });
          }
        });
      },
    },

effects

异步操作调service里两个接口

    *fetch({ payload }, { call, put }) {  // eslint-disable-line
        console.log(payload, chatService.getUser)
        const userInfo = yield call(chatService.getUser);
        console.log(userInfo)
        yield put({
          type: 'save',
          payload: userInfo.data,
        });
      },
      *ask({ payload }, { call, put, select }) {
        const res = yield call(chatService.ask, payload);
        console.log(res)
        yield put({ type: 'askQuestion', payload: {"answer": res.data}});
      },

reducers

同步操作返回最新state

    reducers: {
      save(state, { payload }) {
        console.log(payload)
        return { ...state, userInfo: payload };
      },
      askQuestion(state, { payload }) {
        console.log(payload)
        return { ...state, answer: payload.answer};
      },
      saveQuestion(state, { payload }) {
        console.log(payload)
        return { ...state, question: payload.question};
      },
    },

接下来我们创建mock数据方便测试

为了模拟真实延迟我们让所有的接口都延迟一秒返回

需要用到roadhog-api-doc

文档介绍

import { delay } from 'roadhog-api-doc';

const proxy = {
    'GET /user_infos': {username: 'ddw', s_n: "04599A672", email:"jorgegong@163.com"},
    'POST /answers': (req, res) => {
    res.send({ answer: 'hello', image: '' });
    },
};

// 调用 delay 函数,统一处理
export default delay(proxy, 1000);

这里页面上部分可以参照之前关于hooks的介绍,想看代码的同学可以干住下我的github

传送门

上一篇 下一篇

猜你喜欢

热点阅读