Rev 1 | Rev 140 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
1 | pmbaty | 1 | // network.cpp |
2 | |||
3 | #include "common.h" |
||
4 | |||
5 | |||
6 | // prototypes of local functions |
||
7 | static void StartThread_ListenToServer (void *thread_parms); |
||
8 | static void EvaluateServerReply (player_t *player); |
||
9 | |||
10 | |||
11 | bool PlayerNetwork_Init (player_t *player) |
||
12 | { |
||
13 | // this function initializes the network layer and connects to the chess server |
||
14 | |||
15 | WSADATA wsaData; |
||
16 | struct hostent *hostinfo; |
||
17 | struct sockaddr_in service; |
||
18 | unsigned long nonblocking_mode; |
||
19 | char ascii_hostname[256]; |
||
20 | |||
21 | onlineplayers = NULL; // ensure the online players array is empty |
||
22 | onlineplayer_count = 0; // and reset the online players count |
||
23 | onlineplayers_updated = false; |
||
24 | |||
25 | soughtgames = NULL; // ensure the sought games array is empty |
||
26 | soughtgame_count = 0; // and reset the sought games count |
||
27 | soughtgames_updated = false; |
||
28 | lastsought_time = 0; |
||
29 | |||
30 | chatterchannels = NULL; // ensure the chatter channels array is empty |
||
31 | chatterchannel_count = 0; // and reset the chatter channels count |
||
32 | chatterchannels_updated = false; |
||
33 | selected_chatterchannel = NULL; |
||
34 | |||
35 | PlayerCards_Init (); // initialize the player cards array |
||
36 | Challenges_Init (); // initialize the challenges array |
||
37 | Interlocutors_Init (); // initialize the interlocutors array |
||
38 | |||
39 | server_motd[0] = 0; // reset the server MOTD |
||
40 | |||
41 | player->is_connected = false; // remember we aren't connected yet |
||
42 | player->is_logged_in = false; // and not logged in either |
||
43 | player->is_in_game = false; // we aren't in game either |
||
44 | player->game_number = 0; |
||
45 | player->remaining_seconds = 0; |
||
46 | player->our_socket = INVALID_SOCKET; // and that we have no socket yet |
||
47 | player->name[0] = 0; // player name undefined |
||
48 | |||
49 | // initialize WinSock |
||
50 | if (WSAStartup (MAKEWORD (2, 2), &wsaData) != 0) |
||
51 | { |
||
52 | messagebox.hWndParent = hMainWnd; |
||
53 | wcscpy_s (messagebox.title, WCHAR_SIZEOF (messagebox.title), LOCALIZE (L"ImportantMessage")); |
||
54 | wcscpy_s (messagebox.text, WCHAR_SIZEOF (messagebox.text), LOCALIZE (L"Error_NetworkInitializationFailed")); |
||
55 | messagebox.flags = MB_ICONWARNING | MB_OK; |
||
56 | DialogBox_Message (&messagebox); // display a modeless error message box |
||
57 | |||
58 | PlayerNetwork_Shutdown (player); // on error, shutdown the network and display an error message |
||
59 | return (false); |
||
60 | } |
||
61 | |||
62 | // get the chess server's IP address from the host name |
||
63 | ConvertTo7BitASCII (ascii_hostname, sizeof (ascii_hostname), options.network.server_address); |
||
64 | hostinfo = gethostbyname (ascii_hostname); |
||
65 | if (hostinfo == NULL) |
||
66 | { |
||
67 | messagebox.hWndParent = hMainWnd; |
||
68 | wcscpy_s (messagebox.title, WCHAR_SIZEOF (messagebox.title), LOCALIZE (L"ImportantMessage")); |
||
69 | wcscpy_s (messagebox.text, WCHAR_SIZEOF (messagebox.text), LOCALIZE (L"Error_NetworkInitializationFailed")); |
||
70 | messagebox.flags = MB_ICONWARNING | MB_OK; |
||
71 | DialogBox_Message (&messagebox); // display a modeless error message box |
||
72 | |||
73 | PlayerNetwork_Shutdown (player); // on error, shutdown the network and display an error message |
||
74 | return (false); |
||
75 | } |
||
76 | |||
77 | // fill in the sockaddr server structure with the server hostinfo data |
||
78 | service.sin_family = AF_INET; |
||
79 | service.sin_addr.s_addr = inet_addr (inet_ntoa (*(struct in_addr *) hostinfo->h_addr_list[0])); |
||
80 | service.sin_port = htons (options.network.server_port); |
||
81 | |||
82 | // create our socket |
||
83 | if ((player->our_socket = socket (AF_INET, SOCK_STREAM, IPPROTO_TCP)) == INVALID_SOCKET) |
||
84 | { |
||
85 | messagebox.hWndParent = hMainWnd; |
||
86 | wcscpy_s (messagebox.title, WCHAR_SIZEOF (messagebox.title), LOCALIZE (L"ImportantMessage")); |
||
87 | wcscpy_s (messagebox.text, WCHAR_SIZEOF (messagebox.text), LOCALIZE (L"Error_NetworkInitializationFailed")); |
||
88 | messagebox.flags = MB_ICONWARNING | MB_OK; |
||
89 | DialogBox_Message (&messagebox); // display a modeless error message box |
||
90 | |||
91 | PlayerNetwork_Shutdown (player); // on error, shutdown the network and display an error message |
||
92 | return (false); |
||
93 | } |
||
94 | |||
95 | // connect the game to the chess server using that socket |
||
96 | if (connect (player->our_socket, (struct sockaddr *) &service, sizeof (service)) == -1) |
||
97 | { |
||
98 | messagebox.hWndParent = hMainWnd; |
||
99 | wcscpy_s (messagebox.title, WCHAR_SIZEOF (messagebox.title), LOCALIZE (L"ImportantMessage")); |
||
100 | wcscpy_s (messagebox.text, WCHAR_SIZEOF (messagebox.text), LOCALIZE (L"Error_NetworkInitializationFailed")); |
||
101 | messagebox.flags = MB_ICONWARNING | MB_OK; |
||
102 | DialogBox_Message (&messagebox); // display a modeless error message box |
||
103 | |||
104 | PlayerNetwork_Shutdown (player); // on error, shutdown the network and display an error message |
||
105 | return (false); |
||
106 | } |
||
107 | |||
108 | // set it to be non-blocking (but only AFTER the connection is made) |
||
109 | nonblocking_mode = 1; |
||
110 | if (ioctlsocket (player->our_socket, FIONBIO, &nonblocking_mode) == SOCKET_ERROR) |
||
111 | { |
||
112 | messagebox.hWndParent = hMainWnd; |
||
113 | wcscpy_s (messagebox.title, WCHAR_SIZEOF (messagebox.title), LOCALIZE (L"ImportantMessage")); |
||
114 | wcscpy_s (messagebox.text, WCHAR_SIZEOF (messagebox.text), LOCALIZE (L"Error_NetworkInitializationFailed")); |
||
115 | messagebox.flags = MB_ICONWARNING | MB_OK; |
||
116 | DialogBox_Message (&messagebox); // display a modeless error message box |
||
117 | |||
118 | PlayerNetwork_Shutdown (player); // on error, shutdown the network and display an error message |
||
119 | return (false); |
||
120 | } |
||
121 | |||
122 | // initialize the debug log file |
||
123 | Debug_Init (L"Chess server output.txt"); |
||
124 | |||
125 | player->is_connected = true; // remember we are connected |
||
126 | return (true); // finished, we have a valid, connected socket |
||
127 | } |
||
128 | |||
129 | |||
130 | void PlayerNetwork_Shutdown (player_t *player) |
||
131 | { |
||
132 | // this function shutdowns the network layer |
||
133 | |||
134 | int chatterchannel_index; |
||
135 | |||
136 | // shutdown the socket if it is open. This will disconnect us from the server. |
||
137 | if (player->our_socket != INVALID_SOCKET) |
||
138 | closesocket (player->our_socket); // close the network socket |
||
139 | player->our_socket = INVALID_SOCKET; |
||
140 | |||
141 | player->is_connected = false; // remember we are no longer connected |
||
142 | player->is_logged_in = false; // and not logged in either |
||
143 | player->is_in_game = false; // we aren't in game either |
||
144 | player->game_number = 0; |
||
145 | player->remaining_seconds = 0; |
||
146 | player->name[0] = 0; // player name undefined |
||
147 | |||
148 | server_motd[0] = 0; // reset the server MOTD |
||
149 | |||
150 | // cycle through all chatter channels we know... |
||
151 | for (chatterchannel_index = 0; chatterchannel_index < chatterchannel_count; chatterchannel_index++) |
||
152 | { |
||
153 | SAFE_free ((void **) &chatterchannels[chatterchannel_index].members); // free their member list |
||
154 | chatterchannels[chatterchannel_index].member_count = 0; // reset their members count |
||
155 | } |
||
156 | SAFE_free ((void **) &chatterchannels); // free the chatter channels array |
||
157 | chatterchannel_count = 0; // and reset the chatter channels count |
||
158 | chatterchannels_updated = false; |
||
159 | selected_chatterchannel = NULL; |
||
160 | |||
161 | SAFE_free ((void **) &soughtgames); // free the sought games array |
||
162 | soughtgame_count = 0; // and reset the sought games count |
||
163 | soughtgames_updated = false; |
||
164 | |||
165 | SAFE_free ((void **) &onlineplayers); // free the online players array |
||
166 | onlineplayer_count = 0; // and reset the online players count |
||
167 | onlineplayers_updated = false; |
||
168 | |||
169 | Interlocutors_Shutdown (); // shutdown the interlocutors |
||
170 | Challenges_Shutdown (); // shutdown the challenges |
||
171 | PlayerCards_Shutdown (); // shutdown the player cards |
||
172 | |||
173 | // shutdown all possible opened UI windows |
||
174 | if (IsWindow (hChatterChannelsWnd)) |
||
175 | DestroyWindow (hChatterChannelsWnd); |
||
176 | hChatterChannelsWnd = NULL; |
||
177 | if (IsWindow (hGamesWnd)) |
||
178 | DestroyWindow (hGamesWnd); |
||
179 | hGamesWnd = NULL; |
||
180 | if (IsWindow (hMOTDWnd)) |
||
181 | DestroyWindow (hMOTDWnd); |
||
182 | hMOTDWnd = NULL; |
||
183 | if (IsWindow (hOpponentsWnd)) |
||
184 | DestroyWindow (hOpponentsWnd); |
||
185 | hOpponentsWnd = NULL; |
||
186 | if (IsWindow (hSoughtWnd)) |
||
187 | DestroyWindow (hSoughtWnd); |
||
188 | hSoughtWnd = NULL; |
||
189 | |||
190 | // shutdown WinSock |
||
191 | WSACleanup (); |
||
192 | |||
193 | return; // finished |
||
194 | } |
||
195 | |||
196 | |||
197 | bool PlayerNetwork_Think (player_t *player) |
||
198 | { |
||
199 | // this function is called once per game tick to listen to the network and react to what the server tells us. Returns TRUE |
||
200 | // if we need to update the scene. |
||
201 | |||
202 | player_t *local_player; |
||
203 | char *block_start; |
||
204 | char *block_end; |
||
205 | char *ascii_sendbuffer; // mallocated |
||
206 | wchar_t *widechar_buffer; // mallocated |
||
207 | wchar_t *widechar_line; |
||
208 | int send_retval; |
||
209 | int recv_retval; |
||
210 | int byte_index; |
||
211 | int rewrite_index; |
||
212 | int length; |
||
213 | bool do_update; |
||
214 | |||
215 | // are we NOT connected yet ? |
||
216 | if (!player->is_connected) |
||
217 | return (false); // consistency check: if we aren't connected, don't do anything |
||
218 | |||
219 | // are we NOT logged in yet AND is our socket valid AND is there nothing in the center of the screen yet ? |
||
220 | if (!player->is_logged_in && (player->our_socket != INVALID_SOCKET) && !the_scene.gui.central_text.is_displayed) |
||
221 | { |
||
222 | Scene_SetText (&the_scene.gui.central_text, 50.0f, 40.0f, -1, ALIGN_CENTER, ALIGN_CENTER, ALIGN_CENTER, centermsg_fontindex, RGBA_TO_RGBACOLOR (255, 255, 255, 191), |
||
223 | 999999.0f, true, LOCALIZE (L"Connecting")); // display "connecting" in the middle of the screen |
||
224 | the_scene.gui.want_spinwheel = true; // start spinning wheel |
||
225 | } |
||
226 | |||
227 | do_update = false; // assume we don't need to update the scene until told otherwise |
||
228 | |||
229 | //////////////////////////// |
||
230 | // BEGIN listening to server |
||
231 | |||
232 | // proceed in filling the ascii buffer with what's coming |
||
233 | |||
234 | // get a hand on the end of the player's recv buffer and compute its remaining size |
||
235 | length = strlen (player->ascii_recvbuffer); |
||
236 | |||
237 | // if the server sent us something, append it to what we've received |
||
238 | recv_retval = recv (player->our_socket, &player->ascii_recvbuffer[length], player->recvbuffer_size - length, 0); |
||
239 | if (recv_retval > 0) |
||
240 | { |
||
241 | // parse all received data and eradicate all carriage returns. Also convert %'s to a safer glyph. |
||
242 | rewrite_index = 0; |
||
243 | for (byte_index = 0; byte_index < recv_retval; byte_index++) |
||
244 | if (player->ascii_recvbuffer[length + byte_index] != '\r') |
||
245 | { |
||
246 | if (player->ascii_recvbuffer[length + byte_index] == '%') |
||
247 | player->ascii_recvbuffer[length + rewrite_index] = '¤'; |
||
248 | else |
||
249 | player->ascii_recvbuffer[length + rewrite_index] = player->ascii_recvbuffer[length + byte_index]; |
||
250 | rewrite_index++; |
||
251 | } |
||
252 | player->ascii_recvbuffer[length + rewrite_index] = 0; // terminate the buffer ourselves |
||
253 | length = strlen (player->ascii_recvbuffer); // and update the new recvbuffer's length |
||
254 | } |
||
255 | else if (recv_retval == SOCKET_ERROR) |
||
256 | { |
||
257 | recv_retval = WSAGetLastError (); // ask Windows to be more specific about the error |
||
258 | if (recv_retval == WSAEWOULDBLOCK) |
||
259 | ; // it's okay. We're a non-blocking socket. |
||
260 | else if (recv_retval == WSAETIMEDOUT) |
||
261 | { |
||
262 | messagebox.hWndParent = hMainWnd; |
||
263 | wcscpy_s (messagebox.title, WCHAR_SIZEOF (messagebox.title), LOCALIZE (L"ImportantMessage")); |
||
264 | wcscpy_s (messagebox.text, WCHAR_SIZEOF (messagebox.text), LOCALIZE (L"Error_ConnectionToChessServerLost")); |
||
265 | messagebox.flags = MB_ICONWARNING | MB_OK; |
||
266 | DialogBox_Message (&messagebox); // display a modeless error message box |
||
267 | |||
268 | PlayerNetwork_Shutdown (player); // on error, shutdown the network and display an error message |
||
269 | return (false); |
||
270 | } |
||
271 | } |
||
272 | |||
273 | // END listening to server |
||
274 | ////////////////////////// |
||
275 | |||
276 | ////////////////////////////////// |
||
277 | // BEGIN evaluating server replies |
||
278 | |||
279 | // are we NOT logged in yet ? |
||
280 | if (!player->is_logged_in) |
||
281 | { |
||
282 | // is it a login prompt ? |
||
283 | if ((length > 7) && (strcmp (&player->ascii_recvbuffer[length - 7], "login: ") == 0)) |
||
284 | { |
||
285 | Player_SendBuffer_Add (player, 1000, L"%s\n", options.network.login); // send the login string |
||
286 | player->ascii_recvbuffer[0] = 0; // and discard recvbuffer as we've processed this message |
||
287 | } |
||
288 | |||
289 | // else is it a password prompt ? |
||
290 | else if (((length > 10) && (strcmp (&player->ascii_recvbuffer[length - 10], "password: ") == 0)) |
||
291 | || ((length > 3) && (strcmp (&player->ascii_recvbuffer[length - 3], "\":\n") == 0))) |
||
292 | { |
||
293 | Player_SendBuffer_Add (player, 1000, L"%s\n", options.network.password); // send the password string |
||
294 | player->ascii_recvbuffer[0] = 0; // and discard recvbuffer as we've processed this message |
||
295 | } |
||
296 | |||
297 | // else is it a confirmation that we've successfully logged in ? |
||
298 | else if (((block_start = strstr (player->ascii_recvbuffer, "\n**** Starting FICS session as ")) != NULL) |
||
299 | && (block_end = strstr (player->ascii_recvbuffer, " ****\n"))) |
||
300 | { |
||
301 | player->is_logged_in = true; // remember we are logged in |
||
302 | the_scene.gui.central_text.disappear_time = current_time + 1.0f; // fade the "connecting" phrase out now (FIXME: ugly) |
||
303 | the_scene.gui.want_spinwheel = false; // stop spinning wheel |
||
304 | do_update = true; // and update the scene |
||
305 | |||
306 | block_start += 31; // skip the "\n**** Starting FICS session as " substring |
||
307 | local_player = Player_FindByType (PLAYER_HUMAN); // get a pointer to the local player |
||
308 | |||
309 | // copy out local player's login name, as reported by server, and convert it to wchar_t |
||
310 | for (byte_index = 0; (byte_index < WCHAR_SIZEOF (local_player->name)) && isalpha (block_start[byte_index]); byte_index++) |
||
311 | local_player->name[byte_index] = (wchar_t) block_start[byte_index]; // copy nickname one character after the other. Very dirty conversion. |
||
312 | if (byte_index < WCHAR_SIZEOF (local_player->name)) |
||
313 | local_player->name[byte_index] = 0; // finish the string ourselves |
||
314 | else |
||
315 | local_player->name[WCHAR_SIZEOF (local_player->name) - 1] = 0; // truncate it if neeeded |
||
316 | |||
317 | server_motd[0] = 0; // remember we haven't read the MOTD yet |
||
318 | the_board.reevaluate = true; // reeevaluate the board (to update the title bar) |
||
319 | |||
320 | // see if there's more data to parse after this message, in which case move it in place |
||
321 | length = strlen (block_end + 6); |
||
322 | if (length > 0) |
||
323 | memmove (player->ascii_recvbuffer, block_end + 6, length + 1); |
||
324 | else |
||
325 | player->ascii_recvbuffer[0] = 0; // else discard recvbuffer as we've processed this message |
||
326 | } |
||
327 | } |
||
328 | |||
329 | // else we are logged in |
||
330 | else |
||
331 | { |
||
332 | // see if we have a complete reply block (remember: percent sign was replaced with '¤') |
||
333 | block_end = strstr (player->ascii_recvbuffer, "\nfics¤ "); |
||
334 | |||
335 | // as long as we can find some... |
||
336 | while (block_end != NULL) |
||
337 | { |
||
338 | *block_end = 0; // break the string here |
||
339 | |||
340 | // convert the block to wide char and evaluate it, taking action if needed |
||
341 | ConvertToWideChar (player->recvbuffer, player->recvbuffer_size, player->ascii_recvbuffer); |
||
342 | Debug_Log (L"RECEIVED:[%s]\n", player->recvbuffer); // log what we've received |
||
343 | EvaluateServerReply (player); // and evaluate it |
||
344 | |||
345 | // see if there's more data to parse after this message, in which case move it in place |
||
346 | length = strlen (block_end + 7); |
||
347 | if (length > 0) |
||
348 | memmove (player->ascii_recvbuffer, block_end + 7, length + 1); |
||
349 | else |
||
350 | { |
||
351 | player->ascii_recvbuffer[0] = 0; |
||
352 | break; // it was the last prompt, so discard recvbuffer as we've processed this message |
||
353 | } |
||
354 | block_end = strstr (player->ascii_recvbuffer, "\nfics¤ "); // remember: percent sign was replaced with '¤' |
||
355 | } |
||
356 | } |
||
357 | |||
358 | // END evaluating server replies |
||
359 | //////////////////////////////// |
||
360 | |||
361 | // have we been notified that the current player just changed, has at least one move been played AND is it the remote player's turn now ? |
||
362 | if (the_board.has_playerchanged && (the_board.move_count > 1) && (Board_ColorToMove (&the_board) == player->color)) |
||
363 | { |
||
364 | Debug_Log (L"===Local player just played, sending the chosen move to chess server===\n"); |
||
365 | Player_SendBuffer_Add (player, 1000, L"%s\n", the_board.moves[the_board.move_count - 1].pgntext); // send the move string to the chess server |
||
366 | } |
||
367 | |||
368 | // has the user entered some chatter text ? |
||
369 | if (!the_scene.gui.is_entering_text && (the_scene.gui.entered_ccreply.text != NULL)) |
||
370 | { |
||
371 | // is player logged in AND has a chatter channel been selected ? if so, send chat string to server |
||
372 | if (player->is_logged_in && (selected_chatterchannel != NULL)) |
||
373 | Player_SendBuffer_Add (player, 1000, L"+channel %d\ntell %d %s\n", selected_chatterchannel->id, selected_chatterchannel->id, the_scene.gui.entered_ccreply.text); |
||
374 | |||
375 | // reset the entered text buffer |
||
376 | SAFE_free ((void **) &the_scene.gui.entered_ccreply.text); // reset the entered text buffer |
||
377 | the_scene.gui.entered_ccreply.text_length = 0; // and set its length to zero |
||
378 | } |
||
379 | |||
380 | // does the local player want to cancel its last move ? this can be either HIS move or HIS OPPONENTS's move... |
||
381 | local_player = Player_FindByType (PLAYER_HUMAN); // get a pointer to the local player |
||
382 | if ((local_player != NULL) && local_player->wants_cancel) |
||
383 | { |
||
384 | Player_SendBuffer_Add (player, 1000, L"takeback\n"); // send the takeback request to our opponent |
||
385 | Interlocutor_Notify (Interlocutor_FindOrCreate (player->name), LOCALIZE (L"Chat_TakebackRequestSent"), player->name); // send a notification to its chat window |
||
136 | pmbaty | 386 | the_board.game_state = STATE_PLAYING; // remember the game is now playing (in case we wanted to cancel the closing move of a finished game, this opens the game again) |
1 | pmbaty | 387 | local_player->wants_cancel = false; // don't do this all day long |
388 | } |
||
389 | if (player->wants_cancel) |
||
390 | { |
||
391 | Player_SendBuffer_Add (player, 1000, L"takeback\n"); // send the takeback request to our opponent |
||
392 | Interlocutor_Notify (Interlocutor_FindOrCreate (player->name), LOCALIZE (L"Chat_TakebackRequestSent"), player->name); // send a notification to its chat window |
||
136 | pmbaty | 393 | the_board.game_state = STATE_PLAYING; // remember the game is now playing (in case we wanted to cancel the closing move of a finished game, this opens the game again) |
1 | pmbaty | 394 | player->wants_cancel = false; // don't do this all day long |
395 | } |
||
396 | |||
397 | // if we have something to send, do it |
||
398 | if (!player->sendbuffer_locked && (player->sendbuffer[0] != 0) && (animation_endtime + 1.0f < current_time)) |
||
399 | { |
||
400 | // log what we're sending |
||
401 | Debug_Log (L"SENDING:[%s]\n", player->sendbuffer); |
||
402 | |||
403 | player->sendbuffer_locked = true; // lock the buffer |
||
404 | |||
405 | // now read line per line (mallocate a line buffer as large as necessary) |
||
406 | length = wcslen (player->sendbuffer) + 1 + 1; // +1 for \n, +1 for null terminator |
||
407 | ascii_sendbuffer = (char *) SAFE_malloc (length, sizeof (char), false); |
||
408 | widechar_buffer = (wchar_t *) SAFE_malloc (length, sizeof (wchar_t), false); |
||
409 | widechar_line = player->sendbuffer; // start at the first character |
||
410 | while ((widechar_line = wcsgets (widechar_buffer, length, widechar_line)) != NULL) |
||
411 | { |
||
412 | wcscat_s (widechar_buffer, length, L"\n"); // put the carriage return back |
||
413 | ConvertTo7BitASCII (ascii_sendbuffer, length, widechar_buffer); // convert to ASCII |
||
414 | send_retval = send (player->our_socket, ascii_sendbuffer, strlen (ascii_sendbuffer), 0); // send data |
||
415 | } |
||
416 | SAFE_free ((void **) &widechar_buffer); // we no longer need the line buffer, so free it |
||
417 | SAFE_free ((void **) &ascii_sendbuffer); |
||
418 | |||
419 | player->sendbuffer[0] = 0; // what we had to send has been sent, reset the send buffer |
||
420 | player->sendbuffer_locked = false; // and unlock it |
||
421 | |||
422 | // did send() report an error ? |
||
423 | if (send_retval == SOCKET_ERROR) |
||
424 | { |
||
425 | messagebox.hWndParent = hMainWnd; |
||
426 | wcscpy_s (messagebox.title, WCHAR_SIZEOF (messagebox.title), LOCALIZE (L"ImportantMessage")); |
||
427 | wcscpy_s (messagebox.text, WCHAR_SIZEOF (messagebox.text), LOCALIZE (L"Error_ConnectionToChessServerLost")); |
||
428 | messagebox.flags = MB_ICONERROR | MB_OK; |
||
429 | DialogBox_Message (&messagebox); // display a modeless error message box |
||
430 | |||
431 | PlayerNetwork_Shutdown (player); // on error, shutdown the network and display an error message |
||
432 | return (true); // on error, cancel |
||
433 | } |
||
434 | } |
||
435 | |||
436 | return (do_update); // finished |
||
437 | } |
||
438 | |||
439 | |||
440 | static void EvaluateServerReply (player_t *player) |
||
441 | { |
||
442 | // this function parses a network reply and evaluates it, deciding what to do |
||
443 | |||
444 | ////////////////////// |
||
445 | // OUT OF GAME REPLIES |
||
446 | |||
447 | // the MOTD is the first reply to be evaluated |
||
448 | EvaluateServerReply_MOTD (player); |
||
449 | |||
450 | // evaluate dangerous replies first, i.e. those that contain user-settable text |
||
451 | EvaluateServerReply_Announcement (player); |
||
452 | EvaluateServerReply_ChannelMessage (player); |
||
453 | EvaluateServerReply_PrivateMessage (player); |
||
454 | EvaluateServerReply_Finger (player); |
||
455 | |||
456 | // the less dangerous replies can now be evaluated without ambiguity |
||
457 | EvaluateServerReply_Seek (player); |
||
458 | EvaluateServerReply_Challenge (player); |
||
459 | EvaluateServerReply_ChallengeAccepted (player); |
||
460 | EvaluateServerReply_ChallengeDeclined (player); |
||
461 | EvaluateServerReply_Takeback (player); |
||
462 | EvaluateServerReply_TakebackDeclinedByOther (player); |
||
463 | EvaluateServerReply_TakebackDeclinedByYou (player); |
||
464 | EvaluateServerReply_PlayNotAllowed (player); |
||
465 | EvaluateServerReply_PlayUnexistent (player); |
||
466 | EvaluateServerReply_PlayWrongRating (player); |
||
467 | EvaluateServerReply_ChannelsAndMembers (player); |
||
468 | EvaluateServerReply_SoughtList (player); |
||
469 | EvaluateServerReply_PlayersList (player); |
||
470 | |||
471 | ////////////////// |
||
472 | // IN GAME REPLIES |
||
473 | |||
474 | EvaluateServerReply_GameStarting (player); |
||
475 | EvaluateServerReply_GameState (player); |
||
476 | EvaluateServerReply_GameResults (player); |
||
477 | |||
478 | return; // finished evaluating this reply |
||
479 | } |