整体思路[根据自己的Angular4+
开发经验而来,可能我写的代码有点偏向于Angular
,不过思路更清晰, 同时我会把思路说的非常清楚, 我时刻明白, 如果你不能给一个三岁小孩子讲清楚,那么你就是不会.]
以下的思路, 可以用到任何框架上面, 同时用了width做了屏幕适配, 不会对原有项目代码造成任何影响, 请放心使用
比如Angular中 把ListView当成一个div, 把渲染换成*ngFor, 数据放到 constructor函数里面返回
比如 Vue 中, 把ListView当成一个div, 把渲染换成v-for, 数据放到 data函数里面返回,
比如 React 中, 把ListView当成一个div,直接用数组的一些遍历函数map一类都可以渲染出来组件, React和下面的代码差不多
星级评价组件预览
星级评价组件预览
如果你不想画上3分钟看思路, 那么最下面有代码可以直接用,
- 星级展示思路
传入值, 1-5 任意数字
然后把把数字组成数组, 举例 假如传入的是4,那么会有一个这样的数组,也就是把值都变为true
[true, true, true, true, false]
那么就可以根据这里面的值,改变组件的类名,从而改变了样式
具体实现
- 在constructor拿到传入的级别,类型是数字类型 int
constructor(props) {
super(props);
// @param {星级,如果不传,默认是满级,值为5} this.props.rating
this.rating = this.props.rating <= 5 && this.props.rating >= 0 ? this.props.rating : 5
}
- 导入
RN
组件,同时创建一个变量用来保存,ListView
将要遍历的值
// 导入ListView
import { ListView } from "react-native";
constructor(props) {
super(props);
// @param {星级,如果不传,默认是满级,值为5} this.props.rating
this.rating = this.props.rating <= 5 && this.props.rating >= 0 ? this.props.rating : 5
// @param {是否可编辑,如果不传,默认是不可编辑,值为false} this.props.editable
this.editable = this.props.editable ? this.props.editable : false
// 定义准备被ListView包装的数组 this.ratingArr: Array<object> =[]
this.ratingArr = []
// 定义并声明 `ListView` 的包装函数 this.dSource: any
this.dSource = (new ListView.DataSource({ rowHasChanged: (r1, r2) => r1 !== r2 }))
// 定义被ListView的包装函数 this.dSource处理的数据,这个数据是用来渲染使用的
this.state = { rating: this.dSource.cloneWithRows(this.ratingArr) }
}
- 处理
this.ratingArr
的数据结构
setRatingChange = () => {
// 这里的星级是5级,如果你是其他星级,可以更改,
this.ratingArr = Array.from({ length: 5 }).map((item, index) => {
// 这句话的是,如果传入的等级小于或者等于index+1,
// 那么才返回一个true, true代表的是点亮星星个数
if (this.rating <= index + 1) {
return {
status: true,
index,
// 索引都是0开始的, 因此需要增加1
level: index + 1
}
} else {
return {
status: false,
index,
level: index + 1
}
}
})
}
- 把
this.state.rating
渲染到ListView
中,特别强调[两点]-
ListView
的属性contentContainerStyle
是用来写ListView
样式的,其中,为了防止各种版本之间的缺陷,一点要写上样式flexWrap: "wrap"
- 在
RN
的0.28版本的时候, 改变了内部代码, 从而,造成了, 需要定义即将渲染的元素的alignItems
为"flex-start"
-
render = () => {
return (<View style={styles.vView}>
<View style={[styles.vRatingV, styles.vRating_s]}>
<ListView
contentContainerStyle={styles.vRating}
dataSource={this.state.rating}
// 这里是根据数组里的是否是true进行渲染不同的样式,
// 用了箭头函数, 代表直接return一个组件,采用的是JSX语法
renderRow={(item, index) => <Text
style={[styles.rating, this.rating < item.level ? styles.rating_n : ""]}></Text>}
/>
</View>
</View>)
}
以上就是展示的代码完成了,
- 星级选择思路
- 星级选择的思路也简单, 通过上面的星级展示思路, 想必也清楚了一个核心就是根据传入的星级的等级生成一个含有true,false的值
因此,我们可以根据选择的是哪个星星, 根据该星星的索引对数组this.ratingArr = []
重新计算值, 然后把ListView
的dataSource的值进行重新赋值 - 我们增加一个点击函数事件
render = () => {
return (<View style={styles.vView}>
<View style={[styles.vRatingV, styles.vRating_s]}>
<ListView
contentContainerStyle={styles.vRating}
dataSource={this.state.rating}
renderRow={(item, index) => <Text
// 这里增加一个点击事件`this.selectStar`
// 参数是`item`和`index`, 用来进行选择
onPress={this.selectStar.bind(this, item, index)}
style={[styles.rating, this.rating < item.level ? styles.rating_n : ""]}></Text>}
/>
</View>
</View>)
}
这里是添加的点击事件匿名变量函数
selectStar = (item, Windex) => {
this.rating = item.level
this.setRatingChange()
this.setState({ rating: this.dSource.cloneWithRows(this.ratingArr) })
}
- 以上就完成了点击切换功能
怎么封装一个组件出去呢?
我们可以定义三个导入值, 为了防止报错,(比如没有传入值等), 我们需要定义三个具体的参数,用来说明,分别是
/**
- @param {星级,如果不传,默认是满级,值为5} this.props.rating
- @param {是否可编辑,如果不传,默认是不可编辑,值为false} this.props.editable
- @param {接收的函数,如果不传递,那么不会传递出去,只会什么都不做} this.props.selectStar
*/
- 我们需要在初始化的时候处理这参数, 其中第三个参数this.props.selectStar可以放到使用的时候再判断,
// @param {星级,如果不传,默认是满级,值为5} this.props.rating
this.rating = this.props.rating <= 5 && this.props.rating >= 0 ? this.props.rating : 5
// @param {是否可编辑,如果不传,默认是不可编辑,值为false} this.props.editable
this.editable = this.props.editable ? this.props.editable : false
- 对是否能都编辑,进行判断, 如果传入了可以编辑,那么就不会使用
this.editable
的默认值
同时,我们可以改造一下点击事件函数
selectStar = (item, Windex) => {
// 判断是否传入了可编辑
if (this.editable == true) {
this.rating = item.level
this.setRatingChange()
this.setState({ rating: this.dSource.cloneWithRows(this.ratingArr) })
// 把值传递出去,如果为真,那么可以作为参数传递出去,如果为假,则无法传递出去
this.props.selectStar ? this.props.selectStar(item.level) : ""
return item.level
} else {
return false
}
}
- 我们可以定义一个最大星级
- @param {最大星级,如果不传,默认是5,值为5} this.props.maxRating
- 如果父组件传入的是一个变量,或者用了很多的引用, 那么需要在子组件里面实时更改值,而不是第一次的时候就默认那样子了
componentWillReceiveProps = (nextProps) => {
this.maxRating = nextProps.maxRating ? nextProps.maxRating : 5
this.rating = nextProps.rating ? (nextProps.rating >= 0 && nextProps.rating <= this.maxRating ? nextProps.rating : this.maxRating) : this.maxRating
this.editable = nextProps.editable ? nextProps.editable : false
this.props.selectStar = nextProps.selectStar ? nextProps.selectStar : null
this.initPage()
}
我们最后封装封装代码优化一下,这样子的代码看着才更简洁, 下面附代码,
使用的使用,直接
import Rating from "./rating"
.....N多代码
render = () =>{
return (<View>
<Rating rating={this.state.companyDetail.rating ? this.state.companyDetail.rating / 10 : 5}></Rating>
</View>)
}
/*
* @Description: 星级评论组件
* @version: 0.1.0
* @Company:
* @Author: AmandaYi
* @Date: 2018-10-25
* @LastEditors: AmandaYi
* @LastEditTime: 2018-10-25
*/
/**
* @param {最大星级,如果不传,默认是5,值为5} this.props.maxRating
* @param {星级,如果不传,默认是满级,值为5} this.props.rating
* @param {是否可编辑,如果不传,默认是不可编辑,值为false} this.props.editable
* @param {接收的函数,如果不传递,那么不会传递出去,只会什么都不做} this.props.selectStar
*/
import React, { Component } from "react";
import { View, Text, StyleSheet, ListView } from "react-native";
import globalStyle, {
width,
rx
} from "./variable"
const styles = StyleSheet.create({
vView: {
position: "relative"
},
vRating: {
flexDirection: "row",
flexWrap: "wrap",
width,
},
vRatingV: {
position: "absolute",
top: 0,
left: 0,
},
vRating_s: {
zIndex: 1000
},
vRating_n: {
zIndex: 800
},
rating: {
fontFamily: "iconfont",
fontSize: rx(34),
color: '#ff6600',
marginRight: rx(10),
backgroundColor: "#ffffff",
alignItems: "flex-start",
},
rating_n: {
color: '#cccccc',
}
})
export default class Rating extends Component {
constructor(props) {
super(props);
// 定义总星级
this.maxRating = this.props.maxRating ? this.props.maxRating : 5
// @param {星级,如果不传,默认是满级,值为5} this.props.rating
this.rating = this.props.rating <= this.maxRating && this.props.rating >= 0 ? this.props.rating : this.maxRating
// @param {是否可编辑,如果不传,默认是不可编辑,值为false} this.props.editable
this.editable = this.props.editable ? this.props.editable : false
this.ratingArr = []
this.dSource = (new ListView.DataSource({ rowHasChanged: (r1, r2) => r1 !== r2 }))
this.initPage()
this.state = { rating: this.dSource.cloneWithRows(this.ratingArr) }
this.props.selectStar = null
}
selectStar = (item, Windex) => {
// 判断是否传入了可编辑
if (this.editable == true) {
this.rating = item.level
this.setRatingChange()
this.setState({ rating: this.dSource.cloneWithRows(this.ratingArr) })
// 把值传递出去,如果为真,那么可以作为参数传递出去,如果为假,则无法传递出去
this.props.selectStar ? this.props.selectStar(item.level) : ""
return item.level
} else {
return false
}
}
componentWillReceiveProps = (nextProps) => {
this.maxRating = nextProps.maxRating ? nextProps.maxRating : 5
this.rating = nextProps.rating ? (nextProps.rating >= 0 && nextProps.rating <= this.maxRating ? nextProps.rating : this.maxRating) : this.maxRating
this.editable = nextProps.editable ? nextProps.editable : false
this.props.selectStar = nextProps.selectStar ? nextProps.selectStar : null
this.initPage()
}
// 处理函数
initPage = () => {
this.setRatingChange()
}
// 改变状态
setRatingChange = () => {
this.ratingArr = Array.from({ length: this.maxRating }).map((item, index) => {
if (this.rating <= index + 1) {
return {
status: true,
index,
level: index + 1
}
} else {
return {
status: false,
index,
level: index + 1
}
}
})
}
render = () => {
return (<View style={styles.vView}>
<View style={[styles.vRatingV, styles.vRating_s]}>
<ListView
contentContainerStyle={styles.vRating}
dataSource={this.state.rating}
renderRow={(item, index) => <Text
onPress={this.selectStar.bind(this, item, index)}
style={[styles.rating, this.rating < item.level ? styles.rating_n : ""]}></Text>}
/>
</View>
</View>)
}
}
网友评论