项目目录搭建
1 | npx create-react-app react-jianshu |
1 | // style.js |
##
头部 Header 组件布局
src/common/header/style.js
1 | import styled from 'styled-components' |
src/common/header/index.js
1 | import React, { Component } from 'react' |
App.js
1 | import Header from './common/header' |
使用 iconfont
嵌入头部图标
在 iconfont.cn 查找下载需要的 字体图标;并在 index.js
文件中引入iconfont.css
文件
代码实现参考 Header 组件布局
搜索框动画效果实现
首先设置 onFoucs
和 onBlur
的样式,具体代码参考 Header 组件布局
安装第三方动画模块 npm install react-transition-group --save
在 header/index.js
文件中引入: import { CSSTransition } from 'react-transition-group'
使用 <CSSTransition>
标签包裹 <NavSearch>
组件
设置 .slide-enter
、.slide-enter-active
、.slide-exit
、.slide-exit-active
样式
1 | <CSSTransition |
使用 React-Redux
进行数据管理
下载 npm install redux react-redux --save
src/store/index.js
1 | import { createStore } from 'redux' |
src/store/reducer.js
1 | /* 默认数据 */ |
在 App.js
文件中引入 store:
1 | import store from './store' |
在 header 组件中使用 store 中的数据
引入 connect: import { connect } from 'react-redux'
修改 header 组件中所有的数据引用:将 this.state.focused
修改为 this.props.focused
修改 handleInputFoucs
和 handleInputBlur
方法以及调用 this.props.handleInputFocus
1 | import { connect } from 'react-redux' |
代码优化:Header 组件为 无状态组件
1 | import React from 'react' |
使用 combineReducers 完成对数据的拆分管理
如果将所有的数据全部存放在 reduce.js
中,如果将来要管理的数据变得非常多的时候,是很不方便维护的,这时可以对 reducer
进行拆分:
修改 store/index.js
文件内容,方便 开发工具 的使用
参考:https://github.com/zalmoxisus/redux-devtools-extension
1 | import { createStore, compose } from 'redux' |
对数据进行拆分header/store/reducer.js
1 | // 将 store/reducer.js 文件中的代码剪贴到 header/store/reducer.js 文件中 |
header/store/index.js
1 | // 直接在 reducer.js 中引入 header/store/reducer.js 路径很长,可以这样处理,然后引入这个文件 |
store/reducer.js
1 | // 将拆分出去的数据合并 |
header/index.js
1 | // 修改 header/index.js 文件中的 mapStateToProps |
actionCreators 与 constants 的拆分
header/store/actionCreators.js
1 | import * as constants from './constants' |
header/index.js
1 | import { actionCreators } from './store' |
header/store/constants.js
1 | export const SEARCH_FOCUS = 'header/SEARCH_FOCUS'; |
header/store/reducer.js
1 | import * as constants from './constants' |
header/store/index.js
1 | import reducer from './reducer'; |
使用 lmmutable.js 来管理 store 中的数据
安装: npm install immutable --save
header/store/reducer.js
1 | import { fromJS } from 'immutable' |
header/index.js
1 | const mapStateToProps = (state) => { |
使用 redux-immutable 统一数据格式
安装 npm install redux-immutable --save
store/reducer.js
1 | import { combineReducers } from 'redux-immutable' |
header/index.js
1 | const mapStateToProps = (state) => { |
热门搜索样式布局
header.js
1 | import { SearchInfo, SearchInfoTitle, SearchInfoSwitch, SearchInfoList, SearchInfoItem } from './style' |
AJAX 获取推荐数据
安装: npm install axios --save
将 Header 先恢复为普通组件
1 | class Header extend Component { |
异步请求 ajax 需要放在 redux-thunk
安装:npm install redux-thunk --save
在 store/index.js
文件中引入:import thunk from 'redux-thunk'
1 | import { createStore, compose, applyMiddleware } from 'redux' |
有了 redux-thunk 就可以在组件中进行异步操作了
1 | // header/index.js |
热门搜索换页功能实现
1 | // header/store/reducer.js |
header/index.js
1 | getListArea () { |
换页旋转动画效果实现
1 | // header/index.js |
避免无意义的请求发送
每次点击搜索框都会发送请求
1 | <NavSearch |
在 React 中使用路由功能
路由就是根据 URL 的不同显示不同的内容
安装: npm install react-router-dom --save
1 | // App.js |
首页组件的拆分
src/pages/home/index.js
1 | import React, { Component } from 'react' |
src/pages/home/style.js
1 | import styled from 'styled-components' |
src/pages/detail/index.js
1 | import React, { Component } from 'react' |
App.js
1 | import Home from './pages/home' |
src/pages/home/components/List.js
1 | import React, { Component } from 'react' |
src/pages/home/components/Recommend.js
1 | import React, { Component } from 'react' |
src/pages/home/components/Topic.js
1 | import React, { Component } from 'react' |
src/pages/home/components/Writer.js
1 | import React, { Component } from 'react' |
首页专题区域布局及 reducer 的设计
pages/home/components/Topic.js
1 | import { connect } from 'react-redux' |
pages/home/style.js
1 | export const TopicWrapper = styled.div` |
pages/home/store/reducer.js
管理 Home 页面中的数据
1 | import { fromJS } from 'immutable'; |
src/store/reducer.js
1 | import { reducer as HomeReducer } from '../pages/home/store' |
首页文章列表制作
home/components/List.js
1 | import { ListItem, ListInfo } from '../style' |
home/style.js
1 | export const ListItem = styled.div` |
pages/home/store/reducer.js
1 | const defaultState = formJS({ |
首页推荐部分
home/components/Recommend.js
1 | import { connect } from 'react-redux' |
pages/home/style.js
1 | export const RecommendWrapper = styled.div` |
pages/home/store/reducer.js
1 | const defaultState = formJS({ |
home/components/Writer.js
1 | import { WriterWrapper } from '../style' |
pages/home/style.js
1 | export const WriterWrapper = styled.div` |
首页异步数据获取
api/home.json
1 | { |
src/pages/home/index.js
1 | import { connect } from 'react-redux' |
home/store/reducer.js
1 | switch(action.type) { |
异步操作代码拆分优化
pages/home/index.js
1 | import { actionCreators } from './store' |
pages/home/store/actionCreators.js
1 | import axios from 'axios' // 删除 index.js 中的 axios 引入 |
pages/home/store/constants.js
1 | export const CHANGE_HOME_DATA = 'home/CHANGE_HOME_DATA' |
home/store/reducer.js
1 | import * as constants from './constants' |
pages/home/store/index.js
1 | import * as actionCreators from './actionCreators' |
实现加载更多功能
pages/home/components/List.js
1 | import { LoadMore } from '../style' |
pages/home/store/actionCreators.js
1 | import { fromJS } from 'immutable' |
public/api/homeList.json
1 | { |
返回顶部
pages/home/index.js
1 | import { BackTop } from './style' |
首页性能优化及路由跳转
pages/home/index.js
1 | import React, { PureComponent } from 'react' |
src/app.js
1 | // src/common/header/index.js |
详情页面布局
src/pages/detail/index.js
1 | import { DetailWrapper, Header, Content } from './style' |
src/pages/detail/style.js
1 | import styled from 'styled-components' |
使用 redux 管理详情页面数据
创建文件
1 | // pages/detail/store/actionCreators.js |
src/store/reducer.js
1 | import { reducer as detailReducer } from '../pages/detail/store' |
pages/detail/index.js
1 | import { connect } from 'react-redux' |
异步获取数据
pages/detail/index.js
1 | import { actionCreators } from './store' |
页面路由参数的传递
pages/home/components/List.js
1 | <Link to={'/detail/' + item.get('id')}></Link> |
App.js
1 | <Route path='/detail/:id'> |
pages/detail/index.js
1 | componentDidMount(){ |
登录页面布局
pages/login/index.js
1 | import React, { PureComponent } from 'react' |
pages/login/style.js
1 | import styled from 'styled-components' |
登录功能实现
创建文件
1 | // pages/login/store/actionCreators.js |
src/store/reducer.js
1 | import { reducer as loginReducer } from '../pages/login/store' |
src/common/header/index.js
1 | import { actionCreators as loginActionCreators } from '../../pages/login/store' |
pages/login/index.js
1 | import { Redirect } from 'react-router-dom' |
pages/login/store/actionCreators.js
1 | import axios from 'axios' |
pages/login/store/reducer.js
1 | case constants.CHANGE_LOGIN: |
public/api/login.json
1 | { |
登录鉴权及代码优化
pages/write/index.js
1 | // 复制 login/index.js 中的代码,删除不要的 mapDispatch |
异步组件及withRouter路由方法的使用
安装:npm install react-loadable --save
pages/detail/loadable.js
1 | import React from 'react' |
项目上线流程
删除:public/api
打包:npm run build
将 build 中的所有文件放在后端项目中