1 /* 2 * Regedit find dialog 3 * 4 * This library is free software; you can redistribute it and/or 5 * modify it under the terms of the GNU Lesser General Public 6 * License as published by the Free Software Foundation; either 7 * version 2.1 of the License, or (at your option) any later version. 8 * 9 * This library is distributed in the hope that it will be useful, 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 * Lesser General Public License for more details. 13 * 14 * You should have received a copy of the GNU Lesser General Public 15 * License along with this library; if not, write to the Free Software 16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 17 */ 18 19 #include "regedit.h" 20 21 #define RSF_WHOLESTRING 0x00000001 22 #define RSF_LOOKATKEYS 0x00000002 23 #define RSF_LOOKATVALUES 0x00000004 24 #define RSF_LOOKATDATA 0x00000008 25 #define RSF_MATCHCASE 0x00010000 26 27 static WCHAR s_szFindWhat[256]; 28 static const WCHAR s_szFindFlags[] = L"FindFlags"; 29 static const WCHAR s_szFindFlagsR[] = L"FindFlagsReactOS"; 30 static HWND s_hwndAbortDialog; 31 static BOOL s_bAbort; 32 33 static DWORD s_dwFlags; 34 static WCHAR s_szName[MAX_PATH]; 35 static DWORD s_cbName; 36 static const WCHAR s_empty[] = L""; 37 static const WCHAR s_backslash[] = L"\\"; 38 39 extern VOID SetValueName(HWND hwndLV, LPCWSTR pszValueName); 40 41 BOOL DoEvents(VOID) 42 { 43 MSG msg; 44 if (PeekMessageW(&msg, NULL, 0, 0, PM_REMOVE)) 45 { 46 if (msg.message == WM_QUIT) 47 s_bAbort = TRUE; 48 if (!IsDialogMessageW(s_hwndAbortDialog, &msg)) 49 { 50 TranslateMessage(&msg); 51 DispatchMessageW(&msg); 52 } 53 } 54 return s_bAbort; 55 } 56 57 static LPWSTR lstrstri(LPCWSTR psz1, LPCWSTR psz2) 58 { 59 INT i, cch1, cch2; 60 61 cch1 = wcslen(psz1); 62 cch2 = wcslen(psz2); 63 for(i = 0; i <= cch1 - cch2; i++) 64 { 65 if (CompareStringW(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, 66 psz1 + i, cch2, psz2, cch2) == 2) 67 return (LPWSTR) (psz1 + i); 68 } 69 return NULL; 70 } 71 72 static BOOL CompareName(LPCWSTR pszName1, LPCWSTR pszName2) 73 { 74 if (s_dwFlags & RSF_WHOLESTRING) 75 { 76 if (s_dwFlags & RSF_MATCHCASE) 77 return wcscmp(pszName1, pszName2) == 0; 78 else 79 return _wcsicmp(pszName1, pszName2) == 0; 80 } 81 else 82 { 83 if (s_dwFlags & RSF_MATCHCASE) 84 return wcsstr(pszName1, pszName2) != NULL; 85 else 86 return lstrstri(pszName1, pszName2) != NULL; 87 } 88 } 89 90 static BOOL 91 CompareData( 92 DWORD dwType, 93 LPCWSTR psz1, 94 LPCWSTR psz2) 95 { 96 INT i, cch1 = wcslen(psz1), cch2 = wcslen(psz2); 97 if (dwType == REG_SZ || dwType == REG_EXPAND_SZ) 98 { 99 if (s_dwFlags & RSF_WHOLESTRING) 100 { 101 if (s_dwFlags & RSF_MATCHCASE) 102 return 2 == CompareStringW(LOCALE_SYSTEM_DEFAULT, 0, 103 psz1, cch1, psz2, cch2); 104 else 105 return 2 == CompareStringW(LOCALE_SYSTEM_DEFAULT, 106 NORM_IGNORECASE, psz1, cch1, psz2, cch2); 107 } 108 109 for(i = 0; i <= cch1 - cch2; i++) 110 { 111 if (s_dwFlags & RSF_MATCHCASE) 112 { 113 if (2 == CompareStringW(LOCALE_SYSTEM_DEFAULT, 0, 114 psz1 + i, cch2, psz2, cch2)) 115 return TRUE; 116 } 117 else 118 { 119 if (2 == CompareStringW(LOCALE_SYSTEM_DEFAULT, 120 NORM_IGNORECASE, psz1 + i, cch2, psz2, cch2)) 121 return TRUE; 122 } 123 } 124 } 125 return FALSE; 126 } 127 128 int compare(const void *x, const void *y) 129 { 130 const LPCWSTR *a = (const LPCWSTR *)x; 131 const LPCWSTR *b = (const LPCWSTR *)y; 132 return _wcsicmp(*a, *b); 133 } 134 135 BOOL RegFindRecurse( 136 HKEY hKey, 137 LPCWSTR pszSubKey, 138 LPCWSTR pszValueName, 139 LPWSTR *ppszFoundSubKey, 140 LPWSTR *ppszFoundValueName) 141 { 142 HKEY hSubKey; 143 LONG lResult; 144 WCHAR szSubKey[MAX_PATH]; 145 DWORD i, c, cb, type; 146 BOOL fPast = FALSE; 147 LPWSTR *ppszNames = NULL; 148 LPBYTE pb = NULL; 149 150 if (DoEvents()) 151 return FALSE; 152 153 if(wcslen(pszSubKey) >= _countof(szSubKey)) 154 return FALSE; 155 156 wcscpy(szSubKey, pszSubKey); 157 hSubKey = NULL; 158 159 lResult = RegOpenKeyExW(hKey, szSubKey, 0, KEY_ALL_ACCESS, &hSubKey); 160 if (lResult != ERROR_SUCCESS) 161 return FALSE; 162 163 if (pszValueName == NULL) 164 pszValueName = s_empty; 165 166 lResult = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, NULL, NULL, NULL, 167 &c, NULL, NULL, NULL, NULL); 168 if (lResult != ERROR_SUCCESS) 169 goto err; 170 ppszNames = (LPWSTR *) malloc(c * sizeof(LPWSTR)); 171 if (ppszNames == NULL) 172 goto err; 173 ZeroMemory(ppszNames, c * sizeof(LPWSTR)); 174 175 for(i = 0; i < c; i++) 176 { 177 if (DoEvents()) 178 goto err; 179 180 s_cbName = MAX_PATH * sizeof(WCHAR); 181 lResult = RegEnumValueW(hSubKey, i, s_szName, &s_cbName, NULL, NULL, 182 NULL, &cb); 183 if (lResult == ERROR_NO_MORE_ITEMS) 184 { 185 c = i; 186 break; 187 } 188 if (lResult != ERROR_SUCCESS) 189 goto err; 190 if (s_cbName >= MAX_PATH * sizeof(WCHAR)) 191 continue; 192 193 ppszNames[i] = _wcsdup(s_szName); 194 } 195 196 qsort(ppszNames, c, sizeof(LPWSTR), compare); 197 198 for(i = 0; i < c; i++) 199 { 200 if (DoEvents()) 201 goto err; 202 203 if (!fPast && _wcsicmp(ppszNames[i], pszValueName) == 0) 204 { 205 fPast = TRUE; 206 continue; 207 } 208 if (!fPast) 209 continue; 210 211 if ((s_dwFlags & RSF_LOOKATVALUES) && 212 CompareName(ppszNames[i], s_szFindWhat)) 213 { 214 *ppszFoundSubKey = _wcsdup(szSubKey); 215 if (ppszNames[i][0] == 0) 216 *ppszFoundValueName = NULL; 217 else 218 *ppszFoundValueName = _wcsdup(ppszNames[i]); 219 goto success; 220 } 221 222 lResult = RegQueryValueExW(hSubKey, ppszNames[i], NULL, &type, 223 NULL, &cb); 224 if (lResult != ERROR_SUCCESS) 225 goto err; 226 pb = malloc(cb); 227 if (pb == NULL) 228 goto err; 229 lResult = RegQueryValueExW(hSubKey, ppszNames[i], NULL, &type, 230 pb, &cb); 231 if (lResult != ERROR_SUCCESS) 232 goto err; 233 234 if ((s_dwFlags & RSF_LOOKATDATA) && 235 CompareData(type, (LPWSTR) pb, s_szFindWhat)) 236 { 237 *ppszFoundSubKey = _wcsdup(szSubKey); 238 if (ppszNames[i][0] == 0) 239 *ppszFoundValueName = NULL; 240 else 241 *ppszFoundValueName = _wcsdup(ppszNames[i]); 242 goto success; 243 } 244 free(pb); 245 pb = NULL; 246 } 247 248 if (ppszNames != NULL) 249 { 250 for(i = 0; i < c; i++) 251 free(ppszNames[i]); 252 free(ppszNames); 253 } 254 ppszNames = NULL; 255 256 lResult = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, &c, NULL, NULL, 257 NULL, NULL, NULL, NULL, NULL); 258 if (lResult != ERROR_SUCCESS) 259 goto err; 260 ppszNames = (LPWSTR *) malloc(c * sizeof(LPWSTR)); 261 if (ppszNames == NULL) 262 goto err; 263 ZeroMemory(ppszNames, c * sizeof(LPWSTR)); 264 265 for(i = 0; i < c; i++) 266 { 267 if (DoEvents()) 268 goto err; 269 270 s_cbName = MAX_PATH * sizeof(WCHAR); 271 lResult = RegEnumKeyExW(hSubKey, i, s_szName, &s_cbName, NULL, NULL, 272 NULL, NULL); 273 if (lResult == ERROR_NO_MORE_ITEMS) 274 { 275 c = i; 276 break; 277 } 278 if (lResult != ERROR_SUCCESS) 279 goto err; 280 if (s_cbName >= MAX_PATH * sizeof(WCHAR)) 281 continue; 282 283 ppszNames[i] = _wcsdup(s_szName); 284 } 285 286 qsort(ppszNames, c, sizeof(LPWSTR), compare); 287 288 for(i = 0; i < c; i++) 289 { 290 if (DoEvents()) 291 goto err; 292 293 if ((s_dwFlags & RSF_LOOKATKEYS) && 294 CompareName(ppszNames[i], s_szFindWhat)) 295 { 296 *ppszFoundSubKey = malloc( 297 (wcslen(szSubKey) + wcslen(ppszNames[i]) + 2) * 298 sizeof(WCHAR)); 299 if (*ppszFoundSubKey == NULL) 300 goto err; 301 if (szSubKey[0]) 302 { 303 wcscpy(*ppszFoundSubKey, szSubKey); 304 wcscat(*ppszFoundSubKey, s_backslash); 305 } 306 else 307 **ppszFoundSubKey = 0; 308 wcscat(*ppszFoundSubKey, ppszNames[i]); 309 *ppszFoundValueName = NULL; 310 goto success; 311 } 312 313 if (RegFindRecurse(hSubKey, ppszNames[i], NULL, ppszFoundSubKey, 314 ppszFoundValueName)) 315 { 316 LPWSTR psz = *ppszFoundSubKey; 317 *ppszFoundSubKey = malloc( 318 (wcslen(szSubKey) + wcslen(psz) + 2) * sizeof(WCHAR)); 319 if (*ppszFoundSubKey == NULL) 320 goto err; 321 if (szSubKey[0]) 322 { 323 wcscpy(*ppszFoundSubKey, szSubKey); 324 wcscat(*ppszFoundSubKey, s_backslash); 325 } 326 else 327 **ppszFoundSubKey = 0; 328 wcscat(*ppszFoundSubKey, psz); 329 free(psz); 330 goto success; 331 } 332 } 333 334 err: 335 if (ppszNames != NULL) 336 { 337 for(i = 0; i < c; i++) 338 free(ppszNames[i]); 339 free(ppszNames); 340 } 341 free(pb); 342 RegCloseKey(hSubKey); 343 return FALSE; 344 345 success: 346 if (ppszNames != NULL) 347 { 348 for(i = 0; i < c; i++) 349 free(ppszNames[i]); 350 free(ppszNames); 351 } 352 RegCloseKey(hSubKey); 353 return TRUE; 354 } 355 356 BOOL RegFindWalk( 357 HKEY * phKey, 358 LPCWSTR pszSubKey, 359 LPCWSTR pszValueName, 360 LPWSTR *ppszFoundSubKey, 361 LPWSTR *ppszFoundValueName) 362 { 363 LONG lResult; 364 DWORD i, c; 365 HKEY hBaseKey, hSubKey; 366 WCHAR szKeyName[MAX_PATH]; 367 WCHAR szSubKey[MAX_PATH]; 368 LPWSTR pch; 369 BOOL fPast; 370 LPWSTR *ppszNames = NULL; 371 372 hBaseKey = *phKey; 373 if (RegFindRecurse(hBaseKey, pszSubKey, pszValueName, ppszFoundSubKey, 374 ppszFoundValueName)) 375 return TRUE; 376 377 if (wcslen(pszSubKey) >= MAX_PATH) 378 return FALSE; 379 380 wcscpy(szSubKey, pszSubKey); 381 while(szSubKey[0] != 0) 382 { 383 if (DoEvents()) 384 return FALSE; 385 386 pch = wcsrchr(szSubKey, L'\\'); 387 if (pch == NULL) 388 { 389 wcscpy(szKeyName, szSubKey); 390 szSubKey[0] = 0; 391 hSubKey = hBaseKey; 392 } 393 else 394 { 395 lstrcpynW(szKeyName, pch + 1, MAX_PATH); 396 *pch = 0; 397 lResult = RegOpenKeyExW(hBaseKey, szSubKey, 0, KEY_ALL_ACCESS, 398 &hSubKey); 399 if (lResult != ERROR_SUCCESS) 400 return FALSE; 401 } 402 403 lResult = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, &c, NULL, NULL, 404 NULL, NULL, NULL, NULL, NULL); 405 if (lResult != ERROR_SUCCESS) 406 goto err; 407 408 ppszNames = (LPWSTR *) malloc(c * sizeof(LPWSTR)); 409 if (ppszNames == NULL) 410 goto err; 411 ZeroMemory(ppszNames, c * sizeof(LPWSTR)); 412 413 for(i = 0; i < c; i++) 414 { 415 if (DoEvents()) 416 goto err; 417 418 s_cbName = MAX_PATH * sizeof(WCHAR); 419 lResult = RegEnumKeyExW(hSubKey, i, s_szName, &s_cbName, 420 NULL, NULL, NULL, NULL); 421 if (lResult == ERROR_NO_MORE_ITEMS) 422 { 423 c = i; 424 break; 425 } 426 if (lResult != ERROR_SUCCESS) 427 break; 428 ppszNames[i] = _wcsdup(s_szName); 429 } 430 431 qsort(ppszNames, c, sizeof(LPWSTR), compare); 432 433 fPast = FALSE; 434 for(i = 0; i < c; i++) 435 { 436 if (DoEvents()) 437 goto err; 438 439 if (!fPast && _wcsicmp(ppszNames[i], szKeyName) == 0) 440 { 441 fPast = TRUE; 442 continue; 443 } 444 if (!fPast) 445 continue; 446 447 if ((s_dwFlags & RSF_LOOKATKEYS) && 448 CompareName(ppszNames[i], s_szFindWhat)) 449 { 450 *ppszFoundSubKey = malloc( 451 (wcslen(szSubKey) + wcslen(ppszNames[i]) + 2) * 452 sizeof(WCHAR)); 453 if (*ppszFoundSubKey == NULL) 454 goto err; 455 if (szSubKey[0]) 456 { 457 wcscpy(*ppszFoundSubKey, szSubKey); 458 wcscat(*ppszFoundSubKey, s_backslash); 459 } 460 else 461 **ppszFoundSubKey = 0; 462 wcscat(*ppszFoundSubKey, ppszNames[i]); 463 *ppszFoundValueName = NULL; 464 goto success; 465 } 466 467 if (RegFindRecurse(hSubKey, ppszNames[i], NULL, 468 ppszFoundSubKey, ppszFoundValueName)) 469 { 470 LPWSTR psz = *ppszFoundSubKey; 471 *ppszFoundSubKey = malloc( 472 (wcslen(szSubKey) + wcslen(psz) + 2) * 473 sizeof(WCHAR)); 474 if (*ppszFoundSubKey == NULL) 475 goto err; 476 if (szSubKey[0]) 477 { 478 wcscpy(*ppszFoundSubKey, szSubKey); 479 wcscat(*ppszFoundSubKey, s_backslash); 480 } 481 else 482 **ppszFoundSubKey = 0; 483 wcscat(*ppszFoundSubKey, psz); 484 free(psz); 485 goto success; 486 } 487 } 488 if (ppszNames != NULL) 489 { 490 for(i = 0; i < c; i++) 491 free(ppszNames[i]); 492 free(ppszNames); 493 } 494 ppszNames = NULL; 495 496 if (hBaseKey != hSubKey) 497 RegCloseKey(hSubKey); 498 } 499 500 if (*phKey == HKEY_CLASSES_ROOT) 501 { 502 *phKey = HKEY_CURRENT_USER; 503 if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey, 504 ppszFoundValueName)) 505 return TRUE; 506 } 507 508 if (*phKey == HKEY_CURRENT_USER) 509 { 510 *phKey = HKEY_LOCAL_MACHINE; 511 if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey, 512 ppszFoundValueName)) 513 goto success; 514 } 515 516 if (*phKey == HKEY_LOCAL_MACHINE) 517 { 518 *phKey = HKEY_USERS; 519 if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey, 520 ppszFoundValueName)) 521 goto success; 522 } 523 524 if (*phKey == HKEY_USERS) 525 { 526 *phKey = HKEY_CURRENT_CONFIG; 527 if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey, 528 ppszFoundValueName)) 529 goto success; 530 } 531 532 err: 533 if (ppszNames != NULL) 534 { 535 for(i = 0; i < c; i++) 536 free(ppszNames[i]); 537 free(ppszNames); 538 } 539 if (hBaseKey != hSubKey) 540 RegCloseKey(hSubKey); 541 return FALSE; 542 543 success: 544 if (ppszNames != NULL) 545 { 546 for(i = 0; i < c; i++) 547 free(ppszNames[i]); 548 free(ppszNames); 549 } 550 if (hBaseKey != hSubKey) 551 RegCloseKey(hSubKey); 552 return TRUE; 553 } 554 555 556 static DWORD GetFindFlags(void) 557 { 558 HKEY hKey; 559 DWORD dwType, dwValue, cbData; 560 DWORD dwFlags = RSF_LOOKATKEYS | RSF_LOOKATVALUES | RSF_LOOKATDATA; 561 562 if (RegOpenKeyW(HKEY_CURRENT_USER, g_szGeneralRegKey, &hKey) == ERROR_SUCCESS) 563 { 564 /* Retrieve flags from registry key */ 565 cbData = sizeof(dwValue); 566 if (RegQueryValueExW(hKey, s_szFindFlags, NULL, &dwType, (LPBYTE) &dwValue, &cbData) == ERROR_SUCCESS) 567 { 568 if (dwType == REG_DWORD) 569 dwFlags = (dwFlags & ~0x0000FFFF) | ((dwValue & 0x0000FFFF) << 0); 570 } 571 572 /* Retrieve ReactOS Regedit specific flags from registry key */ 573 cbData = sizeof(dwValue); 574 if (RegQueryValueExW(hKey, s_szFindFlagsR, NULL, &dwType, (LPBYTE) &dwValue, &cbData) == ERROR_SUCCESS) 575 { 576 if (dwType == REG_DWORD) 577 dwFlags = (dwFlags & ~0xFFFF0000) | ((dwValue & 0x0000FFFF) << 16); 578 } 579 580 RegCloseKey(hKey); 581 } 582 return dwFlags; 583 } 584 585 static void SetFindFlags(DWORD dwFlags) 586 { 587 HKEY hKey; 588 DWORD dwDisposition; 589 DWORD dwData; 590 591 if (RegCreateKeyExW(HKEY_CURRENT_USER, g_szGeneralRegKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, &dwDisposition) == ERROR_SUCCESS) 592 { 593 dwData = (dwFlags >> 0) & 0x0000FFFF; 594 RegSetValueExW(hKey, s_szFindFlags, 0, REG_DWORD, (const BYTE *) &dwData, sizeof(dwData)); 595 596 dwData = (dwFlags >> 16) & 0x0000FFFF; 597 RegSetValueExW(hKey, s_szFindFlagsR, 0, REG_DWORD, (const BYTE *) &dwData, sizeof(dwData)); 598 599 RegCloseKey(hKey); 600 } 601 } 602 603 static INT_PTR CALLBACK AbortFindDialogProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) 604 { 605 UNREFERENCED_PARAMETER(lParam); 606 UNREFERENCED_PARAMETER(hDlg); 607 608 switch(uMsg) 609 { 610 case WM_CLOSE: 611 s_bAbort = TRUE; 612 break; 613 614 case WM_COMMAND: 615 switch(HIWORD(wParam)) 616 { 617 case BN_CLICKED: 618 switch(LOWORD(wParam)) 619 { 620 case IDCANCEL: 621 s_bAbort = TRUE; 622 break; 623 } 624 break; 625 } 626 break; 627 } 628 return 0; 629 } 630 631 BOOL FindNext(HWND hWnd) 632 { 633 HKEY hKeyRoot; 634 LPCWSTR pszKeyPath; 635 BOOL fSuccess; 636 WCHAR szFullKey[512]; 637 LPCWSTR pszValueName; 638 LPWSTR pszFoundSubKey, pszFoundValueName; 639 640 if (wcslen(s_szFindWhat) == 0) 641 { 642 FindDialog(hWnd); 643 return TRUE; 644 } 645 646 s_dwFlags = GetFindFlags(); 647 648 pszKeyPath = GetItemPath(g_pChildWnd->hTreeWnd, 0, &hKeyRoot); 649 if (pszKeyPath == NULL) 650 { 651 hKeyRoot = HKEY_CLASSES_ROOT; 652 pszKeyPath = s_empty; 653 } 654 655 /* Create abort find dialog */ 656 s_hwndAbortDialog = CreateDialogW(GetModuleHandle(NULL), 657 MAKEINTRESOURCEW(IDD_FINDING), hWnd, AbortFindDialogProc); 658 if (s_hwndAbortDialog) 659 { 660 ShowWindow(s_hwndAbortDialog, SW_SHOW); 661 UpdateWindow(s_hwndAbortDialog); 662 } 663 s_bAbort = FALSE; 664 665 pszValueName = GetValueName(g_pChildWnd->hListWnd, -1); 666 667 EnableWindow(hFrameWnd, FALSE); 668 EnableWindow(g_pChildWnd->hTreeWnd, FALSE); 669 EnableWindow(g_pChildWnd->hListWnd, FALSE); 670 EnableWindow(g_pChildWnd->hAddressBarWnd, FALSE); 671 672 fSuccess = RegFindWalk(&hKeyRoot, pszKeyPath, pszValueName, 673 &pszFoundSubKey, &pszFoundValueName); 674 675 EnableWindow(hFrameWnd, TRUE); 676 EnableWindow(g_pChildWnd->hTreeWnd, TRUE); 677 EnableWindow(g_pChildWnd->hListWnd, TRUE); 678 EnableWindow(g_pChildWnd->hAddressBarWnd, TRUE); 679 680 if (s_hwndAbortDialog) 681 { 682 DestroyWindow(s_hwndAbortDialog); 683 s_hwndAbortDialog = NULL; 684 } 685 686 if (fSuccess) 687 { 688 GetKeyName(szFullKey, COUNT_OF(szFullKey), hKeyRoot, pszFoundSubKey); 689 SelectNode(g_pChildWnd->hTreeWnd, szFullKey); 690 SetValueName(g_pChildWnd->hListWnd, pszFoundValueName); 691 free(pszFoundSubKey); 692 free(pszFoundValueName); 693 SetFocus(g_pChildWnd->hListWnd); 694 } 695 return fSuccess || s_bAbort; 696 } 697 698 static INT_PTR CALLBACK FindDialogProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) 699 { 700 INT_PTR iResult = 0; 701 HWND hControl; 702 LONG lStyle; 703 DWORD dwFlags; 704 static WCHAR s_szSavedFindValue[256]; 705 706 switch(uMsg) 707 { 708 case WM_INITDIALOG: 709 dwFlags = GetFindFlags(); 710 711 hControl = GetDlgItem(hDlg, IDC_LOOKAT_KEYS); 712 if (hControl) 713 SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_LOOKATKEYS) ? TRUE : FALSE, 0); 714 715 hControl = GetDlgItem(hDlg, IDC_LOOKAT_VALUES); 716 if (hControl) 717 SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_LOOKATVALUES) ? TRUE : FALSE, 0); 718 719 hControl = GetDlgItem(hDlg, IDC_LOOKAT_DATA); 720 if (hControl) 721 SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_LOOKATDATA) ? TRUE : FALSE, 0); 722 723 /* Match whole string */ 724 hControl = GetDlgItem(hDlg, IDC_MATCHSTRING); 725 if (hControl) 726 SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_WHOLESTRING) ? TRUE : FALSE, 0); 727 728 /* Case sensitivity */ 729 hControl = GetDlgItem(hDlg, IDC_MATCHCASE); 730 if (hControl) 731 SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_MATCHCASE) ? TRUE : FALSE, 0); 732 733 hControl = GetDlgItem(hDlg, IDC_FINDWHAT); 734 if (hControl) 735 { 736 SetWindowTextW(hControl, s_szSavedFindValue); 737 SetFocus(hControl); 738 SendMessageW(hControl, EM_SETSEL, 0, -1); 739 } 740 break; 741 742 case WM_CLOSE: 743 EndDialog(hDlg, 0); 744 break; 745 746 case WM_COMMAND: 747 switch(HIWORD(wParam)) 748 { 749 case BN_CLICKED: 750 switch(LOWORD(wParam)) 751 { 752 case IDOK: 753 dwFlags = 0; 754 755 hControl = GetDlgItem(hDlg, IDC_LOOKAT_KEYS); 756 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED)) 757 dwFlags |= RSF_LOOKATKEYS; 758 759 hControl = GetDlgItem(hDlg, IDC_LOOKAT_VALUES); 760 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED)) 761 dwFlags |= RSF_LOOKATVALUES; 762 763 hControl = GetDlgItem(hDlg, IDC_LOOKAT_DATA); 764 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED)) 765 dwFlags |= RSF_LOOKATDATA; 766 767 hControl = GetDlgItem(hDlg, IDC_MATCHSTRING); 768 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED)) 769 dwFlags |= RSF_WHOLESTRING; 770 771 hControl = GetDlgItem(hDlg, IDC_MATCHCASE); 772 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED)) 773 dwFlags |= RSF_MATCHCASE; 774 775 SetFindFlags(dwFlags); 776 777 hControl = GetDlgItem(hDlg, IDC_FINDWHAT); 778 if (hControl) 779 GetWindowTextW(hControl, s_szFindWhat, COUNT_OF(s_szFindWhat)); 780 EndDialog(hDlg, 1); 781 break; 782 783 case IDCANCEL: 784 EndDialog(hDlg, 0); 785 break; 786 } 787 break; 788 789 case EN_CHANGE: 790 switch(LOWORD(wParam)) 791 { 792 case IDC_FINDWHAT: 793 GetWindowTextW((HWND) lParam, s_szSavedFindValue, COUNT_OF(s_szSavedFindValue)); 794 hControl = GetDlgItem(hDlg, IDOK); 795 if (hControl) 796 { 797 lStyle = GetWindowLongPtr(hControl, GWL_STYLE); 798 if (s_szSavedFindValue[0]) 799 lStyle &= ~WS_DISABLED; 800 else 801 lStyle |= WS_DISABLED; 802 SetWindowLongPtr(hControl, GWL_STYLE, lStyle); 803 RedrawWindow(hControl, NULL, NULL, RDW_INVALIDATE); 804 } 805 break; 806 } 807 } 808 break; 809 } 810 return iResult; 811 } 812 813 void FindNextMessageBox(HWND hWnd) 814 { 815 if (!FindNext(hWnd)) 816 { 817 WCHAR msg[128], caption[128]; 818 819 LoadStringW(hInst, IDS_FINISHEDFIND, msg, COUNT_OF(msg)); 820 LoadStringW(hInst, IDS_APP_TITLE, caption, COUNT_OF(caption)); 821 MessageBoxW(hWnd, msg, caption, MB_ICONINFORMATION); 822 } 823 } 824 825 void FindDialog(HWND hWnd) 826 { 827 if (DialogBoxParamW(GetModuleHandle(NULL), MAKEINTRESOURCEW(IDD_FIND), 828 hWnd, FindDialogProc, 0) != 0) 829 { 830 FindNextMessageBox(hWnd); 831 } 832 } 833