https://swimlanes.io/

XState Catalogue (xstate-catalogue.com)

XState Visualizer (stately.ai)

Data Fetching

Simple Data Fetch

Global State

Authentication

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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
import { assign, createMachine, Sender } from 'xstate';

export type AuthenticationMachineContext = {
userDetails?: UserDetails;
};

interface UserDetails {
username: string;
}

export type AuthenticationMachineEvent =
| {
type: 'REPORT_IS_LOGGED_IN';
userDetails: UserDetails;
}
| {
type: 'REPORT_IS_LOGGED_OUT';
}
| {
type: 'LOG_OUT';
}
| {
type: 'LOG_IN';
userDetails: UserDetails;
};

const authenticationMachine = createMachine<
AuthenticationMachineContext,
AuthenticationMachineEvent
>(
{
id: 'authentication',
initial: 'checkingIfLoggedIn',
states: {
checkingIfLoggedIn: {
invoke: {
src: 'checkIfLoggedIn',
onError: {
target: 'loggedOut',
},
},
on: {
REPORT_IS_LOGGED_IN: {
target: 'loggedIn',
actions: 'assignUserDetailsToContext',
},
REPORT_IS_LOGGED_OUT: 'loggedOut',
},
},
loggedIn: {
on: {
LOG_OUT: {
target: 'loggedOut',
},
},
},
loggedOut: {
entry: ['navigateToAuthPage', 'clearUserDetailsFromContext'],
on: {
LOG_IN: {
target: 'loggedIn',
actions: 'assignUserDetailsToContext',
},
},
},
},
},
{
services: {
checkIfLoggedIn: () => async (
send: Sender<AuthenticationMachineEvent>,
) => {
// Perform some async check here
// if (isLoggedIn) {
// send({
// type: "REPORT_IS_LOGGED_IN",
// userDetails: {
// username: "mpocock1",
// },
// });
// } else {
// send({
// type: "REPORT_IS_LOGGED_OUT",
// });
// }
},
},
actions: {
navigateToAuthPage: () => {
// When the user is logged out, we
// should take them to the /auth route
},
assignUserDetailsToContext: assign((context, event) => {
if (event.type !== 'REPORT_IS_LOGGED_IN') {
return {};
}
return {
userDetails: event.userDetails,
};
}),
clearUserDetailsFromContext: assign({
userDetails: undefined,
}),
},
},
);

export default authenticationMachine;