1 /* 2 * Regedit listviews 3 * 4 * Copyright (C) 2002 Robert Dickenson <robd@reactos.org> 5 * 6 * This library is free software; you can redistribute it and/or 7 * modify it under the terms of the GNU Lesser General Public 8 * License as published by the Free Software Foundation; either 9 * version 2.1 of the License, or (at your option) any later version. 10 * 11 * This library is distributed in the hope that it will be useful, 12 * but WITHOUT ANY WARRANTY; without even the implied warranty of 13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 14 * Lesser General Public License for more details. 15 * 16 * You should have received a copy of the GNU Lesser General Public 17 * License along with this library; if not, write to the Free Software 18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 19 */ 20 21 #include "regedit.h" 22 23 #define CX_ICON 16 24 #define CY_ICON 16 25 #define LISTVIEW_NUM_ICONS 2 26 27 int Image_String = 0; 28 int Image_Bin = 0; 29 INT iListViewSelect = -1; 30 31 typedef struct tagLINE_INFO 32 { 33 DWORD dwValType; 34 LPWSTR name; 35 void* val; 36 size_t val_len; 37 } LINE_INFO, *PLINE_INFO; 38 39 typedef struct tagSORT_INFO 40 { 41 INT iSortingColumn; 42 BOOL bSortAscending; 43 } SORT_INFO, *PSORT_INFO; 44 45 /******************************************************************************* 46 * Global and Local Variables: 47 */ 48 49 static INT g_iSortedColumn = 0; 50 51 #define MAX_LIST_COLUMNS (IDS_LIST_COLUMN_LAST - IDS_LIST_COLUMN_FIRST + 1) 52 static const int default_column_widths[MAX_LIST_COLUMNS] = { 35, 25, 40 }; /* in percents */ 53 static const int column_alignment[MAX_LIST_COLUMNS] = { LVCFMT_LEFT, LVCFMT_LEFT, LVCFMT_LEFT }; 54 55 LPCWSTR GetValueName(HWND hwndLV, int iStartAt) 56 { 57 int item; 58 LVITEMW LVItem; 59 PLINE_INFO lineinfo; 60 61 /* 62 if a new item is inserted, then no allocation, 63 otherwise the heap block will be lost! 64 */ 65 item = ListView_GetNextItem(hwndLV, iStartAt, LVNI_SELECTED); 66 if (item == -1) return NULL; 67 68 /* 69 Should be always TRUE anyways 70 */ 71 LVItem.iItem = item; 72 LVItem.iSubItem = 0; 73 LVItem.mask = LVIF_PARAM; 74 if (ListView_GetItem(hwndLV, &LVItem) == FALSE) 75 return NULL; 76 77 lineinfo = (PLINE_INFO)LVItem.lParam; 78 if (lineinfo == NULL) 79 return NULL; 80 81 return lineinfo->name; 82 } 83 84 VOID SetValueName(HWND hwndLV, LPCWSTR pszValueName) 85 { 86 INT i, c; 87 LVFINDINFOW fi; 88 89 c = ListView_GetItemCount(hwndLV); 90 for(i = 0; i < c; i++) 91 { 92 ListView_SetItemState(hwndLV, i, 0, LVIS_FOCUSED | LVIS_SELECTED); 93 } 94 if (pszValueName == NULL) 95 i = 0; 96 else 97 { 98 fi.flags = LVFI_STRING; 99 fi.psz = pszValueName; 100 i = ListView_FindItem(hwndLV, -1, &fi); 101 } 102 ListView_SetItemState(hwndLV, i, LVIS_FOCUSED | LVIS_SELECTED, 103 LVIS_FOCUSED | LVIS_SELECTED); 104 iListViewSelect = i; 105 } 106 107 BOOL IsDefaultValue(HWND hwndLV, int i) 108 { 109 PLINE_INFO lineinfo; 110 LVITEMW Item; 111 112 Item.mask = LVIF_PARAM; 113 Item.iItem = i; 114 if(ListView_GetItem(hwndLV, &Item)) 115 { 116 lineinfo = (PLINE_INFO)Item.lParam; 117 return lineinfo && (!lineinfo->name || !wcscmp(lineinfo->name, L"")); 118 } 119 return FALSE; 120 } 121 122 /******************************************************************************* 123 * Local module support methods 124 */ 125 static void AddEntryToList(HWND hwndLV, LPWSTR Name, DWORD dwValType, void* ValBuf, DWORD dwCount, int Position, BOOL ValExists) 126 { 127 PLINE_INFO linfo; 128 LVITEMW item; 129 int index; 130 131 linfo = (PLINE_INFO)HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(LINE_INFO) + dwCount); 132 linfo->dwValType = dwValType; 133 linfo->val_len = dwCount; 134 if (dwCount > 0) 135 { 136 memcpy(&linfo[1], ValBuf, dwCount); 137 linfo->val = &linfo[1]; 138 } 139 linfo->name = _wcsdup(Name); 140 141 item.mask = LVIF_TEXT | LVIF_PARAM | LVIF_IMAGE; 142 item.iItem = (Position == -1 ? 0: Position); 143 item.iSubItem = 0; 144 item.state = 0; 145 item.stateMask = 0; 146 item.pszText = Name; 147 item.cchTextMax = (int)wcslen(item.pszText); 148 if (item.cchTextMax == 0) 149 item.pszText = LPSTR_TEXTCALLBACK; 150 item.iImage = 0; 151 item.lParam = (LPARAM)linfo; 152 switch(dwValType) 153 { 154 case REG_SZ: 155 case REG_EXPAND_SZ: 156 case REG_MULTI_SZ: 157 item.iImage = Image_String; 158 break; 159 default: 160 item.iImage = Image_Bin; 161 break; 162 } 163 164 /* item.lParam = (LPARAM)ValBuf; */ 165 #if (_WIN32_IE >= 0x0300) 166 item.iIndent = 0; 167 #endif 168 169 index = ListView_InsertItem(hwndLV, &item); 170 if (index != -1) 171 { 172 switch (dwValType) 173 { 174 case REG_SZ: 175 case REG_EXPAND_SZ: 176 if(dwCount > 0) 177 { 178 ListView_SetItemText(hwndLV, index, 2, ValBuf); 179 } 180 else if(!ValExists) 181 { 182 WCHAR buffer[255]; 183 /* load (value not set) string */ 184 LoadStringW(hInst, IDS_VALUE_NOT_SET, buffer, COUNT_OF(buffer)); 185 ListView_SetItemText(hwndLV, index, 2, buffer); 186 } 187 break; 188 case REG_MULTI_SZ: 189 { 190 LPWSTR src, str; 191 if(dwCount >= 2) 192 { 193 src = (LPWSTR)ValBuf; 194 str = HeapAlloc(GetProcessHeap(), 0, dwCount + sizeof(WCHAR)); 195 if(str != NULL) 196 { 197 *str = L'\0'; 198 /* concatenate all srings */ 199 while(*src != L'\0') 200 { 201 wcscat(str, src); 202 wcscat(str, L" "); 203 src += wcslen(src) + 1; 204 } 205 ListView_SetItemText(hwndLV, index, 2, str); 206 HeapFree(GetProcessHeap(), 0, str); 207 } 208 else 209 ListView_SetItemText(hwndLV, index, 2, L""); 210 } 211 else 212 ListView_SetItemText(hwndLV, index, 2, L""); 213 } 214 break; 215 case REG_DWORD: 216 { 217 WCHAR buf[200]; 218 if(dwCount == sizeof(DWORD)) 219 { 220 wsprintf(buf, L"0x%08x (%u)", *(DWORD*)ValBuf, *(DWORD*)ValBuf); 221 } 222 else 223 { 224 LoadStringW(hInst, IDS_INVALID_DWORD, buf, COUNT_OF(buf)); 225 } 226 ListView_SetItemText(hwndLV, index, 2, buf); 227 } 228 /* lpsRes = convertHexToDWORDStr(lpbData, dwLen); */ 229 break; 230 default: 231 { 232 unsigned int i; 233 LPBYTE pData = (LPBYTE)ValBuf; 234 LPWSTR strBinary; 235 if(dwCount > 0) 236 { 237 strBinary = HeapAlloc(GetProcessHeap(), 0, (dwCount * sizeof(WCHAR) * 3) + sizeof(WCHAR)); 238 for (i = 0; i < dwCount; i++) 239 { 240 wsprintf( strBinary + i*3, L"%02X ", pData[i] ); 241 } 242 strBinary[dwCount * 3] = 0; 243 ListView_SetItemText(hwndLV, index, 2, strBinary); 244 HeapFree(GetProcessHeap(), 0, strBinary); 245 } 246 else 247 { 248 WCHAR szText[128]; 249 LoadStringW(hInst, IDS_BINARY_EMPTY, szText, COUNT_OF(szText)); 250 ListView_SetItemText(hwndLV, index, 2, szText); 251 } 252 } 253 break; 254 } 255 } 256 } 257 258 static BOOL CreateListColumns(HWND hWndListView, INT cxTotal) 259 { 260 WCHAR szText[50]; 261 int index; 262 LVCOLUMN lvC; 263 264 /* Create columns. */ 265 lvC.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT | LVCF_SUBITEM; 266 lvC.pszText = szText; 267 268 /* Load the column labels from the resource file. */ 269 for (index = 0; index < MAX_LIST_COLUMNS; index++) 270 { 271 lvC.iSubItem = index; 272 lvC.cx = (cxTotal * default_column_widths[index]) / 100; 273 lvC.fmt = column_alignment[index]; 274 LoadStringW(hInst, IDS_LIST_COLUMN_FIRST + index, szText, COUNT_OF(szText)); 275 if (ListView_InsertColumn(hWndListView, index, &lvC) == -1) return FALSE; 276 } 277 return TRUE; 278 } 279 280 static BOOL InitListViewImageLists(HWND hwndLV) 281 { 282 HIMAGELIST himl; /* handle to image list */ 283 HICON hico; /* handle to icon */ 284 285 /* Create the image list. */ 286 if ((himl = ImageList_Create(CX_ICON, CY_ICON, 287 ILC_MASK, 0, LISTVIEW_NUM_ICONS)) == NULL) 288 { 289 return FALSE; 290 } 291 292 hico = LoadIconW(hInst, MAKEINTRESOURCEW(IDI_BIN)); 293 Image_Bin = ImageList_AddIcon(himl, hico); 294 295 hico = LoadIconW(hInst, MAKEINTRESOURCEW(IDI_STRING)); 296 Image_String = ImageList_AddIcon(himl, hico); 297 298 /* Fail if not all of the images were added. */ 299 if (ImageList_GetImageCount(himl) < LISTVIEW_NUM_ICONS) 300 { 301 return FALSE; 302 } 303 304 /* Associate the image list with the tree view control. */ 305 (void)ListView_SetImageList(hwndLV, himl, LVSIL_SMALL); 306 307 return TRUE; 308 } 309 310 /* OnGetDispInfo - processes the LVN_GETDISPINFO notification message. */ 311 312 static void OnGetDispInfo(NMLVDISPINFO* plvdi) 313 { 314 static WCHAR buffer[200]; 315 316 plvdi->item.pszText = NULL; 317 plvdi->item.cchTextMax = 0; 318 319 switch (plvdi->item.iSubItem) 320 { 321 case 0: 322 LoadStringW(hInst, IDS_DEFAULT_VALUE_NAME, buffer, COUNT_OF(buffer)); 323 plvdi->item.pszText = buffer; 324 break; 325 case 1: 326 switch (((LINE_INFO*)plvdi->item.lParam)->dwValType) 327 { 328 case REG_NONE: 329 plvdi->item.pszText = L"REG_NONE"; 330 break; 331 case REG_SZ: 332 plvdi->item.pszText = L"REG_SZ"; 333 break; 334 case REG_EXPAND_SZ: 335 plvdi->item.pszText = L"REG_EXPAND_SZ"; 336 break; 337 case REG_BINARY: 338 plvdi->item.pszText = L"REG_BINARY"; 339 break; 340 case REG_DWORD: /* REG_DWORD_LITTLE_ENDIAN */ 341 plvdi->item.pszText = L"REG_DWORD"; 342 break; 343 case REG_DWORD_BIG_ENDIAN: 344 plvdi->item.pszText = L"REG_DWORD_BIG_ENDIAN"; 345 break; 346 case REG_LINK: 347 plvdi->item.pszText = L"REG_LINK"; 348 break; 349 case REG_MULTI_SZ: 350 plvdi->item.pszText = L"REG_MULTI_SZ"; 351 break; 352 case REG_RESOURCE_LIST: 353 plvdi->item.pszText = L"REG_RESOURCE_LIST"; 354 break; 355 case REG_FULL_RESOURCE_DESCRIPTOR: 356 plvdi->item.pszText = L"REG_FULL_RESOURCE_DESCRIPTOR"; 357 break; 358 case REG_RESOURCE_REQUIREMENTS_LIST: 359 plvdi->item.pszText = L"REG_RESOURCE_REQUIREMENTS_LIST"; 360 break; 361 case REG_QWORD: /* REG_QWORD_LITTLE_ENDIAN */ 362 plvdi->item.pszText = L"REG_QWORD"; 363 break; 364 default: 365 { 366 WCHAR buf2[200]; 367 LoadStringW(hInst, IDS_UNKNOWN_TYPE, buf2, COUNT_OF(buf2)); 368 wsprintf(buffer, buf2, ((LINE_INFO*)plvdi->item.lParam)->dwValType); 369 plvdi->item.pszText = buffer; 370 break; 371 } 372 } 373 break; 374 case 3: 375 plvdi->item.pszText = L""; 376 break; 377 } 378 } 379 380 static int CALLBACK CompareFunc(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort) 381 { 382 PSORT_INFO pSortInfo = (PSORT_INFO)lParamSort; 383 LINE_INFO *l, *r; 384 DWORD dw1, dw2; 385 DWORDLONG qw1, qw2; 386 387 l = (LINE_INFO*)lParam1; 388 r = (LINE_INFO*)lParam2; 389 390 if (pSortInfo->iSortingColumn == 1 && l->dwValType != r->dwValType) 391 { 392 /* Sort by type */ 393 if (pSortInfo->bSortAscending) 394 return ((int)l->dwValType - (int)r->dwValType); 395 else 396 return ((int)r->dwValType - (int)l->dwValType); 397 } 398 if (pSortInfo->iSortingColumn == 2) 399 { 400 /* Sort by value */ 401 if (l->dwValType != r->dwValType) 402 { 403 if (pSortInfo->bSortAscending) 404 return ((int)l->dwValType - (int)r->dwValType); 405 else 406 return ((int)r->dwValType - (int)l->dwValType); 407 } 408 409 if (l->val == NULL && r->val == NULL) 410 return 0; 411 412 if (pSortInfo->bSortAscending) 413 { 414 if (l->val == NULL) 415 return -1; 416 if (r->val == NULL) 417 return 1; 418 } 419 else 420 { 421 if (l->val == NULL) 422 return 1; 423 if (r->val == NULL) 424 return -1; 425 } 426 427 switch(l->dwValType) 428 { 429 case REG_DWORD: 430 { 431 dw1 = *(DWORD*)l->val; 432 dw2 = *(DWORD*)r->val; 433 if (pSortInfo->bSortAscending) 434 // return (dw1 > dw2 ? 1 : -1); 435 return ((int)dw1 - (int)dw2); 436 else 437 // return (dw1 > dw2 ? -1 : 1); 438 return ((int)dw2 - (int)dw1); 439 } 440 441 case REG_QWORD: /* REG_QWORD_LITTLE_ENDIAN */ 442 { 443 qw1 = *(DWORDLONG*)l->val; 444 qw2 = *(DWORDLONG*)r->val; 445 if (pSortInfo->bSortAscending) 446 // return (qw1 > qw2 ? 1 : -1); 447 return ((int)qw1 - (int)qw2); 448 else 449 // return (qw1 > qw2 ? -1 : 1); 450 return ((int)qw2 - (int)qw1); 451 } 452 453 default: 454 { 455 INT nCompare = 0; 456 457 if (pSortInfo->bSortAscending) 458 { 459 nCompare = memcmp(l->val, r->val, min(l->val_len, r->val_len)); 460 if (nCompare == 0) 461 nCompare = l->val_len - r->val_len; 462 } 463 else 464 { 465 nCompare = memcmp(r->val, l->val, min(r->val_len, l->val_len)); 466 if (nCompare == 0) 467 nCompare = r->val_len - l->val_len; 468 } 469 470 return nCompare; 471 } 472 } 473 } 474 475 /* Sort by name */ 476 return (pSortInfo->bSortAscending ? StrCmpLogicalW(l->name, r->name) : StrCmpLogicalW(r->name, l->name)); 477 } 478 479 static BOOL ListView_Sort(HWND hListView, int iSortingColumn, int iSortedColumn) 480 { 481 if (!(GetWindowLongPtr(hListView, GWL_STYLE) & LVS_NOSORTHEADER) && 482 (iSortingColumn >= 0) ) 483 { 484 BOOL bSortAscending; 485 SORT_INFO SortInfo; 486 487 HWND hHeader = ListView_GetHeader(hListView); 488 HDITEM hColumn = {0}; 489 490 /* If we are sorting according to another column, uninitialize the old one */ 491 if ( (iSortedColumn >= 0) && (iSortingColumn != iSortedColumn) ) 492 { 493 hColumn.mask = HDI_FORMAT; 494 Header_GetItem(hHeader, iSortedColumn, &hColumn); 495 hColumn.fmt &= ~(HDF_SORTUP | HDF_SORTDOWN); 496 Header_SetItem(hHeader, iSortedColumn, &hColumn); 497 } 498 499 /* Get the sorting state of the new column */ 500 hColumn.mask = HDI_FORMAT; 501 Header_GetItem(hHeader, iSortingColumn, &hColumn); 502 503 /* 504 * Check whether we are sorting the list because the user clicked 505 * on a column, or because we are refreshing the list: 506 * 507 * iSortedColumn >= 0 - User clicked on a column; holds the 508 * old sorting column index. 509 * iSortedColumn < 0 - List being refreshed. 510 */ 511 if (iSortedColumn >= 0) 512 { 513 /* Invert the sorting direction */ 514 bSortAscending = ((hColumn.fmt & HDF_SORTUP) == 0); 515 } 516 else 517 { 518 /* 519 * If the sorting state of the column is uninitialized, 520 * initialize it by default to ascending sorting. 521 */ 522 if ((hColumn.fmt & (HDF_SORTUP | HDF_SORTDOWN)) == 0) 523 hColumn.fmt |= HDF_SORTUP; 524 525 /* Keep the same sorting direction */ 526 bSortAscending = ((hColumn.fmt & HDF_SORTUP) != 0); 527 } 528 529 /* Set the new column sorting state */ 530 hColumn.fmt &= ~(bSortAscending ? HDF_SORTDOWN : HDF_SORTUP ); 531 hColumn.fmt |= (bSortAscending ? HDF_SORTUP : HDF_SORTDOWN); 532 Header_SetItem(hHeader, iSortingColumn, &hColumn); 533 534 /* Sort the list */ 535 SortInfo.iSortingColumn = iSortingColumn; 536 SortInfo.bSortAscending = bSortAscending; 537 return ListView_SortItems(hListView, CompareFunc, (LPARAM)&SortInfo); 538 } 539 else 540 return TRUE; 541 } 542 543 BOOL ListWndNotifyProc(HWND hWnd, WPARAM wParam, LPARAM lParam, BOOL *Result) 544 { 545 NMLVDISPINFO* Info; 546 int iSortingColumn; 547 UNREFERENCED_PARAMETER(wParam); 548 *Result = TRUE; 549 switch (((LPNMHDR)lParam)->code) 550 { 551 case LVN_GETDISPINFO: 552 OnGetDispInfo((NMLVDISPINFO*)lParam); 553 return TRUE; 554 case LVN_COLUMNCLICK: 555 iSortingColumn = ((LPNMLISTVIEW)lParam)->iSubItem; 556 (void)ListView_Sort(hWnd, iSortingColumn, g_iSortedColumn); 557 g_iSortedColumn = iSortingColumn; 558 return TRUE; 559 case NM_DBLCLK: 560 case NM_RETURN: 561 { 562 SendMessageW(hFrameWnd, WM_COMMAND, MAKEWPARAM(ID_EDIT_MODIFY, 0), 0); 563 } 564 return TRUE; 565 case NM_SETFOCUS: 566 g_pChildWnd->nFocusPanel = 0; 567 break; 568 case LVN_BEGINLABELEDIT: 569 Info = (NMLVDISPINFO*)lParam; 570 if(Info) 571 { 572 PLINE_INFO lineinfo = (PLINE_INFO)Info->item.lParam; 573 if(!lineinfo->name || !wcscmp(lineinfo->name, L"")) 574 { 575 *Result = TRUE; 576 } 577 else 578 { 579 *Result = FALSE; 580 } 581 } 582 else 583 *Result = TRUE; 584 return TRUE; 585 case LVN_ENDLABELEDIT: 586 Info = (NMLVDISPINFO*)lParam; 587 if(Info && Info->item.pszText) 588 { 589 PLINE_INFO lineinfo = (PLINE_INFO)Info->item.lParam; 590 if(!lineinfo->name || !wcscmp(lineinfo->name, L"")) 591 { 592 *Result = FALSE; 593 } 594 else 595 { 596 if(wcslen(Info->item.pszText) == 0) 597 { 598 WCHAR msg[128], caption[128]; 599 600 LoadStringW(hInst, IDS_ERR_RENVAL_TOEMPTY, msg, COUNT_OF(msg)); 601 LoadStringW(hInst, IDS_ERR_RENVAL_CAPTION, caption, COUNT_OF(caption)); 602 MessageBoxW(0, msg, caption, 0); 603 *Result = TRUE; 604 } 605 else 606 { 607 HKEY hKeyRoot; 608 LPCWSTR keyPath; 609 LONG lResult; 610 611 keyPath = GetItemPath(g_pChildWnd->hTreeWnd, 0, &hKeyRoot); 612 lResult = RenameValue(hKeyRoot, keyPath, Info->item.pszText, lineinfo->name); 613 lineinfo->name = realloc(lineinfo->name, (wcslen(Info->item.pszText)+1)*sizeof(WCHAR)); 614 if (lineinfo->name != NULL) 615 wcscpy(lineinfo->name, Info->item.pszText); 616 617 *Result = TRUE; 618 return (lResult == ERROR_SUCCESS); 619 } 620 } 621 } 622 else 623 *Result = TRUE; 624 625 return TRUE; 626 } 627 return FALSE; 628 } 629 630 HWND CreateListView(HWND hwndParent, HMENU id, INT cx) 631 { 632 RECT rcClient; 633 HWND hwndLV; 634 635 /* Get the dimensions of the parent window's client area, and create the list view control. */ 636 GetClientRect(hwndParent, &rcClient); 637 hwndLV = CreateWindowExW(WS_EX_CLIENTEDGE, WC_LISTVIEW, L"List View", 638 WS_VISIBLE | WS_CHILD | WS_TABSTOP | LVS_REPORT | LVS_EDITLABELS | LVS_SHOWSELALWAYS, 639 0, 0, rcClient.right, rcClient.bottom, 640 hwndParent, id, hInst, NULL); 641 if (!hwndLV) return NULL; 642 643 /* Initialize the image list, and add items to the control. */ 644 if (!CreateListColumns(hwndLV, cx)) goto fail; 645 if (!InitListViewImageLists(hwndLV)) goto fail; 646 647 return hwndLV; 648 fail: 649 DestroyWindow(hwndLV); 650 return NULL; 651 } 652 653 void DestroyListView(HWND hwndLV) 654 { 655 INT count, i; 656 LVITEMW item; 657 658 count = ListView_GetItemCount(hwndLV); 659 for (i = 0; i < count; i++) 660 { 661 item.mask = LVIF_PARAM; 662 item.iItem = i; 663 (void)ListView_GetItem(hwndLV, &item); 664 free(((LINE_INFO*)item.lParam)->name); 665 HeapFree(GetProcessHeap(), 0, (void*)item.lParam); 666 } 667 668 } 669 670 BOOL RefreshListView(HWND hwndLV, HKEY hKey, LPCWSTR keyPath) 671 { 672 DWORD max_sub_key_len; 673 DWORD max_val_name_len; 674 DWORD max_val_size; 675 DWORD val_count; 676 HKEY hNewKey; 677 LONG errCode; 678 INT i, c; 679 BOOL AddedDefault = FALSE; 680 681 if (!hwndLV) return FALSE; 682 683 (void)ListView_EditLabel(hwndLV, -1); 684 685 SendMessageW(hwndLV, WM_SETREDRAW, FALSE, 0); 686 DestroyListView(hwndLV); 687 688 (void)ListView_DeleteAllItems(hwndLV); 689 690 if(!hKey) return FALSE; 691 692 errCode = RegOpenKeyExW(hKey, keyPath, 0, KEY_READ, &hNewKey); 693 if (errCode != ERROR_SUCCESS) return FALSE; 694 695 /* get size information and resize the buffers if necessary */ 696 errCode = RegQueryInfoKeyW(hNewKey, NULL, NULL, NULL, NULL, &max_sub_key_len, NULL, 697 &val_count, &max_val_name_len, &max_val_size, NULL, NULL); 698 699 if (errCode == ERROR_SUCCESS) 700 { 701 WCHAR* ValName = HeapAlloc(GetProcessHeap(), 0, ++max_val_name_len * sizeof(WCHAR)); 702 DWORD dwValNameLen = max_val_name_len; 703 BYTE* ValBuf = HeapAlloc(GetProcessHeap(), 0, max_val_size + sizeof(WCHAR)); 704 DWORD dwValSize = max_val_size; 705 DWORD dwIndex = 0L; 706 DWORD dwValType; 707 /* if (RegQueryValueExW(hNewKey, NULL, NULL, &dwValType, ValBuf, &dwValSize) == ERROR_SUCCESS) { */ 708 /* AddEntryToList(hwndLV, L"(Default)", dwValType, ValBuf, dwValSize); */ 709 /* } */ 710 /* dwValSize = max_val_size; */ 711 while (RegEnumValueW(hNewKey, dwIndex, ValName, &dwValNameLen, NULL, &dwValType, ValBuf, &dwValSize) == ERROR_SUCCESS) 712 { 713 /* Add a terminating 0 character. Usually this is only necessary for strings. */ 714 ValBuf[dwValSize] = ValBuf[dwValSize + 1] = 0; 715 716 AddEntryToList(hwndLV, ValName, dwValType, ValBuf, dwValSize, -1, TRUE); 717 dwValNameLen = max_val_name_len; 718 dwValSize = max_val_size; 719 dwValType = 0L; 720 ++dwIndex; 721 if(!wcscmp(ValName, L"")) 722 { 723 AddedDefault = TRUE; 724 } 725 } 726 HeapFree(GetProcessHeap(), 0, ValBuf); 727 HeapFree(GetProcessHeap(), 0, ValName); 728 } 729 RegCloseKey(hNewKey); 730 731 if(!AddedDefault) 732 { 733 AddEntryToList(hwndLV, L"", REG_SZ, NULL, 0, 0, FALSE); 734 } 735 c = ListView_GetItemCount(hwndLV); 736 for(i = 0; i < c; i++) 737 { 738 ListView_SetItemState(hwndLV, i, 0, LVIS_FOCUSED | LVIS_SELECTED); 739 } 740 ListView_SetItemState(hwndLV, iListViewSelect, 741 LVIS_FOCUSED | LVIS_SELECTED, 742 LVIS_FOCUSED | LVIS_SELECTED); 743 (void)ListView_Sort(hwndLV, g_iSortedColumn, -1); 744 SendMessageW(hwndLV, WM_SETREDRAW, TRUE, 0); 745 746 return TRUE; 747 } 748