4 * Copyright 2014 Giannis Adamopoulos
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.
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.
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
22 #include "CMergedFolder.h"
24 WINE_DEFAULT_DEBUG_CHANNEL(CStartMenu
);
26 // TODO: declare these GUIDs and interfaces in the right place (whatever that may be)
27 IID IID_IAugmentedShellFolder
= { 0x91EA3F8C, 0xC99B, 0x11D0, { 0x98, 0x15, 0x00, 0xC0, 0x4F, 0xD9, 0x19, 0x72 } };
28 IID IID_IAugmentedShellFolder2
= { 0x8DB3B3F4, 0x6CFE, 0x11D1, { 0x8A, 0xE9, 0x00, 0xC0, 0x4F, 0xD9, 0x18, 0xD0 } };
29 CLSID CLSID_MergedFolder
= { 0x26FDC864, 0xBE88, 0x46E7, { 0x92, 0x35, 0x03, 0x2D, 0x8E, 0xA5, 0x16, 0x2E } };
31 //#define TEST_TRACKPOPUPMENU_SUBMENUS
34 /* NOTE: The following constants *MUST NOT* be changed because
35 they're hardcoded and need to be the exact values
36 in order to get the start menu to work! */
38 #define IDM_LOGOFF 402
39 #define IDM_UNDOCKCOMPUTER 410
40 #define IDM_TASKBARANDSTARTMENU 413
41 #define IDM_LASTSTARTMENU_SEPARATOR 450
42 #define IDM_DOCUMENTS 501
43 #define IDM_HELPANDSUPPORT 503
44 #define IDM_PROGRAMS 504
45 #define IDM_CONTROLPANEL 505
46 #define IDM_SHUTDOWN 506
47 #define IDM_FAVORITES 507
48 #define IDM_SETTINGS 508
49 #define IDM_PRINTERSANDFAXES 510
50 #define IDM_SEARCH 520
51 #define IDM_SYNCHRONIZE 553
52 #define IDM_NETWORKCONNECTIONS 557
53 #define IDM_DISCONNECT 5000
54 #define IDM_SECURITY 5001
58 * 1. append the start menu contents from all users
59 * 2. implement the context menu for start menu entries (programs, control panel, network connetions, printers)
60 * 3. filter out programs folder from the shell folder part of the start menu
61 * 4. showing the programs start menu is SLOW compared to windows. this needs some investigation
64 class CShellMenuCallback
:
65 public CComObjectRootEx
<CComMultiThreadModelNoCS
>,
66 public IShellMenuCallback
71 CComPtr
<IShellMenu
> m_pShellMenu
;
72 CComPtr
<IBandSite
> m_pBandSite
;
73 CComPtr
<IDeskBar
> m_pDeskBar
;
74 CComPtr
<ITrayPriv
> m_pTrayPriv
;
75 CComPtr
<IShellFolder
> m_psfPrograms
;
77 LPITEMIDLIST m_pidlPrograms
;
87 hr
= IUnknown_GetSite(m_pDeskBar
, IID_PPV_ARG(ITrayPriv
, &m_pTrayPriv
));
88 hr
= IUnknown_GetWindow(m_pTrayPriv
, &m_hwndTray
);
89 hr
= m_pTrayPriv
->AppendMenuW(&hmenu
);
90 #ifndef TEST_TRACKPOPUPMENU_SUBMENUS
91 hr
= m_pShellMenu
->SetMenu(hmenu
, NULL
, SMSET_BOTTOM
);
93 hr
= m_pShellMenu
->SetMenu(hmenu
, m_hwndTray
, SMSET_BOTTOM
);
99 HRESULT
OnGetInfo(LPSMDATA psmd
, SMINFO
*psminfo
)
105 // Smaller "24x24" icons used for the start menu
106 // The bitmaps are still 32x32, but the image is centered
107 case IDM_FAVORITES
: iconIndex
= -322; break;
108 case IDM_SEARCH
: iconIndex
= -323; break;
109 case IDM_HELPANDSUPPORT
: iconIndex
= -324; break;
110 case IDM_LOGOFF
: iconIndex
= -325; break;
111 case IDM_PROGRAMS
: iconIndex
= -326; break;
112 case IDM_DOCUMENTS
: iconIndex
= -327; break;
113 case IDM_RUN
: iconIndex
= -328; break;
114 case IDM_SHUTDOWN
: iconIndex
= -329; break;
115 case IDM_SETTINGS
: iconIndex
= -330; break;
117 case IDM_CONTROLPANEL
: iconIndex
= -22; break;
118 case IDM_NETWORKCONNECTIONS
: iconIndex
= -257; break;
119 case IDM_PRINTERSANDFAXES
: iconIndex
= -138; break;
120 case IDM_TASKBARANDSTARTMENU
: iconIndex
= -40; break;
121 //case IDM_SECURITY: iconIndex = -21; break;
122 //case IDM_SYNCHRONIZE: iconIndex = -21; break;
123 //case IDM_DISCONNECT: iconIndex = -21; break;
124 //case IDM_UNDOCKCOMPUTER: iconIndex = -21; break;
131 if ((psminfo
->dwMask
& SMIM_TYPE
) != 0)
132 psminfo
->dwType
= SMIT_STRING
;
133 if ((psminfo
->dwMask
& SMIM_ICON
) != 0)
134 psminfo
->iIcon
= Shell_GetCachedImageIndex(L
"shell32.dll", iconIndex
, FALSE
);
135 if ((psminfo
->dwMask
& SMIM_FLAGS
) != 0)
136 psminfo
->dwFlags
|= SMIF_ICON
;
137 #ifdef TEST_TRACKPOPUPMENU_SUBMENUS
138 if ((psminfo
->dwMask
& SMIM_FLAGS
) != 0)
139 psminfo
->dwFlags
|= SMIF_TRACKPOPUP
;
144 if ((psminfo
->dwMask
& SMIM_TYPE
) != 0)
145 psminfo
->dwType
= SMIT_SEPARATOR
;
150 HRESULT
OnGetSubMenu(LPSMDATA psmd
, REFIID iid
, void ** pv
)
154 IShellMenu
*pShellMenu
;
156 #if USE_SYSTEM_MENUBAND
157 hr
= CoCreateInstance(CLSID_MenuBand
,
159 CLSCTX_INPROC_SERVER
,
160 IID_PPV_ARG(IShellMenu
, &pShellMenu
));
162 hr
= CMenuBand_Constructor(IID_PPV_ARG(IShellMenu
, &pShellMenu
));
164 if (FAILED_UNEXPECTEDLY(hr
))
167 hr
= pShellMenu
->Initialize(this, 0, ANCESTORDEFAULT
, SMINIT_VERTICAL
);
171 case IDM_PROGRAMS
: csidl
= CSIDL_PROGRAMS
; break;
172 case IDM_FAVORITES
: csidl
= CSIDL_FAVORITES
; break;
173 case IDM_DOCUMENTS
: csidl
= CSIDL_RECENT
; break;
178 IShellFolder
*psfStartMenu
;
180 if (csidl
== CSIDL_PROGRAMS
&& m_psfPrograms
)
182 psfStartMenu
= m_psfPrograms
;
186 LPITEMIDLIST pidlStartMenu
;
187 IShellFolder
*psfDestop
;
188 hr
= SHGetFolderLocation(NULL
, csidl
, 0, 0, &pidlStartMenu
);
189 hr
= SHGetDesktopFolder(&psfDestop
);
190 hr
= psfDestop
->BindToObject(pidlStartMenu
, NULL
, IID_PPV_ARG(IShellFolder
, &psfStartMenu
));
193 hr
= pShellMenu
->SetShellFolder(psfStartMenu
, NULL
, NULL
, 0);
198 mii
.cbSize
= sizeof(mii
);
199 mii
.fMask
= MIIM_SUBMENU
;
200 if (GetMenuItemInfoW(psmd
->hmenu
, psmd
->uId
, FALSE
, &mii
))
202 hr
= pShellMenu
->SetMenu(mii
.hSubMenu
, NULL
, SMSET_BOTTOM
);
205 return pShellMenu
->QueryInterface(iid
, pv
);
208 HRESULT
OnGetContextMenu(LPSMDATA psmd
, REFIID iid
, void ** pv
)
210 if (psmd
->uId
== IDM_PROGRAMS
||
211 psmd
->uId
== IDM_CONTROLPANEL
||
212 psmd
->uId
== IDM_NETWORKCONNECTIONS
||
213 psmd
->uId
== IDM_PRINTERSANDFAXES
)
221 HRESULT
OnGetObject(LPSMDATA psmd
, REFIID iid
, void ** pv
)
223 if (IsEqualIID(iid
, IID_IShellMenu
))
224 return OnGetSubMenu(psmd
, iid
, pv
);
225 else if (IsEqualIID(iid
, IID_IContextMenu
))
226 return OnGetContextMenu(psmd
, iid
, pv
);
231 HRESULT
OnExec(LPSMDATA psmd
)
233 // HACK: Instead of running explorer.exe with the path, we should be using ShellExecute to "open" the path directly!
234 // Remove once ShellExecute can handle CLSID path components.
236 if (psmd
->uId
== IDM_CONTROLPANEL
)
237 ShellExecuteW(NULL
, NULL
, L
"explorer.exe", L
"::{20D04FE0-3AEA-1069-A2D8-08002B30309D}\\::{21EC2020-3AEA-1069-A2DD-08002B30309D}", NULL
, SW_SHOWNORMAL
);
238 else if (psmd
->uId
== IDM_NETWORKCONNECTIONS
)
239 ShellExecuteW(NULL
, NULL
, L
"explorer.exe", L
"::{20D04FE0-3AEA-1069-A2D8-08002B30309D}\\::{21EC2020-3AEA-1069-A2DD-08002B30309D}\\::{7007ACC7-3202-11D1-AAD2-00805FC1270E}", NULL
, SW_SHOWNORMAL
);
240 else if (psmd
->uId
== IDM_PRINTERSANDFAXES
)
241 ShellExecuteW(NULL
, NULL
, L
"explorer.exe", L
"::{20D04FE0-3AEA-1069-A2D8-08002B30309D}\\::{21EC2020-3AEA-1069-A2DD-08002B30309D}\\::{2227A280-3AEA-1069-A2DE-08002B30309D}", NULL
, SW_SHOWNORMAL
);
243 PostMessageW(m_hwndTray
, WM_COMMAND
, psmd
->uId
, 0);
250 DECLARE_NOT_AGGREGATABLE(CShellMenuCallback
)
251 DECLARE_PROTECT_FINAL_CONSTRUCT()
252 BEGIN_COM_MAP(CShellMenuCallback
)
253 COM_INTERFACE_ENTRY_IID(IID_IShellMenuCallback
, IShellMenuCallback
)
257 IShellMenu
* pShellMenu
,
258 IBandSite
* pBandSite
,
261 m_pShellMenu
= pShellMenu
;
262 m_pBandSite
= pBandSite
;
263 m_pDeskBar
= pDeskBar
;
266 ~CShellMenuCallback()
270 HRESULT
_SetProgramsFolder(IShellFolder
* psf
, LPITEMIDLIST pidl
)
273 m_pidlPrograms
= pidl
;
277 HRESULT STDMETHODCALLTYPE
CallbackSM(
288 return OnGetInfo(psmd
, reinterpret_cast<SMINFO
*>(lParam
));
290 return OnGetObject(psmd
, *reinterpret_cast<IID
*>(wParam
), reinterpret_cast<void **>(lParam
));
294 m_pTrayPriv
->Execute(psmd
->psf
, psmd
->pidlItem
);
296 case 0x10000000: // _FilterPIDL from CMenuSFToolbar
297 if (psmd
->psf
->CompareIDs(0, psmd
->pidlItem
, m_pidlPrograms
) == 0)
306 HRESULT
BindToDesktop(LPCITEMIDLIST pidl
, IShellFolder
** ppsfResult
)
309 CComPtr
<IShellFolder
> psfDesktop
;
313 hr
= SHGetDesktopFolder(&psfDesktop
);
317 hr
= psfDesktop
->BindToObject(pidl
, NULL
, IID_PPV_ARG(IShellFolder
, ppsfResult
));
322 static HRESULT
GetStartMenuFolder(IShellFolder
** ppsfStartMenu
)
325 LPITEMIDLIST pidlUserStartMenu
;
326 LPITEMIDLIST pidlCommonStartMenu
;
327 CComPtr
<IShellFolder
> psfUserStartMenu
;
328 CComPtr
<IShellFolder
> psfCommonStartMenu
;
329 CComPtr
<IAugmentedShellFolder
> pasf
;
331 *ppsfStartMenu
= NULL
;
333 hr
= SHGetSpecialFolderLocation(NULL
, CSIDL_STARTMENU
, &pidlUserStartMenu
);
336 WARN("Failed to get the USER start menu folder. Trying to run with just the COMMON one.\n");
338 hr
= SHGetSpecialFolderLocation(NULL
, CSIDL_COMMON_STARTMENU
, &pidlCommonStartMenu
);
339 if (FAILED_UNEXPECTEDLY(hr
))
342 TRACE("COMMON start menu obtained.\n");
343 hr
= BindToDesktop(pidlCommonStartMenu
, ppsfStartMenu
);
344 ILFree(pidlCommonStartMenu
);
348 hr
= SHGetSpecialFolderLocation(NULL
, CSIDL_COMMON_STARTMENU
, &pidlCommonStartMenu
);
349 if (FAILED_UNEXPECTEDLY(hr
))
351 WARN("Failed to get the COMMON start menu folder. Will use only the USER contents.\n");
352 hr
= BindToDesktop(pidlUserStartMenu
, ppsfStartMenu
);
353 ILFree(pidlUserStartMenu
);
357 TRACE("Both COMMON and USER statr menu folders obtained, merging them...\n");
359 hr
= BindToDesktop(pidlUserStartMenu
, &psfUserStartMenu
);
360 if (FAILED_UNEXPECTEDLY(hr
))
363 hr
= BindToDesktop(pidlCommonStartMenu
, &psfCommonStartMenu
);
364 if (FAILED_UNEXPECTEDLY(hr
))
367 #if !USE_SYSTEM_MERGED_FOLDERS
368 hr
= CMergedFolder_Constructor(IID_PPV_ARG(IAugmentedShellFolder
, &pasf
));
370 hr
= CoCreateInstance(CLSID_MergedFolder
, NULL
, CLSCTX_INPROC_SERVER
, IID_PPV_ARG(IAugmentedShellFolder
, &pasf
));
372 if (FAILED_UNEXPECTEDLY(hr
))
374 *ppsfStartMenu
= psfUserStartMenu
.Detach();
375 ILFree(pidlCommonStartMenu
);
376 ILFree(pidlUserStartMenu
);
380 hr
= pasf
->AddNameSpace(NULL
, psfUserStartMenu
, pidlUserStartMenu
, 0xFF00);
381 if (FAILED_UNEXPECTEDLY(hr
))
384 hr
= pasf
->AddNameSpace(NULL
, psfCommonStartMenu
, pidlCommonStartMenu
, 0);
385 if (FAILED_UNEXPECTEDLY(hr
))
388 hr
= pasf
->QueryInterface(IID_PPV_ARG(IShellFolder
, ppsfStartMenu
));
391 ILFree(pidlCommonStartMenu
);
392 ILFree(pidlUserStartMenu
);
399 CStartMenu_Constructor(REFIID riid
, void **ppv
)
401 CComPtr
<IShellMenu
> pShellMenu
;
402 CComPtr
<IBandSite
> pBandSite
;
403 CComPtr
<IDeskBar
> pDeskBar
;
408 LPITEMIDLIST pidlProgramsAbsolute
;
409 LPITEMIDLIST pidlPrograms
;
410 CComPtr
<IShellFolder
> psfPrograms
;
412 #if USE_SYSTEM_MENUBAND
413 hr
= CoCreateInstance(CLSID_MenuBand
,
415 CLSCTX_INPROC_SERVER
,
416 IID_PPV_ARG(IShellMenu
, &pShellMenu
));
418 hr
= CMenuBand_Constructor(IID_PPV_ARG(IShellMenu
, &pShellMenu
));
420 if (FAILED_UNEXPECTEDLY(hr
))
423 #if USE_SYSTEM_MENUSITE
424 hr
= CoCreateInstance(CLSID_MenuBandSite
,
426 CLSCTX_INPROC_SERVER
,
427 IID_PPV_ARG(IBandSite
, &pBandSite
));
429 hr
= CMenuSite_Constructor(IID_PPV_ARG(IBandSite
, &pBandSite
));
431 if (FAILED_UNEXPECTEDLY(hr
))
434 #if USE_SYSTEM_MENUDESKBAR
435 hr
= CoCreateInstance(CLSID_MenuDeskBar
,
437 CLSCTX_INPROC_SERVER
,
438 IID_PPV_ARG(IDeskBar
, &pDeskBar
));
440 hr
= CMenuDeskBar_Constructor(IID_PPV_ARG(IDeskBar
, &pDeskBar
));
442 if (FAILED_UNEXPECTEDLY(hr
))
445 CComObject
<CShellMenuCallback
> *pCallback
;
446 hr
= CComObject
<CShellMenuCallback
>::CreateInstance(&pCallback
);
447 if (FAILED_UNEXPECTEDLY(hr
))
449 pCallback
->AddRef(); // CreateInstance returns object with 0 ref count */
450 pCallback
->Initialize(pShellMenu
, pBandSite
, pDeskBar
);
452 pShellMenu
->Initialize(pCallback
, (UINT
) -1, 0, SMINIT_TOPLEVEL
| SMINIT_VERTICAL
);
453 if (FAILED_UNEXPECTEDLY(hr
))
456 hr
= GetStartMenuFolder(&psf
);
457 if (FAILED_UNEXPECTEDLY(hr
))
460 hr
= SHGetSpecialFolderLocation(NULL
, CSIDL_PROGRAMS
, &pidlProgramsAbsolute
);
463 WARN("USER Programs folder not found.");
464 hr
= SHGetSpecialFolderLocation(NULL
, CSIDL_COMMON_PROGRAMS
, &pidlProgramsAbsolute
);
465 if (FAILED_UNEXPECTEDLY(hr
))
469 pidlPrograms
= ILClone(ILFindLastID(pidlProgramsAbsolute
));
470 ILFree(pidlProgramsAbsolute
);
472 hr
= psf
->BindToObject(pidlPrograms
, NULL
, IID_PPV_ARG(IShellFolder
, &psfPrograms
));
473 if (FAILED_UNEXPECTEDLY(hr
))
476 hr
= pCallback
->_SetProgramsFolder(psfPrograms
, pidlPrograms
);
477 if (FAILED_UNEXPECTEDLY(hr
))
480 hr
= pShellMenu
->SetShellFolder(psf
, NULL
, NULL
, 0);
481 if (FAILED_UNEXPECTEDLY(hr
))
484 hr
= pDeskBar
->SetClient(pBandSite
);
485 if (FAILED_UNEXPECTEDLY(hr
))
488 hr
= pBandSite
->AddBand(pShellMenu
);
489 if (FAILED_UNEXPECTEDLY(hr
))
492 return pDeskBar
->QueryInterface(riid
, ppv
);