API:点这里,感谢
项目地址:Github
pc主页.png 移动端主页.png 内容详情.png 评论.png依赖
- 框架:react
- 路由:react-router 4.0
- HTTP请求:whatwg-fetch(主要是保证fetch兼容性)
- 设备判断:react-responsive
- UI:antd(主要是PC端)
- CSS预处理:stylus
- 移动端布局:flexible.js + rem
遇到的问题以及解决方法
API问题
1. API不支持跨域
那位大大的API是不支持跨域的,于是自己使用koa2+request-promise+koa-cors进行了后台API转发 ,代码如下,在源码的server/server.js中
const Koa = require('koa');
const cors = require('koa-cors');
const rp = require('request-promise');
const app = new Koa();
const basrUrl = "http://news-at.zhihu.com";
app.use(cors());
const main = async (ctx, next) => {
const pathname = ctx.request.path;
ctx.response.type = 'json';
ctx.response.body = JSON.parse(await rp(basrUrl + pathname));
};
app.use(main);
app.listen(9999);
console.log('app started at port 9999...');
启动了服务后,对自己的本地服务器进行请求即可,只需要将原API的“http://news-at.zhihu.com”改为“http://localhost:9999”。
2.知乎图片防盗链问题
“知乎的图片可能会通过请求头的referer参数判断,如果不是指定的域名会返回403,如果精通后台的同学,可以去访问这些图片来缓存这些图片,我是搜索到了一个相对简单一些的办法,点击链接,主要用到的是Images.weserv.nl这个网站,可以缓存图片,而且可以修改图片的尺寸大小。
具体是我自定义了一个过滤器,在需要用到地方,把知乎的url替换成这个图片代理网站的url,这样图片就可以显示了。”
const replaceUrl = (srcUrl) => {
return srcUrl.replace(/http\w{0,1}:\/\/p/g, 'https://images.weserv.nl/?url=p')
}
然后对返回的data直接进行如下操作:
data = JSON.parse(replaceUrl(JSON.stringify(data)));
data中的图片就可以正常引用了。
参考
- 使用vue完成知乎日报web版
这个博主做的vue版十分完善,使用vue的同学可以看看。
React/React Router4.0相关问题
1.React Router为何在url相同参数不同的情况下跳转但是并不刷新页面?
案情描述:当我从“日常心理学”主题日报跳转至“用户推荐日报”时,组件内部数据没有刷新,仍然为“日常心理学”的数据,而非“用户推荐日报”数据。
路由一:/topic/abc
路由二:/topic/efg
即路由一跳转至路由二组件没有重新去获取数据,其实这个问题在vue-router里面也有,vue中是通过watch监测路由参数变化判断组件是否需要刷新。
解决方案:我们一般是在componentDidMount中获取数据,当我从路由一跳转至路由二,仍然为同一组件,所以并没有重新Mount。
也就是说,同级路由相同组件跳转的情况下,componentDidMount方法仅仅在第一次Mount的时候触发了,路由跳转并没有触发该方法。
但是路由跳转之后 props.params 变了, 我们可以在componentWillReceiveProps中进行第二次的数据拉取,于是会触发更新过程。
参考:
2. 如何保持路由组件keep-alive,跳转返回后不重新渲染
案情描述:假如我有以下两种需求:
-
有一个下拉加载更多列表页,当列表项达到1000项,点击列表项到详细页,当点击返回按钮时,保持列表页的一千条记录。
-
把下拉加载更改为翻页,假设不是通过改变URL,只是单纯的ajax请求,击列表项到详细页,当点击返回按钮时,返回离开时的那一页。
其实我之前也写过篇文章,戳这里
某一天我很无聊,正在刷知乎日报,我要完成一个目标:从2017-10-07号一直看到2017-01-01。
我一直看到了10-05的信息,此时列表已经较长了,点进去,看完了,返回,组件重新渲染,滚动条还原,数据还原,我又下拉到三天前,又看了一篇文章,返回,重复上述操作。
随着列表越来越长,我每一次返回的代价都是巨大的,我都要重新拉回到之前看到的地方,且数据还要重新加载,因为每次返回只能返回latest信息,无法返回你通过ajax下拉加载beforeNews的信息。
很显然,这个问题不解决,用户体验绝对负分。
解决方案:
方式一:
把加载页面的必要信息放URL里,后退就是浏览器的后退,回退时组件依然会重新渲染,componentDidMount的时候根据这些url的信息去请求API。数据保持离开时最新的状态。
这种方法在翻页情况下使用固然可以,但是像下拉加载那样,是多次请求的数据集合,就无法通过这种方式解决,而且后退时依旧进行了一次http请求。
方式二:
- 在componentWillUnMount方法中存储当前已经加载的数据到本地(sessionStorage,localStorage)
- 详情回退后,从本地取出数据,进行渲染(所以componentDidMount内部要判断本地是否已经存在数据,如果存在则return,不存在从服务器获取数据)
- 这种方式回退时不需要再次进行http请求,也能满足下拉的情况,我选择的就是这种方式。
参考:
3.后退时记录滚动条位置
案情描述:同上
恢复数据还不够,只是省下了我加载的时间,而我还是要拉到最底部,所以滚动条的位置也需要恢复。
思路同上,将当前的scrollTop记录在一个全局变量中(我是作为LatestNews组件的属性,为了让需要滚动的组件获取到),在componentWillUnMount方法中存储当前的滚动条位置(sessionStorage,localStorage)。
当后退返回时取出位置赋值给LatestNews.scrollPoint,确认数据更新后(一定要确认数据更新后,setState方法是异步方法),在setState的回调中根据Latest.scrollPoint滚动需要滚动的组件。
参考
4. 当state的属性为数组,如何setState比较好
这个主要是push,unshift方法的返回值造成state没有赋值成功,最好先用slice把数组copy以下,用copy的数组去push(data),然后再去setState.
const _storiesQue = this.state.storiesQue.slice();
_storiesQue.push(data);
this.setState({
storiesQue:_storiesQue
});
这里有一篇文章写的很好
移动端适配问题
1.react-responsive
npm install react-responsive --save
然后在index.js中
import MediaQuery from 'react-responsive';
ReactDOM.render(
<Router>
<div>
<MediaQuery query='(min-device-width:1224px)'>
<PCIndex/>
</MediaQuery>
<MediaQuery query='(max-device-width:1224px)'>
<MBIndex/>
</MediaQuery>
</div>
</Router>,
document.getElementById('root')
);
在设置了query参数后,就可以根据设备的宽度来决定渲染PC组件还是移动端组件,独立开发即可,当然,要考虑组件的复用。
2.flexible.js+rem布局
关于flexible.js+rem布局有疑问的可以看下面的文章
参考:
使用lib-flexible.js
首先:
npm install lib-flexible --save
然后在index.js中,即可使用rem进行适配了。
import 'lib-flexible';
将px转为rem的几种方法
- 可以通过开发工具的插件,例如sublime上面就有插件
- webpack的px2rem-loader,对stylus文件无效
- 使用定义好的预处理器(stylus,sass,less)的方法
以下是stylus的示例
//定义一个变量和一个mixin
$baseFontSize = 16; //默认基准font-size
px2rem(name, px){
{name}: px / $baseFontSize * 1rem;
}
// 使用示例:
.container {
px2rem('height', 240);
}
// stylus翻译结果:
.container {
height: 3.2rem;
}
参考:
2. Drawer组件遮罩层滑动组织底部窗体滚动
因为PC端用的是ant-design,对移动端不是很友好,所以移动端基本是自己手写(除了引入了antd的轮播图(Carousel)和折叠面板(Collapse),不过真的丑)。
Drawer组件,如下图所示:
Drawer.gif设计思路是:Drawer组件先通过绝对定位移出视窗外,当点击menu按钮时,添加class,改变left以及top的值,当然得有transition营造动画效果(还可以通过transform:translateZ(0)硬件加速),同时渲染遮罩层mask,宽高均为100%.
z-index层级:Drawer>mask>index
案件描述:当我不小心滑动到遮罩层mask的时候,底部list也在滑动,这显然不是我想要的,这里主要是看了张鑫旭大佬的文章。我就不细说了。
参考
关于组件复用
移动端以及PC端的news_detail/comments均是基于公共的模块news_content/comments_content/comment/avatar进行二次包装。
以内容详情页news_detail举例。
PC端内容详情页news_detail这一模块中包含评论comments模块,而移动端包含一个news_header头部模块。
所以采取对共有模块news_content进行引用,同时PC端引用comments模块进行组合,移动端引用news_header模块进行组合,在各自的news_detail里进行数据的获取,共有模块只负责展示,属于stateless组件。
这样一来,即使完全删除PC文件或者移动端文件,剩下的一方依然可以正常使用,二者之间不存在相互关联引用。
关于打包(未解决)
记得要将开发时对webpack.config.dev.js做的操作也要对webpack.config.prod.js操作一遍。
比如我开发时引入了stylus,自己也配置了webpack.config.dev.js的环境[stylus, stylus-loader ],在开发环境下没有问题。
但是打包之后stylus文件全部失效,一看好好的躺在static/media文件夹呢,说明生产环境的webpack没有配置好。再对webpack.config.prod.js里面的loaders也添加一遍就好了,其它同理。
然后:
npm run build
如何在github-page上展示呢?
戳这里,虽然说的是vue项目,但是build后操作一模一样,除了将命令中dist=>build
上传后会发现只有在github根路径才能正常访问,而不是仓库下面的路径,很坑啊。。。不是很懂这个原因啊,恳请高手替我解答一下,资源路径没问题,就是路由匹配不正确。
理想情况下应该是这个路径可以直接访问
https://nickname.github.io/project/[index.html]
但是这个gh-pages的的路径进去是一个header(PC端),路由'/'的组件没有渲染出来,点了以下header的首页'/topics'的链接,一切恢复正常,但是url却变成了下面这个:
https://nickname.github.io/topics
项目路径被吃了,这个路径直接访问又是404,难道react-router写path的时候只能相对于域名根路径?
我看别人都是打包后资源加载问题,我发现直接打包资源路径没有问题,难道我开发时要把path由'/a'改为'/project/a'???
如果有成功在gh-pages上展示项目的同学,请务必在评论中留下您的解决方案,真的非常感谢!
参考
后续操作:完善细节,自己写折叠面板。
后话
如果我的文章给您了一点帮助,希望可以去github上给我一颗star。
能够在国庆短短几天从不知道react是啥到做个小东西出来,一直都是因为有这么多热爱分享的人,站在巨人的肩膀上!
但是仍然有很多不足,请指正。
项目地址:狂戳这里
⭐⭐⭐
其他:
我的博客:点这里
网友评论