react-native城市列表组件

前端之家收集整理的这篇文章主要介绍了react-native城市列表组件前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

城市列表选择是很多app共有的功能,比如典型的美图app。那么对于React Native怎么实现呢?

要实现上面的效果,首先需要对界面的组成简单分析,界面的数据主要由当前城市,历史访问城市和热门城市组成,所以我们在提供Json数据的时候就需要将数据分为至少3部分。

  1. const ALL_CITY_LIST = DATA_JSON.allCityList;
  2. const HOT_CITY_LIST = DATA_JSON.hotCityList;
  3. const LAST_VISIT_CITY_LIST = DATA_JSON.lastVisitCityList;

而要实现字母索引功能,我们需要自定义一个控件,实现和数据的绑定关系,自定义组件代码如下:
CityIndexListView.js

  1. 'use strict';
  2. import React,{Component} from 'react';
  3. import {
  4. StyleSheet,View,Text,TouchableOpacity,ListView,Dimensions,} from 'react-native';
  5.  
  6. import Toast,{DURATION} from './ToastUtil'
  7.  
  8. const SECTIONHEIGHT = 30;
  9. const ROWHEIGHT = 40;
  10. const ROWHEIGHT_Box = 40;
  11. var totalheight = []; //每个字母对应的城市和字母的总高度
  12.  
  13. const {width,height} = Dimensions.get('window');
  14.  
  15. var that;
  16.  
  17. const key_now = '当前';
  18. const key_last_visit = '最近';
  19. const key_hot = '热门';
  20.  
  21. export default class CityIndexListView extends Component {
  22.  
  23. constructor(props) {
  24. super(props);
  25.  
  26. var getSectionData = (dataBlob,sectionID) => {
  27. return sectionID;
  28. };
  29. var getRowData = (dataBlob,sectionID,rowID) => {
  30. return dataBlob[sectionID][rowID];
  31. };
  32.  
  33. let ALL_CITY_LIST = this.props.allCityList;
  34. let CURRENT_CITY_LIST = this.props.nowCityList;
  35. let LAST_VISIT_CITY_LIST = this.props.lastVisitCityList;
  36. let HOT_CITY_LIST = this.props.hotCityList;
  37.  
  38. let letterList = this._getSortLetters(ALL_CITY_LIST);
  39.  
  40. let dataBlob = {};
  41. dataBlob[key_now] = CURRENT_CITY_LIST;
  42. dataBlob[key_last_visit] = LAST_VISIT_CITY_LIST;
  43. dataBlob[key_hot] = HOT_CITY_LIST;
  44.  
  45. ALL_CITY_LIST.map(cityJson => {
  46. let key = cityJson.sortLetters.toUpperCase();
  47.  
  48. if (dataBlob[key]) {
  49. let subList = dataBlob[key];
  50. subList.push(cityJson);
  51. } else {
  52. let subList = [];
  53. subList.push(cityJson);
  54. dataBlob[key] = subList;
  55. }
  56. });
  57.  
  58. let sectionIDs = Object.keys(dataBlob);
  59. let rowIDs = sectionIDs.map(sectionID => {
  60. let thisRow = [];
  61. let count = dataBlob[sectionID].length;
  62. for (let ii = 0; ii < count; ii++) {
  63. thisRow.push(ii);
  64. }
  65.  
  66. let eachheight = SECTIONHEIGHT + ROWHEIGHT * thisRow.length;
  67. if (sectionID === key_hot || sectionID === key_now || sectionID === key_last_visit) {
  68. let rowNum = (thisRow.length % 3 === 0)
  69. ? (thisRow.length / 3)
  70. : parseInt(thisRow.length / 3) + 1;
  71.  
  72. console.log('sectionIDs===>' + sectionIDs + ",rowNum=====>" + rowNum);
  73.  
  74. eachheight = SECTIONHEIGHT + ROWHEIGHT_Box * rowNum;
  75. }
  76.  
  77. totalheight.push(eachheight);
  78.  
  79. return thisRow;
  80. });
  81.  
  82.  
  83. let ds = new ListView.DataSource({
  84. getRowData: getRowData,getSectionHeaderData: getSectionData,rowHasChanged: (row1,row2) => row1 !== row2,sectionHeaderHasChanged: (s1,s2) => s1 !== s2
  85. });
  86.  
  87. this.state = {
  88. dataSource: ds.cloneWithRowsAndSections(dataBlob,sectionIDs,rowIDs),letters: sectionIDs
  89. };
  90.  
  91. that = this;
  92. }
  93.  
  94. _getSortLetters(dataList) {
  95. let list = [];
  96.  
  97. for (let j = 0; j < dataList.length; j++) {
  98. let sortLetters = dataList[j].sortLetters.toUpperCase();
  99.  
  100. let exist = false;
  101. for (let xx = 0; xx < list.length; xx++) {
  102. if (list[xx] === sortLetters) {
  103. exist = true;
  104. }
  105. if (exist) {
  106. break;
  107. }
  108. }
  109. if (!exist) {
  110. list.push(sortLetters);
  111. }
  112. }
  113.  
  114. return list;
  115. }
  116.  
  117. _cityNameClick(cityJson) {
  118. // alert('选择了城市====》' + cityJson.id + '#####' + cityJson.name);
  119. this.props.onSelectCity(cityJson);
  120. }
  121.  
  122. _scrollTo(index,letter) {
  123. this.refs.toast.close();
  124. let position = 0;
  125. for (let i = 0; i < index; i++) {
  126. position += totalheight[i]
  127. }
  128. this._listView.scrollTo({y: position});
  129. this.refs.toast.show(letter,DURATION.LENGTH_SHORT);
  130. }
  131.  
  132. _renderRightLetters(letter,index) {
  133. return (
  134. <TouchableOpacity key={'letter_idx_' + index} activeOpacity={0.6} onPress={() => {
  135. this._scrollTo(index,letter)
  136. }}>
  137. <View style={styles.letter}>
  138. <Text style={styles.letterText}>{letter}</Text>
  139. </View>
  140. </TouchableOpacity>
  141. );
  142. }
  143.  
  144. _renderListBox(cityJson,rowId) {
  145. return (
  146. <TouchableOpacity key={'list_item_' + cityJson.id} style={styles.rowViewBox} onPress={() => {
  147. that._cityNameClick(cityJson)
  148. }}>
  149. <View style={styles.rowdataBox}>
  150. <Text style={styles.rowDataTextBox}>{cityJson.name}</Text>
  151. </View>
  152. </TouchableOpacity>
  153. );
  154. }
  155.  
  156. _renderListRow(cityJson,rowId) {
  157. console.log('rowId===>' + rowId + ",cityJson====>" + JSON.stringify(cityJson));
  158. if (rowId === key_now || rowId === key_hot || rowId === key_last_visit) {
  159. return that._renderListBox(cityJson,rowId);
  160. }
  161.  
  162. return (
  163. <TouchableOpacity key={'list_item_' + cityJson.id} style={styles.rowView} onPress={() => {
  164. that._cityNameClick(cityJson)
  165. }}>
  166. <View style={styles.rowdata}>
  167. <Text style={styles.rowdatatext}>{cityJson.name}</Text>
  168. </View>
  169. </TouchableOpacity>
  170. )
  171. }
  172.  
  173. _renderListSectionHeader(sectionData,sectionID) {
  174. return (
  175. <View style={styles.sectionView}>
  176. <Text style={styles.sectionText}>
  177. {sectionData}
  178. </Text>
  179. </View>
  180. );
  181. }
  182.  
  183. render() {
  184. return (
  185. <View style={styles.container}>
  186. <View style={styles.listContainner}>
  187. <ListView ref={listView => this._listView = listView}
  188. contentContainerStyle={styles.contentContainer} dataSource={this.state.dataSource}
  189. renderRow={this._renderListRow} renderSectionHeader={this._renderListSectionHeader}
  190. enableEmptySections={true} initialListSize={500}/>
  191. <View style={styles.letters}>
  192. {this.state.letters.map((letter,index) => this._renderRightLetters(letter,index))} </View> </View> <Toast ref="toast" position='top' positionValue={200} fadeInDuration={750} fadeOutDuration={1000} opacity={0.8}/> </View> ) } } const styles = StyleSheet.create({ container: { // paddingTop: 50,flex: 1,flexDirection: 'column',backgroundColor: '#F4F4F4',},listContainner: { height: Dimensions.get('window').height,marginBottom: 10 },contentContainer: { flexDirection: 'row',width: width,backgroundColor: 'white',justifyContent: 'flex-start',flexWrap: 'wrap' },letters: { position: 'absolute',height: height,top: 0,bottom: 0,right: 10,backgroundColor: 'transparent',// justifyContent: 'flex-start',// alignItems: 'flex-start' alignItems: 'center',justifyContent: 'center' },letter: { height: height * 4 / 100,width: width * 4 / 50,justifyContent: 'center',alignItems: 'center' },letterText: { textAlign: 'center',fontSize: height * 1.1 / 50,color: '#e75404' },sectionView: { paddingTop: 5,paddingBottom: 5,height: 30,paddingLeft: 10,backgroundColor: '#F4F4F4' },sectionText: { color: '#e75404',fontWeight: 'bold' },rowView: { height: ROWHEIGHT,paddingRight: 10,borderBottomColor: '#F4F4F4',borderBottomWidth: 0.5 },rowdata: { paddingTop: 10,paddingBottom: 2 },rowdatatext: { color: 'gray',width: width },rowViewBox: { height: ROWHEIGHT_Box,width: (width - 30) / 3,flexDirection: 'row',backgroundColor: '#ffffff' },rowdataBox: { borderWidth: 1,borderColor: '#DBDBDB',marginTop: 5,marginBottom: 5,paddingBottom: 2,marginLeft: 10,marginRight: 10,rowDataTextBox: { marginTop: 5,height: 20 } });

然后在头部还需要实现一个搜索框。
SearchBox.js

  1. 'use strict';
  2. import React,{Component} from 'react';
  3. import {
  4. View,TextInput,StyleSheet,Platform,} from 'react-native';
  5.  
  6. export default class SearchBox extends Component {
  7. constructor(props) {
  8. super(props);
  9. this.state = {
  10. value: ''
  11. };
  12.  
  13. }
  14.  
  15. onEndEditingKeyword(vv) {
  16. console.log(vv);
  17. }
  18.  
  19. onChanegeTextKeyword(vv) {
  20. console.log('onChanegeTextKeyword',vv);
  21.  
  22. this.setState({value: vv});
  23. this.props.onChanegeTextKeyword(vv);
  24. }
  25.  
  26. render() {
  27. return (
  28. <View style={styles.container}>
  29. <View style={styles.inputBox}>
  30. <View style={styles.inputIcon}>
  31. </View>
  32. <TextInput ref="keyword" autoCapitalize="none" value={this.props.keyword}
  33. onChangeText={this.onChanegeTextKeyword.bind(this)} returnKeyType="search" maxLength={20}
  34. style={styles.inputText} underlineColorAndroid="transparent"
  35. placeholder={'输入城市名或拼音查询'}/>
  36. </View>
  37. </View>
  38. )
  39. }
  40. }
  41.  
  42. const styles = StyleSheet.create({
  43. container: {
  44. marginTop: 5,backgroundColor: '#ffffff',height: Platform.OS === 'ios'
  45. ? 35
  46. : 45,borderBottomWidth: StyleSheet.hairlineWidth,borderBottomColor: '#cdcdcd',paddingBottom: 5
  47. },inputBox: {
  48. height: Platform.OS === 'ios'
  49. ? 30
  50. : 40,marginLeft: 5,marginRight: 5,backgroundColor: '#E6E7E8'
  51. },inputIcon: {
  52. margin: Platform.OS === 'ios'
  53. ? 5
  54. : 10
  55. },inputText: {
  56. alignSelf: 'flex-end',marginTop: Platform.OS === 'ios'
  57. ? 0
  58. : 0,height: Platform.OS === 'ios'
  59. ? 30
  60. : 40,marginLeft: 2,fontSize: 12,lineHeight: 30,textAlignVertical: 'bottom',textDecorationLine: 'none'
  61. }
  62. });

最终效果


最后是界面的绘制,这里就不多说了,大家可以下载源码自行查看。源码地址:http://download.csdn.net/detail/xiangzhihong8/9905924

猜你在找的React相关文章