1 #include "precomp.h" 2 3 #define NDEBUG 4 #include <debug.h> 5 6 static HINSTANCE hInstance; 7 8 #ifdef UNICODE 9 typedef INT_PTR (WINAPI *PDEVICEPROPERTIES)(HWND,LPCWSTR,LPCWSTR,BOOL); 10 #define FUNC_DEVICEPROPERTIES "DevicePropertiesW" 11 #else 12 typedef INT_PTR (WINAPI *PDEVICEPROPERTIES)(HWND,LPCSTR,LPCSTR,BOOL); 13 #define FUNC_DEVICEPROPERTIES "DevicePropertiesA" 14 #endif 15 16 static LPTSTR 17 GetMonitorDevInstID(LPCTSTR lpDeviceID) 18 { 19 /* FIXME: Implement, allocate returned string with LocalAlloc! */ 20 return NULL; 21 } 22 23 static VOID 24 ShowMonitorProperties(PDESKMONITOR This) 25 { 26 HMODULE hDevMgr; 27 PDEVICEPROPERTIES pDeviceProperties; 28 LPTSTR lpDevInstID; 29 30 if (This->SelMonitor != NULL) 31 { 32 lpDevInstID = GetMonitorDevInstID(This->SelMonitor->dd.DeviceID); 33 if (lpDevInstID != NULL) 34 { 35 hDevMgr = LoadLibrary(TEXT("devmgr.dll")); 36 if (hDevMgr != NULL) 37 { 38 pDeviceProperties = (PDEVICEPROPERTIES)GetProcAddress(hDevMgr, 39 FUNC_DEVICEPROPERTIES); 40 if (pDeviceProperties != NULL) 41 { 42 pDeviceProperties(This->hwndDlg, 43 NULL, 44 This->SelMonitor->dd.DeviceID, 45 FALSE); 46 } 47 48 FreeLibrary(hDevMgr); 49 } 50 51 LocalFree((HLOCAL)lpDevInstID); 52 } 53 } 54 } 55 56 static VOID 57 UpdateMonitorSelection(PDESKMONITOR This) 58 { 59 INT i; 60 61 if (This->dwMonitorCount > 1) 62 { 63 This->SelMonitor = NULL; 64 65 i = (INT)SendDlgItemMessage(This->hwndDlg, 66 IDC_MONITORLIST, 67 LB_GETCURSEL, 68 0, 69 0); 70 if (i >= 0) 71 { 72 This->SelMonitor = (PDESKMONINFO)SendDlgItemMessage(This->hwndDlg, 73 IDC_MONITORLIST, 74 LB_GETITEMDATA, 75 (WPARAM)i, 76 0); 77 } 78 } 79 else 80 This->SelMonitor = This->Monitors; 81 82 EnableWindow(GetDlgItem(This->hwndDlg, 83 IDC_MONITORPROPERTIES), 84 This->SelMonitor != NULL); 85 } 86 87 static VOID 88 UpdatePruningControls(PDESKMONITOR This) 89 { 90 EnableWindow(GetDlgItem(This->hwndDlg, 91 IDC_PRUNINGCHECK), 92 This->bModesPruned && !This->bKeyIsReadOnly); 93 CheckDlgButton(This->hwndDlg, 94 IDC_PRUNINGCHECK, 95 (This->bModesPruned && This->bPruningOn) ? BST_CHECKED : BST_UNCHECKED); 96 } 97 98 static VOID 99 GetPruningSettings(PDESKMONITOR This) 100 { 101 BOOL bModesPruned = FALSE, bKeyIsReadOnly = FALSE, bPruningOn = FALSE; 102 103 if (This->DeskExtInterface != NULL) 104 { 105 This->DeskExtInterface->GetPruningMode(This->DeskExtInterface->Context, 106 &bModesPruned, 107 &bKeyIsReadOnly, 108 &bPruningOn); 109 } 110 111 /* Check the boolean values against zero before assigning it to the bitfields! */ 112 This->bModesPruned = (bModesPruned != FALSE); 113 This->bKeyIsReadOnly = (bKeyIsReadOnly != FALSE); 114 This->bPruningOn = (bPruningOn != FALSE); 115 116 UpdatePruningControls(This); 117 } 118 119 static VOID 120 UpdateRefreshFrequencyList(PDESKMONITOR This) 121 { 122 PDEVMODEW lpCurrentMode, lpMode; 123 TCHAR szBuffer[64]; 124 DWORD dwIndex; 125 INT i; 126 BOOL bHasDef = FALSE; 127 BOOL bAdded = FALSE; 128 129 /* Fill the refresh rate combo box */ 130 SendDlgItemMessage(This->hwndDlg, 131 IDC_REFRESHRATE, 132 CB_RESETCONTENT, 133 0, 134 0); 135 136 lpCurrentMode = This->DeskExtInterface->GetCurrentMode(This->DeskExtInterface->Context); 137 dwIndex = 0; 138 139 do 140 { 141 lpMode = This->DeskExtInterface->EnumAllModes(This->DeskExtInterface->Context, 142 dwIndex++); 143 if (lpMode != NULL && 144 lpMode->dmBitsPerPel == lpCurrentMode->dmBitsPerPel && 145 lpMode->dmPelsWidth == lpCurrentMode->dmPelsWidth && 146 lpMode->dmPelsHeight == lpCurrentMode->dmPelsHeight) 147 { 148 /* We're only interested in refresh rates for the current resolution and color depth */ 149 150 if (lpMode->dmDisplayFrequency <= 1) 151 { 152 /* Default hardware frequency */ 153 if (bHasDef) 154 continue; 155 156 bHasDef = TRUE; 157 158 if (!LoadString(hInstance, 159 IDS_USEDEFFRQUENCY, 160 szBuffer, 161 sizeof(szBuffer) / sizeof(szBuffer[0]))) 162 { 163 szBuffer[0] = TEXT('\0'); 164 } 165 } 166 else 167 { 168 TCHAR szFmt[64]; 169 170 if (!LoadString(hInstance, 171 IDS_FREQFMT, 172 szFmt, 173 sizeof(szFmt) / sizeof(szFmt[0]))) 174 { 175 szFmt[0] = TEXT('\0'); 176 } 177 178 _sntprintf(szBuffer, 179 sizeof(szBuffer) / sizeof(szBuffer[0]), 180 szFmt, 181 lpMode->dmDisplayFrequency); 182 } 183 184 i = (INT)SendDlgItemMessage(This->hwndDlg, 185 IDC_REFRESHRATE, 186 CB_ADDSTRING, 187 0, 188 (LPARAM)szBuffer); 189 if (i >= 0) 190 { 191 bAdded = TRUE; 192 193 SendDlgItemMessage(This->hwndDlg, 194 IDC_REFRESHRATE, 195 CB_SETITEMDATA, 196 (WPARAM)i, 197 (LPARAM)lpMode); 198 199 if (lpMode->dmDisplayFrequency == lpCurrentMode->dmDisplayFrequency) 200 { 201 SendDlgItemMessage(This->hwndDlg, 202 IDC_REFRESHRATE, 203 CB_SETCURSEL, 204 (WPARAM)i, 205 0); 206 } 207 } 208 } 209 210 } while (lpMode != NULL); 211 212 EnableWindow(GetDlgItem(This->hwndDlg, 213 IDS_MONITORSETTINGSGROUP), 214 bAdded); 215 EnableWindow(GetDlgItem(This->hwndDlg, 216 IDS_REFRESHRATELABEL), 217 bAdded); 218 EnableWindow(GetDlgItem(This->hwndDlg, 219 IDC_REFRESHRATE), 220 bAdded); 221 222 GetPruningSettings(This); 223 } 224 225 static VOID 226 InitMonitorDialog(PDESKMONITOR This) 227 { 228 PDESKMONINFO pmi, pminext, *pmilink; 229 DISPLAY_DEVICE dd; 230 BOOL bRet; 231 INT i; 232 DWORD dwIndex; 233 234 /* Free all allocated monitors */ 235 pmi = This->Monitors; 236 This->Monitors = NULL; 237 while (pmi != NULL) 238 { 239 pminext = pmi->Next; 240 LocalFree((HLOCAL)pmi); 241 pmi = pminext; 242 } 243 244 This->SelMonitor = NULL; 245 This->dwMonitorCount = 0; 246 247 if (This->lpDisplayDevice != NULL) 248 LocalFree((HLOCAL)This->lpDisplayDevice); 249 250 This->lpDisplayDevice = QueryDeskCplString(This->pdtobj, 251 RegisterClipboardFormat(DESK_EXT_DISPLAYDEVICE)); 252 253 if (This->DeskExtInterface != NULL) 254 { 255 if (This->lpDisplayDevice != NULL) 256 { 257 /* Enumerate all monitors */ 258 dwIndex = 0; 259 pmilink = &This->Monitors; 260 261 do 262 { 263 dd.cb = sizeof(dd); 264 bRet = EnumDisplayDevices(This->lpDisplayDevice, 265 dwIndex++, 266 &dd, 267 0); 268 if (bRet) 269 { 270 pmi = LocalAlloc(LMEM_FIXED, 271 sizeof(*pmi)); 272 if (pmi != NULL) 273 { 274 CopyMemory(&pmi->dd, 275 &dd, 276 sizeof(dd)); 277 pmi->Next = NULL; 278 *pmilink = pmi; 279 pmilink = &pmi->Next; 280 281 This->dwMonitorCount++; 282 } 283 } 284 } while (bRet); 285 } 286 287 This->lpDevModeOnInit = This->DeskExtInterface->GetCurrentMode(This->DeskExtInterface->Context); 288 } 289 else 290 This->lpDevModeOnInit = NULL; 291 292 This->lpSelDevMode = This->lpDevModeOnInit; 293 294 /* Setup the UI depending on how many monitors are attached */ 295 if (This->dwMonitorCount == 0) 296 { 297 LPTSTR lpMonitorName; 298 299 /* This is a fallback, let's hope that desk.cpl can provide us with a monitor name */ 300 lpMonitorName = QueryDeskCplString(This->pdtobj, 301 RegisterClipboardFormat(DESK_EXT_MONITORNAME)); 302 303 SetDlgItemText(This->hwndDlg, 304 IDC_MONITORNAME, 305 lpMonitorName); 306 307 if (lpMonitorName != NULL) 308 LocalFree((HLOCAL)lpMonitorName); 309 } 310 else if (This->dwMonitorCount == 1) 311 { 312 This->SelMonitor = This->Monitors; 313 SetDlgItemText(This->hwndDlg, 314 IDC_MONITORNAME, 315 This->Monitors->dd.DeviceString); 316 } 317 else 318 { 319 SendDlgItemMessage(This->hwndDlg, 320 IDC_MONITORLIST, 321 LB_RESETCONTENT, 322 0, 323 0); 324 325 pmi = This->Monitors; 326 while (pmi != NULL) 327 { 328 i = (INT)SendDlgItemMessage(This->hwndDlg, 329 IDC_MONITORLIST, 330 LB_ADDSTRING, 331 0, 332 (LPARAM)pmi->dd.DeviceString); 333 if (i >= 0) 334 { 335 SendDlgItemMessage(This->hwndDlg, 336 IDC_MONITORLIST, 337 LB_SETITEMDATA, 338 (WPARAM)i, 339 (LPARAM)pmi); 340 341 if (This->SelMonitor == NULL) 342 { 343 SendDlgItemMessage(This->hwndDlg, 344 IDC_MONITORLIST, 345 LB_SETCURSEL, 346 (WPARAM)i, 347 0); 348 349 This->SelMonitor = pmi; 350 } 351 } 352 353 pmi = pmi->Next; 354 } 355 } 356 357 /* Show/Hide controls */ 358 ShowWindow(GetDlgItem(This->hwndDlg, 359 IDC_MONITORNAME), 360 (This->dwMonitorCount <= 1 ? SW_SHOW : SW_HIDE)); 361 ShowWindow(GetDlgItem(This->hwndDlg, 362 IDC_MONITORLIST), 363 (This->dwMonitorCount > 1 ? SW_SHOW : SW_HIDE)); 364 365 UpdateRefreshFrequencyList(This); 366 UpdateMonitorSelection(This); 367 } 368 369 static VOID 370 UpdatePruningSelection(PDESKMONITOR This) 371 { 372 BOOL bPruningOn; 373 374 if (This->DeskExtInterface != NULL && This->bModesPruned && !This->bKeyIsReadOnly) 375 { 376 bPruningOn = IsDlgButtonChecked(This->hwndDlg, 377 IDC_PRUNINGCHECK) != BST_UNCHECKED; 378 379 if (bPruningOn != This->bPruningOn) 380 { 381 /* Tell desk.cpl to turn on/off pruning mode */ 382 This->bPruningOn = bPruningOn; 383 This->DeskExtInterface->SetPruningMode(This->DeskExtInterface->Context, 384 bPruningOn); 385 386 /* Fill the refresh rate combobox again, we now receive a filtered 387 or unfiltered device mode list from desk.cpl (depending on whether 388 pruning is active or not) */ 389 UpdateRefreshFrequencyList(This); 390 391 (void)PropSheet_Changed(GetParent(This->hwndDlg), 392 This->hwndDlg); 393 } 394 } 395 } 396 397 static VOID 398 UpdateRefreshRateSelection(PDESKMONITOR This) 399 { 400 PDEVMODEW lpCurrentDevMode; 401 INT i; 402 403 if (This->DeskExtInterface != NULL) 404 { 405 i = (INT)SendDlgItemMessage(This->hwndDlg, 406 IDC_REFRESHRATE, 407 CB_GETCURSEL, 408 0, 409 0); 410 if (i >= 0) 411 { 412 lpCurrentDevMode = This->lpSelDevMode; 413 This->lpSelDevMode = (PDEVMODEW)SendDlgItemMessage(This->hwndDlg, 414 IDC_REFRESHRATE, 415 CB_GETITEMDATA, 416 (WPARAM)i, 417 0); 418 419 if (This->lpSelDevMode != NULL && This->lpSelDevMode != lpCurrentDevMode) 420 { 421 This->DeskExtInterface->SetCurrentMode(This->DeskExtInterface->Context, 422 This->lpSelDevMode); 423 424 UpdateRefreshFrequencyList(This); 425 426 (void)PropSheet_Changed(GetParent(This->hwndDlg), 427 This->hwndDlg); 428 } 429 } 430 } 431 } 432 433 static LONG 434 ApplyMonitorChanges(PDESKMONITOR This) 435 { 436 LONG lChangeRet; 437 438 if (This->DeskExtInterface != NULL) 439 { 440 /* Change the display settings through desk.cpl */ 441 lChangeRet = DeskCplExtDisplaySaveSettings(This->DeskExtInterface, 442 This->hwndDlg); 443 if (lChangeRet == DISP_CHANGE_SUCCESSFUL) 444 { 445 /* Save the new mode */ 446 This->lpDevModeOnInit = This->DeskExtInterface->GetCurrentMode(This->DeskExtInterface->Context); 447 This->lpSelDevMode = This->lpDevModeOnInit; 448 return PSNRET_NOERROR; 449 } 450 else if (lChangeRet == DISP_CHANGE_RESTART) 451 { 452 /* Notify desk.cpl that the user needs to reboot */ 453 PropSheet_RestartWindows(GetParent(This->hwndDlg)); 454 return PSNRET_NOERROR; 455 } 456 } 457 458 InitMonitorDialog(This); 459 460 return PSNRET_INVALID_NOCHANGEPAGE; 461 } 462 463 static VOID 464 ResetMonitorChanges(PDESKMONITOR This) 465 { 466 if (This->DeskExtInterface != NULL && This->lpDevModeOnInit != NULL) 467 { 468 This->DeskExtInterface->SetCurrentMode(This->DeskExtInterface->Context, 469 This->lpDevModeOnInit); 470 } 471 } 472 473 static INT_PTR CALLBACK 474 MonitorDlgProc(HWND hwndDlg, 475 UINT uMsg, 476 WPARAM wParam, 477 LPARAM lParam) 478 { 479 PDESKMONITOR This; 480 INT_PTR Ret = 0; 481 482 if (uMsg != WM_INITDIALOG) 483 { 484 This = (PDESKMONITOR)GetWindowLongPtr(hwndDlg, 485 DWL_USER); 486 } 487 488 switch (uMsg) 489 { 490 case WM_INITDIALOG: 491 This = (PDESKMONITOR)((LPCPROPSHEETPAGE)lParam)->lParam; 492 This->hwndDlg = hwndDlg; 493 SetWindowLongPtr(hwndDlg, 494 DWL_USER, 495 (LONG_PTR)This); 496 497 InitMonitorDialog(This); 498 Ret = TRUE; 499 break; 500 501 case WM_COMMAND: 502 switch (LOWORD(wParam)) 503 { 504 case IDC_MONITORPROPERTIES: 505 ShowMonitorProperties(This); 506 break; 507 508 case IDC_MONITORLIST: 509 if (HIWORD(wParam) == LBN_SELCHANGE) 510 UpdateMonitorSelection(This); 511 break; 512 513 case IDC_PRUNINGCHECK: 514 if (HIWORD(wParam) == BN_CLICKED) 515 UpdatePruningSelection(This); 516 break; 517 518 case IDC_REFRESHRATE: 519 if (HIWORD(wParam) == CBN_SELCHANGE) 520 UpdateRefreshRateSelection(This); 521 break; 522 } 523 break; 524 525 case WM_NOTIFY: 526 { 527 NMHDR *nmh = (NMHDR *)lParam; 528 529 switch (nmh->code) 530 { 531 case PSN_APPLY: 532 { 533 SetWindowLong(hwndDlg, 534 DWL_MSGRESULT, 535 ApplyMonitorChanges(This)); 536 break; 537 } 538 539 case PSN_RESET: 540 ResetMonitorChanges(This); 541 break; 542 543 case PSN_SETACTIVE: 544 UpdateRefreshFrequencyList(This); 545 break; 546 } 547 break; 548 } 549 } 550 551 return Ret; 552 } 553 554 static VOID 555 IDeskMonitor_Destroy(PDESKMONITOR This) 556 { 557 PDESKMONINFO pmi, pminext; 558 559 if (This->pdtobj != NULL) 560 { 561 IDataObject_Release(This->pdtobj); 562 This->pdtobj = NULL; 563 } 564 565 if (This->DeskExtInterface != NULL) 566 { 567 LocalFree((HLOCAL)This->DeskExtInterface); 568 This->DeskExtInterface = NULL; 569 } 570 571 if (This->lpDisplayDevice != NULL) 572 { 573 LocalFree((HLOCAL)This->lpDisplayDevice); 574 This->lpDisplayDevice = NULL; 575 } 576 577 /* Free all monitors */ 578 pmi = This->Monitors; 579 This->Monitors = NULL; 580 while (pmi != NULL) 581 { 582 pminext = pmi->Next; 583 LocalFree((HLOCAL)pmi); 584 pmi = pminext; 585 } 586 } 587 588 ULONG 589 IDeskMonitor_AddRef(PDESKMONITOR This) 590 { 591 ULONG ret; 592 593 ret = InterlockedIncrement((PLONG)&This->ref); 594 if (ret == 1) 595 InterlockedIncrement(&dll_refs); 596 597 return ret; 598 } 599 600 ULONG 601 IDeskMonitor_Release(PDESKMONITOR This) 602 { 603 ULONG ret; 604 605 ret = InterlockedDecrement((PLONG)&This->ref); 606 if (ret == 0) 607 { 608 IDeskMonitor_Destroy(This); 609 InterlockedDecrement(&dll_refs); 610 611 HeapFree(GetProcessHeap(), 612 0, 613 This); 614 } 615 616 return ret; 617 } 618 619 HRESULT STDMETHODCALLTYPE 620 IDeskMonitor_QueryInterface(PDESKMONITOR This, 621 REFIID iid, 622 PVOID *pvObject) 623 { 624 *pvObject = NULL; 625 626 if (IsEqualIID(iid, 627 &IID_IShellPropSheetExt) || 628 IsEqualIID(iid, 629 &IID_IUnknown)) 630 { 631 *pvObject = impl_to_interface(This, IShellPropSheetExt); 632 } 633 else if (IsEqualIID(iid, 634 &IID_IShellExtInit)) 635 { 636 *pvObject = impl_to_interface(This, IShellExtInit); 637 } 638 else if (IsEqualIID(iid, 639 &IID_IClassFactory)) 640 { 641 *pvObject = impl_to_interface(This, IClassFactory); 642 } 643 else 644 { 645 DPRINT1("IDeskMonitor::QueryInterface(%p,%p): E_NOINTERFACE\n", iid, pvObject); 646 return E_NOINTERFACE; 647 } 648 649 IDeskMonitor_AddRef(This); 650 return S_OK; 651 } 652 653 HRESULT 654 IDeskMonitor_Initialize(PDESKMONITOR This, 655 LPCITEMIDLIST pidlFolder, 656 IDataObject *pdtobj, 657 HKEY hkeyProgID) 658 { 659 DPRINT1("IDeskMonitor::Initialize(%p,%p,%p)\n", pidlFolder, pdtobj, hkeyProgID); 660 661 if (pdtobj != NULL) 662 { 663 IDataObject_AddRef(pdtobj); 664 This->pdtobj = pdtobj; 665 666 /* Get a copy of the desk.cpl extension interface */ 667 This->DeskExtInterface = QueryDeskCplExtInterface(This->pdtobj); 668 if (This->DeskExtInterface != NULL) 669 return S_OK; 670 } 671 672 return S_FALSE; 673 } 674 675 HRESULT 676 IDeskMonitor_AddPages(PDESKMONITOR This, 677 LPFNADDPROPSHEETPAGE pfnAddPage, 678 LPARAM lParam) 679 { 680 HPROPSHEETPAGE hpsp; 681 PROPSHEETPAGE psp; 682 683 DPRINT1("IDeskMonitor::AddPages(%p,%p)\n", pfnAddPage, lParam); 684 685 psp.dwSize = sizeof(psp); 686 psp.dwFlags = PSP_DEFAULT; 687 psp.hInstance = hInstance; 688 psp.pszTemplate = MAKEINTRESOURCE(IDD_MONITOR); 689 psp.pfnDlgProc = MonitorDlgProc; 690 psp.lParam = (LPARAM)This; 691 692 hpsp = CreatePropertySheetPage(&psp); 693 if (hpsp != NULL && pfnAddPage(hpsp, lParam)) 694 return S_OK; 695 696 return S_FALSE; 697 } 698 699 HRESULT 700 IDeskMonitor_ReplacePage(PDESKMONITOR This, 701 EXPPS uPageID, 702 LPFNADDPROPSHEETPAGE pfnReplacePage, 703 LPARAM lParam) 704 { 705 DPRINT1("IDeskMonitor::ReplacePage(%u,%p,%p)\n", uPageID, pfnReplacePage, lParam); 706 return E_NOTIMPL; 707 } 708 709 HRESULT 710 IDeskMonitor_Constructor(REFIID riid, 711 LPVOID *ppv) 712 { 713 PDESKMONITOR This; 714 HRESULT hRet = E_OUTOFMEMORY; 715 716 DPRINT1("IDeskMonitor::Constructor(%p,%p)\n", riid, ppv); 717 718 This = HeapAlloc(GetProcessHeap(), 719 0, 720 sizeof(*This)); 721 if (This != NULL) 722 { 723 ZeroMemory(This, 724 sizeof(*This)); 725 726 IDeskMonitor_InitIface(This); 727 728 hRet = IDeskMonitor_QueryInterface(This, 729 riid, 730 ppv); 731 if (!SUCCEEDED(hRet)) 732 IDeskMonitor_Release(This); 733 } 734 735 return hRet; 736 } 737 738 BOOL STDCALL 739 DllMain(HINSTANCE hinstDLL, 740 DWORD dwReason, 741 LPVOID lpvReserved) 742 { 743 switch (dwReason) 744 { 745 case DLL_PROCESS_ATTACH: 746 hInstance = hinstDLL; 747 DisableThreadLibraryCalls(hInstance); 748 break; 749 } 750 751 return TRUE; 752 } 753