xref: /reactos/base/applications/regedit/find.c (revision 4572aad1)
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_cchName;
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     lResult = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, NULL, NULL, NULL,
164                               &c, NULL, NULL, NULL, NULL);
165     if (lResult != ERROR_SUCCESS)
166         goto err;
167     ppszNames = (LPWSTR *) malloc(c * sizeof(LPWSTR));
168     if (ppszNames == NULL)
169         goto err;
170     ZeroMemory(ppszNames, c * sizeof(LPWSTR));
171 
172     for(i = 0; i < c; i++)
173     {
174         if (DoEvents())
175             goto err;
176 
177         s_cchName = _countof(s_szName);
178         lResult = RegEnumValueW(hSubKey, i, s_szName, &s_cchName, NULL, NULL,
179                                NULL, &cb);
180         if (lResult == ERROR_NO_MORE_ITEMS)
181         {
182             c = i;
183             break;
184         }
185         if (lResult != ERROR_SUCCESS)
186             goto err;
187         if (s_cchName >= _countof(s_szName))
188             continue;
189 
190         ppszNames[i] = _wcsdup(s_szName);
191     }
192 
193     qsort(ppszNames, c, sizeof(LPWSTR), compare);
194 
195     if (pszValueName == NULL)
196         pszValueName = ppszNames[0];
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_cchName = _countof(s_szName);
271         lResult = RegEnumKeyExW(hSubKey, i, s_szName, &s_cchName, 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_cchName >= _countof(s_szName))
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 
374     if (wcslen(pszSubKey) >= _countof(szSubKey))
375         return FALSE;
376 
377     if (RegFindRecurse(hBaseKey, pszSubKey, pszValueName, ppszFoundSubKey,
378                        ppszFoundValueName))
379         return TRUE;
380 
381     wcscpy(szSubKey, pszSubKey);
382     while(szSubKey[0] != 0)
383     {
384         if (DoEvents())
385             return FALSE;
386 
387         pch = wcsrchr(szSubKey, L'\\');
388         if (pch == NULL)
389         {
390             wcscpy(szKeyName, szSubKey);
391             szSubKey[0] = 0;
392             hSubKey = hBaseKey;
393         }
394         else
395         {
396             lstrcpynW(szKeyName, pch + 1, MAX_PATH);
397             *pch = 0;
398             lResult = RegOpenKeyExW(hBaseKey, szSubKey, 0, KEY_ALL_ACCESS,
399                                    &hSubKey);
400             if (lResult != ERROR_SUCCESS)
401                 return FALSE;
402         }
403 
404         lResult = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, &c, NULL, NULL,
405                                   NULL, NULL, NULL, NULL, NULL);
406         if (lResult != ERROR_SUCCESS)
407             goto err;
408 
409         ppszNames = (LPWSTR *) malloc(c * sizeof(LPWSTR));
410         if (ppszNames == NULL)
411             goto err;
412         ZeroMemory(ppszNames, c * sizeof(LPWSTR));
413 
414         for(i = 0; i < c; i++)
415         {
416             if (DoEvents())
417                 goto err;
418 
419             s_cchName = _countof(s_szName);
420             lResult = RegEnumKeyExW(hSubKey, i, s_szName, &s_cchName,
421                                     NULL, NULL, NULL, NULL);
422             if (lResult == ERROR_NO_MORE_ITEMS)
423             {
424                 c = i;
425                 break;
426             }
427             if (lResult != ERROR_SUCCESS)
428                 break;
429             ppszNames[i] = _wcsdup(s_szName);
430         }
431 
432         qsort(ppszNames, c, sizeof(LPWSTR), compare);
433 
434         fPast = FALSE;
435         for(i = 0; i < c; i++)
436         {
437             if (DoEvents())
438                 goto err;
439 
440             if (!fPast && _wcsicmp(ppszNames[i], szKeyName) == 0)
441             {
442                 fPast = TRUE;
443                 continue;
444             }
445             if (!fPast)
446                 continue;
447 
448             if ((s_dwFlags & RSF_LOOKATKEYS) &&
449                     CompareName(ppszNames[i], s_szFindWhat))
450             {
451                 *ppszFoundSubKey = malloc(
452                                        (wcslen(szSubKey) + wcslen(ppszNames[i]) + 2) *
453                                        sizeof(WCHAR));
454                 if (*ppszFoundSubKey == NULL)
455                     goto err;
456                 if (szSubKey[0])
457                 {
458                     wcscpy(*ppszFoundSubKey, szSubKey);
459                     wcscat(*ppszFoundSubKey, s_backslash);
460                 }
461                 else
462                     **ppszFoundSubKey = 0;
463                 wcscat(*ppszFoundSubKey, ppszNames[i]);
464                 *ppszFoundValueName = NULL;
465                 goto success;
466             }
467 
468             if (RegFindRecurse(hSubKey, ppszNames[i], NULL,
469                                ppszFoundSubKey, ppszFoundValueName))
470             {
471                 LPWSTR psz = *ppszFoundSubKey;
472                 *ppszFoundSubKey = malloc(
473                                        (wcslen(szSubKey) + wcslen(psz) + 2) *
474                                        sizeof(WCHAR));
475                 if (*ppszFoundSubKey == NULL)
476                     goto err;
477                 if (szSubKey[0])
478                 {
479                     wcscpy(*ppszFoundSubKey, szSubKey);
480                     wcscat(*ppszFoundSubKey, s_backslash);
481                 }
482                 else
483                     **ppszFoundSubKey = 0;
484                 wcscat(*ppszFoundSubKey, psz);
485                 free(psz);
486                 goto success;
487             }
488         }
489         if (ppszNames != NULL)
490         {
491             for(i = 0; i < c; i++)
492                 free(ppszNames[i]);
493             free(ppszNames);
494         }
495         ppszNames = NULL;
496 
497         if (hBaseKey != hSubKey)
498             RegCloseKey(hSubKey);
499     }
500 
501     if (*phKey == HKEY_CLASSES_ROOT)
502     {
503         *phKey = HKEY_CURRENT_USER;
504         if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey,
505                            ppszFoundValueName))
506             return TRUE;
507     }
508 
509     if (*phKey == HKEY_CURRENT_USER)
510     {
511         *phKey = HKEY_LOCAL_MACHINE;
512         if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey,
513                            ppszFoundValueName))
514             goto success;
515     }
516 
517     if (*phKey == HKEY_LOCAL_MACHINE)
518     {
519         *phKey = HKEY_USERS;
520         if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey,
521                            ppszFoundValueName))
522             goto success;
523     }
524 
525     if (*phKey == HKEY_USERS)
526     {
527         *phKey = HKEY_CURRENT_CONFIG;
528         if (RegFindRecurse(*phKey, s_empty, NULL, ppszFoundSubKey,
529                            ppszFoundValueName))
530             goto success;
531     }
532 
533 err:
534     if (ppszNames != NULL)
535     {
536         for(i = 0; i < c; i++)
537             free(ppszNames[i]);
538         free(ppszNames);
539     }
540     if (hBaseKey != hSubKey)
541         RegCloseKey(hSubKey);
542     return FALSE;
543 
544 success:
545     if (ppszNames != NULL)
546     {
547         for(i = 0; i < c; i++)
548             free(ppszNames[i]);
549         free(ppszNames);
550     }
551     if (hBaseKey != hSubKey)
552         RegCloseKey(hSubKey);
553     return TRUE;
554 }
555 
556 
557 static DWORD GetFindFlags(void)
558 {
559     HKEY hKey;
560     DWORD dwType, dwValue, cbData;
561     DWORD dwFlags = RSF_LOOKATKEYS | RSF_LOOKATVALUES | RSF_LOOKATDATA;
562 
563     if (RegOpenKeyW(HKEY_CURRENT_USER, g_szGeneralRegKey, &hKey) == ERROR_SUCCESS)
564     {
565         /* Retrieve flags from registry key */
566         cbData = sizeof(dwValue);
567         if (RegQueryValueExW(hKey, s_szFindFlags, NULL, &dwType, (LPBYTE) &dwValue, &cbData) == ERROR_SUCCESS)
568         {
569             if (dwType == REG_DWORD)
570                 dwFlags = (dwFlags & ~0x0000FFFF) | ((dwValue & 0x0000FFFF) << 0);
571         }
572 
573         /* Retrieve ReactOS Regedit specific flags from registry key */
574         cbData = sizeof(dwValue);
575         if (RegQueryValueExW(hKey, s_szFindFlagsR, NULL, &dwType, (LPBYTE) &dwValue, &cbData) == ERROR_SUCCESS)
576         {
577             if (dwType == REG_DWORD)
578                 dwFlags = (dwFlags & ~0xFFFF0000) | ((dwValue & 0x0000FFFF) << 16);
579         }
580 
581         RegCloseKey(hKey);
582     }
583     return dwFlags;
584 }
585 
586 static void SetFindFlags(DWORD dwFlags)
587 {
588     HKEY hKey;
589     DWORD dwDisposition;
590     DWORD dwData;
591 
592     if (RegCreateKeyExW(HKEY_CURRENT_USER, g_szGeneralRegKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, &dwDisposition) == ERROR_SUCCESS)
593     {
594         dwData = (dwFlags >> 0) & 0x0000FFFF;
595         RegSetValueExW(hKey, s_szFindFlags, 0, REG_DWORD, (const BYTE *) &dwData, sizeof(dwData));
596 
597         dwData = (dwFlags >> 16) & 0x0000FFFF;
598         RegSetValueExW(hKey, s_szFindFlagsR, 0, REG_DWORD, (const BYTE *) &dwData, sizeof(dwData));
599 
600         RegCloseKey(hKey);
601     }
602 }
603 
604 static INT_PTR CALLBACK AbortFindDialogProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
605 {
606     UNREFERENCED_PARAMETER(lParam);
607     UNREFERENCED_PARAMETER(hDlg);
608 
609     switch(uMsg)
610     {
611     case WM_CLOSE:
612         s_bAbort = TRUE;
613         break;
614 
615     case WM_COMMAND:
616         switch(HIWORD(wParam))
617         {
618         case BN_CLICKED:
619             switch(LOWORD(wParam))
620             {
621             case IDCANCEL:
622                 s_bAbort = TRUE;
623                 break;
624             }
625             break;
626         }
627         break;
628     }
629     return 0;
630 }
631 
632 BOOL FindNext(HWND hWnd)
633 {
634     HKEY hKeyRoot;
635     LPCWSTR pszKeyPath;
636     BOOL fSuccess;
637     WCHAR szFullKey[512];
638     LPCWSTR pszValueName;
639     LPWSTR pszFoundSubKey, pszFoundValueName;
640 
641     if (wcslen(s_szFindWhat) == 0)
642     {
643         FindDialog(hWnd);
644         return TRUE;
645     }
646 
647     s_dwFlags = GetFindFlags();
648 
649     pszKeyPath = GetItemPath(g_pChildWnd->hTreeWnd, 0, &hKeyRoot);
650     if (pszKeyPath == NULL)
651     {
652         hKeyRoot = HKEY_CLASSES_ROOT;
653         pszKeyPath = s_empty;
654     }
655 
656     /* Create abort find dialog */
657     s_hwndAbortDialog = CreateDialogW(GetModuleHandle(NULL),
658                                      MAKEINTRESOURCEW(IDD_FINDING), hWnd, AbortFindDialogProc);
659     if (s_hwndAbortDialog)
660     {
661         ShowWindow(s_hwndAbortDialog, SW_SHOW);
662         UpdateWindow(s_hwndAbortDialog);
663     }
664     s_bAbort = FALSE;
665 
666     pszValueName = GetValueName(g_pChildWnd->hListWnd, -1);
667 
668     EnableWindow(hFrameWnd, FALSE);
669     EnableWindow(g_pChildWnd->hTreeWnd, FALSE);
670     EnableWindow(g_pChildWnd->hListWnd, FALSE);
671     EnableWindow(g_pChildWnd->hAddressBarWnd, FALSE);
672 
673     fSuccess = RegFindWalk(&hKeyRoot, pszKeyPath, pszValueName,
674                            &pszFoundSubKey, &pszFoundValueName);
675 
676     EnableWindow(hFrameWnd, TRUE);
677     EnableWindow(g_pChildWnd->hTreeWnd, TRUE);
678     EnableWindow(g_pChildWnd->hListWnd, TRUE);
679     EnableWindow(g_pChildWnd->hAddressBarWnd, TRUE);
680 
681     if (s_hwndAbortDialog)
682     {
683         DestroyWindow(s_hwndAbortDialog);
684         s_hwndAbortDialog = NULL;
685     }
686 
687     if (fSuccess)
688     {
689         GetKeyName(szFullKey, COUNT_OF(szFullKey), hKeyRoot, pszFoundSubKey);
690         SelectNode(g_pChildWnd->hTreeWnd, szFullKey);
691         free(pszFoundSubKey);
692 
693         if (pszFoundValueName != NULL)
694         {
695             SetValueName(g_pChildWnd->hListWnd, pszFoundValueName);
696             free(pszFoundValueName);
697             SetFocus(g_pChildWnd->hListWnd);
698         }
699         else
700         {
701             SetFocus(g_pChildWnd->hTreeWnd);
702         }
703     }
704     return fSuccess || s_bAbort;
705 }
706 
707 static INT_PTR CALLBACK FindDialogProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
708 {
709     INT_PTR iResult = 0;
710     HWND hControl;
711     LONG lStyle;
712     DWORD dwFlags;
713     static WCHAR s_szSavedFindValue[256];
714 
715     switch(uMsg)
716     {
717     case WM_INITDIALOG:
718         dwFlags = GetFindFlags();
719 
720         hControl = GetDlgItem(hDlg, IDC_LOOKAT_KEYS);
721         if (hControl)
722             SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_LOOKATKEYS) ? TRUE : FALSE, 0);
723 
724         hControl = GetDlgItem(hDlg, IDC_LOOKAT_VALUES);
725         if (hControl)
726             SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_LOOKATVALUES) ? TRUE : FALSE, 0);
727 
728         hControl = GetDlgItem(hDlg, IDC_LOOKAT_DATA);
729         if (hControl)
730             SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_LOOKATDATA) ? TRUE : FALSE, 0);
731 
732         /* Match whole string */
733         hControl = GetDlgItem(hDlg, IDC_MATCHSTRING);
734         if (hControl)
735             SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_WHOLESTRING) ? TRUE : FALSE, 0);
736 
737         /* Case sensitivity */
738         hControl = GetDlgItem(hDlg, IDC_MATCHCASE);
739         if (hControl)
740             SendMessageW(hControl, BM_SETCHECK, (dwFlags & RSF_MATCHCASE) ? TRUE : FALSE, 0);
741 
742         hControl = GetDlgItem(hDlg, IDC_FINDWHAT);
743         if (hControl)
744         {
745             SetWindowTextW(hControl, s_szSavedFindValue);
746             SetFocus(hControl);
747             SendMessageW(hControl, EM_SETSEL, 0, -1);
748         }
749         break;
750 
751     case WM_CLOSE:
752         EndDialog(hDlg, 0);
753         break;
754 
755     case WM_COMMAND:
756         switch(HIWORD(wParam))
757         {
758         case BN_CLICKED:
759             switch(LOWORD(wParam))
760             {
761             case IDOK:
762                 dwFlags = 0;
763 
764                 hControl = GetDlgItem(hDlg, IDC_LOOKAT_KEYS);
765                 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED))
766                     dwFlags |= RSF_LOOKATKEYS;
767 
768                 hControl = GetDlgItem(hDlg, IDC_LOOKAT_VALUES);
769                 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED))
770                     dwFlags |= RSF_LOOKATVALUES;
771 
772                 hControl = GetDlgItem(hDlg, IDC_LOOKAT_DATA);
773                 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED))
774                     dwFlags |= RSF_LOOKATDATA;
775 
776                 hControl = GetDlgItem(hDlg, IDC_MATCHSTRING);
777                 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED))
778                     dwFlags |= RSF_WHOLESTRING;
779 
780                 hControl = GetDlgItem(hDlg, IDC_MATCHCASE);
781                 if (hControl && (SendMessageW(hControl, BM_GETCHECK, 0, 0) == BST_CHECKED))
782                     dwFlags |= RSF_MATCHCASE;
783 
784                 SetFindFlags(dwFlags);
785 
786                 hControl = GetDlgItem(hDlg, IDC_FINDWHAT);
787                 if (hControl)
788                     GetWindowTextW(hControl, s_szFindWhat, COUNT_OF(s_szFindWhat));
789                 EndDialog(hDlg, 1);
790                 break;
791 
792             case IDCANCEL:
793                 EndDialog(hDlg, 0);
794                 break;
795             }
796             break;
797 
798         case EN_CHANGE:
799             switch(LOWORD(wParam))
800             {
801             case IDC_FINDWHAT:
802                 GetWindowTextW((HWND) lParam, s_szSavedFindValue, COUNT_OF(s_szSavedFindValue));
803                 hControl = GetDlgItem(hDlg, IDOK);
804                 if (hControl)
805                 {
806                     lStyle = GetWindowLongPtr(hControl, GWL_STYLE);
807                     if (s_szSavedFindValue[0])
808                         lStyle &= ~WS_DISABLED;
809                     else
810                         lStyle |= WS_DISABLED;
811                     SetWindowLongPtr(hControl, GWL_STYLE, lStyle);
812                     RedrawWindow(hControl, NULL, NULL, RDW_INVALIDATE);
813                 }
814                 break;
815             }
816         }
817         break;
818     }
819     return iResult;
820 }
821 
822 void FindNextMessageBox(HWND hWnd)
823 {
824     if (!FindNext(hWnd))
825     {
826         WCHAR msg[128], caption[128];
827 
828         LoadStringW(hInst, IDS_FINISHEDFIND, msg, COUNT_OF(msg));
829         LoadStringW(hInst, IDS_APP_TITLE, caption, COUNT_OF(caption));
830         MessageBoxW(hWnd, msg, caption, MB_ICONINFORMATION);
831     }
832 }
833 
834 void FindDialog(HWND hWnd)
835 {
836     if (DialogBoxParamW(GetModuleHandle(NULL), MAKEINTRESOURCEW(IDD_FIND),
837                        hWnd, FindDialogProc, 0) != 0)
838     {
839         FindNextMessageBox(hWnd);
840     }
841 }
842