1 /****************************************************************************** 2 * [ maze ] ... 3 * 4 * modified: [ 03-08-15 ] Ge van Geldorp <ge@gse.nl> 5 * ported to ReactOS 6 * modified: [ 94-10-8 ] Ge van Geldorp <Ge.vanGeldorp@lr.tudelft.nl> 7 * ported to MS Windows 8 * modified: [ 3-7-93 ] Jamie Zawinski <jwz@lucid.com> 9 * added the XRoger logo, cleaned up resources, made 10 * grid size a parameter. 11 * modified: [ 3-3-93 ] Jim Randell <jmr@mddjmr.fc.hp.com> 12 * Added the colour stuff and integrated it with jwz's 13 * screenhack stuff. There's still some work that could 14 * be done on this, particularly allowing a resource to 15 * specify how big the squares are. 16 * modified: [ 10-4-88 ] Richard Hess ...!uunet!cimshop!rhess 17 * [ Revised primary execution loop within main()... 18 * [ Extended X event handler, check_events()... 19 * modified: [ 1-29-88 ] Dave Lemke lemke@sun.com 20 * [ Hacked for X11... 21 * [ Note the word "hacked" -- this is extremely ugly, but at 22 * [ least it does the job. NOT a good programming example 23 * [ for X. 24 * original: [ 6/21/85 ] Martin Weiss Sun Microsystems [ SunView ] 25 * 26 ****************************************************************************** 27 Copyright 1988 by Sun Microsystems, Inc. Mountain View, CA. 28 29 All Rights Reserved 30 31 Permission to use, copy, modify, and distribute this software and its 32 documentation for any purpose and without fee is hereby granted, 33 provided that the above copyright notice appear in all copies and that 34 both that copyright notice and this permission notice appear in 35 supporting documentation, and that the names of Sun or MIT not be 36 used in advertising or publicity pertaining to distribution of the 37 software without specific prior written permission. Sun and M.I.T. 38 make no representations about the suitability of this software for 39 any purpose. It is provided "as is" without any express or implied warranty. 40 41 SUN DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING 42 ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 43 PURPOSE. IN NO EVENT SHALL SUN BE LIABLE FOR ANY SPECIAL, INDIRECT 44 OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS 45 OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE 46 OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE 47 OR PERFORMANCE OF THIS SOFTWARE. 48 *****************************************************************************/ 49 50 #define STRICT 51 52 #include <windows.h> /* required for all Windows applications */ 53 #include <scrnsave.h> 54 #include <tchar.h> 55 #include <stdlib.h> 56 #include <commctrl.h> 57 #include <string.h> 58 #include <time.h> 59 #include "resource.h" 60 61 #define APPNAME _T("Maze") 62 63 LRESULT CALLBACK ScreenSaverProc(HWND hWnd, UINT message, WPARAM uParam, LPARAM lParam); 64 static int choose_door(); 65 static long backup(); 66 static void draw_wall(); 67 static void draw_solid_square(int, int, int, HDC, HBRUSH); 68 static void enter_square(int, HDC, HBRUSH); 69 70 extern HINSTANCE hMainInstance; /* current instance */ 71 HBRUSH hBrushDead; 72 HBRUSH hBrushLiving; 73 HPEN hPenWall; 74 HDC hDC; 75 76 static int solve_delay, pre_solve_delay, post_solve_delay, size; 77 78 #define MAX_MAZE_SIZE_X ((unsigned long) 1000) // Dynamic detection? 79 #define MAX_MAZE_SIZE_Y ((unsigned long) 1000) // Dynamic detection? 80 81 #define MOVE_LIST_SIZE (MAX_MAZE_SIZE_X * MAX_MAZE_SIZE_Y) 82 83 #define WALL_TOP 0x8000 84 #define WALL_RIGHT 0x4000 85 #define WALL_BOTTOM 0x2000 86 #define WALL_LEFT 0x1000 87 88 #define DOOR_IN_TOP 0x800 89 #define DOOR_IN_RIGHT 0x400 90 #define DOOR_IN_BOTTOM 0x200 91 #define DOOR_IN_LEFT 0x100 92 #define DOOR_IN_ANY 0xF00 93 94 #define DOOR_OUT_TOP 0x80 95 #define DOOR_OUT_RIGHT 0x40 96 #define DOOR_OUT_BOTTOM 0x20 97 #define DOOR_OUT_LEFT 0x10 98 99 #define START_SQUARE 0x2 100 #define END_SQUARE 0x1 101 102 #define border_x (0) 103 #define border_y (0) 104 105 #define get_random(x) (rand() % (x)) 106 107 static unsigned short maze[MAX_MAZE_SIZE_X][MAX_MAZE_SIZE_Y]; 108 109 static struct { 110 unsigned int x; 111 unsigned int y; 112 unsigned int dir; 113 } move_list[MOVE_LIST_SIZE], save_path[MOVE_LIST_SIZE], path[MOVE_LIST_SIZE]; 114 115 static int maze_size_x, maze_size_y; 116 static long sqnum, path_length; 117 static int cur_sq_x, cur_sq_y; 118 static int start_x, start_y, start_dir, end_x, end_y, end_dir; 119 static int grid_width, grid_height; 120 static int bw; 121 static int state = 1, pathi = 0; 122 static LPCWSTR registryPath = _T("Software\\Microsoft\\ScreenSavers\\mazescr"); 123 124 static void SetDefaults() 125 { 126 size = 10; 127 pre_solve_delay = 5000; 128 post_solve_delay = 5000; 129 solve_delay = 1; 130 } 131 132 static void ReadRegistry() 133 { 134 LONG result; 135 HKEY skey; 136 DWORD valuetype, valuesize, val_size, val_presd, val_postsd, val_sd; 137 138 SetDefaults(); 139 140 result = RegOpenKeyEx(HKEY_CURRENT_USER, registryPath, 0, KEY_READ, &skey); 141 if(result != ERROR_SUCCESS) 142 return; 143 144 valuesize = sizeof(DWORD); 145 146 result = RegQueryValueEx(skey, _T("size"), NULL, &valuetype, (LPBYTE)&val_size, &valuesize); 147 if(result == ERROR_SUCCESS) 148 size = val_size; 149 result = RegQueryValueEx(skey, _T("pre_solve_delay"), NULL, &valuetype, (LPBYTE)&val_presd, &valuesize); 150 if(result == ERROR_SUCCESS) 151 pre_solve_delay = val_presd; 152 result = RegQueryValueEx(skey, _T("post_solve_delay"), NULL, &valuetype, (LPBYTE)&val_postsd, &valuesize); 153 if(result == ERROR_SUCCESS) 154 post_solve_delay = val_postsd; 155 result = RegQueryValueEx(skey, _T("solve_delay"), NULL, &valuetype, (LPBYTE)&val_sd, &valuesize); 156 if(result == ERROR_SUCCESS) 157 solve_delay = val_sd; 158 159 RegCloseKey(skey); 160 } 161 162 static void WriteRegistry() 163 { 164 LONG result; 165 HKEY skey; 166 DWORD disp; 167 168 result = RegCreateKeyEx(HKEY_CURRENT_USER, registryPath, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &skey, &disp); 169 if(result != ERROR_SUCCESS) 170 return; 171 172 RegSetValueEx(skey, _T("size"), 0, REG_DWORD, (LPBYTE)&size, sizeof(size)); 173 RegSetValueEx(skey, _T("pre_solve_delay"), 0, REG_DWORD, (LPBYTE)&pre_solve_delay, sizeof(pre_solve_delay)); 174 RegSetValueEx(skey, _T("post_solve_delay"), 0, REG_DWORD, (LPBYTE)&post_solve_delay, sizeof(post_solve_delay)); 175 RegSetValueEx(skey, _T("solve_delay"), 0, REG_DWORD, (LPBYTE)&solve_delay, sizeof(solve_delay)); 176 177 RegCloseKey(skey); 178 } 179 180 static void set_maze_sizes(width, height) 181 int width, height; 182 { 183 maze_size_x = (width -1)/ grid_width; 184 maze_size_y = (height-1) / grid_height; 185 if (maze_size_x > MAX_MAZE_SIZE_X) 186 maze_size_x = MAX_MAZE_SIZE_X; 187 if (maze_size_y > MAX_MAZE_SIZE_Y) 188 maze_size_y = MAX_MAZE_SIZE_Y; 189 } 190 191 static void initialize_maze() /* draw the surrounding wall and start/end squares */ 192 { 193 register int i, j, wall; 194 195 /* initialize all squares */ 196 for (i = 0; i < maze_size_x; i++) { 197 for (j = 0; j < maze_size_y; j++) { 198 maze[i][j] = 0; 199 } 200 } 201 202 /* top wall */ 203 for (i = 0; i < maze_size_x; i++) { 204 maze[i][0] |= WALL_TOP; 205 } 206 207 /* right wall */ 208 for (j = 0; j < maze_size_y; j++) { 209 maze[maze_size_x - 1][j] |= WALL_RIGHT; 210 } 211 212 /* bottom wall */ 213 for (i = 0; i < maze_size_x; i++) { 214 maze[i][maze_size_y - 1] |= WALL_BOTTOM; 215 } 216 217 /* left wall */ 218 for (j = 0; j < maze_size_y; j++) { 219 maze[0][j] |= WALL_LEFT; 220 } 221 222 /* set start square */ 223 wall = get_random(4); 224 switch (wall) { 225 case 0: 226 i = get_random(maze_size_x); 227 j = 0; 228 break; 229 case 1: 230 i = maze_size_x - 1; 231 j = get_random(maze_size_y); 232 break; 233 case 2: 234 i = get_random(maze_size_x); 235 j = maze_size_y - 1; 236 break; 237 case 3: 238 i = 0; 239 j = get_random(maze_size_y); 240 break; 241 } 242 maze[i][j] |= START_SQUARE; 243 maze[i][j] |= (DOOR_IN_TOP >> wall); 244 maze[i][j] &= ~(WALL_TOP >> wall); 245 cur_sq_x = i; 246 cur_sq_y = j; 247 start_x = i; 248 start_y = j; 249 start_dir = wall; 250 sqnum = 0; 251 252 /* set end square */ 253 wall = (wall + 2) % 4; 254 switch (wall) { 255 case 0: 256 i = get_random(maze_size_x); 257 j = 0; 258 break; 259 case 1: 260 i = maze_size_x - 1; 261 j = get_random(maze_size_y); 262 break; 263 case 2: 264 i = get_random(maze_size_x); 265 j = maze_size_y - 1; 266 break; 267 case 3: 268 i = 0; 269 j = get_random(maze_size_y); 270 break; 271 } 272 maze[i][j] |= END_SQUARE; 273 maze[i][j] |= (DOOR_OUT_TOP >> wall); 274 maze[i][j] &= ~(WALL_TOP >> wall); 275 end_x = i; 276 end_y = j; 277 end_dir = wall; 278 } 279 280 static void create_maze(HWND hWnd) /* create a maze layout given the initialized maze */ 281 { 282 register int i, newdoor = 0; 283 284 do { 285 move_list[sqnum].x = cur_sq_x; 286 move_list[sqnum].y = cur_sq_y; 287 move_list[sqnum].dir = newdoor; 288 while ((newdoor = choose_door(hDC)) == -1) { /* pick a door */ 289 if (backup() == -1) { /* no more doors ... backup */ 290 return; /* done ... return */ 291 } 292 } 293 294 /* mark the out door */ 295 maze[cur_sq_x][cur_sq_y] |= (DOOR_OUT_TOP >> newdoor); 296 297 switch (newdoor) { 298 case 0: cur_sq_y--; 299 break; 300 case 1: cur_sq_x++; 301 break; 302 case 2: cur_sq_y++; 303 break; 304 case 3: cur_sq_x--; 305 break; 306 } 307 sqnum++; 308 309 /* mark the in door */ 310 maze[cur_sq_x][cur_sq_y] |= (DOOR_IN_TOP >> ((newdoor + 2) % 4)); 311 312 /* if end square set path length and save path */ 313 if (maze[cur_sq_x][cur_sq_y] & END_SQUARE) { 314 path_length = sqnum; 315 for (i = 0; i < path_length; i++) { 316 save_path[i].x = move_list[i].x; 317 save_path[i].y = move_list[i].y; 318 save_path[i].dir = move_list[i].dir; 319 } 320 } 321 } while (1); 322 } 323 324 static int choose_door(HDC hDC) /* pick a new path */ 325 { 326 int candidates[3]; 327 register int num_candidates; 328 329 num_candidates = 0; 330 331 /* top wall */ 332 if (maze[cur_sq_x][cur_sq_y] & DOOR_IN_TOP) 333 goto rightwall; 334 if (maze[cur_sq_x][cur_sq_y] & DOOR_OUT_TOP) 335 goto rightwall; 336 if (maze[cur_sq_x][cur_sq_y] & WALL_TOP) 337 goto rightwall; 338 if (maze[cur_sq_x][cur_sq_y - 1] & DOOR_IN_ANY) { 339 maze[cur_sq_x][cur_sq_y] |= WALL_TOP; 340 maze[cur_sq_x][cur_sq_y - 1] |= WALL_BOTTOM; 341 draw_wall(cur_sq_x, cur_sq_y, 0, hDC); 342 goto rightwall; 343 } 344 candidates[num_candidates++] = 0; 345 346 rightwall: 347 /* right wall */ 348 if (maze[cur_sq_x][cur_sq_y] & DOOR_IN_RIGHT) 349 goto bottomwall; 350 if (maze[cur_sq_x][cur_sq_y] & DOOR_OUT_RIGHT) 351 goto bottomwall; 352 if (maze[cur_sq_x][cur_sq_y] & WALL_RIGHT) 353 goto bottomwall; 354 if (maze[cur_sq_x + 1][cur_sq_y] & DOOR_IN_ANY) { 355 maze[cur_sq_x][cur_sq_y] |= WALL_RIGHT; 356 maze[cur_sq_x + 1][cur_sq_y] |= WALL_LEFT; 357 draw_wall(cur_sq_x, cur_sq_y, 1, hDC); 358 goto bottomwall; 359 } 360 candidates[num_candidates++] = 1; 361 362 bottomwall: 363 /* bottom wall */ 364 if (maze[cur_sq_x][cur_sq_y] & DOOR_IN_BOTTOM) 365 goto leftwall; 366 if (maze[cur_sq_x][cur_sq_y] & DOOR_OUT_BOTTOM) 367 goto leftwall; 368 if (maze[cur_sq_x][cur_sq_y] & WALL_BOTTOM) 369 goto leftwall; 370 if (maze[cur_sq_x][cur_sq_y + 1] & DOOR_IN_ANY) { 371 maze[cur_sq_x][cur_sq_y] |= WALL_BOTTOM; 372 maze[cur_sq_x][cur_sq_y + 1] |= WALL_TOP; 373 draw_wall(cur_sq_x, cur_sq_y, 2, hDC); 374 goto leftwall; 375 } 376 candidates[num_candidates++] = 2; 377 378 leftwall: 379 /* left wall */ 380 if (maze[cur_sq_x][cur_sq_y] & DOOR_IN_LEFT) 381 goto donewall; 382 if (maze[cur_sq_x][cur_sq_y] & DOOR_OUT_LEFT) 383 goto donewall; 384 if (maze[cur_sq_x][cur_sq_y] & WALL_LEFT) 385 goto donewall; 386 if (maze[cur_sq_x - 1][cur_sq_y] & DOOR_IN_ANY) { 387 maze[cur_sq_x][cur_sq_y] |= WALL_LEFT; 388 maze[cur_sq_x - 1][cur_sq_y] |= WALL_RIGHT; 389 draw_wall(cur_sq_x, cur_sq_y, 3, hDC); 390 goto donewall; 391 } 392 candidates[num_candidates++] = 3; 393 394 donewall: 395 if (num_candidates == 0) 396 return -1; 397 if (num_candidates == 1) 398 return candidates[0]; 399 return candidates[get_random(num_candidates)]; 400 401 } 402 403 static long backup() /* back up a move */ 404 { 405 sqnum--; 406 if (0 <= sqnum) { 407 cur_sq_x = move_list[sqnum].x; 408 cur_sq_y = move_list[sqnum].y; 409 } 410 return sqnum; 411 } 412 413 static void draw_solid_square(i, j, dir, hDC, hBrush) /* draw a solid square in a square */ 414 register int i, j, dir; 415 HDC hDC; 416 HBRUSH hBrush; 417 { 418 RECT rc; 419 420 switch (dir) { 421 case 0: 422 rc.left = border_x + bw + grid_width * i; 423 rc.right = rc.left + grid_width - (bw + bw); 424 rc.top = border_y - bw + grid_height * j; 425 rc.bottom = rc.top + grid_height; 426 break; 427 case 1: 428 rc.left = border_x + bw + grid_width * i; 429 rc.right = rc.left + grid_width; 430 rc.top = border_y + bw + grid_height * j; 431 rc.bottom = rc.top + grid_height - (bw + bw); 432 break; 433 case 2: 434 rc.left = border_x + bw + grid_width * i; 435 rc.right = rc.left + grid_width - (bw + bw); 436 rc.top = border_y + bw + grid_height * j; 437 rc.bottom = rc.top + grid_height; 438 break; 439 case 3: 440 rc.left = border_x - bw + grid_width * i; 441 rc.right = rc.left + grid_width; 442 rc.top = border_y + bw + grid_height * j; 443 rc.bottom = rc.top + grid_height - (bw + bw); 444 break; 445 } 446 (void) FillRect(hDC, &rc, hBrush); 447 } 448 449 static void draw_maze_border(HWND hWnd) /* draw the maze outline */ 450 { 451 register int i, j; 452 HBRUSH hBrush; 453 454 SelectObject(hDC, hPenWall); 455 456 for (i = 0; i < maze_size_x; i++) { 457 if (maze[i][0] & WALL_TOP) { 458 MoveToEx(hDC, border_x + grid_width * i, border_y, NULL); 459 (void) LineTo(hDC, border_x + grid_width * (i + 1) - 1, border_y); 460 } 461 if ((maze[i][maze_size_y - 1] & WALL_BOTTOM)) { 462 MoveToEx(hDC, border_x + grid_width * i, border_y + grid_height * (maze_size_y) -1, NULL); 463 (void) LineTo(hDC, border_x + grid_width * (i + 1) - 1, border_y + grid_height * (maze_size_y) -1); 464 } 465 } 466 for (j = 0; j < maze_size_y; j++) { 467 if (maze[maze_size_x - 1][j] & WALL_RIGHT) { 468 MoveToEx(hDC, border_x + grid_width * maze_size_x - 1, border_y + grid_height * j, NULL); 469 (void) LineTo(hDC, border_x + grid_width * maze_size_x - 1, border_y + grid_height * (j + 1) - 1); 470 } 471 if (maze[0][j] & WALL_LEFT) { 472 MoveToEx(hDC, border_x, border_y + grid_height * j, NULL); 473 (void) LineTo(hDC, border_x, border_y + grid_height * (j + 1) - 1); 474 } 475 } 476 477 hBrush = GetStockObject(WHITE_BRUSH); 478 draw_solid_square(start_x, start_y, start_dir, hDC, hBrush); 479 draw_solid_square(end_x, end_y, end_dir, hDC, hBrush); 480 } 481 482 static void draw_wall(i, j, dir, hDC) /* draw a single wall */ 483 register int i, j, dir; 484 HDC hDC; 485 { 486 SelectObject(hDC, hPenWall); 487 488 switch (dir) { 489 case 0: 490 MoveToEx(hDC, border_x + grid_width * i, border_y + grid_height * j, NULL); 491 (void) LineTo(hDC, border_x + grid_width * (i + 1), border_y + grid_height * j); 492 break; 493 case 1: 494 MoveToEx(hDC, border_x + grid_width * (i + 1), border_y + grid_height * j, NULL); 495 (void) LineTo(hDC, border_x + grid_width * (i + 1), border_y + grid_height * (j + 1)); 496 break; 497 case 2: 498 MoveToEx(hDC, border_x + grid_width * i, border_y + grid_height * (j + 1), NULL); 499 (void) LineTo(hDC, border_x + grid_width * (i + 1), border_y + grid_height * (j + 1)); 500 break; 501 case 3: 502 MoveToEx(hDC, border_x + grid_width * i, border_y + grid_height * j, NULL); 503 (void) LineTo(hDC, border_x + grid_width * i, border_y + grid_height * (j + 1)); 504 break; 505 } 506 } 507 508 static void begin_solve_maze(HWND hWnd) /* solve it with graphical feedback */ 509 { 510 /* plug up the surrounding wall */ 511 maze[start_x][start_y] |= (WALL_TOP >> start_dir); 512 maze[end_x][end_y] |= (WALL_TOP >> end_dir); 513 514 /* initialize search path */ 515 pathi = 0; 516 path[pathi].x = end_x; 517 path[pathi].y = end_y; 518 path[pathi].dir = -1; 519 } 520 521 static int solve_maze(HWND hWnd) /* solve it with graphical feedback */ 522 { 523 int ret; 524 int action_done; 525 526 do { 527 action_done = 1; 528 if (++path[pathi].dir >= 4) { 529 pathi--; 530 draw_solid_square((int) (path[pathi].x), (int) (path[pathi].y), (int) (path[pathi].dir), hDC, hBrushDead); 531 ret = 0; 532 } 533 else if (!(maze[path[pathi].x][path[pathi].y] & (WALL_TOP >> path[pathi].dir)) && 534 ((pathi == 0) || ((path[pathi].dir != (int) (path[pathi - 1].dir + 2) % 4)))) { 535 enter_square(pathi, hDC, hBrushLiving); 536 pathi++; 537 if (maze[path[pathi].x][path[pathi].y] & START_SQUARE) { 538 539 ret = 1; 540 } 541 else { 542 ret = 0; 543 } 544 } 545 else { 546 action_done = 0; 547 } 548 } while (!action_done); 549 return ret; 550 } 551 552 static void enter_square(int n, HDC hDC, HBRUSH hBrush) /* move into a neighboring square */ 553 { 554 draw_solid_square((int) path[n].x, (int) path[n].y, (int) path[n].dir, hDC, hBrush); 555 556 path[n + 1].dir = -1; 557 switch (path[n].dir) { 558 case 0: path[n + 1].x = path[n].x; 559 path[n + 1].y = path[n].y - 1; 560 break; 561 case 1: path[n + 1].x = path[n].x + 1; 562 path[n + 1].y = path[n].y; 563 break; 564 case 2: path[n + 1].x = path[n].x; 565 path[n + 1].y = path[n].y + 1; 566 break; 567 case 3: path[n + 1].x = path[n].x - 1; 568 path[n + 1].y = path[n].y; 569 break; 570 } 571 } 572 573 static void start_timer(HWND hWnd, int iTimeout) 574 { 575 SetTimer(hWnd, 1, iTimeout, NULL); 576 } 577 578 static BOOL OnCreate(HWND hWnd, LPCREATESTRUCT lpCreateStruct) 579 { 580 srand((unsigned) time(NULL)); 581 582 ReadRegistry(); 583 584 if (size < 2) { 585 size = 7 + (rand() % 30); 586 } 587 grid_width = grid_height = size; 588 bw = (size > 6 ? 3 : (size - 1) / 2); 589 590 #if 0 591 /* FIXME Pattern brushes not yet implemented in ReactOS */ 592 { 593 static long grayPattern [] = { 594 0x55555555, 595 0xaaaaaaaa, 596 0x55555555, 597 0xaaaaaaaa, 598 0x55555555, 599 0xaaaaaaaa, 600 0x55555555, 601 0xaaaaaaaa 602 }; 603 static RGBQUAD argbq [] = { 604 { 0, 0, 255, 0 }, 605 { 255, 255, 255, 0 } 606 }; 607 BITMAPINFO *pbmi; 608 609 pbmi = malloc(sizeof(BITMAPINFOHEADER) + sizeof(argbq) + sizeof(grayPattern)); 610 pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER); 611 pbmi->bmiHeader.biWidth = 8; 612 pbmi->bmiHeader.biHeight = 8; 613 pbmi->bmiHeader.biPlanes = 1; 614 pbmi->bmiHeader.biBitCount = 1; 615 pbmi->bmiHeader.biCompression = BI_RGB; 616 (void) memcpy(pbmi->bmiColors, argbq, sizeof(argbq)); 617 (void) memcpy(pbmi->bmiColors + 2, grayPattern, sizeof(grayPattern)); 618 hBrushDead = CreateDIBPatternBrushPt(pbmi, DIB_RGB_COLORS); 619 // hBrushDead = CreateHatchBrush(HS_DIAGCROSS, RGB(255, 0, 0)); 620 free(pbmi); 621 } 622 #else 623 hBrushDead = CreateSolidBrush(RGB(255, 0, 0)); 624 #endif 625 hBrushLiving = CreateSolidBrush(RGB(0, 255, 0)); 626 hPenWall = CreatePen(PS_SOLID, 3, RGB(150, 150, 150)); 627 628 hDC = GetDC(hWnd); 629 630 start_timer(hWnd, 1); 631 632 return TRUE; 633 } 634 635 BOOL WINAPI AboutProc(HWND hWnd, UINT message, WPARAM wparam, LPARAM lparam) 636 { 637 switch(message){ 638 case WM_COMMAND: 639 switch(LOWORD(wparam)){ 640 case IDOK: 641 EndDialog(hWnd, LOWORD(wparam)); 642 return TRUE; 643 } 644 } 645 return FALSE; 646 } 647 648 static void ReadSettings(HWND hWnd) 649 { 650 size = SendDlgItemMessage(hWnd, IDC_SLIDER_SIZE, TBM_GETPOS, 0, 0); 651 SetDlgItemInt(hWnd, IDC_TEXT_SIZE, size, FALSE); 652 653 pre_solve_delay = SendDlgItemMessage(hWnd, IDC_SLIDER_PRESD, TBM_GETPOS, 0, 0); 654 SetDlgItemInt(hWnd, IDC_TEXT_PRESD, pre_solve_delay, FALSE); 655 656 post_solve_delay = SendDlgItemMessage(hWnd, IDC_SLIDER_POSTSD, TBM_GETPOS, 0, 0); 657 SetDlgItemInt(hWnd, IDC_TEXT_POSTSD, post_solve_delay, FALSE); 658 659 solve_delay = SendDlgItemMessage(hWnd, IDC_SLIDER_SD, TBM_GETPOS, 0, 0); 660 SetDlgItemInt(hWnd, IDC_TEXT_SD, solve_delay, FALSE); 661 } 662 663 LRESULT CALLBACK ScreenSaverProc( 664 HWND hWnd, // window handle 665 UINT message, // type of message 666 WPARAM wParam, // additional information 667 LPARAM lParam) // additional information 668 { 669 switch (message) 670 { 671 case WM_CREATE: 672 OnCreate(hWnd, (LPCREATESTRUCT) lParam); 673 break; 674 case WM_SIZE: 675 set_maze_sizes(LOWORD(lParam), HIWORD(lParam)); 676 break; 677 case WM_TIMER: 678 switch (state) 679 { 680 case 2: 681 begin_solve_maze(hWnd); 682 683 state = 3; 684 685 start_timer(hWnd, solve_delay); 686 break; 687 688 case 3: 689 if (!solve_maze(hWnd)) 690 { 691 start_timer(hWnd, solve_delay); 692 } 693 else 694 { 695 state = 1; 696 start_timer(hWnd, post_solve_delay); 697 } 698 break; 699 700 default: 701 initialize_maze(); 702 703 SendMessage(hWnd, WM_ERASEBKGND, (WPARAM) hDC, (LPARAM) 0); 704 draw_maze_border(hWnd); 705 706 create_maze(hWnd); 707 708 state = 2; 709 710 start_timer(hWnd, pre_solve_delay); 711 break; 712 } 713 break; 714 715 case WM_DESTROY: // message: window being destroyed 716 DeleteObject(hBrushLiving); 717 DeleteObject(hBrushDead); 718 ReleaseDC(hWnd, hDC); 719 break; 720 721 default: // Passes it on if unproccessed 722 return DefScreenSaverProc(hWnd, message, wParam, lParam); 723 } 724 return 0; 725 } 726 727 BOOL WINAPI ScreenSaverConfigureDialog(HWND hWnd, UINT message, WPARAM wparam, LPARAM lparam) 728 { 729 switch (message) 730 { 731 case WM_INITDIALOG: 732 ReadRegistry(); 733 //Set slider ranges 734 SendDlgItemMessage(hWnd, IDC_SLIDER_SIZE, TBM_SETRANGE, FALSE, MAKELPARAM(5, 64)); 735 SendDlgItemMessage(hWnd, IDC_SLIDER_PRESD, TBM_SETRANGE, FALSE, MAKELPARAM(1, 10000)); 736 SendDlgItemMessage(hWnd, IDC_SLIDER_POSTSD, TBM_SETRANGE, FALSE, MAKELPARAM(1, 10000)); 737 SendDlgItemMessage(hWnd, IDC_SLIDER_SD, TBM_SETRANGE, FALSE, MAKELPARAM(1, 10000)); 738 //Set current values to slider 739 SendDlgItemMessage(hWnd, IDC_SLIDER_SIZE, TBM_SETPOS, TRUE, size); 740 SendDlgItemMessage(hWnd, IDC_SLIDER_PRESD, TBM_SETPOS, TRUE, pre_solve_delay); 741 SendDlgItemMessage(hWnd, IDC_SLIDER_POSTSD, TBM_SETPOS, TRUE, post_solve_delay); 742 SendDlgItemMessage(hWnd, IDC_SLIDER_SD, TBM_SETPOS, TRUE, solve_delay); 743 //Set current values to texts 744 SetDlgItemInt(hWnd, IDC_TEXT_SIZE, size, FALSE); 745 SetDlgItemInt(hWnd, IDC_TEXT_PRESD, pre_solve_delay, FALSE); 746 SetDlgItemInt(hWnd, IDC_TEXT_POSTSD, post_solve_delay, FALSE); 747 SetDlgItemInt(hWnd, IDC_TEXT_SD, solve_delay, FALSE); 748 return TRUE; 749 case WM_COMMAND: 750 switch (LOWORD(wparam)) 751 { 752 case IDOK: 753 WriteRegistry(); 754 EndDialog(hWnd, TRUE); 755 return TRUE; 756 case IDCANCEL: 757 EndDialog(hWnd, TRUE); 758 break; 759 case IDABOUT: 760 DialogBox(hMainInstance, MAKEINTRESOURCE(IDD_DLG_ABOUT), hWnd, (DLGPROC)AboutProc); 761 break; 762 } 763 case WM_HSCROLL: 764 ReadSettings(hWnd); 765 return TRUE; 766 } 767 return FALSE; 768 } 769 770 BOOL WINAPI RegisterDialogClasses(HANDLE hmodule) 771 { 772 return TRUE; 773 } 774