无法以平面列表格式显示数组

时间:2020-08-06 19:54:30

标签: javascript css reactjs react-native react-native-flatlist

长话短说,我正在尝试创建一个视频游戏列表。我有一个自动完成库,可将游戏标题与我在输入框中所写的内容相匹配。

我添加了一个按钮,onPress触发了一个函数,该函数将'query'中的内容(即我在输入框中所写的内容)推入名为myGamesArray的数组中

然后,我尝试使用Flatlist以列表形式显示myGamesArray。但是由于某种原因,它没有显示出来。

我的代码有点混乱,因为我发现在输入框旁边放置一个按钮的解决方案是将flex方向更改为“行”

我在FlatList下方添加了一个按钮,只是为了检查它是否会显示在输入框下方(确实如此),但由于某种原因,flatlist不会显示

这是我所有的App.js

/*This is an example of AutoComplete Input/ AutoSuggestion Input*/
import React, { Component } from 'react';
//import react in our code.
import { StyleSheet, Text, TouchableOpacity, View, Image, Alert, FlatList } from 'react-native';
//import all the components we are going to use.
import Autocomplete from 'react-native-autocomplete-input';
import { Button, List, Container, ListItem, Header } from 'native-base';
//import Autocomplete component


//Demo base API to get the data for the Autocomplete suggestion
class App extends Component {
  constructor(props) {
    super(props);
    //Initialization of state
    //games will contain the array of suggestion
    //query will have the input from the autocomplete input
    this.state = {
      myGamesArray: [],
      games: [],
      query: '',
    };
  }
  componentDidMount() {
    //First method to be called after components mount
    //fetch the data from the server for the suggestion
    fetch('https://api.rawg.io/api/games?page=1&platforms=18', {
        "method": "GET",
        "headers": {
          "x-rapidapi-host": "rawg-video-games-database.p.rapidapi.com",
          "x-rapidapi-key": "495a18eab9msh50938d62f12fc40p1a3b83jsnac8ffeb4469f"
        }
      })
      .then(res => res.json())
      .then(json => {
        const { results: games } = json;
        this.setState({ games });
        //setting the data in the games state
      });
  }

  findGame(query) {
    //method called everytime when we change the value of the input
    if (query === '') {
      //if the query is null then return blank
      return [];
    }
 
    const { games } = this.state;
    //making a case insensitive regular expression to get similar value from the game json
    const regex = new RegExp(`${query.trim()}`, 'i');

    //return the filtered game array according the query from the input
    return games.filter(game => game.name.search(regex) >= 0);
  }  

AddItemsToArray=()=>{
 
      //Adding Items To Array.
     this.state.myGamesArray.push( this.state.query.toString() )
 
      // Showing the complete Array on Screen Using Alert (just to check if it's infact inside the array)
      Alert.alert(this.state.myGamesArray.toString());
 
  }
 
  render() {
    const { query } = this.state;
    const games = this.findGame(query);
    const comp = (a, b) => a.toLowerCase().trim() === b.toLowerCase().trim();
 
    return (
      <View>
               
      <View style={styles.container}>
        <Autocomplete
          style={styles.autocompleteContainer1}
          autoCapitalize="none"
          autoCorrect={false}
          //data to show in suggestion
          data={games.length === 1 && comp(query, games[0].name) ? [] : games}
          //default value if you want to set something in input
          defaultValue={query}
          /*onchange of the text changing the state of the query which will trigger
          the findGame method to show the suggestions*/
          onChangeText={text => this.setState({ query: text })}
          placeholder="Selecione os jogos que voce quer receber!"
          renderItem={({ item }) => (
            //you can change the view you want to show in suggestion from here
            //I GET ERROR WHEN TRYING TO ERASE (PS4) IN TEXT BOX ***NEED TO CHECK THIS
            <View style={styles.iconContainer} >
              <TouchableOpacity onPress={() => this.setState({ query: item.name})}
              style={styles.itemContainer} >
              <View>
              <Image
                style={styles.profilepic}
                  source={{uri: item.background_image}}
                />
               </View>
              <Text style={styles.itemText}>
                {item.name} (PS4)
              </Text>
              </TouchableOpacity>
            </View>
          )}
          />
          <Button
            style={styles.addButton}
            icon='plus'
            title="Click Here To Add Value To Array"
            onPress={() => this.AddItemsToArray()}
          />
          <View style={styles.descriptionContainer}>
          
            {games.length > 0 ? (
              //Text inside input box
              <Text style={styles.infoText}>{this.state.query}</Text>
            ) : (
              <Text style={styles.infoText}>{this.state.query}</Text>                    
              )
          } 
          </View>  
      </View>

      <View style={styles.container2}> 
          <Button
          full></Button> 
        
        <FlatList
          data={this.state.myGamesArray}
            renderItem={({ item }) => (
              <ListItem>{item}</ListItem>
            )}
          />
          
    </View>  
      
      </View>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    backgroundColor: '#fff',
    flex: 1,
    flexDirection:'row',
    padding: 16,
    marginTop: 40,
    justifyContent: 'center',
  },
  container2: {
    backgroundColor: '#fff',
    flex: 1,
    marginTop: 100,
    alignItems: 'center',
    justifyContent: 'center',
  },

  autocompleteContainer1: {
    borderWidth: 1,
    backgroundColor: "#fff",
    borderColor: '#7843FF',
    height: 50,
    width:300
  },
  itemContainer: {
    flex:1,
    flexDirection: 'row',
    borderWidth:0.5,
    borderColor: '#7843FF',
    paddingVertical: 5,
    alignItems: 'center',
    paddingRight: 60
  },
  addButton: {
  width:50,
  height: 50,
  borderColor: '#7843FF',
  borderWidth: 0.5,
  backgroundColor: '#fff'
  },
  descriptionContainer: {
    flex: 1,
    justifyContent: 'center',
    borderRadius: 10,
    color: '#fff',
    borderColor: '#7843FF'
  },
  itemText: {
    fontSize: 15,
    marginLeft: 10,
    marginRight:30,
    marginVertical:10,
    color: '#000',
    textAlign: 'left',
    justifyContent: 'center'
    
  },
  infoText: {
    textAlign: 'center',
    color: '#000',
  },
  profilepic: {
    flex: 3,
    height: 60,
    width: 60,
    marginLeft:10,
    borderRadius: 100,
  },
});
export default App;

1 个答案:

答案 0 :(得分:0)

问题

this.state.myGamesArray.push(this.state.query.toString())是状态突变。您实际上需要 进行状态更新并返回新的数组对象引用。

解决方案

使用功能状态更新来访问先前的状态对象,然后将旧数组浅复制到新数组引用中,并将新查询值附加到末尾。

此外,在使更新入队后立即记录日志或尝试以其他方式引用更新后的状态将无法正常工作,状态为const并且在渲染周期之间已更新,因此它将括起来当前状态。

AddItemsToArray = () => {
  //Adding Items To Array.
  this.setState(prevState => {
      const { myGamesArray, query } = prevState;
      return {
        myGamesArray: [...myGamesArray, query.toString()],
      };
    },
    // Use setState callback to alert with the updated state
    () => {
      // Showing the complete Array on Screen Using Alert (just to check if it's in fact inside the array)
      Alert.alert(this.state.myGamesArray.toString());
    },
  );
}