This repository was archived by the owner on Aug 17, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathUsersProvider.tsx
95 lines (79 loc) · 2.21 KB
/
UsersProvider.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
import * as React from "react";
import { ReactNode, useEffect, useState } from "react";
import { useSocket, useSocketMessage } from "@/providers/Socket/SockerProvider";
import {
User,
UsersOutgoing,
UsersOutgoingType,
} from "@/providers/Users/users-server";
import { useTimeout } from "usehooks-ts";
type UsersContextType = {
thisUser: string | null;
users: User[];
sendName: (name: string) => void;
};
const UsersContext = React.createContext<UsersContextType | undefined>(
undefined,
);
function UsersProvider({ children }: { children: ReactNode }) {
const { sendJson } = useSocket<UsersOutgoingType>();
const [users, setUsers] = useState<User[]>([]);
const [thisUser, setThisUser] = useState<string | null>(null);
useTimeout(() => {
if (!thisUser) {
throw new Error("Could not get the current user id?");
}
}, 3000);
useSocketMessage<string>(setThisUser, UsersOutgoing.userId);
useSocketMessage<User[]>(setUsers, UsersOutgoing.users);
useSocketMessage<User>((obj) => {
setUsers((users) => [...users, obj]);
}, UsersOutgoing.newUser);
useSocketMessage<string>((userId) => {
setUsers((users) => users.filter((user) => user.id !== userId));
}, UsersOutgoing.removeUser);
useSocketMessage<User>((obj) => {
setUsers((users) => {
return users.map((user) => {
if (user.id === obj.id && obj.name) {
return {
...user,
name: obj.name,
};
}
return user;
});
});
}, UsersOutgoing.nameChanged);
// on unload we tell the server to remove the user
useEffect(() => {
return () => {
if (thisUser) {
sendJson({
removeUser: thisUser,
});
}
};
}, [sendJson, thisUser]);
const sendName = (name: string) => {
return sendJson({
changeName: name,
});
};
const value = {
thisUser,
users,
sendName,
};
return (
<UsersContext.Provider value={value}>{children}</UsersContext.Provider>
);
}
function useUsers() {
const context = React.useContext(UsersContext);
if (context === undefined) {
throw new Error("useUsers must be used within a UsersProvider");
}
return context;
}
export { UsersProvider, useUsers };