ReactNative 的 ListView 在数据更新方面有点奇怪。
我有一个对象数组(名称:Item),它们有 3 个属性:
- 标识(整数)
- 名称(字符串)
- 启用(布尔)
项目数组显示在 ListView 中。每行显示项目的名称和表示启用状态的复选标记(红色为真,灰色为假)。在我的示例中,它被简化为一个简单的字符串“true”/“false”。
每行都嵌入在带有 onPress-EventHandler 的 TouchableOpacity 中,用于切换所代表项目的启用状态。
不幸的是,ListView 不会更新该行。
在调试 rowHasChanged-Event 时,事实证明 rowHasChanged-Event 永远无法确定更改的状态,因为在切换启用属性后两行都已经获得了新状态,而没有对 UI 进行任何更新。
这是我的代码:
项目类:Item.js
export default class Item {
constructor(id, name, enabled) {
this._id = id;
this._name = name;
this._enabled = enabled;
}
get id() {return this._id}
get name() {return this._name}
set name(value) {this._name = value}
get enabled() {return this._enabled}
set enabled(value) {this._enabled = value}
}
列表视图:
'use strict';
import React, {PropTypes, Component} from 'react';
import {View,TouchableOpacity, TouchableHighlight,
Text, ListView, StyleSheet} from 'react-native';
import Item from './Item';
class ItemListView extends Component {
constructor(props) {
super(props);
this.updateListView = this.updateListView.bind(this);
this.toggleItemEnabled = this.toggleItemEnabled.bind(this);
this.state = {
dataSource: new ListView.DataSource({
rowHasChanged: (row1, row2) => row1 != row2
}),
};
}
componentDidMount() {
this.updateListView(this.props.items)
}
componentWillReceiveProps(nextProps) {
this.updateListView(nextProps.items)
}
updateListView(items) {
this.setState({
dataSource: this.state.dataSource.cloneWithRows(
items.slice() // copy items to a new array
),
});
}
toggleItemEnabled(item) {
item.enabled = !item.enabled;
this.updateListView(this.props.items);
}
renderItem(item) {
console.log('Render', item.enabled, item.name)
return (
<TouchableOpacity style={{flex:1}} onPress={ () => this.toggleItemEnabled(item) }>
<View style={s.row}>
<Text>{item.name + ' ' + item.enabled}</Text>
</View>
</TouchableOpacity>
)
}
render() {
return (
<View style={s.container}>
<ListView
style={s.listView}
dataSource={this.state.dataSource}
renderRow={(item) => this.renderItem(item)}
/>
</View>
)
}
}
ItemListView.propTypes = {
items: PropTypes.array
};
ItemListView.defaultProps = {
items: [],
};
export default ItemListView;
const s = StyleSheet.create({
container: {
},
row: {
flex: 1,
flexDirection: 'row',
justifyContent: 'space-between',
alignItems: 'center',
marginTop: 12,
paddingBottom: 12,
paddingHorizontal: 30,
borderBottomWidth: 1,
borderBottomColor: 'gray',
},
innerContainer: {
alignItems: 'center',
},
});
如何解决这种行为,当行数据更改时 ListView 会更新?