react好租客项目Day06-axios封装封装搜索导航栏组件实现筛选功能

axios优化&环境变量

目标

  • 能够通过 axios.create() 方法来构建axios实例对象,并且配置baseURL
  • 能够知道 .env.development 和 .env.production 两个文件的作用
  • 能够配置开发环境变量
  • 能够在代码中引入配置的环境变量的值

每一次我们请求接口的时候,每一次都需要写相同的baseUrl。例如:http://localhost:8080,这样太繁琐,所以我们可以对网络请求进行优化,接口域名、图片域名、分为开发环境和生产环境,直接写在代码中,项目发布时,很难替换

配置统一的URL(★★)

axios.defaults.baseURL = 'http://localhost:8080'
// 或者
const instance = axios.create({baseURL: 'http://localhost:8080'
})

配置生产环境和开发环境(★★★)

// 通过脚手架的环境变量来解决 开发环境
在开发环境变量文件 .env.development 中,配置 REACT_APP_URL= http://localhost:8080// 通过脚手架的环境变量解决, 生成环境
在生产环境变量文件 .env.production 中,配置 REACT_APP_URL=线上接口地址

使用环境变量(★★★)

在react官网中,有详细说明环境变量的配置

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6ugvaX4h-1575111776952)(images/环境变量.png)]

在里面会发现对应的文件还有个 .local 后缀的文件,这个文件的优先级更高,以下就是你输入不同命令,执行文件的优先级

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-crEAa03x-1575111776953)(images/环境变量优先级.png)]

  • 在项目根目录中创建文件 .env.development

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RdcFacOY-1575111776954)(images/环境变量配置-01.jpg)]

  • 在该文件中添加环境变量 REACT_APP_URL(注意:环境变量约定REACT_APP 开头),设置 REACT_APP_URL=http://localhost:8080
REACT_APP_URL =http://localhost:8080
  • 重新启动脚手架,脚手架在运行的时候就会解析这个文件
  • 在utils/url.js 中,创建 BASE_URL 变量,设置值为 process.env.REACT_APP_URL
  • 导出BASE_URL
// 配置baseURL
export const BASE_URL = process.env.REACT_APP_URL
  • 在我们页面引入就能使用了
import {BASE_URL} from '../../utils/url.js'

axios 优化(★★★)

  • 在utils/api.js 中,导入 axios和BASE_URL
  • 调用 axios.create() 方法创建一个axios实例
  • 给 create 方法,添加配置baseURL,值为 BASE_URL
  • 导出API对象
import axios from 'axios'
import { BASE_URL } from './url.js'// 创建axios的配置文件,里面配置baseURL路径
const config = {baseURL: BASE_URL
}// 根据create 方法来构建axios对象
const instance = axios.create(config)export { instance }
  • 导入API,代替之前直接利用axois请求的代码
import {instance} from '../../utils/api.js'

列表找房功能

目标

  • 说出抽取搜索导航栏的好处
  • 能够说出抽取搜索导航栏的思路,并且参照老师代码能够实现抽取功能
  • 能够说出条件筛选功能的大致思路,Filter组件作用,FilterTitle组件作用,FilterPicker组件作用,FilterFooter组件作用
  • 能够参照老师代码实现FilterTitle组件中的逻辑代码
  • 能够参照老师代码实现FilterPicker组件的逻辑代码

功能分析

  • 搜索导航栏组件的封装
  • 条件筛选栏组件封装
  • 条件筛选栏吸顶功能
  • 房屋列表

顶部搜索导航栏

封装搜索导航栏组件(★★★)

  • 在components 目录中创建组件 SearchHeader/index.js
  • 把之前写过的结构拷贝到这个文件中
  • 然后把跟首页相关的数据去掉,标题,城市名称
  • 通过props来进行传递
import { Flex } from 'antd-mobile';
import React from 'react'
import {withRouter} from 'react-router-dom'
import './index.scss'
import PropTypes from 'prop-types'function SearchHeader({ history, cityName}) {return ({/* 左侧白色区域 */}{/* 位置 */} history.push('/citylist')}>{cityName}{/* 搜索表单 */} history.push('/search')}>请输入小区或地址{/* 右侧地图图标 */} history.push('/map')} />)
}
// 设置校验
SearchHeader.propTypes = {cityName: PropTypes.string.isRequired
}
export default withRouter(SearchHeader)

把搜索导航栏引入到HouseList中,调整相应样式

import React from "react";
import SearchHeader from "../../components/SearchHeader";let {label} = JSON.parse(localStorage.getItem('localCity'))export default class HouseList extends React.Component {componentDidMount(){console.log('houseList')}render() {return ( );}
}
  • 在找房页面SearHeader组件基础上,调整结构

    • 我们需要SearHeader组件样式,所以我们还需要传递className的属性进去,调整一下SearchHeader组件
    function SearchHeader({ history, cityName, className }) {return (// search-box 这个样式不能去掉,所以我们可以先通过数组的方式,添加多个类名,然后利用 join 方法转成字符串...)
    }
    
  • 给SearchHeader组件传递className属性,来调整组件样式,让其适应找房页面效果,下面是HouseList的头布局

 this.props.history.go(-1)}> 

  • 创建 index.module.css,设置相应的样式,修改了一些组件中的全局样式,所以我们需要通过 :global来设置
/* 覆盖 searchHeader的样式 */.header {height: 45px;background-color: #f5f6f5;padding: 0 10px;
}/* 控制左侧小箭头 */.header :global(.icon-back) {font-size: 16px!important;color: #999;
}/* 控制右侧的图标 */.header :global(.icon-map) {color: #00ae66;
}/* 控制search输入框 */.header :global(.search) {height: 30px;
}.searchHeader {position: relative;top: 0;padding: 0;
}

条件筛选

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-YOwKeIdg-1575111776956)(images/找房页面分析.png)]

结构分析

  • 父组件:Filter
  • 子组件:FilterTitle 标题菜单组件
  • 子组件:FilterPicker 前三个菜单对应的内容组件
  • 子组件:FilterMore 最后一个菜单对应的内容组件

功能分析

  • 点击FilterTitle组件菜单,展开该条件筛选对话框,被点击的标题高亮
  • 点击取消按钮或空白区域,隐藏对话框,取消标题高亮
  • 选择筛选条件后,点击确定按钮,隐藏对话框,当前标题高亮
  • 打开对话框时,如果有选择的条件,那么默认显示已选择的条件
  • 打开对话框已经隐藏对话框有动画效果
  • 吸顶功能

FilterTitle组件实现(★★★)

思路

  • 根据标题菜单数据,渲染标题列表
  • 标题可以被点击
  • 标题高亮
    • 点击时高亮
    • 有筛选条件选中时
    • 标题高亮状态:提升至父组件Filter中,由父组件提供高亮状态,子组件通过props接受状态来实现高亮
    • 原则:单一数据源,也就是说,状态只应该有一个组件提供并且提供操作状态的方法,其他组件直接使用组件中状态和操作状态的方法即可

步骤

  • 通过props接受,高亮状态对象 titleSelectedStatus
  • 遍历titleList数组,渲染标题列表
  • 判断高亮对象中当前标题是否高亮,如果是,添加高亮类
// FilterTitle
// 条件筛选栏标题数组:
const titleList = [{ title: "区域", type: "area" },{ title: "方式", type: "mode" },{ title: "租金", type: "price" },{ title: "筛选", type: "more" }
];export default function FilterTitle({titleSelectedStatus}) {return ({/* 遍历标题数组 */}{titleList.map(item => {// 获取父组件传递过来的状态let isSelected = titleSelectedStatus[item.type];return ({/* 选中类名: selected */}{" "}{item.title} );})});
}// Filter
/*** 标题高亮状态*/
const titleSelectedStatus = {area: false,mode: false,price: true,more: false
}
export default class Filter extends Component {state = {titleSelectedStatus};render() {let { titleSelectedStatus } = this.state;return ({/* 前三个菜单的遮罩层 */}{/* 标题栏 */}{" "}...);}
}
  • 给标题项绑定单击事件,在事件中调用父组件传过来的方法 onClick
  • 将当前标题type,通过onClick的参数,传递给父组件
  • 父组件中接受到当前type,修改改标题的选中状态为true
// Filter 
/*** 标题高亮状态*/
const titleSelectedStatus = {area: false,mode: false,price: false,more: false
};export default class Filter extends Component {...// 父元素提供子元素调用的函数onTitleClick = type => {this.setState({titleSelectedStatus: {...titleSelectedStatus,[type]: true}});};render() {let { titleSelectedStatus } = this.state;return ({" "}{/* 前三个菜单的遮罩层 */}{" "}{/* 标题栏 */}{" "}...{" "});}
}// FilterTitle
export default function FilterTitle({titleSelectedStatus,onClick}) {return ({/* 遍历标题数组 */}{titleList.map(item => {...return ( onClick(item.type)}>{/* 选中类名: selected */}{" "}{item.title} );})});
}

FilterPicker 组件(★★★)

思路分析

  • 点击前三个标题展示该组件,点击取消的时候隐藏
  • 使用PickerView组件来实现页面效果
  • 获取到PickerView组件中,选中的筛选条件值
  • 点击确定按钮,隐藏该组件,将获取到的筛选条件值传递给父组件
  • 展示或隐藏对话框的状态:由父组件提供,通过props传递给子组件
  • 筛选条件数据:由父组件提供(因为所有筛选条件是通过一个接口来获取的),通过props传递给子组件

使用步骤

定义openType,实现FilterPicker显示隐藏
  • 在Filter组件中,提供组件展示或隐藏的状态:openType
  state = {...// 控制FilterPicker或 FilterMore组件的展示和隐藏openType: ""};
  • 在render方法中判断 openType的值为 area/mode/price 时,就显示 FilterPicker组件,以及遮罩层
{/* 前三个菜单的遮罩层 */} 
{ openType === "area" || openType === "mode" || openType === "price" ? () : ("")}
...
{/* 前三个菜单对应的内容: */}
{openType === "area" || openType === "mode" || openType === "price" ? () : ("")}
  • 在 onTitleClick方法中,修改状态 openType为当前 type,展示对话框
// 父元素提供子元素调用的函数
onTitleClick = type => {this.setState({titleSelectedStatus: {...titleSelectedStatus,[type]: true},openType: type});
};
  • 在Filter组件中,提供onCancel方法(作为取消按钮和遮罩层的事件)
// 取消,
onCancel = () => {// 隐藏对话框this.setState({openType:''})
};
  • 在onCancel方法中,修改状态 openType为空,隐藏对话框
  • 讲onCancel通过props传递给FilterPicker组件,在取消按钮的单击事件中调用该方法
  • 在Filter组件中,提供onSave方法,作为确定按钮的事件处理
// 在父组件 Filter中定义 确定和取消的函数
// 取消
onCancel = () => {// 隐藏对话框this.setState({openType: ""});
};
// 保存,隐藏对话框
onSave = () => {this.setState({openType: ""});
};
// 传递给FilterPicker
render(){return (...)
}// 在FilterPicker里面进行一次中转,最后这个按钮是在FilterFooter里面
render() {let { onCancel ,onSave} = this.props;return (<>{/* 选择器组件: */}{/* 底部按钮 */});
}// 在FilterFooter里面调用
function FilterFooter({cancelText = '取消',okText = '确定',onCancel,onOk,className
}) {return ({/* 取消按钮 */}{cancelText}{/* 确定按钮 */}{okText})
}
获取筛选条件数据
  • 在Filter组件中,发送请求,获取所有筛选条件数据
  • 将数据保存为状态:filtersData
// 获取筛选数据async getFilterData() {let { value } = JSON.parse(localStorage.getItem("localCity"));let res = await instance.get(`/houses/condition?id=${value}`);console.log(res);this.setState({filtersData: res.data.body});}
  • 封装方法 renderFilterPicker 来渲染FilterPicker组件
// 渲染FilterPicker组件的方法renderFilterPicker() {const { openType } = this.state;if (openType !== "area" && openType !== "mode" && openType !== "price")return null;return ;}
render(){return (...{this.renderFilterPicker()})
}
  • 在方法中,根据openType的类型,从filtersData中获取需要的数据
  • 讲数据通过props传递给FilterPicker组件
// 渲染FilterPicker组件的方法renderFilterPicker() {const {openType,filtersData: { area, subway, rentType, price }} = this.state;if (openType !== "area" && openType !== "mode" && openType !== "price")return null;// 拼接数据let data = [];// pickerView显示的列数let cols = 3switch (openType) {case "area":// 区域数据data = [area, subway];break;case "mode":// 方式数据data = rentType;cols = 1break;case "price":// 租金数据data = price;cols =1break;}return ();}
  • FilterPicker组件接收到数据后,讲其作为PickerView组件的data
export default class FilterPicker extends Component {render() {let { onCancel, onSave, data ,cols} = this.props;return (<>{/* 选择器组件: */}{/* 底部按钮 */});}
}


本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!

相关文章