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
20 #include "shellmenu.h"
22 #include "CMergedFolder.h"
24 WINE_DEFAULT_DEBUG_CHANNEL(CStartMenu
);
26 //#define TEST_TRACKPOPUPMENU_SUBMENUS
29 /* NOTE: The following constants *MUST NOT* be changed because
30 they're hardcoded and need to be the exact values
31 in order to get the start menu to work! */
33 #define IDM_LOGOFF 402
34 #define IDM_UNDOCKCOMPUTER 410
35 #define IDM_TASKBARANDSTARTMENU 413
36 #define IDM_LASTSTARTMENU_SEPARATOR 450
37 #define IDM_DOCUMENTS 501
38 #define IDM_HELPANDSUPPORT 503
39 #define IDM_PROGRAMS 504
40 #define IDM_CONTROLPANEL 505
41 #define IDM_SHUTDOWN 506
42 #define IDM_FAVORITES 507
43 #define IDM_SETTINGS 508
44 #define IDM_PRINTERSANDFAXES 510
45 #define IDM_SEARCH 520
46 #define IDM_SYNCHRONIZE 553
47 #define IDM_NETWORKCONNECTIONS 557
48 #define IDM_DISCONNECT 5000
49 #define IDM_SECURITY 5001
53 * 1. append the start menu contents from all users
54 * 2. implement the context menu for start menu entries (programs, control panel, network connetions, printers)
55 * 3. filter out programs folder from the shell folder part of the start menu
56 * 4. showing the programs start menu is SLOW compared to windows. this needs some investigation
59 class CShellMenuCallback
:
60 public CComObjectRootEx
<CComMultiThreadModelNoCS
>,
61 public IShellMenuCallback
66 CComPtr
<IShellMenu
> m_pShellMenu
;
67 CComPtr
<IBandSite
> m_pBandSite
;
68 CComPtr
<IDeskBar
> m_pDeskBar
;
69 CComPtr
<ITrayPriv
> m_pTrayPriv
;
70 CComPtr
<IShellFolder
> m_psfPrograms
;
72 LPITEMIDLIST m_pidlPrograms
;
82 hr
= IUnknown_GetSite(m_pDeskBar
, IID_PPV_ARG(ITrayPriv
, &m_pTrayPriv
));
83 if (FAILED_UNEXPECTEDLY(hr
))
86 hr
= IUnknown_GetWindow(m_pTrayPriv
, &m_hwndTray
);
87 if (FAILED_UNEXPECTEDLY(hr
))
90 hr
= m_pTrayPriv
->AppendMenuW(&hmenu
);
91 if (FAILED_UNEXPECTEDLY(hr
))
94 hr
= m_pShellMenu
->SetMenu(hmenu
, NULL
, SMSET_BOTTOM
);
95 if (FAILED_UNEXPECTEDLY(hr
))
101 HRESULT
OnGetInfo(LPSMDATA psmd
, SMINFO
*psminfo
)
107 // Smaller "24x24" icons used for the start menu
108 // The bitmaps are still 32x32, but the image is centered
109 case IDM_FAVORITES
: iconIndex
= -322; break;
110 case IDM_SEARCH
: iconIndex
= -323; break;
111 case IDM_HELPANDSUPPORT
: iconIndex
= -324; break;
112 case IDM_LOGOFF
: iconIndex
= -325; break;
113 case IDM_PROGRAMS
: iconIndex
= -326; break;
114 case IDM_DOCUMENTS
: iconIndex
= -327; break;
115 case IDM_RUN
: iconIndex
= -328; break;
116 case IDM_SHUTDOWN
: iconIndex
= -329; break;
117 case IDM_SETTINGS
: iconIndex
= -330; break;
119 case IDM_CONTROLPANEL
: iconIndex
= -22; break;
120 case IDM_NETWORKCONNECTIONS
: iconIndex
= -257; break;
121 case IDM_PRINTERSANDFAXES
: iconIndex
= -138; break;
122 case IDM_TASKBARANDSTARTMENU
: iconIndex
= -40; break;
123 //case IDM_SECURITY: iconIndex = -21; break;
124 //case IDM_SYNCHRONIZE: iconIndex = -21; break;
125 //case IDM_DISCONNECT: iconIndex = -21; break;
126 //case IDM_UNDOCKCOMPUTER: iconIndex = -21; break;
133 if ((psminfo
->dwMask
& SMIM_TYPE
) != 0)
134 psminfo
->dwType
= SMIT_STRING
;
135 if ((psminfo
->dwMask
& SMIM_ICON
) != 0)
136 psminfo
->iIcon
= Shell_GetCachedImageIndex(L
"shell32.dll", iconIndex
, FALSE
);
137 if ((psminfo
->dwMask
& SMIM_FLAGS
) != 0)
138 psminfo
->dwFlags
|= SMIF_ICON
;
139 #ifdef TEST_TRACKPOPUPMENU_SUBMENUS
140 if ((psminfo
->dwMask
& SMIM_FLAGS
) != 0)
141 psminfo
->dwFlags
|= SMIF_TRACKPOPUP
;
146 if ((psminfo
->dwMask
& SMIM_TYPE
) != 0)
147 psminfo
->dwType
= SMIT_SEPARATOR
;
152 HRESULT
OnGetSubMenu(LPSMDATA psmd
, REFIID iid
, void ** pv
)
156 IShellMenu
*pShellMenu
;
158 #if USE_SYSTEM_MENUBAND
159 hr
= CoCreateInstance(CLSID_MenuBand
,
161 CLSCTX_INPROC_SERVER
,
162 IID_PPV_ARG(IShellMenu
, &pShellMenu
));
164 hr
= CMenuBand_Constructor(IID_PPV_ARG(IShellMenu
, &pShellMenu
));
166 if (FAILED_UNEXPECTEDLY(hr
))
169 hr
= pShellMenu
->Initialize(this, 0, ANCESTORDEFAULT
, SMINIT_VERTICAL
);
173 case IDM_PROGRAMS
: csidl
= CSIDL_PROGRAMS
; break;
174 case IDM_FAVORITES
: csidl
= CSIDL_FAVORITES
; break;
175 case IDM_DOCUMENTS
: csidl
= CSIDL_RECENT
; break;
180 IShellFolder
*psfStartMenu
;
182 if (csidl
== CSIDL_PROGRAMS
&& m_psfPrograms
)
184 psfStartMenu
= m_psfPrograms
;
188 LPITEMIDLIST pidlStartMenu
;
189 IShellFolder
*psfDestop
;
190 hr
= SHGetFolderLocation(NULL
, csidl
, 0, 0, &pidlStartMenu
);
191 hr
= SHGetDesktopFolder(&psfDestop
);
192 hr
= psfDestop
->BindToObject(pidlStartMenu
, NULL
, IID_PPV_ARG(IShellFolder
, &psfStartMenu
));
195 hr
= pShellMenu
->SetShellFolder(psfStartMenu
, NULL
, NULL
, 0);
200 mii
.cbSize
= sizeof(mii
);
201 mii
.fMask
= MIIM_SUBMENU
;
202 if (GetMenuItemInfoW(psmd
->hmenu
, psmd
->uId
, FALSE
, &mii
))
204 hr
= pShellMenu
->SetMenu(mii
.hSubMenu
, NULL
, SMSET_BOTTOM
);
207 return pShellMenu
->QueryInterface(iid
, pv
);
210 HRESULT
OnGetContextMenu(LPSMDATA psmd
, REFIID iid
, void ** pv
)
212 if (psmd
->uId
== IDM_PROGRAMS
||
213 psmd
->uId
== IDM_CONTROLPANEL
||
214 psmd
->uId
== IDM_NETWORKCONNECTIONS
||
215 psmd
->uId
== IDM_PRINTERSANDFAXES
)
223 HRESULT
OnGetObject(LPSMDATA psmd
, REFIID iid
, void ** pv
)
225 if (IsEqualIID(iid
, IID_IShellMenu
))
226 return OnGetSubMenu(psmd
, iid
, pv
);
227 else if (IsEqualIID(iid
, IID_IContextMenu
))
228 return OnGetContextMenu(psmd
, iid
, pv
);
233 HRESULT
OnExec(LPSMDATA psmd
)
235 // HACK: Because our ShellExecute can't handle CLSID components in paths, we can't launch the paths using the "open" verb.
236 // FIXME: Change this back to using the path as the filename and the "open" verb, once ShellExecute can handle CLSID path components.
238 if (psmd
->uId
== IDM_CONTROLPANEL
)
239 ShellExecuteW(NULL
, NULL
, L
"explorer.exe", L
"::{20D04FE0-3AEA-1069-A2D8-08002B30309D}\\::{21EC2020-3AEA-1069-A2DD-08002B30309D}", NULL
, SW_SHOWNORMAL
);
240 else if (psmd
->uId
== IDM_NETWORKCONNECTIONS
)
241 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
);
242 else if (psmd
->uId
== IDM_PRINTERSANDFAXES
)
243 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
);
245 PostMessageW(m_hwndTray
, WM_COMMAND
, psmd
->uId
, 0);
252 DECLARE_NOT_AGGREGATABLE(CShellMenuCallback
)
253 DECLARE_PROTECT_FINAL_CONSTRUCT()
254 BEGIN_COM_MAP(CShellMenuCallback
)
255 COM_INTERFACE_ENTRY_IID(IID_IShellMenuCallback
, IShellMenuCallback
)
259 IShellMenu
* pShellMenu
,
260 IBandSite
* pBandSite
,
263 m_pShellMenu
= pShellMenu
;
264 m_pBandSite
= pBandSite
;
265 m_pDeskBar
= pDeskBar
;
268 ~CShellMenuCallback()
272 HRESULT
_SetProgramsFolder(IShellFolder
* psf
, LPITEMIDLIST pidl
)
275 m_pidlPrograms
= pidl
;
279 HRESULT STDMETHODCALLTYPE
CallbackSM(
290 return OnGetInfo(psmd
, reinterpret_cast<SMINFO
*>(lParam
));
292 return OnGetObject(psmd
, *reinterpret_cast<IID
*>(wParam
), reinterpret_cast<void **>(lParam
));
296 m_pTrayPriv
->Execute(psmd
->psf
, psmd
->pidlItem
);
298 case 0x10000000: // _FilterPIDL from CMenuSFToolbar
299 if (psmd
->psf
->CompareIDs(0, psmd
->pidlItem
, m_pidlPrograms
) == 0)
308 HRESULT
BindToDesktop(LPCITEMIDLIST pidl
, IShellFolder
** ppsfResult
)
311 CComPtr
<IShellFolder
> psfDesktop
;
315 hr
= SHGetDesktopFolder(&psfDesktop
);
319 hr
= psfDesktop
->BindToObject(pidl
, NULL
, IID_PPV_ARG(IShellFolder
, ppsfResult
));
324 static HRESULT
GetMergedFolder(int folder1
, int folder2
, IShellFolder
** ppsfStartMenu
)
327 LPITEMIDLIST pidlUserStartMenu
;
328 LPITEMIDLIST pidlCommonStartMenu
;
329 CComPtr
<IShellFolder
> psfUserStartMenu
;
330 CComPtr
<IShellFolder
> psfCommonStartMenu
;
331 CComPtr
<IAugmentedShellFolder
> pasf
;
333 *ppsfStartMenu
= NULL
;
335 hr
= SHGetSpecialFolderLocation(NULL
, folder1
, &pidlUserStartMenu
);
338 WARN("Failed to get the USER start menu folder. Trying to run with just the COMMON one.\n");
340 hr
= SHGetSpecialFolderLocation(NULL
, folder2
, &pidlCommonStartMenu
);
341 if (FAILED_UNEXPECTEDLY(hr
))
344 TRACE("COMMON start menu obtained.\n");
345 hr
= BindToDesktop(pidlCommonStartMenu
, ppsfStartMenu
);
346 ILFree(pidlCommonStartMenu
);
350 hr
= SHGetSpecialFolderLocation(NULL
, folder2
, &pidlCommonStartMenu
);
351 if (FAILED_UNEXPECTEDLY(hr
))
356 WARN("Failed to get the COMMON start menu folder. Will use only the USER contents.\n");
357 hr
= BindToDesktop(pidlUserStartMenu
, ppsfStartMenu
);
358 ILFree(pidlUserStartMenu
);
362 TRACE("Both COMMON and USER statr menu folders obtained, merging them...\n");
364 hr
= BindToDesktop(pidlUserStartMenu
, &psfUserStartMenu
);
365 if (FAILED_UNEXPECTEDLY(hr
))
368 hr
= BindToDesktop(pidlCommonStartMenu
, &psfCommonStartMenu
);
369 if (FAILED_UNEXPECTEDLY(hr
))
372 #if !USE_SYSTEM_MERGED_FOLDERS
373 hr
= CMergedFolder_Constructor(IID_PPV_ARG(IAugmentedShellFolder
, &pasf
));
375 hr
= CoCreateInstance(CLSID_MergedFolder
, NULL
, CLSCTX_INPROC_SERVER
, IID_PPV_ARG(IAugmentedShellFolder
, &pasf
));
377 if (FAILED_UNEXPECTEDLY(hr
))
379 *ppsfStartMenu
= psfUserStartMenu
.Detach();
380 ILFree(pidlCommonStartMenu
);
381 ILFree(pidlUserStartMenu
);
385 hr
= pasf
->AddNameSpace(NULL
, psfUserStartMenu
, pidlUserStartMenu
, 0xFF00);
386 if (FAILED_UNEXPECTEDLY(hr
))
389 hr
= pasf
->AddNameSpace(NULL
, psfCommonStartMenu
, pidlCommonStartMenu
, 0);
390 if (FAILED_UNEXPECTEDLY(hr
))
393 hr
= pasf
->QueryInterface(IID_PPV_ARG(IShellFolder
, ppsfStartMenu
));
396 ILFree(pidlCommonStartMenu
);
397 ILFree(pidlUserStartMenu
);
402 static HRESULT
GetStartMenuFolder(IShellFolder
** ppsfStartMenu
)
404 return GetMergedFolder(CSIDL_STARTMENU
, CSIDL_COMMON_STARTMENU
, ppsfStartMenu
);
407 static HRESULT
GetProgramsFolder(IShellFolder
** ppsfStartMenu
)
409 return GetMergedFolder(CSIDL_PROGRAMS
, CSIDL_COMMON_PROGRAMS
, ppsfStartMenu
);
414 CStartMenu_Constructor(REFIID riid
, void **ppv
)
416 CComPtr
<IShellMenu
> pShellMenu
;
417 CComPtr
<IBandSite
> pBandSite
;
418 CComPtr
<IDeskBar
> pDeskBar
;
423 LPITEMIDLIST pidlProgramsAbsolute
;
424 LPITEMIDLIST pidlPrograms
;
425 CComPtr
<IShellFolder
> psfPrograms
;
427 #if USE_SYSTEM_MENUBAND
428 hr
= CoCreateInstance(CLSID_MenuBand
,
430 CLSCTX_INPROC_SERVER
,
431 IID_PPV_ARG(IShellMenu
, &pShellMenu
));
433 hr
= CMenuBand_Constructor(IID_PPV_ARG(IShellMenu
, &pShellMenu
));
435 if (FAILED_UNEXPECTEDLY(hr
))
438 #if USE_SYSTEM_MENUSITE
439 hr
= CoCreateInstance(CLSID_MenuBandSite
,
441 CLSCTX_INPROC_SERVER
,
442 IID_PPV_ARG(IBandSite
, &pBandSite
));
444 hr
= CMenuSite_Constructor(IID_PPV_ARG(IBandSite
, &pBandSite
));
446 if (FAILED_UNEXPECTEDLY(hr
))
449 #if USE_SYSTEM_MENUDESKBAR
450 hr
= CoCreateInstance(CLSID_MenuDeskBar
,
452 CLSCTX_INPROC_SERVER
,
453 IID_PPV_ARG(IDeskBar
, &pDeskBar
));
455 hr
= CMenuDeskBar_Constructor(IID_PPV_ARG(IDeskBar
, &pDeskBar
));
457 if (FAILED_UNEXPECTEDLY(hr
))
460 CComObject
<CShellMenuCallback
> *pCallback
;
461 hr
= CComObject
<CShellMenuCallback
>::CreateInstance(&pCallback
);
462 if (FAILED_UNEXPECTEDLY(hr
))
464 pCallback
->AddRef(); // CreateInstance returns object with 0 ref count */
465 pCallback
->Initialize(pShellMenu
, pBandSite
, pDeskBar
);
467 pShellMenu
->Initialize(pCallback
, (UINT
) -1, 0, SMINIT_TOPLEVEL
| SMINIT_VERTICAL
);
468 if (FAILED_UNEXPECTEDLY(hr
))
471 hr
= GetStartMenuFolder(&psf
);
472 if (FAILED_UNEXPECTEDLY(hr
))
476 hr
= SHGetSpecialFolderLocation(NULL
, CSIDL_PROGRAMS
, &pidlProgramsAbsolute
);
479 WARN("USER Programs folder not found.");
480 hr
= SHGetSpecialFolderLocation(NULL
, CSIDL_COMMON_PROGRAMS
, &pidlProgramsAbsolute
);
481 if (FAILED_UNEXPECTEDLY(hr
))
485 LPCITEMIDLIST pcidlPrograms
;
486 CComPtr
<IShellFolder
> psfParent
;
488 TCHAR szDisplayName
[MAX_PATH
];
490 hr
= SHBindToParent(pidlProgramsAbsolute
, IID_PPV_ARG(IShellFolder
, &psfParent
), &pcidlPrograms
);
494 hr
= psfParent
->GetDisplayNameOf(pcidlPrograms
, SHGDN_NORMAL
, &str
);
498 StrRetToBuf(&str
, pcidlPrograms
, szDisplayName
, _countof(szDisplayName
));
499 ILFree(pidlProgramsAbsolute
);
501 hr
= psf
->ParseDisplayName(NULL
, NULL
, szDisplayName
, NULL
, &pidlPrograms
, NULL
);
506 hr
= GetProgramsFolder(&psfPrograms
);
507 if (FAILED_UNEXPECTEDLY(hr
))
510 hr
= pCallback
->_SetProgramsFolder(psfPrograms
, pidlPrograms
);
511 if (FAILED_UNEXPECTEDLY(hr
))
514 hr
= pShellMenu
->SetShellFolder(psf
, NULL
, NULL
, 0);
515 if (FAILED_UNEXPECTEDLY(hr
))
518 hr
= pDeskBar
->SetClient(pBandSite
);
519 if (FAILED_UNEXPECTEDLY(hr
))
522 hr
= pBandSite
->AddBand(pShellMenu
);
523 if (FAILED_UNEXPECTEDLY(hr
))
526 return pDeskBar
->QueryInterface(riid
, ppv
);