第一次使用react,还是习惯性的去找了合适的移动开发框架,直接在此基础上迭代开发。
优点是能快速及时上手一个项目,边开发应用边学习,能做出一个可以用,可以看的东西。
缺点是不能系统的学习,她的特性不能系统全面地掌握以及与之相关的技术底层能用但是原理模糊。
整个项目虽然功能实现,但是是在一知半解状态做出来,边开发边解决问题,边重构迭代,react的优势并没有完全发挥出来。
希望系统学习之后能进行一次整体重构。
使用这个就是因为以前用过,上手快,而且她快速,稳定,但是需要购买证书。
速度较快,对超大文件读写速度飞快。
速度一般,更新快,占用资源较大。
react + redux + react-router + less/sass + ES6 + webpack
react全家桶,react使用的标配,一个个做简单的说明。
使用create-react-app构建React开发环境,node以及npm已配置。
npm慢可以使用淘宝定制的cnpm,设置如下:
npm install -g cnpm --registry=https://registry.npm.taobao.org
npm config set registry https://registry.npm.taobao.org
create-react-app demo安装
create-react-app my-app
cnpm install -g create-react-app
cd my-app/
npm start
启动之后,http://localhost:3000/
接下来可以在此基础上,学习或者实验react相关…
采用声明式,高效而且灵活的用来构建用户界面的框架。
官方文档 https://react.docschina.org
react-dom提供操作DOM的扩展库。
通过babel解析JSX语法代码转为纯JS语法代码的库。
面向组件编程的(组件化编码开发),最后得到标签代码,同时组件之间可以进行通信,数据传递也非常灵活。
关于组件通信方式,下面单独做了实验与总结。
组件通讯方式
组件通信方式汇总
其他
组件上加上prop-types进行验证
optionalArray: PropTypes.array,
optionalBool: PropTypes.bool,
optionalFunc: PropTypes.func,
optionalNumber: PropTypes.number,
optionalObject: PropTypes.object,
optionalString: PropTypes.string,
optionalSymbol: PropTypes.symbol
虽然使用的脚手架中整合了redux,但是在使用中,为了赶进度没有使用,重构的时候也把这部分也加上了。
只有遇到 React 实在解决不了的问题,你才需要 Redux 。
redux简单做了下总结。
Redux简单汇总
用于构建路由,主要有Router,IndexRoute,Route,Link,IndexLink这几个组件,以及hashHistory,browserHistory。
npm install --save react-router
项目部署的时候,原配置createHashHistory出现一片白问题,后来更换了createBrowserHistory,重新部署后可访问。
区别
browserHistory 其实使用的是 HTML5 的 History API,浏览器提供相应的接口来修改浏览器的历史记录;
hashHistory 是通过改变地址后面的 hash来改变浏览器的历史记录。
import createHistory from 'history/createBrowserHistory'
从 React Router 中导入
import { BrowserRouter as Router, Route } from 'react-router-dom';
react-router嵌套路由
Router包裹Route,
当url变化的时候,Router将会匹配到指定的路由,然后渲染路由绑定的组件。
Route用来显式地把路由映射到应用的组件结构上。 用path指定url,用component指定路由命中url后需要渲染的那个组件。
如下:
<Router basename={AppCfg.app.BaseName}>
<App>
<Route exact path="/" component={CreateComponent(Default)} />
<Route path="/home" component={CreateComponent(Home)} />
<Route path="/userinfo" component={CreateComponent(UserInfo)} />
</App>
</Router>
通过Provider将redux绑定到react,同时用Provider包裹路由,路由控制器就可以访问store。
<AppContainer warnings={false}>
<Provider store={ApiClientStore} key="provider">
<Router history={history}>
{RootElement}
</Router>
</Provider>
</AppContainer>
Switch
Switch的特点是从上往下读,只要有一个匹配成功,就不会往下读(Switch是由包容性变成排他性的一个重要组件)
在具体页面跳转中使用switch
import { Switch } from 'react-router-dom';
const { location } = route || {};
const { key } = location || {};
<Switch key={key} location={location}>
{this.props.children}
</Switch>
其他
React Router 提供一个 routerWillLeave生命周期钩子,这使得 React 组件可以拦截正在发生的跳转,或在离开 route 前提示用户。routerWillLeave 返回值有以下两种:
return false 取消此次跳转
return 返回提示信息,在离开 route 前提示用户进行确认。
import { Lifecycle } from 'react-router'
const Home = React.createClass({
// 假设 Home 是一个 route 组件,它可能会使用
// Lifecycle mixin 去获得一个 routerWillLeave 方法。
mixins: [ Lifecycle ],
routerWillLeave(nextLocation) {
if (!this.state.isSaved)
return 'Your work is not saved! Are you sure you want to leave?'
},
// ...
})
Links
Link在内部做了一个操作,把标签变成了a标签。
touser
Redirect组件(重定向)
from(使用绝对路径)
exact(精准匹配)
to(跳转路径)
本次开发中使用sass,由于使用了react-weui样式使用较少了。
SASS是一种CSS的开发工具,使CSS的开发,变得简单和可维护。
可以查看阮一峰老师关于此教程。
http://www.ruanyifeng.com/blog/2012/06/sass.html
声明方式
var 、function 、let 、const 、import、 class。
ES6中明确规定,如果区块中存在let和const命令,这个区块对这些命令声明的变量,从一开始就形成了封闭作用域,凡是在声明之前就使用这些变量就会报错。
箭头函数
相当于匿名函数,使用“箭头”(=>)定义函数。
无参箭头函数
setInterval(() => this.nextSlide(), 2000);
一个参数箭头函数
x => {console.log(x*x)}
多个参数箭头函数
参数需要用括号()括起来
'use strict'
var arr = [10, 20, 1, 2];
arr.sort((x, y) => {
if(x-y>0){
return 1;
}else{
return -1;
}
});
console.log(arr); // [1, 2, 10, 20]
可变参数
…rest
(x, y, ...rest) => {
var i, sum = x + y;
for (i=0; i<rest.length; i++) {
sum += rest[i];
}
return sum;
}
其他
Es6箭头函数说明挺全也很简明
https://www.cnblogs.com/snandy/p/4403111.html
ES6 ESLint 是一个语法规则和代码风格的检查工具,可以用来保证写出语法正确、风格统一的代码。
重构时候原来的代码都有问题,前后空格、无效的代码以及分号等等各种问题。
一开始使用eslint确实会影响进度,但是对于规范代码以及语法有很好的引导,虽然麻烦点,最后的使用效果不错。
类似如下错误,根据提示修改即可。
(1) expected parenteses around arrow function argument having a body with curly braces arrow-parens
箭头函数参数周围的预期括号具有带大括号的主体
(2) Empty components are self-closing
组件内部空时,不需要自己关闭,如下:
<a href={link}>
<div className="slider__slide" //此处
data-active={active}
style={slideStyle}
onTouchStart={(ev) => { this.clickCarousel(carouselId, ev); }}
onTouchEnd={(ev) => { this.touchEnd(ev); }
}></div> //此处
</a>
<a href={link}>
<div
className="slider__slide" //换行
data-active={active}
style={slideStyle}
onTouchStart={(ev) => { this.clickCarousel(carouselId, ev); }}
onTouchEnd={(ev) => { this.touchEnd(ev); }
}/> //空的div去除
</a>
WebPack可以看做是模块打包机:它做的事情是,分析你的项目结构,找到JavaScript模块以及其它的一些浏览器不能直接运行的拓展语言(Scss,TypeScript等),并将其转换和打包为合适的格式供浏览器使用
webpack与Gulp的区别
Grunt和Gulp的工作方式是:在一个配置文件中,指明对某些文件进行类似编译,组合,压缩等任务的具体步骤,工具之后可以自动替你完成这些任务。
Webpack的工作方式是:把你的项目当做一个整体,通过一个给定的主文件(如:index.js),Webpack将从这个文件开始找到你的项目的所有依赖文件,使用loaders处理它们,最后打包为一个(或多个)浏览器可识别的JavaScript文件。
Webpack的处理速度更快更直接,能打包更多不同类型的文件
这个不详细写了,要再实践总结下,但是这篇文章写得很好,值得学习。
https://segmentfault.com/a/1190000006178770?utm_source=tag-newest
UI库的选择上,找了好几个,鉴于微信公众号网页的兼容性,最后选择了react-weui。
Material-UI是一个实现了Google’s Material Design设计规范的react组件库。
蚂蚁金服开发的一个基于react的UI组件库
API
https://weui.github.io/react-weui/docs
1.安装
npm install [email protected] react-weui --save
2.使用
在App.js中引入模块。
import 'weui';
import 'react-weui/build/packages/react-weui.css';
实际使用中,import 相应的组件即可,以首页的tab切换页为例,如下:
import { Tab, TabBody, TabBar, TabBarItem, Article } from 'react-weui';
详细页面如下
<Tab>
<TabBody>
<Article style={{ display: this.state.tab == 0 ? null : 'none' }}>
<h1>Page 1</h1>
<section>
<h2 className="title">Heading</h2>
<section>
<h3>1.1 Title</h3>
<p>1111</p>
</section>
</section>
</Article>
<Article style={{ display: this.state.tab == 1 ? null : 'none' }}>
<h1>Page 2</h1>
<section>
<h2 className="title">Heading</h2>
<section>
<h3>2.1 Title</h3>
<p>222</p>
</section>
</section>
</Article>
</TabBody>
<TabBar>
<TabBarItem
active={this.state.tab == 0}
icon={<img src={this.state.urlmain} alt="购票主页" />}
onClick={(e) => this.handleTab(e, 0)}
label="主页"
/>
<TabBarItem
active={this.state.tab == 1}
icon={<img src={this.state.urlmine} alt="个人中心" />}
onClick={(e) => this.handleTab(e, 1)}
label="个人中心"
/>
</TabBar>
</Tab>
效果如下图
react-weui提供了一系列的组件,可以方便引入和使用,基本可以满足项目中的应用。
其实一开始对微信里使用支付宝支付这个感觉有点喧宾夺主了。但是微信里还真的可以这么干,也挺有意思的。总之,用户需求至上。
不过支付宝给出的解决方案里真正使用有点另起炉灶的意思。
1.需要引入相关的js文件作为支撑ap.js。
(function(){var b={};var a={};a.PADCHAR="=";a.ALPHA="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";a.makeDOMException=function(){var f,d;try{return new DOMException(DOMException.INVALID_CHARACTER_ERR)}catch(d){var c=new Error("DOM Exception 5");c.code=c.number=5;c.name=c.description="INVALID_CHARACTER_ERR";c.toString=function(){return"Error: "+c.name+": "+c.message};return c}};a.getbyte64=function(e,d){var c=a.ALPHA.indexOf(e.charAt(d));if(c===-1){throw a.makeDOMException()}return c};a.decode=function(f){f=""+f;var j=a.getbyte64;var h,e,g;var d=f.length;if(d===0){return f}if(d%4!==0){throw a.makeDOMException()}h=0;if(f.charAt(d-1)===a.PADCHAR){h=1;if(f.charAt(d-2)===a.PADCHAR){h=2}d-=4}var c=[];for(e=0;e<d;e+=4){g=(j(f,e)<<18)|(j(f,e+1)<<12)|(j(f,e+2)<<6)|j(f,e+3);c.push(String.fromCharCode(g>>16,(g>>8)&255,g&255))}switch(h){case 1:g=(j(f,e)<<18)|(j(f,e+1)<<12)|(j(f,e+2)<<6);c.push(String.fromCharCode(g>>16,(g>>8)&255));break;case 2:g=(j(f,e)<<18)|(j(f,e+1)<<12);c.push(String.fromCharCode(g>>16));break}return c.join("")};a.getbyte=function(e,d){var c=e.charCodeAt(d);if(c>255){throw a.makeDOMException()}return c};a.encode=function(f){if(arguments.length!==1){throw new SyntaxError("Not enough arguments")}var g=a.PADCHAR;var h=a.ALPHA;var k=a.getbyte;var e,j;var c=[];f=""+f;var d=f.length-f.length%3;if(f.length===0){return f}for(e=0;e<d;e+=3){j=(k(f,e)<<16)|(k(f,e+1)<<8)|k(f,e+2);c.push(h.charAt(j>>18));c.push(h.charAt((j>>12)&63));c.push(h.charAt((j>>6)&63));c.push(h.charAt(j&63))}switch(f.length-d){case 1:j=k(f,e)<<16;c.push(h.charAt(j>>18)+h.charAt((j>>12)&63)+g+g);break;case 2:j=(k(f,e)<<16)|(k(f,e+1)<<8);c.push(h.charAt(j>>18)+h.charAt((j>>12)&63)+h.charAt((j>>6)&63)+g);break}return c.join("")};b.pay=function(d){var c=encodeURIComponent(a.encode(d));location.href="pay.htm?goto="+c};b.decode=function(c){return a.decode(decodeURIComponent(c))};window._AP=b})();
2.支付宝生成签名
请求生成支付宝签名返回参数作为支付宝支付的参数。
let params = new FormData();
params.append('loginId', user.id);
params.append('orderNo', orderNo);
ApiClientUtil.post(ApiInfo.Alipay, params).then((json) => {
if(json.success) {
const gotoUrl='https://openapi.alipay.com/gateway.do?'+ json.body;
window._AP.pay(gotoUrl);
}else {
Alert.error('支付宝支付遇到问题,请重新尝试', {position: 'top',
offset: 40,onRouteClose: true, timeout: 2000});
return;
}
})
3.跳转到pay.htm
这个页面当用户选择支付宝账户点击确认支付时,页面跳转到此页面提示用户打开浏览器,进而完成支付的页面,在新的浏览器网页中完成支付宝支付。
请在菜单中选择在浏览器中打开,以完成支付…
<html>
<head>
<meta charset="utf-8"
">
<title>支付提示title>
<meta name="apple-mobile-web-app-capable" content="yes"/>
<meta name="apple-mobile-web-app-status-bar-style" content="black"/>
<meta name="format-detection" content="telephone=no"/>
<meta name="format-detection" content="email=no"/>
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=0"/>
<style>
*, :before, :after {
-webkit-tap-highlight-color: rgba(0, 0, 0, 0)
}
body, div, dl, dt, dd, ul, ol, li, h1, h2, h3, h4, h5, h6, form, fieldset, legend, input, textarea, p, blockquote, th, td {
margin: 0;
padding: 0
}
table {
border-collapse: collapse;
border-spacing: 0
}
fieldset, img {
border: 0
}
li {
list-style: none
}
caption, th {
text-align: left
}
q:before, q:after {
content: ""
}
input:password {
ime-mode: disabled
}
:focus {
outline: 0
}
html, body {
-webkit-touch-callout: none;
touch-callout: none;
-webkit-user-select: none;
user-select: none;
-webkit-tap-highlight-color: transparent;
tap-highlight-color: transparent;
height: 100%;
margin: 0;
padding: 0;
text-align: center;
font-size: 15px;
font-weight: 300;
font-family: "Helvetica Neue", Helvetica, Arial, "Lucida Grande", sans-serif
}
a {
text-decoration: none
}
body {
background: #F4F4F8
}
.weixin-tip {
display: none;
-webkit-box-sizing: border-box;
box-sizing: border-box;
position: absolute;
top: 15px;
right: 20px;
width: 265px;
padding: 55px 0 0;
text-align: left;
background: url() no-repeat right top;
background-size: 45px 68px
}
.weixin-tip-img {
display: none;
padding: 110px 0 0
}
.weixin-tip-img::after {
display: block;
margin: 15px auto;
content: ' ';
background-size: cover
}
.weixin-tip-img.iphone::after {
width: 150px;
height: 150px;
background-image: url()
}
.weixin-tip-img.android::after {
width: 173px;
height: 240px;
background-image: url()
}
style>
head>
<body>
<div class="J-weixin-tip weixin-tip">
<div class="weixin-tip-content">
请在菜单中选择在浏览器中打开,<br/>
以完成支付
div>
div>
<div class="J-weixin-tip-img weixin-tip-img">div>
<div>
<br/>
<a href="/" name="" style="font-size: 16px;color: green;">支付完成,点此返回主页a>
div>
<script type="text/javascript" src="ap.js">script>
<script>
if (location.hash.indexOf('error') != -1) {
alert('参数错误,请检查');
} else {
var ua = navigator.userAgent.toLowerCase();
var tip = document.querySelector(".weixin-tip");
var tipImg = document.querySelector(".J-weixin-tip-img");
if (ua.indexOf('micromessenger') != -1) {
tip.style.display = 'block';
tipImg.style.display = 'block';
if (ua.indexOf('iphone') != -1 || ua.indexOf('ipad') != -1 || ua.indexOf('ipod') != -1) {
tipImg.className = 'J-weixin-tip-img weixin-tip-img iphone'
} else {
tipImg.className = 'J-weixin-tip-img weixin-tip-img android'
}
} else {
var getQueryString = function (url, name) {
var reg = new RegExp("(^|\\?|&)" + name + "=([^&]*)(\\s|&|$)", "i");
if (reg.test(url)) return RegExp.$2.replace(/\+/g, " ");
};
var param = getQueryString(location.href, 'goto') || '';
location.href = param != '' ? _AP.decode(param) : 'pay.htm#error';
}
}
script>
body>
html>
参考这个实现
https://blog.csdn.net/wyk304443164/article/details/72845409
首先,用户选择微信支付,传入订单号。
let _that = this;
Pay.done(order.bookNum, _that, 1, is_true => {
if (is_true) {
Alert.success('支付成功', {
position: 'top',
offset: 40,
onRouteClose: true,
timeout: 1500,
onClose: function () {
Utility.toPage('order');
}
});
} else {
Alert.error('微信支付失败,请重新尝试', {position: 'top',
offset: 40,onRouteClose: true, timeout: 2000});
return;
}
});
Pay组件支付实现
支付过程中先由微信生成预支付信息,用户支付完成后执行成功提示等操作。同时,后台接口提供订单回调,修改订单状态业务等。
import 'react-jweixin';
let Pay = {};
/**
* 支付的回调
*/
let payCallback;
/**
* 开始支付--这个方法其实是从后台请求微信支付签名。成功继续,失败回调
* @param _order_id 传bookNum即可
* @param openid 微信公众号openid
* @param _thisObj 传this即可
* @param _payType 支付类型--暂时只有微信
* @param _callback 回调 true or false
*/
Pay.done = function (_order_id, _thisObj, _payType, _callback) {
let params = new FormData();
params.append('searchParam.bookNum', _order_id);
params.append('searchParam.useFlag', 0); //0新商户
let openid = sessionStorage.getItem('openid');
params.append('searchParam.openid', openid);
if (typeof _callback === "function") {
payCallback = _callback;
}
ApiClientUtil.post(ApiInfo.PaySign, params).then((data) => {
if(data.head.success) {
this.doneResponse(data.body);
}
}).catch(() => {
payCallback(false);
});
};
Pay.doneResponse = function (result) {
this.callPay(result);
};
Pay.callPay = function (code) {
if (typeof WeixinJSBridge === "undefined") {
if (document.addEventListener) {
document.addEventListener('WeixinJSBridgeReady', this.jsApiCall, false);
} else if (document.attachEvent) {
document.attachEvent('WeixinJSBridgeReady', this.jsApiCall);
document.attachEvent('onWeixinJSBridgeReady', this.jsApiCall);
}
} else {
this.jsApiCall(code);
}
};
Pay.jsApiCall = function (code) {
WeixinJSBridge.invoke('getBrandWCPayRequest', code, function (res) {
if (!Utility.isNull(res) && !Utility.isNull(res.err_msg) && res.err_msg.indexOf('ok') > 0) {
payCallback(true);
} else {
payCallback(false);
}
});
};
export { Pay as default }
React-Bmap API
https://huiyan-fe.github.io/react-bmap/examples/
React-BMap只是利用了React组件的生命周期,来调用对应的百度地图JavaScript Api的方法,比如在componentDidMount和componentDidUpdate的时候在地图上添加覆盖物,componentWillUnmount的时候移除覆盖物,React对应的render渲染函数模块返回的是null。所以这里面地图相关的dom并不是react渲染的,真正创建地图之类的还是使用百度地图JavaScript Api,React-BMap只是利用了React组件的写法来封装百度地图JavaScript Api,使我们在使用React的时候能更方便的使用百度地图JavaScript Api。
申请百度地图账号
首先去百度地图开放平台官网注册成为开发者.
http://lbsyun.baidu.com/
创建应用,获取移动账户秘钥。
npm install react-bmap
首页引入api
引入组件
import { Map, Marker, Polyline, PointLabel, NavigationControl, MarkerList } from 'react-bmap';
具体实现实例
基本地图标记
<Map center={center} zoom="13">
{
mapdata.map((item, index) => {
if (isFlag==0) {//
return (
<Marker key={index} position={{lng: item.longitude,lat: item.latitude}} >
<img style={{height:'3rem',width:'3rem'}} src={require('./images/air.png')} />
<div style={{fontSize:'0.8rem',color:'red',fontWeight: 'bold', width:'30rem'}}>{item.num} </div>
</Marker>
)
}
})
}
<NavigationControl />
</Map>
二、其他实例
<Map center={center} zoom="17" style={{height: '44rem'}}>
{/* */}
{/* */} // 绘制标记点
{
markers.map((marker, index) => {
return (
<Marker position={{lng: marker.lng, lat: marker.lat}} offset={new window.BMap.Size(-15, -20)}>
<img src={require('./images/start_pos.png')} onClick={(event) => this.handleClick(marker, event)}/>
<div id={'marker'+marker.porder} style={{fontSize:'0.8rem',color:'red', display:'none'}}>{marker.name} </div>
</Marker>
)
})
}
<Polyline strokeColor='blue' path={ markers} /> //绘制线路
<NavigationControl anchor="BMAP_ANCHOR_TOP_RIGHT"/>
</Map>
API
https://www.npmjs.com/package/qrcode-react
npm install qrcode-react
引入组件
import QRCode from 'qrcode-react';
具体使用
<Dialog type="ios" show={this.state.showQR} buttons={this.state.passStyle.buttons} >
<QRCode size={236} value={ qrstr } />
</Dialog>
react的提示组件,界面和样式可根据实际需要设置。
npm install react-s-alert --save
在APP中引入样式与库文件
import Alert from 'react-s-alert';
require('react-s-alert/dist/s-alert-default.css');
require('react-s-alert/dist/s-alert-css-effects/genie.css');
具体使用
import Alert from 'react-s-alert';
警告提示
Alert.warning('请选择', {position: 'top',
offset: 40,onRouteClose: false, timeout: 1500});
错误提示
Alert.error(json.head.msg,
{position: 'top', offset: 40,onRouteClose: false, timeout: 2000,
onClose: function () {
Utility.toPage('order');
}
});
onClose事件处理提示完之后的操作。
react的确认提示框组件,也可以根据自己需要修改样式。
npm install react-confirm-alert --save
引入相关样式与组件
import { confirmAlert } from 'react-confirm-alert'; // Import
import 'react-confirm-alert/src/react-confirm-alert.css' // Import css
以退票提示为例,在退票操作之前,询问用户是否确认退票(或者常见的删除前确认提示)。
confirmAlert({
title: '操作提示',
message: '退票将产生['+fee+']元手续费,您确认执行退款操作吗?',
buttons: [
{
label: '确认',
onClick: () => this.refund() // 执行退票功能
},
{
label: '取消',
onClick: () => console.log('取消了操作...')
}
]
})
关于提示的样式可以根据API来设置或者到源码里改改界面颜色。
API
https://www.npmjs.com/package/react-confirm-alert
做完这一个项目,对react以及相关组件了解了个大概,但现在看代码还是很杂很乱,看着闹心。
为了赶进度,一些react特性没有好好地应用,还是原始的JavaScript方式。
这次基本总结的同时也开始进行了项目重构,希望可以发挥react的优势,优化代码,优化调用以及实现方式等,争取年前完成。
文献资料
从零搭建React全家桶框架教程
https://github.com/brickspert/react-family
https://react.docschina.org/docs/thinking-in-react.html