1 /* 2 * PROJECT: ReactOS Device Manager 3 * LICENSE: GPL - See COPYING in the top level directory 4 * FILE: dll/win32/devmgr/devmgmt/DeviceView.cpp 5 * PURPOSE: Implements the tree view which contains the devices 6 * COPYRIGHT: Copyright 2015 Ged Murphy <gedmurphy@reactos.org> 7 */ 8 9 10 11 #include "precomp.h" 12 #include "devmgmt.h" 13 #include "DeviceView.h" 14 15 16 // DATA ********************************************/ 17 18 #define CLASS_NAME_LEN 256 19 #define CLASS_DESC_LEN 256 20 #define ROOT_NAME_SIZE MAX_COMPUTERNAME_LENGTH + 1 21 22 23 typedef VOID(WINAPI *PADDHARDWAREWIZARD)(HWND hwnd, LPWSTR lpName); 24 25 struct RefreshThreadData 26 { 27 CDeviceView *This; 28 BOOL ScanForChanges; 29 BOOL UpdateView; 30 }; 31 32 33 // PUBLIC METHODS ************************************/ 34 35 CDeviceView::CDeviceView( 36 HWND hMainWnd 37 ) : 38 m_hMainWnd(hMainWnd), 39 m_hTreeView(NULL), 40 m_hPropertyDialog(NULL), 41 m_hMenu(NULL), 42 m_ViewType(DevicesByType), 43 m_ShowHidden(false), 44 m_RootNode(NULL) 45 { 46 ZeroMemory(&m_ImageListData, sizeof(SP_CLASSIMAGELIST_DATA)); 47 } 48 49 CDeviceView::~CDeviceView(void) 50 { 51 } 52 53 bool 54 CDeviceView::Initialize() 55 { 56 // Get the device image list 57 m_ImageListData.cbSize = sizeof(SP_CLASSIMAGELIST_DATA); 58 BOOL bSuccess = SetupDiGetClassImageList(&m_ImageListData); 59 if (bSuccess == FALSE) 60 return false; 61 62 // Create the main treeview 63 m_hTreeView = CreateWindowExW(WS_EX_CLIENTEDGE, 64 WC_TREEVIEW, 65 NULL, 66 WS_CHILD | WS_VISIBLE | WS_BORDER | TVS_HASLINES | 67 TVS_HASBUTTONS | TVS_SHOWSELALWAYS | TVS_LINESATROOT, 68 0, 0, 0, 0, 69 m_hMainWnd, 70 (HMENU)IDC_TREEVIEW, 71 g_hThisInstance, 72 NULL); 73 if (m_hTreeView) 74 { 75 // Set the image list against the treeview 76 (void)TreeView_SetImageList(m_hTreeView, 77 m_ImageListData.ImageList, 78 TVSIL_NORMAL); 79 80 // Give the treeview arrows instead of +/- boxes (on Win7) 81 SetWindowTheme(m_hTreeView, L"explorer", NULL); 82 83 // Create the root node 84 m_RootNode = new CRootNode(&m_ImageListData); 85 m_RootNode->SetupNode(); 86 } 87 88 89 90 return !!(m_hTreeView); 91 } 92 93 bool 94 CDeviceView::Uninitialize() 95 { 96 EmptyDeviceView(); 97 98 if (m_ImageListData.ImageList != NULL) 99 { 100 SetupDiDestroyClassImageList(&m_ImageListData); 101 ZeroMemory(&m_ImageListData, sizeof(SP_CLASSIMAGELIST_DATA)); 102 } 103 104 return true; 105 } 106 107 LRESULT 108 CDeviceView::OnSize( 109 _In_ int x, 110 _In_ int y, 111 _In_ int cx, 112 _In_ int cy 113 ) 114 { 115 // Resize the treeview 116 SetWindowPos(m_hTreeView, 117 NULL, 118 x, 119 y, 120 cx, 121 cy, 122 SWP_NOZORDER); 123 124 return 0; 125 } 126 127 LRESULT 128 CDeviceView::OnRightClick( 129 _In_ LPNMHDR NmHdr 130 ) 131 { 132 TVHITTESTINFO hitInfo; 133 HTREEITEM hItem; 134 135 GetCursorPos(&hitInfo.pt); 136 ScreenToClient(m_hTreeView, &hitInfo.pt); 137 138 hItem = TreeView_HitTest(m_hTreeView, &hitInfo); 139 if (hItem != NULL && (hitInfo.flags & (TVHT_ONITEM | TVHT_ONITEMICON))) 140 { 141 TreeView_SelectItem(m_hTreeView, hItem); 142 } 143 144 return 0; 145 } 146 147 LRESULT 148 CDeviceView::OnContextMenu( 149 _In_ LPARAM lParam 150 ) 151 { 152 HTREEITEM hSelected = TreeView_GetSelection(m_hTreeView); 153 154 RECT rc; 155 if (TreeView_GetItemRect(m_hTreeView, 156 hSelected, 157 &rc, 158 TRUE)) 159 { 160 POINT pt; 161 if (GetCursorPos(&pt) && 162 ScreenToClient(m_hTreeView, &pt) && 163 PtInRect(&rc, pt)) 164 { 165 CNode *Node = GetSelectedNode(); 166 if (Node) 167 { 168 // Create the context menu 169 HMENU hContextMenu = CreatePopupMenu(); 170 171 // Add the actions for this node 172 BuildActionMenuForNode(hContextMenu, Node, false); 173 174 INT xPos = GET_X_LPARAM(lParam); 175 INT yPos = GET_Y_LPARAM(lParam); 176 177 // Display the menu 178 TrackPopupMenuEx(hContextMenu, 179 TPM_RIGHTBUTTON, 180 xPos, 181 yPos, 182 m_hMainWnd, 183 NULL); 184 185 DestroyMenu(hContextMenu); 186 } 187 } 188 } 189 190 return 0; 191 } 192 193 194 void 195 CDeviceView::Refresh( 196 _In_ ViewType Type, 197 _In_ bool ScanForChanges, 198 _In_ bool UpdateView 199 ) 200 { 201 // Enum devices on a separate thread to keep the gui responsive 202 203 m_ViewType = Type; 204 205 RefreshThreadData *ThreadData; 206 ThreadData = new RefreshThreadData; 207 ThreadData->This = this; 208 ThreadData->ScanForChanges = ScanForChanges; 209 ThreadData->UpdateView = UpdateView; 210 211 HANDLE hThread; 212 hThread = (HANDLE)_beginthreadex(NULL, 213 0, 214 RefreshThread, 215 ThreadData, 216 0, 217 NULL); 218 if (hThread) CloseHandle(hThread); 219 } 220 221 LRESULT 222 CDeviceView::OnAction( 223 _In_ UINT Action 224 ) 225 { 226 switch (Action) 227 { 228 case IDC_PROPERTIES: 229 { 230 DisplayPropertySheet(); 231 break; 232 } 233 234 case IDC_SCAN_HARDWARE: 235 { 236 Refresh(GetCurrentView(), 237 true, 238 true); 239 break; 240 } 241 242 case IDC_ENABLE_DRV: 243 { 244 bool NeedsReboot; 245 if (EnableSelectedDevice(true, NeedsReboot) && 246 NeedsReboot) 247 { 248 MessageBox(m_hMainWnd, L"Rebooting", L"Enable", MB_OK); 249 } 250 break; 251 } 252 253 case IDC_DISABLE_DRV: 254 { 255 bool NeedsReboot; 256 EnableSelectedDevice(false, NeedsReboot); 257 break; 258 } 259 260 case IDC_UPDATE_DRV: 261 { 262 bool NeedsReboot; 263 UpdateSelectedDevice(NeedsReboot); 264 break; 265 } 266 267 case IDC_UNINSTALL_DRV: 268 { 269 UninstallSelectedDevice(); 270 break; 271 } 272 273 case IDC_ADD_HARDWARE: 274 { 275 RunAddHardwareWizard(); 276 break; 277 } 278 } 279 280 return 0; 281 } 282 283 void 284 CDeviceView::DisplayPropertySheet() 285 { 286 CNode *Node = GetSelectedNode(); 287 if (Node && Node->HasProperties()) 288 { 289 DevicePropertiesExW(m_hTreeView, 290 NULL, 291 Node->GetDeviceId(), 292 1,//DPF_EXTENDED, 293 FALSE); 294 } 295 } 296 297 void 298 CDeviceView::SetFocus() 299 { 300 ::SetFocus(m_hTreeView); 301 } 302 303 bool 304 CDeviceView::CreateActionMenu( 305 _In_ HMENU OwnerMenu, 306 _In_ bool MainMenu 307 ) 308 { 309 CNode *Node = GetSelectedNode(); 310 if (Node) 311 { 312 BuildActionMenuForNode(OwnerMenu, Node, MainMenu); 313 return true; 314 } 315 316 return false; 317 } 318 319 CNode* 320 CDeviceView::GetSelectedNode() 321 { 322 TV_ITEM TvItem; 323 TvItem.hItem = TreeView_GetSelection(m_hTreeView); 324 return GetNode(&TvItem); 325 } 326 327 328 329 // PRIVATE METHODS *******************************************/ 330 331 bool 332 CDeviceView::AddRootDevice() 333 { 334 m_hTreeRoot = InsertIntoTreeView(NULL, m_RootNode); 335 return (m_hTreeRoot != NULL); 336 } 337 338 bool 339 CDeviceView::GetNextClass( 340 _In_ ULONG ClassIndex, 341 _Out_ LPGUID ClassGuid, 342 _Out_ HDEVINFO *hDevInfo 343 ) 344 { 345 CONFIGRET cr; 346 347 // Get the next class in the list 348 cr = CM_Enumerate_Classes(ClassIndex, 349 ClassGuid, 350 0); 351 if (cr != CR_SUCCESS) 352 return false; 353 354 // Check if this is the unknown class 355 if (IsEqualGUID(*ClassGuid, GUID_DEVCLASS_UNKNOWN)) 356 { 357 // Get device info for all devices 358 *hDevInfo = SetupDiGetClassDevsW(NULL, 359 NULL, 360 NULL, 361 DIGCF_ALLCLASSES); 362 } 363 else 364 { 365 // We only want the devices for this class 366 *hDevInfo = SetupDiGetClassDevsW(ClassGuid, 367 NULL, 368 NULL, 369 DIGCF_PRESENT); 370 } 371 372 return (hDevInfo != INVALID_HANDLE_VALUE); 373 } 374 375 unsigned int __stdcall CDeviceView::RefreshThread(void *Param) 376 { 377 RefreshThreadData *ThreadData = (RefreshThreadData *)Param; 378 CDeviceView *This = ThreadData->This; 379 380 // Get a copy of the currently selected node 381 CNode *LastSelectedNode = This->GetSelectedNode(); 382 if (LastSelectedNode == nullptr || (LastSelectedNode->GetNodeType() == RootNode)) 383 { 384 LastSelectedNode = new CRootNode(*This->m_RootNode); 385 } 386 else if (LastSelectedNode->GetNodeType() == ClassNode) 387 { 388 LastSelectedNode = new CClassNode(*dynamic_cast<CClassNode *>(LastSelectedNode)); 389 } 390 else if (LastSelectedNode->GetNodeType() == DeviceNode) 391 { 392 LastSelectedNode = new CDeviceNode(*dynamic_cast<CDeviceNode *>(LastSelectedNode)); 393 } 394 395 // Empty the treeview 396 This->EmptyDeviceView(); 397 398 // Re-add the root node to the tree 399 if (This->AddRootDevice() == false) 400 return 0; 401 402 // Refresh the devices only if requested 403 if (ThreadData->ScanForChanges) 404 { 405 This->RefreshDeviceList(); 406 } 407 408 // display the type of view the user wants 409 switch (This->m_ViewType) 410 { 411 case DevicesByType: 412 (void)This->ListDevicesByType(); 413 break; 414 415 case DevicesByConnection: 416 (VOID)This->ListDevicesByConnection(); 417 break; 418 419 case ResourcesByType: 420 break; 421 422 case ResourcesByConnection: 423 break; 424 } 425 426 This->SelectNode(LastSelectedNode); 427 428 delete ThreadData; 429 430 return 0; 431 } 432 433 434 bool 435 CDeviceView::ListDevicesByType() 436 { 437 CClassNode *ClassNode; 438 CDeviceNode *DeviceNode; 439 HDEVINFO hDevInfo; 440 HTREEITEM hTreeItem = NULL; 441 GUID ClassGuid; 442 INT ClassIndex; 443 BOOL bClassSuccess, bSuccess; 444 445 ClassIndex = 0; 446 do 447 { 448 // Loop through all the device classes 449 bClassSuccess = GetNextClass(ClassIndex, &ClassGuid, &hDevInfo); 450 if (bClassSuccess) 451 { 452 bool bClassUnknown = false; 453 bool AddedParent = false; 454 INT DeviceIndex = 0; 455 bool MoreItems = false; 456 457 // Get the cached class node 458 ClassNode = GetClassNode(&ClassGuid); 459 if (ClassNode == nullptr) 460 { 461 ATLASSERT(FALSE); 462 ClassIndex++; 463 continue; 464 } 465 466 // Set a flag is this is the (special case) unknown class 467 if (IsEqualGUID(ClassGuid, GUID_DEVCLASS_UNKNOWN)) 468 bClassUnknown = true; 469 470 // Check if this is a hidden class 471 if (IsEqualGUID(ClassGuid, GUID_DEVCLASS_LEGACYDRIVER) || 472 IsEqualGUID(ClassGuid, GUID_DEVCLASS_VOLUME)) 473 { 474 // Ignore this device if we aren't displaying hidden devices 475 if (m_ShowHidden == FALSE) 476 { 477 ClassIndex++; 478 continue; 479 } 480 } 481 482 do 483 { 484 // Get a handle to all the devices in this class 485 SP_DEVINFO_DATA DeviceInfoData; 486 ZeroMemory(&DeviceInfoData, sizeof(SP_DEVINFO_DATA)); 487 DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA); 488 bSuccess = SetupDiEnumDeviceInfo(hDevInfo, 489 DeviceIndex, 490 &DeviceInfoData); 491 if (bSuccess == FALSE && GetLastError() == ERROR_NO_MORE_ITEMS) 492 MoreItems = false; 493 494 if (bSuccess) 495 { 496 MoreItems = true; 497 498 // The unknown class handle contains all devices on the system, 499 // and we're just looking for the ones with a null GUID 500 if (bClassUnknown) 501 { 502 if (IsEqualGUID(DeviceInfoData.ClassGuid, GUID_NULL) == FALSE) 503 { 504 // This is a known device, we aren't interested in it 505 DeviceIndex++; 506 continue; 507 } 508 } 509 510 // Get the cached device node 511 DeviceNode = GetDeviceNode(DeviceInfoData.DevInst); 512 if (DeviceNode == nullptr) 513 { 514 ATLASSERT(bClassUnknown == true); 515 DeviceIndex++; 516 continue; 517 } 518 519 // Check if this is a hidden device 520 if (DeviceNode->IsHidden()) 521 { 522 // Ignore this device if we aren't displaying hidden devices 523 if (m_ShowHidden == FALSE) 524 { 525 DeviceIndex++; 526 continue; 527 } 528 } 529 530 // We have a device, we need to add the parent if it hasn't yet been added 531 if (AddedParent == false) 532 { 533 // Insert the new class under the root item 534 hTreeItem = InsertIntoTreeView(m_hTreeRoot, 535 ClassNode); 536 AddedParent = true; 537 } 538 539 // Add the device under the class item node 540 (void)InsertIntoTreeView(hTreeItem, DeviceNode); 541 542 // Expand the class if it has a problem device 543 if (DeviceNode->HasProblem()) 544 { 545 (void)TreeView_Expand(m_hTreeView, 546 hTreeItem, 547 TVE_EXPAND); 548 } 549 } 550 551 DeviceIndex++; 552 553 } while (MoreItems); 554 555 // If this class has devices, sort them alphabetically 556 if (AddedParent == true) 557 { 558 (void)TreeView_SortChildren(m_hTreeView, 559 hTreeItem, 560 0); 561 } 562 } 563 564 ClassIndex++; 565 566 } while (bClassSuccess); 567 568 // Sort the classes alphabetically 569 (void)TreeView_SortChildren(m_hTreeView, 570 m_hTreeRoot, 571 0); 572 573 // Expand the root item 574 (void)TreeView_Expand(m_hTreeView, 575 m_hTreeRoot, 576 TVE_EXPAND); 577 578 // Pre-select the root item 579 (VOID)TreeView_SelectItem(m_hTreeView, 580 m_hTreeRoot); 581 582 return 0; 583 } 584 585 bool 586 CDeviceView::ListDevicesByConnection() 587 { 588 // Walk the device tree and add all the devices 589 (void)RecurseChildDevices(m_RootNode->GetDeviceInst(), m_hTreeRoot); 590 591 // Expand the root item 592 (void)TreeView_Expand(m_hTreeView, 593 m_hTreeRoot, 594 TVE_EXPAND); 595 596 return true; 597 } 598 599 bool 600 CDeviceView::RecurseChildDevices( 601 _In_ DEVINST ParentDevice, 602 _In_ HTREEITEM hParentTreeItem 603 ) 604 { 605 HTREEITEM hDevItem = NULL; 606 DEVINST Device; 607 bool HasProblem = false; 608 bool bSuccess; 609 610 // Check if the parent has any child devices 611 if (GetChildDevice(ParentDevice, &Device) == FALSE) 612 return true; 613 614 // Get the cached device node 615 CDeviceNode *DeviceNode; 616 DeviceNode = dynamic_cast<CDeviceNode *>(GetDeviceNode(Device)); 617 if (DeviceNode == nullptr) 618 { 619 ATLASSERT(FALSE); 620 return false; 621 } 622 623 // Don't show hidden devices if not requested 624 if ((m_ShowHidden == TRUE) || (!(DeviceNode->IsHidden()))) 625 { 626 // Add this device to the tree under its parent 627 hDevItem = InsertIntoTreeView(hParentTreeItem, 628 DeviceNode); 629 if (hDevItem) 630 { 631 // Check if this child has any children itself 632 if (!RecurseChildDevices(Device, hDevItem)) 633 HasProblem = true; 634 } 635 636 if (DeviceNode->HasProblem()) 637 { 638 HasProblem = true; 639 } 640 } 641 642 643 // Check for siblings 644 for (;;) 645 { 646 // Check if the parent device has anything at the same level 647 bSuccess = GetSiblingDevice(Device, &Device); 648 if (bSuccess == FALSE) 649 break; 650 651 DeviceNode = dynamic_cast<CDeviceNode *>(GetDeviceNode(Device)); 652 if (DeviceNode == nullptr) 653 { 654 ATLASSERT(FALSE); 655 } 656 657 // Don't show hidden devices if not requested 658 if ((m_ShowHidden == TRUE) || (!(DeviceNode->IsHidden()))) 659 { 660 if (DeviceNode->HasProblem()) 661 { 662 HasProblem = true; 663 } 664 665 // Add this device to the tree under its parent 666 hDevItem = InsertIntoTreeView(hParentTreeItem, 667 DeviceNode); 668 if (hDevItem) 669 { 670 // Check if this child has any children itself 671 if (!RecurseChildDevices(Device, hDevItem)) 672 HasProblem = true; 673 } 674 } 675 } 676 677 (void)TreeView_SortChildren(m_hTreeView, 678 hParentTreeItem, 679 0); 680 681 // Expand the class if it has a problem device 682 if (HasProblem == true) 683 { 684 (void)TreeView_Expand(m_hTreeView, 685 hParentTreeItem, 686 TVE_EXPAND); 687 } 688 689 // If there was a problem, expand the ancestors 690 if (HasProblem) 691 return false; 692 693 return true; 694 } 695 696 bool 697 CDeviceView::EnableSelectedDevice( 698 _In_ bool Enable, 699 _Out_ bool &NeedsReboot 700 ) 701 { 702 CDeviceNode *Node = dynamic_cast<CDeviceNode *>(GetSelectedNode()); 703 if (Node == nullptr) 704 return false; 705 706 if (Enable == false) 707 { 708 CAtlStringW str; 709 if (str.LoadStringW(g_hThisInstance, IDS_CONFIRM_DISABLE)) 710 { 711 if (MessageBoxW(m_hMainWnd, 712 str, 713 Node->GetDisplayName(), 714 MB_YESNO | MB_ICONWARNING | MB_DEFBUTTON2) != IDYES) 715 { 716 return false; 717 } 718 } 719 } 720 721 return Node->EnableDevice(Enable, NeedsReboot); 722 } 723 724 bool 725 CDeviceView::UpdateSelectedDevice( 726 _Out_ bool &NeedsReboot 727 ) 728 { 729 CDeviceNode *Node = dynamic_cast<CDeviceNode *>(GetSelectedNode()); 730 if (Node == nullptr) 731 return false; 732 733 DWORD dwReboot; 734 if (InstallDevInst(m_hMainWnd, Node->GetDeviceId(), TRUE, &dwReboot)) 735 { 736 NeedsReboot = false; 737 return true; 738 } 739 740 return false; 741 } 742 743 bool 744 CDeviceView::UninstallSelectedDevice( 745 ) 746 { 747 CDeviceNode *Node = dynamic_cast<CDeviceNode *>(GetSelectedNode()); 748 if (Node == nullptr) 749 return false; 750 751 CAtlStringW str; 752 if (str.LoadStringW(g_hThisInstance, IDS_CONFIRM_UNINSTALL)) 753 { 754 if (MessageBoxW(m_hMainWnd, 755 str, 756 Node->GetDisplayName(), 757 MB_YESNO | MB_ICONWARNING | MB_DEFBUTTON2) != IDYES) 758 { 759 return false; 760 } 761 } 762 763 return Node->UninstallDevice(); 764 } 765 766 bool 767 CDeviceView::RunAddHardwareWizard() 768 { 769 PADDHARDWAREWIZARD pAddHardwareWizard; 770 HMODULE hModule; 771 772 hModule = LoadLibraryW(L"hdwwiz.cpl"); 773 if (hModule == NULL) 774 return false; 775 776 pAddHardwareWizard = (PADDHARDWAREWIZARD)GetProcAddress(hModule, 777 "AddHardwareWizard"); 778 if (pAddHardwareWizard == NULL) 779 { 780 FreeLibrary(hModule); 781 return false; 782 } 783 784 pAddHardwareWizard(m_hMainWnd, NULL); 785 786 FreeLibrary(hModule); 787 return true; 788 } 789 790 bool 791 CDeviceView::GetChildDevice( 792 _In_ DEVINST ParentDevInst, 793 _Out_ PDEVINST DevInst 794 ) 795 { 796 CONFIGRET cr; 797 cr = CM_Get_Child(DevInst, 798 ParentDevInst, 799 0); 800 return (cr == CR_SUCCESS); 801 } 802 803 bool 804 CDeviceView::GetSiblingDevice( 805 _In_ DEVINST PrevDevice, 806 _Out_ PDEVINST DevInst 807 ) 808 { 809 CONFIGRET cr; 810 cr = CM_Get_Sibling(DevInst, 811 PrevDevice, 812 0); 813 return (cr == CR_SUCCESS); 814 } 815 816 HTREEITEM 817 CDeviceView::InsertIntoTreeView( 818 _In_opt_ HTREEITEM hParent, 819 _In_ CNode *Node 820 ) 821 { 822 LPWSTR lpLabel; 823 lpLabel = Node->GetDisplayName(); 824 825 TV_ITEMW tvi; 826 TV_INSERTSTRUCT tvins; 827 ZeroMemory(&tvi, sizeof(tvi)); 828 ZeroMemory(&tvins, sizeof(tvins)); 829 830 tvi.mask = TVIF_TEXT | TVIF_PARAM | TVIF_IMAGE | TVIF_SELECTEDIMAGE; 831 tvi.pszText = lpLabel; 832 tvi.cchTextMax = wcslen(lpLabel); 833 tvi.lParam = (LPARAM)Node; 834 tvi.iImage = Node->GetClassImage(); 835 tvi.iSelectedImage = Node->GetClassImage(); 836 837 // try to cast it to a device node. This will only succeed if it's the correct type 838 CDeviceNode *DeviceNode = dynamic_cast<CDeviceNode *>(Node); 839 if (DeviceNode && DeviceNode->GetOverlayImage()) 840 { 841 tvi.mask |= TVIF_STATE; 842 tvi.stateMask = TVIS_OVERLAYMASK; 843 tvi.state = INDEXTOOVERLAYMASK(DeviceNode->GetOverlayImage()); 844 } 845 846 tvins.item = tvi; 847 tvins.hParent = hParent; 848 849 return TreeView_InsertItem(m_hTreeView, &tvins); 850 } 851 852 void 853 CDeviceView::BuildActionMenuForNode( 854 _In_ HMENU OwnerMenu, 855 _In_ CNode *Node, 856 _In_ bool MainMenu 857 ) 858 { 859 // Create a separator structure 860 MENUITEMINFOW MenuSeparator = { 0 }; 861 MenuSeparator.cbSize = sizeof(MENUITEMINFOW); 862 MenuSeparator.fType = MFT_SEPARATOR; 863 864 // Setup the 865 MENUITEMINFOW MenuItemInfo = { 0 }; 866 MenuItemInfo.cbSize = sizeof(MENUITEMINFOW); 867 MenuItemInfo.fMask = MIIM_ID | MIIM_STRING | MIIM_DATA | MIIM_SUBMENU; 868 MenuItemInfo.fType = MFT_STRING; 869 870 CAtlStringW String; 871 int i = 0; 872 873 // Device nodes have extra data 874 if (Node->GetNodeType() == DeviceNode) 875 { 876 CDeviceNode *DeviceNode = dynamic_cast<CDeviceNode *>(Node); 877 878 if (DeviceNode->CanUpdate()) 879 { 880 String.LoadStringW(g_hThisInstance, IDS_MENU_UPDATE); 881 MenuItemInfo.wID = IDC_UPDATE_DRV; 882 MenuItemInfo.dwTypeData = String.GetBuffer(); 883 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuItemInfo); 884 i++; 885 } 886 887 if (DeviceNode->IsDisabled()) 888 { 889 String.LoadStringW(g_hThisInstance, IDS_MENU_ENABLE); 890 MenuItemInfo.wID = IDC_ENABLE_DRV; 891 MenuItemInfo.dwTypeData = String.GetBuffer(); 892 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuItemInfo); 893 i++; 894 } 895 896 if (DeviceNode->CanDisable() && !DeviceNode->IsDisabled()) 897 { 898 String.LoadStringW(g_hThisInstance, IDS_MENU_DISABLE); 899 MenuItemInfo.wID = IDC_DISABLE_DRV; 900 MenuItemInfo.dwTypeData = String.GetBuffer(); 901 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuItemInfo); 902 i++; 903 } 904 905 if (DeviceNode->CanUninstall()) 906 { 907 String.LoadStringW(g_hThisInstance, IDS_MENU_UNINSTALL); 908 MenuItemInfo.wID = IDC_UNINSTALL_DRV; 909 MenuItemInfo.dwTypeData = String.GetBuffer(); 910 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuItemInfo); 911 i++; 912 } 913 914 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuSeparator); 915 i++; 916 } 917 918 // All nodes have the scan option 919 String.LoadStringW(g_hThisInstance, IDS_MENU_SCAN); 920 MenuItemInfo.wID = IDC_SCAN_HARDWARE; 921 MenuItemInfo.dwTypeData = String.GetBuffer(); 922 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuItemInfo); 923 i++; 924 925 if ((Node->GetNodeType() == RootNode) || (MainMenu == true)) 926 { 927 String.LoadStringW(g_hThisInstance, IDS_MENU_ADD); 928 MenuItemInfo.wID = IDC_ADD_HARDWARE; 929 MenuItemInfo.dwTypeData = String.GetBuffer(); 930 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuItemInfo); 931 i++; 932 } 933 934 if (Node->HasProperties()) 935 { 936 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuSeparator); 937 i++; 938 939 String.LoadStringW(g_hThisInstance, IDS_MENU_PROPERTIES); 940 MenuItemInfo.wID = IDC_PROPERTIES; 941 MenuItemInfo.dwTypeData = String.GetBuffer(); 942 InsertMenuItemW(OwnerMenu, i, TRUE, &MenuItemInfo); 943 i++; 944 945 SetMenuDefaultItem(OwnerMenu, IDC_PROPERTIES, FALSE); 946 } 947 } 948 949 HTREEITEM 950 CDeviceView::RecurseFindDevice( 951 _In_ HTREEITEM hParentItem, 952 _In_ CNode *Node 953 ) 954 { 955 HTREEITEM FoundItem; 956 HTREEITEM hItem; 957 TVITEMW tvItem; 958 CNode *FoundNode; 959 960 // Check if this node has any children 961 hItem = TreeView_GetChild(m_hTreeView, hParentItem); 962 if (hItem == NULL) 963 return NULL; 964 965 // The lParam contains the node pointer data 966 tvItem.hItem = hItem; 967 tvItem.mask = TVIF_PARAM; 968 if (TreeView_GetItem(m_hTreeView, &tvItem) && 969 tvItem.lParam != NULL) 970 { 971 // check for a matching node 972 FoundNode = reinterpret_cast<CNode *>(tvItem.lParam); 973 if ((FoundNode->GetNodeType() == Node->GetNodeType()) && 974 (IsEqualGUID(*FoundNode->GetClassGuid(), *Node->GetClassGuid()))) 975 { 976 // check if this is a class node, or a device with matching ID's 977 if ((FoundNode->GetNodeType() == ClassNode) || 978 (wcscmp(FoundNode->GetDeviceId(), Node->GetDeviceId()) == 0)) 979 { 980 return hItem; 981 } 982 } 983 } 984 985 // This node may have its own children 986 FoundItem = RecurseFindDevice(hItem, Node); 987 if (FoundItem) 988 return FoundItem; 989 990 // Loop all the siblings 991 for (;;) 992 { 993 // Get the next item at this level 994 hItem = TreeView_GetNextSibling(m_hTreeView, hItem); 995 if (hItem == NULL) 996 break; 997 998 // The lParam contains the node pointer data 999 tvItem.hItem = hItem; 1000 tvItem.mask = TVIF_PARAM; 1001 if (TreeView_GetItem(m_hTreeView, &tvItem)) 1002 { 1003 // check for a matching class 1004 FoundNode = reinterpret_cast<CNode *>(tvItem.lParam); 1005 if ((FoundNode->GetNodeType() == Node->GetNodeType()) && 1006 (IsEqualGUID(*FoundNode->GetClassGuid(), *Node->GetClassGuid()))) 1007 { 1008 // check if this is a class node, or a device with matching ID's 1009 if ((FoundNode->GetNodeType() == ClassNode) || 1010 (wcscmp(FoundNode->GetDeviceId(), Node->GetDeviceId()) == 0)) 1011 { 1012 return hItem; 1013 } 1014 } 1015 } 1016 1017 // This node may have its own children 1018 FoundItem = RecurseFindDevice(hItem, Node); 1019 if (FoundItem) 1020 return FoundItem; 1021 } 1022 1023 return hItem; 1024 } 1025 1026 void 1027 CDeviceView::SelectNode( 1028 _In_ CNode *Node 1029 ) 1030 { 1031 HTREEITEM hRoot, hItem; 1032 1033 // Check if there are any items in the tree 1034 hRoot = TreeView_GetRoot(m_hTreeView); 1035 if (hRoot == NULL) 1036 return; 1037 1038 // If we don't want to set select a node, just select root 1039 if (Node == nullptr || Node->GetNodeType() == RootNode) 1040 { 1041 TreeView_SelectItem(m_hTreeView, hRoot); 1042 return; 1043 } 1044 1045 // Scan the tree looking for the node we want 1046 hItem = RecurseFindDevice(hRoot, Node); 1047 if (hItem) 1048 { 1049 TreeView_SelectItem(m_hTreeView, hItem); 1050 } 1051 else 1052 { 1053 TreeView_SelectItem(m_hTreeView, hRoot); 1054 } 1055 } 1056 1057 1058 void 1059 CDeviceView::EmptyDeviceView() 1060 { 1061 (VOID)TreeView_DeleteAllItems(m_hTreeView); 1062 } 1063 1064 1065 CClassNode* 1066 CDeviceView::GetClassNode( 1067 _In_ LPGUID ClassGuid 1068 ) 1069 { 1070 POSITION Pos; 1071 CClassNode *Node = nullptr; 1072 1073 Pos = m_ClassNodeList.GetHeadPosition(); 1074 if (Pos == NULL) 1075 return nullptr; 1076 1077 do 1078 { 1079 Node = m_ClassNodeList.GetNext(Pos); 1080 if (IsEqualGUID(*Node->GetClassGuid(), *ClassGuid)) 1081 { 1082 ATLASSERT(Node->GetNodeType() == ClassNode); 1083 break; 1084 } 1085 1086 Node = nullptr; 1087 1088 } while (Pos != NULL); 1089 1090 return Node; 1091 } 1092 1093 CDeviceNode* 1094 CDeviceView::GetDeviceNode( 1095 _In_ DEVINST Device 1096 ) 1097 { 1098 POSITION Pos; 1099 CDeviceNode *Node = nullptr; 1100 1101 Pos = m_DeviceNodeList.GetHeadPosition(); 1102 if (Pos == NULL) 1103 return nullptr; 1104 1105 do 1106 { 1107 Node = m_DeviceNodeList.GetNext(Pos); 1108 if (Node->GetDeviceInst() == Device) 1109 { 1110 ATLASSERT(Node->GetNodeType() == DeviceNode); 1111 break; 1112 } 1113 1114 Node = nullptr; 1115 1116 } while (Pos != NULL); 1117 1118 return Node; 1119 } 1120 1121 CNode* CDeviceView::GetNode( 1122 _In_ LPTV_ITEMW TvItem 1123 ) 1124 { 1125 TvItem->mask = TVIF_PARAM; 1126 if (TreeView_GetItem(m_hTreeView, TvItem)) 1127 { 1128 return (CNode *)TvItem->lParam; 1129 } 1130 return nullptr; 1131 } 1132 1133 void 1134 CDeviceView::EmptyLists() 1135 { 1136 CNode *Node; 1137 1138 while (!m_ClassNodeList.IsEmpty()) 1139 { 1140 Node = m_ClassNodeList.RemoveTail(); 1141 delete Node; 1142 } 1143 1144 while (!m_DeviceNodeList.IsEmpty()) 1145 { 1146 Node = m_DeviceNodeList.RemoveTail(); 1147 delete Node; 1148 } 1149 } 1150 1151 bool 1152 CDeviceView::RefreshDeviceList() 1153 { 1154 GUID ClassGuid; 1155 CClassNode *ClassNode; 1156 CDeviceNode *DeviceNode; 1157 HDEVINFO hDevInfo; 1158 SP_DEVINFO_DATA DeviceInfoData; 1159 DWORD i; 1160 BOOL Success; 1161 1162 ULONG ClassIndex = 0; 1163 1164 EmptyLists(); 1165 1166 if (m_RootNode) delete m_RootNode; 1167 m_RootNode = new CRootNode(&m_ImageListData); 1168 m_RootNode->SetupNode(); 1169 1170 // Loop through all the classes 1171 do 1172 { 1173 Success = GetNextClass(ClassIndex, &ClassGuid, &hDevInfo); 1174 if (Success) 1175 { 1176 // Create a new class node and add it to the list 1177 ClassNode = new CClassNode(&ClassGuid, &m_ImageListData); 1178 if (ClassNode->SetupNode()) 1179 { 1180 m_ClassNodeList.AddTail(ClassNode); 1181 } 1182 1183 SetupDiDestroyDeviceInfoList(hDevInfo); 1184 } 1185 ClassIndex++; 1186 } while (Success); 1187 1188 // Get all the devices on the local machine 1189 hDevInfo = SetupDiGetClassDevsW(NULL, 1190 0, 1191 0, 1192 DIGCF_PRESENT | DIGCF_ALLCLASSES); 1193 if (hDevInfo == INVALID_HANDLE_VALUE) 1194 { 1195 return false; 1196 } 1197 1198 // loop though all the devices 1199 DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA); 1200 for (i = 0;; i++) 1201 { 1202 // Get the devinst for this device 1203 Success = SetupDiEnumDeviceInfo(hDevInfo, i, &DeviceInfoData); 1204 if (Success == FALSE) 1205 break; 1206 1207 // create a new device node and add it to the list 1208 DeviceNode = new CDeviceNode(DeviceInfoData.DevInst, &m_ImageListData); 1209 if (DeviceNode->SetupNode()) 1210 { 1211 m_DeviceNodeList.AddTail(DeviceNode); 1212 } 1213 else 1214 { 1215 ATLASSERT(FALSE); 1216 } 1217 } 1218 1219 SetupDiDestroyDeviceInfoList(hDevInfo); 1220 1221 return TRUE; 1222 }