Как правильно обрабатывать блок извлечения данных из Firebase, который работает асинхронно? - PullRequest
0 голосов
/ 20 января 2019

Я создаю собственное приложение, в котором пользователи могут создавать комнаты.

Пользователи заполняют форму с новым именем комнаты, и когда они нажимают кнопку «Отправить», запускается thunk, который проверяет, есть ли уже комната с таким именем в моей базе данных Firebase.Если комната уже существует, метод в моем компоненте устанавливает ошибку в локальном состоянии.

Моя проблема заключается в том, что перед выполнением thunk выполняется логика, проверяющая, существует ли ошибка или нет.и будет перенаправлять на страницу комнаты, несмотря ни на что.

Я попытался сделать функцию создания комнаты асинхронной и ждать завершения, но это не работает.Есть ли способ, чтобы другие функции не запускались до тех пор, пока не завершится работа?

//thunk

export const createRoom = (newRoom = {}, errorFn, clearErrorFn) => {
  console.log('creating room thunk')
  return (dispatch, getState) => {
   const room = {
    name: newRoom.name,
    prompt: newRoom.prompt,
    owner: newRoom.owner,
    people: newRoom.owner,
    ideas: "ideas"
 }

//checking whether the room exists
database.ref('rooms').once('value', (snapshot) => {
  const rooms = [];
  snapshot.forEach((childSnapshot) => {
    rooms.push(childSnapshot.val().name)
  })
  if (!rooms.includes(room.name)) {
    database.ref(`rooms/${room.name}`).set(room).then(() => {
      database.ref(`rooms/${room.name}/prompt`).set(room.prompt).then(() => {
        database.ref(`rooms/${room.name}/owner`).set(room.owner).then(() => {
          database.ref(`rooms/${room.name}/people/${room.people}`).set(room.people).then(() => {
            database.ref(`rooms/${room.name}/ideas/`).set(room.ideas)
          })
        })
      })
    })
    clearErrorFn()
    dispatch(afterRoomCreate(room))
  } else {
    console.log('Room name not available!')
    return errorFn('Room name not available!')
  }
})

}}

//component
    import React from 'react';
import KeyboardSpacer from 'react-native-keyboard-spacer';
import { connect } from 'react-redux';
import {
    ScrollView,
    StyleSheet,
    View,
    TextInput,
    Text,
    Button,
    KeyboardAvoidingView
} from 'react-native';

import { createRoom, joinRoom } from '../redux/reducers/rooms/actions'

class HomeScreen extends React.Component {
    static navigationOptions = {
        title: 'Undecided!',
    };
    constructor(props) {
        super(props)
        this.state = {
            createRoomName: "",
            joinRoomName: '',
            newRoomPurpose: '',
            newRoomUserName: '',
            joinRoomUserName: '',
            createError: "",
            joinError: "",
            yesCreateError: false,
            hasRun: false
        }
        this.showCreateError = this.showCreateError.bind(this)
        this.showJoinError = this.showJoinError.bind(this)
        this.clearCreateError = this.clearCreateError.bind(this)
        this.handleCreateRoom = this.handleCreateRoom.bind(this)
    }

    async handleCreateRoom() {
const newRoom = {
  name: this.state.createRoomName,
  prompt: this.state.newRoomPurpose,
  owner: this.state.newRoomUserName
}

        this.props.createNewRoom(newRoom, this.showCreateError, this.clearCreateError)
        if (this.state.createError === "") {
            const { navigate } = this.props.navigation;
            navigate('Room',
                { roomName: this.state.createRoomName, owner: this.state.newRoomUserName, user: this.state.newRoomUserName, prompt: this.state.newRoomPurpose }
            )
        } else {
            this.setState({
                createError: ""
            })
        }

    }


    showCreateError = (createError) => {
        this.setState({
            createError,
            hasRun: true
        });
    }

    showJoinError = (joinError) => {
        this.setState({
            joinError,
            hasRun: true
        });
    }

    clearCreateError = () => {
        this.setState({
            createError: "",
            hasRun: true
        })
    }


    render() {
        return (
            <KeyboardAvoidingView behavior="padding" style={styles.container}>
                <ScrollView>
                    {/* create room button */}
                    <View style={styles.sectionContainer}>
                        <View>
                            <Text style={styles.homePageHeaders}>Create a New Room</Text>
                        </View>
                        <View>
                            <TextInput
                                style={styles.homePageInputs}
                                onChangeText={(text) => this.setState({ createRoomName: text })}
                                value={this.state.createRoomName}
                                placeholder="Enter New Room Name"
                            />
                            <Text style={styles.errorText}>{this.state.createError}</Text>
                            <TextInput
                                style={styles.homePageInputs}
                                onChangeText={(text) => this.setState({ newRoomPurpose: text })}
                                value={this.state.newRoomPurpose}
                                placeholder="What is the purpose of this room?"
                            />
                            <TextInput
                                style={styles.homePageInputs}
                                onChangeText={(text) => this.setState({ newRoomUserName: text })}
                                value={this.state.newRoomUserName}
                                placeholder="What is your name?"
                            />
                        </View>
                        <View style={styles.homePageButtons}>
                            <Button
                                onPress={this.handleCreateRoom}
                                title="Create Room"
                                accessibilityLabel="Create Room"
                                color="white"
                                style={styles.homePageButtons}
                            />
                        </View>
                    </View>

                    {/* end of create room button */}

                    {/* join room input & button */}
                    <View style={styles.sectionContainer}>
                        <View>
                            <Text style={styles.homePageHeaders}>Enter Existing Room</Text>
                        </View>
                        <View>
                            <TextInput
                                style={styles.homePageInputs}
                                onChangeText={(text) => this.setState({ joinRoomName: text })}
                                value={this.state.joinRoomName}
                                placeholder="Input name of existing room"
                            />
                            <TextInput
                                style={styles.homePageInputs}
                                onChangeText={(text) => this.setState({ joinRoomUserName: text })}
                                value={this.state.joinRoomUserName}
                                placeholder="What is your name?"
                            />

                        </View>

                        <View style={styles.homePageButtons}>
                            <Button
                                onPress={this.handleCreateRoom}
                                title="Join Room"
                                accessibilityLabel="Join Room"
                                color="white"
                            />
                        </View>
                    </View>


                    {/* end of join room input & button */}
                </ScrollView>
            </KeyboardAvoidingView>
        );
    }

}

const mapDispatchToProps = (dispatch) => ({
    createNewRoom: (room, error, clearError) => dispatch(createRoom(room, error, clearError)),
    joinRoom: (data) => dispatch(joinRoom(data, showJoinError))
})


export default connect(null, mapDispatchToProps)(HomeScreen)
...