Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ルーム一覧の取得とルーム作成およびルーム入室をgraphQLからwebsocketでできるように変更 #21

Merged
merged 20 commits into from
Jan 24, 2021
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -98,8 +98,8 @@ export default function App(): ReactElement {

export type RootStackParamList = {
Home: undefined;
Room: { id: number };
CreateRoom: undefined;
Room: { roomid: number; endpoint: string };
CreateRoom: { endpoint: string };
RoomList: undefined;
EditDeck: undefined;
Preferences: undefined;
Expand Down
162 changes: 94 additions & 68 deletions src/screens/create-room/index.tsx
Original file line number Diff line number Diff line change
@@ -1,44 +1,58 @@
import React, { ReactElement } from "react";
import {
StyleSheet,
View,
Text,
Button,
ScrollView,
TextInput,
} from "react-native";
import React, { ReactElement, useEffect, useRef, useState } from "react";
import { StyleSheet, View, Text } from "react-native";
import { Button, Input } from "react-native-elements";
import { StackNavigationProp } from "@react-navigation/stack";
import { RouteProp } from "@react-navigation/native";
import { RootStackParamList } from "../../../App";
import { gql, useMutation } from "@apollo/client";
import { Formik } from "formik";
import * as Yup from "yup";

const CREATE_ROOM = gql`
mutation CreateRoom($name: String!, $player: String!) {
createRoom(name: $name, player: $player) {
id
name
players
}
}
`;

export default function CreateRoomScreen({
route,
navigation,
}: {
route: CreateRoomScreenRouteProp;
navigation: CreateRoomScreenNavigationProp;
}): ReactElement {
const [createRoom] = useMutation(CREATE_ROOM, {
onCompleted: (data) => {
console.log(data.createRoom.id);
navigation.navigate("Room", { id: data.createRoom.id });
},
});
// const [createRoom] = useMutation(CREATE_ROOM, {
// onCompleted: (data) => {
// console.log(data.createRoom.id);
// navigation.navigate("Room", { id: data.createRoom.id });
// },
// });
const { endpoint } = route.params;
const websocket = useRef<WebSocket | null>(null);
const [exists, setExists] = useState(false);

useEffect(() => {
websocket.current = new WebSocket(`ws://${endpoint}/ws`);
websocket.current.onmessage = (event) => {
bana118 marked this conversation as resolved.
Show resolved Hide resolved
if (event.data.startsWith("{")) {
const json = JSON.parse(event.data);
if (json.status === "ok") {
setExists(false);
navigation.navigate("Room", {
roomid: json.data.id,
endpoint: endpoint,
});
} else {
setExists(true);
}
}
};
return () => {
if (websocket.current != null) {
websocket.current.close();
}
};
}, []);

const onSubmit = async (values: { name: string }) => {
// データ送信
console.log(values);
createRoom({ variables: { name: values.name, player: "piypiyo" } });
if (websocket.current != null) {
websocket.current.send(`/create ${values.name}`);
}
};

const schema = Yup.object().shape({
Expand All @@ -48,51 +62,62 @@ export default function CreateRoomScreen({
.required("ルーム名を入力してください"),
});
return (
<ScrollView contentContainerStyle={styles.container}>
<View>
<Formik
initialValues={{
name: "",
}}
validateOnMount
validationSchema={schema}
onSubmit={(values) => onSubmit(values)}
>
{({
handleSubmit,
handleChange,
handleBlur,
isValid,
isSubmitting,
values,
errors,
touched,
}) => (
<>
<View>
{errors.name && touched.name ? (
<Text>{errors.name}</Text>
) : null}
<TextInput
value={values.name}
onChangeText={handleChange("name")}
onBlur={handleBlur("name")}
placeholder="ルーム名を入力してください"
/>
</View>
<Button
title="Submit"
onPress={() => handleSubmit()}
disabled={!isValid || isSubmitting}
<View style={styles.container}>
<Formik
initialValues={{
name: "",
}}
validateOnMount
validationSchema={schema}
onSubmit={(values) => onSubmit(values)}
>
{({
handleSubmit,
handleChange,
handleBlur,
isValid,
isSubmitting,
values,
errors,
touched,
}) => (
<>
<View>
{errors.name && touched.name ? <Text>{errors.name}</Text> : null}
</View>
{exists ? (
<Input
label="新規ルーム名"
value={values.name}
onChangeText={handleChange("name")}
onBlur={handleBlur("name")}
errorMessage="既に存在するルーム名です"
errorStyle={{ color: "red" }}
placeholder="ルーム名を入力してください"
/>
) : (
<Input
label="新規ルーム名"
value={values.name}
onChangeText={handleChange("name")}
onBlur={handleBlur("name")}
placeholder="ルーム名を入力してください"
/>
</>
)}
</Formik>
</View>
</ScrollView>
)}
<Button
title="Submit"
onPress={() => handleSubmit()}
disabled={!isValid || isSubmitting}
style={styles.button}
/>
</>
)}
</Formik>
</View>
);
}

type CreateRoomScreenRouteProp = RouteProp<RootStackParamList, "CreateRoom">;
type CreateRoomScreenNavigationProp = StackNavigationProp<
RootStackParamList,
"CreateRoom"
Expand All @@ -105,4 +130,5 @@ const styles = StyleSheet.create({
alignItems: "center",
justifyContent: "center",
},
button: { margin: 10 },
});
11 changes: 5 additions & 6 deletions src/screens/home/index.tsx
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
import { StatusBar } from "expo-status-bar";
import React, { ReactElement, useState, useEffect } from "react";
import { StyleSheet, View, TextInput } from "react-native";
import { StyleSheet, View } from "react-native";
import AsyncStorage from "@react-native-async-storage/async-storage";
import { Text } from "react-native-elements";
import { Input } from "react-native-elements";

const DEFAULT_ENDPOINT = "127.0.0.1";
export const DEFAULT_ENDPOINT = "127.0.0.1";
export default function HomeScreen(): ReactElement {
const [endpoint, setEndpoint] = useState(DEFAULT_ENDPOINT);
useEffect(() => {
Expand Down Expand Up @@ -34,8 +34,8 @@ export default function HomeScreen(): ReactElement {
// TODO ユーザーに関する情報(ユーザーアイコン,ユーザーネーム)の実装
return (
<View style={styles.container}>
<Text>サーバーアドレス</Text>
<TextInput
<Input
label="サーバーアドレス"
style={styles.input}
onChangeText={(input) => setEndpoint(input)}
value={endpoint}
Expand All @@ -52,7 +52,6 @@ const styles = StyleSheet.create({
justifyContent: "center",
},
input: {
width: "50%",
borderWidth: 1,
borderColor: "#ccc",
},
Expand Down
Loading