1 /* 2 * Shell Desktop 3 * 4 * Copyright 2008 Thomas Bluemel 5 * Copyright 2020 Katayama Hirofumi MZ 6 * 7 * This library is free software; you can redistribute it and/or 8 * modify it under the terms of the GNU Lesser General Public 9 * License as published by the Free Software Foundation; either 10 * version 2.1 of the License, or (at your option) any later version. 11 * 12 * This library is distributed in the hope that it will be useful, 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15 * Lesser General Public License for more details. 16 * 17 * You should have received a copy of the GNU Lesser General Public 18 * License along with this library; if not, write to the Free Software 19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 20 */ 21 22 #include "shelldesktop.h" 23 24 // Support for multiple monitors is disabled till LVM_SETWORKAREAS gets implemented 25 #ifdef MULTIMONITOR_SUPPORT 26 #include <atlcoll.h> 27 #endif 28 29 WINE_DEFAULT_DEBUG_CHANNEL(desktop); 30 31 static const WCHAR szProgmanClassName[] = L"Progman"; 32 static const WCHAR szProgmanWindowName[] = L"Program Manager"; 33 34 class CDesktopBrowser : 35 public CWindowImpl<CDesktopBrowser, CWindow, CFrameWinTraits>, 36 public CComObjectRootEx<CComMultiThreadModelNoCS>, 37 public IShellBrowser, 38 public IServiceProvider 39 { 40 private: 41 HACCEL m_hAccel; 42 HWND m_hWndShellView; 43 CComPtr<IShellDesktopTray> m_Tray; 44 CComPtr<IShellView> m_ShellView; 45 46 CComPtr<IOleWindow> m_ChangeNotifyServer; 47 HWND m_hwndChangeNotifyServer; 48 49 LRESULT _NotifyTray(UINT uMsg, WPARAM wParam, LPARAM lParam); 50 HRESULT _Resize(); 51 52 public: 53 CDesktopBrowser(); 54 ~CDesktopBrowser(); 55 HRESULT Initialize(IShellDesktopTray *ShellDeskx); 56 57 // *** IOleWindow methods *** 58 virtual HRESULT STDMETHODCALLTYPE GetWindow(HWND *lphwnd); 59 virtual HRESULT STDMETHODCALLTYPE ContextSensitiveHelp(BOOL fEnterMode); 60 61 // *** IShellBrowser methods *** 62 virtual HRESULT STDMETHODCALLTYPE InsertMenusSB(HMENU hmenuShared, LPOLEMENUGROUPWIDTHS lpMenuWidths); 63 virtual HRESULT STDMETHODCALLTYPE SetMenuSB(HMENU hmenuShared, HOLEMENU holemenuRes, HWND hwndActiveObject); 64 virtual HRESULT STDMETHODCALLTYPE RemoveMenusSB(HMENU hmenuShared); 65 virtual HRESULT STDMETHODCALLTYPE SetStatusTextSB(LPCOLESTR pszStatusText); 66 virtual HRESULT STDMETHODCALLTYPE EnableModelessSB(BOOL fEnable); 67 virtual HRESULT STDMETHODCALLTYPE TranslateAcceleratorSB(MSG *pmsg, WORD wID); 68 virtual HRESULT STDMETHODCALLTYPE BrowseObject(LPCITEMIDLIST pidl, UINT wFlags); 69 virtual HRESULT STDMETHODCALLTYPE GetViewStateStream(DWORD grfMode, IStream **ppStrm); 70 virtual HRESULT STDMETHODCALLTYPE GetControlWindow(UINT id, HWND *lphwnd); 71 virtual HRESULT STDMETHODCALLTYPE SendControlMsg(UINT id, UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT *pret); 72 virtual HRESULT STDMETHODCALLTYPE QueryActiveShellView(struct IShellView **ppshv); 73 virtual HRESULT STDMETHODCALLTYPE OnViewWindowActive(struct IShellView *ppshv); 74 virtual HRESULT STDMETHODCALLTYPE SetToolbarItems(LPTBBUTTON lpButtons, UINT nButtons, UINT uFlags); 75 76 // *** IServiceProvider methods *** 77 virtual HRESULT STDMETHODCALLTYPE QueryService(REFGUID guidService, REFIID riid, void **ppvObject); 78 79 // message handlers 80 LRESULT OnEraseBkgnd(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 81 LRESULT OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 82 LRESULT OnSettingChange(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 83 LRESULT OnClose(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 84 LRESULT OnOpenNewWindow(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 85 LRESULT OnCommand(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 86 LRESULT OnSetFocus(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 87 LRESULT OnGetChangeNotifyServer(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled); 88 89 DECLARE_WND_CLASS_EX(szProgmanClassName, CS_DBLCLKS, COLOR_DESKTOP) 90 91 BEGIN_MSG_MAP(CBaseBar) 92 MESSAGE_HANDLER(WM_ERASEBKGND, OnEraseBkgnd) 93 MESSAGE_HANDLER(WM_SIZE, OnSize) 94 MESSAGE_HANDLER(WM_SYSCOLORCHANGE, OnSettingChange) 95 MESSAGE_HANDLER(WM_SETTINGCHANGE, OnSettingChange) 96 MESSAGE_HANDLER(WM_CLOSE, OnClose) 97 MESSAGE_HANDLER(WM_EXPLORER_OPEN_NEW_WINDOW, OnOpenNewWindow) 98 MESSAGE_HANDLER(WM_COMMAND, OnCommand) 99 MESSAGE_HANDLER(WM_SETFOCUS, OnSetFocus) 100 MESSAGE_HANDLER(WM_DESKTOP_GET_CNOTIFY_SERVER, OnGetChangeNotifyServer) 101 END_MSG_MAP() 102 103 BEGIN_COM_MAP(CDesktopBrowser) 104 COM_INTERFACE_ENTRY_IID(IID_IOleWindow, IOleWindow) 105 COM_INTERFACE_ENTRY_IID(IID_IShellBrowser, IShellBrowser) 106 COM_INTERFACE_ENTRY_IID(IID_IServiceProvider, IServiceProvider) 107 END_COM_MAP() 108 }; 109 110 CDesktopBrowser::CDesktopBrowser(): 111 m_hAccel(NULL), 112 m_hWndShellView(NULL), 113 m_hwndChangeNotifyServer(NULL) 114 { 115 } 116 117 CDesktopBrowser::~CDesktopBrowser() 118 { 119 if (m_ShellView.p != NULL && m_hWndShellView != NULL) 120 { 121 m_ShellView->DestroyViewWindow(); 122 } 123 124 if (m_hwndChangeNotifyServer) 125 { 126 ::DestroyWindow(m_hwndChangeNotifyServer); 127 } 128 } 129 130 #ifdef MULTIMONITOR_SUPPORT 131 BOOL CALLBACK MonitorEnumProc( 132 _In_ HMONITOR hMonitor, 133 _In_ HDC hdcMonitor, 134 _In_ LPRECT lprcMonitor, 135 _In_ LPARAM dwData 136 ) 137 { 138 CAtlList<RECT> *list = (CAtlList<RECT>*)dwData; 139 MONITORINFO MonitorInfo; 140 MonitorInfo.cbSize = sizeof(MonitorInfo); 141 if (::GetMonitorInfoW(hMonitor, &MonitorInfo)) 142 { 143 list->AddTail(MonitorInfo.rcWork); 144 } 145 146 return TRUE; 147 } 148 #endif 149 150 HRESULT CDesktopBrowser::_Resize() 151 { 152 RECT rcNewSize; 153 154 #ifdef MULTIMONITOR_SUPPORT 155 156 UINT cMonitors = GetSystemMetrics(SM_CMONITORS); 157 if (cMonitors == 1) 158 { 159 SystemParametersInfoW(SPI_GETWORKAREA, 0, &rcNewSize, 0); 160 } 161 else 162 { 163 SetRect(&rcNewSize, 164 GetSystemMetrics(SM_XVIRTUALSCREEN), 165 GetSystemMetrics(SM_YVIRTUALSCREEN), 166 GetSystemMetrics(SM_XVIRTUALSCREEN) + GetSystemMetrics(SM_CXVIRTUALSCREEN), 167 GetSystemMetrics(SM_YVIRTUALSCREEN) + GetSystemMetrics(SM_CYVIRTUALSCREEN)); 168 } 169 170 ::MoveWindow(m_hWnd, rcNewSize.left, rcNewSize.top, rcNewSize.right - rcNewSize.left, rcNewSize.bottom - rcNewSize.top, TRUE); 171 ::MoveWindow(m_hWndShellView, 0, 0, rcNewSize.right - rcNewSize.left, rcNewSize.bottom - rcNewSize.top, TRUE); 172 173 if (cMonitors != 1) 174 { 175 CAtlList<RECT> list; 176 EnumDisplayMonitors(NULL, NULL, MonitorEnumProc, (LPARAM)&list); 177 RECT* prcWorkAreas = new RECT[list.GetCount()]; 178 int i = 0; 179 for (POSITION it = list.GetHeadPosition(); it; list.GetNext(it)) 180 prcWorkAreas[i++] = list.GetAt(it); 181 182 HWND hwndListView = FindWindowExW(m_hWndShellView, NULL, WC_LISTVIEW, NULL); 183 184 ::SendMessageW(hwndListView, LVM_SETWORKAREAS , i, (LPARAM)prcWorkAreas); 185 } 186 187 #else 188 SystemParametersInfoW(SPI_GETWORKAREA, 0, &rcNewSize, 0); 189 ::MoveWindow(m_hWnd, rcNewSize.left, rcNewSize.top, rcNewSize.right - rcNewSize.left, rcNewSize.bottom - rcNewSize.top, TRUE); 190 ::MoveWindow(m_hWndShellView, 0, 0, rcNewSize.right - rcNewSize.left, rcNewSize.bottom - rcNewSize.top, TRUE); 191 192 #endif 193 return S_OK; 194 } 195 196 HRESULT CDesktopBrowser::Initialize(IShellDesktopTray *ShellDesk) 197 { 198 CComPtr<IShellFolder> psfDesktop; 199 HRESULT hRet; 200 hRet = SHGetDesktopFolder(&psfDesktop); 201 if (FAILED_UNEXPECTEDLY(hRet)) 202 return hRet; 203 204 m_Tray = ShellDesk; 205 206 Create(NULL, NULL, szProgmanWindowName, WS_POPUP | WS_CLIPSIBLINGS | WS_CLIPCHILDREN, WS_EX_TOOLWINDOW); 207 if (!m_hWnd) 208 return E_FAIL; 209 210 CSFV csfv = {sizeof(CSFV), psfDesktop}; 211 hRet = SHCreateShellFolderViewEx(&csfv, &m_ShellView); 212 if (FAILED_UNEXPECTEDLY(hRet)) 213 return hRet; 214 215 m_Tray->RegisterDesktopWindow(m_hWnd); 216 if (FAILED_UNEXPECTEDLY(hRet)) 217 return hRet; 218 219 FOLDERSETTINGS fs; 220 RECT rcShellView = {0,0,0,0}; 221 fs.ViewMode = FVM_ICON; 222 fs.fFlags = FWF_DESKTOP | FWF_NOCLIENTEDGE | FWF_NOSCROLL | FWF_TRANSPARENT; 223 hRet = m_ShellView->CreateViewWindow(NULL, &fs, (IShellBrowser *)this, &rcShellView, &m_hWndShellView); 224 if (FAILED_UNEXPECTEDLY(hRet)) 225 return hRet; 226 227 _Resize(); 228 229 HWND hwndListView = FindWindowExW(m_hWndShellView, NULL, WC_LISTVIEW, NULL); 230 231 m_hAccel = LoadAcceleratorsW(shell32_hInstance, MAKEINTRESOURCEW(IDA_DESKBROWSER)); 232 233 #if 1 234 /* A Windows8+ specific hack */ 235 ::ShowWindow(m_hWndShellView, SW_SHOW); 236 ::ShowWindow(hwndListView, SW_SHOW); 237 #endif 238 ShowWindow(SW_SHOW); 239 UpdateWindow(); 240 241 return hRet; 242 } 243 244 HRESULT STDMETHODCALLTYPE CDesktopBrowser::GetWindow(HWND *lphwnd) 245 { 246 if (lphwnd == NULL) 247 return E_POINTER; 248 *lphwnd = m_hWnd; 249 return S_OK; 250 } 251 252 HRESULT STDMETHODCALLTYPE CDesktopBrowser::ContextSensitiveHelp(BOOL fEnterMode) 253 { 254 return E_NOTIMPL; 255 } 256 257 HRESULT STDMETHODCALLTYPE CDesktopBrowser::InsertMenusSB(HMENU hmenuShared, LPOLEMENUGROUPWIDTHS lpMenuWidths) 258 { 259 return E_NOTIMPL; 260 } 261 262 HRESULT STDMETHODCALLTYPE CDesktopBrowser::SetMenuSB(HMENU hmenuShared, HOLEMENU holemenuRes, HWND hwndActiveObject) 263 { 264 return E_NOTIMPL; 265 } 266 267 HRESULT STDMETHODCALLTYPE CDesktopBrowser::RemoveMenusSB(HMENU hmenuShared) 268 { 269 return E_NOTIMPL; 270 } 271 272 HRESULT STDMETHODCALLTYPE CDesktopBrowser::SetStatusTextSB(LPCOLESTR lpszStatusText) 273 { 274 return E_NOTIMPL; 275 } 276 277 HRESULT STDMETHODCALLTYPE CDesktopBrowser::EnableModelessSB(BOOL fEnable) 278 { 279 return E_NOTIMPL; 280 } 281 282 HRESULT STDMETHODCALLTYPE CDesktopBrowser::TranslateAcceleratorSB(LPMSG lpmsg, WORD wID) 283 { 284 if (!::TranslateAcceleratorW(m_hWnd, m_hAccel, lpmsg)) 285 return S_FALSE; 286 return S_OK; 287 } 288 289 HRESULT STDMETHODCALLTYPE CDesktopBrowser::BrowseObject(LPCITEMIDLIST pidl, UINT wFlags) 290 { 291 /* 292 * We should use IShellWindows interface here in order to attempt to 293 * find an open shell window that shows the requested pidl and activate it 294 */ 295 296 DWORD dwFlags = ((wFlags & SBSP_EXPLOREMODE) != 0) ? SH_EXPLORER_CMDLINE_FLAG_E : 0; 297 return SHOpenNewFrame(ILClone(pidl), NULL, 0, dwFlags); 298 } 299 300 HRESULT STDMETHODCALLTYPE CDesktopBrowser::GetViewStateStream(DWORD grfMode, IStream **ppStrm) 301 { 302 return E_NOTIMPL; 303 } 304 305 HRESULT STDMETHODCALLTYPE CDesktopBrowser::GetControlWindow(UINT id, HWND *lphwnd) 306 { 307 if (lphwnd == NULL) 308 return E_POINTER; 309 return E_NOTIMPL; 310 } 311 312 HRESULT STDMETHODCALLTYPE CDesktopBrowser::SendControlMsg(UINT id, UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT *pret) 313 { 314 if (pret == NULL) 315 return E_POINTER; 316 return E_NOTIMPL; 317 } 318 319 HRESULT STDMETHODCALLTYPE CDesktopBrowser::QueryActiveShellView(IShellView **ppshv) 320 { 321 if (ppshv == NULL) 322 return E_POINTER; 323 *ppshv = m_ShellView; 324 if (*ppshv != NULL) 325 (*ppshv)->AddRef(); 326 327 return S_OK; 328 } 329 330 HRESULT STDMETHODCALLTYPE CDesktopBrowser::OnViewWindowActive(IShellView *ppshv) 331 { 332 return E_NOTIMPL; 333 } 334 335 HRESULT STDMETHODCALLTYPE CDesktopBrowser::SetToolbarItems(LPTBBUTTON lpButtons, UINT nButtons, UINT uFlags) 336 { 337 return E_NOTIMPL; 338 } 339 340 HRESULT STDMETHODCALLTYPE CDesktopBrowser::QueryService(REFGUID guidService, REFIID riid, PVOID *ppv) 341 { 342 /* FIXME - handle guidService */ 343 return QueryInterface(riid, ppv); 344 } 345 346 LRESULT CDesktopBrowser::_NotifyTray(UINT uMsg, WPARAM wParam, LPARAM lParam) 347 { 348 HWND hWndTray; 349 HRESULT hRet; 350 351 hRet = m_Tray->GetTrayWindow(&hWndTray); 352 if (SUCCEEDED(hRet)) 353 ::PostMessageW(hWndTray, uMsg, wParam, lParam); 354 355 return 0; 356 } 357 358 LRESULT CDesktopBrowser::OnCommand(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 359 { 360 switch (LOWORD(wParam)) 361 { 362 case FCIDM_DESKBROWSER_CLOSE: 363 return _NotifyTray(TWM_DOEXITWINDOWS, 0, 0); 364 case FCIDM_DESKBROWSER_FOCUS: 365 if (GetKeyState(VK_SHIFT)) 366 return _NotifyTray(TWM_CYCLEFOCUS, 1, 0xFFFFFFFF); 367 else 368 return _NotifyTray(TWM_CYCLEFOCUS, 1, 1); 369 case FCIDM_DESKBROWSER_SEARCH: 370 SHFindFiles(NULL, NULL); 371 break; 372 case FCIDM_DESKBROWSER_REFRESH: 373 if (m_ShellView) 374 m_ShellView->Refresh(); 375 break; 376 } 377 378 return 0; 379 } 380 381 382 LRESULT CDesktopBrowser::OnEraseBkgnd(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 383 { 384 return (LRESULT)PaintDesktop((HDC)wParam); 385 } 386 387 LRESULT CDesktopBrowser::OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 388 { 389 if (wParam == SIZE_MINIMIZED) 390 { 391 /* Hey, we're the desktop!!! */ 392 ::ShowWindow(m_hWnd, SW_RESTORE); 393 } 394 395 ::InvalidateRect(m_hWndShellView, NULL, TRUE); 396 397 return 0; 398 } 399 400 LRESULT CDesktopBrowser::OnSettingChange(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 401 { 402 if (uMsg == WM_SETTINGCHANGE /* == WM_WININICHANGE */ && 403 lstrcmpiW((LPCWSTR)lParam, L"Environment") == 0) 404 { 405 LPVOID lpEnvironment; 406 RegenerateUserEnvironment(&lpEnvironment, TRUE); 407 } 408 409 if (m_hWndShellView) 410 { 411 /* Forward the message */ 412 ::SendMessageW(m_hWndShellView, uMsg, wParam, lParam); 413 } 414 415 if (uMsg == WM_SETTINGCHANGE && wParam == SPI_SETWORKAREA && m_hWndShellView != NULL) 416 { 417 _Resize(); 418 } 419 420 return 0; 421 } 422 423 LRESULT CDesktopBrowser::OnClose(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 424 { 425 return _NotifyTray(TWM_DOEXITWINDOWS, 0, 0); 426 } 427 428 LRESULT CDesktopBrowser::OnOpenNewWindow(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 429 { 430 TRACE("Proxy Desktop message 1035 received.\n"); 431 SHOnCWMCommandLine((HANDLE)lParam); 432 return 0; 433 } 434 435 LRESULT CDesktopBrowser::OnSetFocus(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 436 { 437 ::SetFocus(m_hWndShellView); 438 return 0; 439 } 440 441 // Message WM_DESKTOP_GET_CNOTIFY_SERVER: Get or create the change notification server. 442 // wParam: BOOL bCreate; The flag whether it creates or not. 443 // lParam: Ignored. 444 // return: The window handle of the server window. 445 LRESULT CDesktopBrowser::OnGetChangeNotifyServer(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) 446 { 447 BOOL bCreate = (BOOL)wParam; 448 if (bCreate && !::IsWindow(m_hwndChangeNotifyServer)) 449 { 450 HRESULT hres = CChangeNotifyServer_CreateInstance(IID_PPV_ARG(IOleWindow, &m_ChangeNotifyServer)); 451 if (FAILED_UNEXPECTEDLY(hres)) 452 return NULL; 453 454 hres = m_ChangeNotifyServer->GetWindow(&m_hwndChangeNotifyServer); 455 if (FAILED_UNEXPECTEDLY(hres)) 456 return NULL; 457 } 458 return (LRESULT)m_hwndChangeNotifyServer; 459 } 460 461 HRESULT CDesktopBrowser_CreateInstance(IShellDesktopTray *Tray, REFIID riid, void **ppv) 462 { 463 return ShellObjectCreatorInit<CDesktopBrowser, IShellDesktopTray*>(Tray, riid, ppv); 464 } 465 466 /************************************************************************* 467 * SHCreateDesktop [SHELL32.200] 468 * 469 */ 470 HANDLE WINAPI SHCreateDesktop(IShellDesktopTray *Tray) 471 { 472 if (Tray == NULL) 473 { 474 SetLastError(ERROR_INVALID_PARAMETER); 475 return NULL; 476 } 477 478 CComPtr<IShellBrowser> Browser; 479 HRESULT hr = CDesktopBrowser_CreateInstance(Tray, IID_PPV_ARG(IShellBrowser, &Browser)); 480 if (FAILED_UNEXPECTEDLY(hr)) 481 return NULL; 482 483 return static_cast<HANDLE>(Browser.Detach()); 484 } 485 486 /************************************************************************* 487 * SHCreateDesktop [SHELL32.201] 488 * 489 */ 490 BOOL WINAPI SHDesktopMessageLoop(HANDLE hDesktop) 491 { 492 if (hDesktop == NULL) 493 { 494 SetLastError(ERROR_INVALID_PARAMETER); 495 return FALSE; 496 } 497 498 MSG Msg; 499 BOOL bRet; 500 501 CComPtr<IShellBrowser> browser; 502 CComPtr<IShellView> shellView; 503 504 browser.Attach(static_cast<IShellBrowser*>(hDesktop)); 505 HRESULT hr = browser->QueryActiveShellView(&shellView); 506 if (FAILED_UNEXPECTEDLY(hr)) 507 return FALSE; 508 509 while ((bRet = ::GetMessageW(&Msg, NULL, 0, 0)) != 0) 510 { 511 if (bRet != -1) 512 { 513 if (shellView->TranslateAcceleratorW(&Msg) != S_OK) 514 { 515 ::TranslateMessage(&Msg); 516 ::DispatchMessageW(&Msg); 517 } 518 } 519 } 520 521 return TRUE; 522 } 523