2 * PROJECT: ReactOS shell extensions
3 * LICENSE: GPL - See COPYING in the top level directory
4 * FILE: dll/shellext/qcklnch/CISFBand.cpp
5 * PURPOSE: Quick Launch Toolbar (Taskbar Shell Extension)
6 * PROGRAMMERS: Shriraj Sawant a.k.a SR13 <sr.official@hotmail.com>
10 #include <commoncontrols.h>
12 #define GET_X_LPARAM(lp) ((int)(short)LOWORD(lp))
13 #define GET_Y_LPARAM(lp) ((int)(short)HIWORD(lp))
15 WINE_DEFAULT_DEBUG_CHANNEL(qcklnch
);
21 * Checks whether the given PIDL is of Desktop folder or not.
26 * @return True if PIDL is of Desktop, otherwise false.
29 BOOL WINAPI
_ILIsDesktop(LPCITEMIDLIST pidl
)
31 return (pidl
== NULL
|| pidl
->mkid
.cb
== 0);
34 //*****************************************************************************************
37 CISFBand::CISFBand() :
52 * @name CreateSimpleToolbar
54 * Creates a toolbar and fills it up with buttons for enumerated objects.
57 * Handle to the parent window, which receives the appropriate messages from child toolbar.
59 * @return The error code.
62 HRESULT
CISFBand::CreateSimpleToolbar(HWND hWndParent
)
64 // Declare and initialize local constants.
65 const DWORD buttonStyles
= BTNS_AUTOSIZE
;
67 // Create the toolbar.
68 m_hWnd
= CreateWindowEx(0, TOOLBARCLASSNAME
, NULL
,
69 WS_CHILD
| TBSTYLE_FLAT
| TBSTYLE_LIST
| CCS_NORESIZE
| CCS_NODIVIDER
, CW_USEDEFAULT
, CW_USEDEFAULT
, 0, 0,
70 hWndParent
, NULL
, 0, NULL
);
74 // Set the image list.
76 HRESULT hr
= SHGetImageList(SHIL_SMALL
, IID_IImageList
, (void**)&piml
);
77 if (FAILED_UNEXPECTEDLY(hr
))
82 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
85 CComPtr
<IEnumIDList
> pEndl
;
88 hr
= m_pISF
->EnumObjects(0, SHCONTF_FOLDERS
, &pEndl
);
89 if (FAILED_UNEXPECTEDLY(hr
))
95 for (int i
=0; pEndl
->Next(1, &pidl
, NULL
) != S_FALSE
; i
++)
98 int index
= SHMapPIDLToSystemImageListIndex(m_pISF
, pidl
, NULL
);
99 hr
= m_pISF
->GetDisplayNameOf(pidl
, SHGDN_NORMAL
, &stret
);
100 if (FAILED_UNEXPECTEDLY(hr
))
102 strcpyW(sz
, L
"<Unknown-Name>");
105 StrRetToBuf(&stret
, pidl
, sz
, _countof(sz
));
107 TBBUTTON tb
= { MAKELONG(index
, 0), i
, TBSTATE_ENABLED
, buttonStyles
,{ 0 }, (DWORD_PTR
)pidl
, (INT_PTR
)sz
};
108 SendMessage(m_hWnd
, TB_INSERTBUTTONW
, i
, (LPARAM
)&tb
);
111 // Resize the toolbar, and then show it.
112 SendMessage(m_hWnd
, TB_AUTOSIZE
, 0, 0);
117 /*****************************************************************************/
119 // *** IObjectWithSite ***
120 STDMETHODIMP
CISFBand::SetSite(IUnknown
*pUnkSite
)
125 TRACE("CISFBand::SetSite(0x%p)\n", pUnkSite
);
127 hr
= IUnknown_GetWindow(pUnkSite
, &hwndParent
);
130 TRACE("Querying site window failed: 0x%x\n", hr
);
135 hr
= CreateSimpleToolbar(hwndParent
);
136 if (FAILED_UNEXPECTEDLY(hr
))
142 STDMETHODIMP
CISFBand::GetSite(IN REFIID riid
, OUT VOID
**ppvSite
)
144 TRACE("CISFBand::GetSite(0x%p,0x%p)\n", riid
, ppvSite
);
149 hr
= m_Site
->QueryInterface(riid
, ppvSite
);
150 if (FAILED(hr
)) return hr
;
157 /*****************************************************************************/
159 STDMETHODIMP
CISFBand::GetWindow(OUT HWND
*phwnd
)
170 STDMETHODIMP
CISFBand::ContextSensitiveHelp(IN BOOL fEnterMode
)
172 /* FIXME: Implement */
176 STDMETHODIMP
CISFBand::ShowDW(IN BOOL bShow
)
180 ShowWindow(bShow
? SW_SHOW
: SW_HIDE
);
187 STDMETHODIMP
CISFBand::CloseDW(IN DWORD dwReserved
)
194 for (int i
= 0; SendMessage(m_hWnd
, TB_GETBUTTON
, i
, (LPARAM
)&tb
); i
++)
196 CoTaskMemFree((LPITEMIDLIST
)tb
.dwData
);
207 STDMETHODIMP
CISFBand::ResizeBorderDW(LPCRECT prcBorder
, IUnknown
*punkToolbarSite
, BOOL fReserved
)
209 /* No need to implement this method */
214 STDMETHODIMP
CISFBand::GetBandInfo(IN DWORD dwBandID
, IN DWORD dwViewMode
, IN OUT DESKBANDINFO
*pdbi
)
216 TRACE("CTaskBand::GetBandInfo(0x%x,0x%x,0x%p) hWnd=0x%p\n", dwBandID
, dwViewMode
, pdbi
, m_hWnd
);
228 GetWindowRect(&actualRect
);
229 actualSize
.x
= actualRect
.right
- actualRect
.left
;
230 actualSize
.y
= actualRect
.bottom
- actualRect
.top
;
232 // Obtain the ideal size, to be used as min and max
233 SendMessageW(m_hWnd
, TB_AUTOSIZE
, 0, 0);
234 SendMessageW(m_hWnd
, TB_GETMAXSIZE
, 0, reinterpret_cast<LPARAM
>(&maxSize
));
237 SendMessageW(m_hWnd
, TB_GETIDEALSIZE
, FALSE
, reinterpret_cast<LPARAM
>(&idealSize
));
239 // Obtain the button size, to be used as the integral size
240 DWORD size
= SendMessageW(m_hWnd
, TB_GETBUTTONSIZE
, 0, 0);
241 itemSize
.x
= GET_X_LPARAM(size
);
242 itemSize
.y
= GET_Y_LPARAM(size
);
244 if (pdbi
->dwMask
& DBIM_MINSIZE
)
246 pdbi
->ptMinSize
.x
= -1;
247 pdbi
->ptMinSize
.y
= idealSize
.y
;
249 if (pdbi
->dwMask
& DBIM_MAXSIZE
)
251 pdbi
->ptMaxSize
= maxSize
;
253 if (pdbi
->dwMask
& DBIM_INTEGRAL
)
255 pdbi
->ptIntegral
= itemSize
;
257 if (pdbi
->dwMask
& DBIM_ACTUAL
)
259 pdbi
->ptActual
= actualSize
;
261 if (pdbi
->dwMask
& DBIM_TITLE
)
262 wcscpy(pdbi
->wszTitle
, L
"Quick Launch");
263 if (pdbi
->dwMask
& DBIM_MODEFLAGS
)
265 pdbi
->dwModeFlags
= DBIMF_NORMAL
| DBIMF_VARIABLEHEIGHT
| DBIMF_USECHEVRON
| DBIMF_NOMARGINS
| DBIMF_BKCOLOR
| DBIMF_ADDTOFRONT
;
267 if (pdbi
->dwMask
& DBIM_BKCOLOR
)
268 pdbi
->dwMask
&= ~DBIM_BKCOLOR
;
276 /*****************************************************************************/
277 // *** IPersistStream ***
278 STDMETHODIMP
CISFBand::GetClassID(OUT CLSID
*pClassID
)
280 TRACE("CISFBand::GetClassID(0x%p)\n", pClassID
);
281 /* We're going to return the (internal!) CLSID of the quick launch band */
282 *pClassID
= CLSID_QuickLaunchBand
;
287 STDMETHODIMP
CISFBand::IsDirty()
289 /* The object hasn't changed since the last save! */
294 STDMETHODIMP
CISFBand::Load(IN IStream
*pStm
)
296 TRACE("CISFBand::Load called\n");
302 STDMETHODIMP
CISFBand::Save(IN IStream
*pStm
, IN BOOL fClearDirty
)
309 STDMETHODIMP
CISFBand::GetSizeMax(OUT ULARGE_INTEGER
*pcbSize
)
311 TRACE("CISFBand::GetSizeMax called\n");
316 /*****************************************************************************/
317 // *** IWinEventHandler ***
318 STDMETHODIMP
CISFBand::ContainsWindow(IN HWND hWnd
)
320 if (hWnd
== m_hWnd
|| IsChild(hWnd
))
322 TRACE("CISFBand::ContainsWindow(0x%p) returns S_OK\n", hWnd
);
329 STDMETHODIMP
CISFBand::OnWinEvent(HWND hWnd
, UINT uMsg
, WPARAM wParam
, LPARAM lParam
, LRESULT
*theResult
)
336 bool chk
= SendMessage(m_hWnd
, TB_GETBUTTON
, LOWORD(wParam
), (LPARAM
)&tb
);
338 SHInvokeDefaultCommand(m_hWnd
, m_pISF
, (LPITEMIDLIST
)tb
.dwData
);
345 switch (((LPNMHDR
)lParam
)->code
)
350 POINT pt
= ((LPNMMOUSE
)lParam
)->pt
;
351 CComPtr
<IContextMenu
> picm
;
352 HMENU fmenu
= CreatePopupMenu();
355 bool chk
= SendMessage(m_hWnd
, TB_GETBUTTON
, ((LPNMMOUSE
)lParam
)->dwItemSpec
, (LPARAM
)&tb
);
356 LPITEMIDLIST pidl
= (LPITEMIDLIST
)tb
.dwData
;
361 hr
= m_pISF
->GetUIObjectOf(m_hWnd
, 1, &pidl
, IID_NULL_PPV_ARG(IContextMenu
, &picm
));
362 if (FAILED_UNEXPECTEDLY(hr
))
365 hr
= picm
->QueryContextMenu(fmenu
, 0, 1, 0x7FFF, CMF_DEFAULTONLY
);
366 if (FAILED_UNEXPECTEDLY(hr
))
369 int id
= TrackPopupMenuEx(fmenu
, TPM_LEFTALIGN
| TPM_BOTTOMALIGN
| TPM_RETURNCMD
, pt
.x
, pt
.y
, m_hWnd
, 0);
372 CMINVOKECOMMANDINFOEX info
= { 0 };
373 info
.cbSize
= sizeof(info
);
374 info
.fMask
= CMIC_MASK_PTINVOKE
;
375 if (GetKeyState(VK_CONTROL
) < 0)
377 info
.fMask
|= CMIC_MASK_CONTROL_DOWN
;
379 if (GetKeyState(VK_SHIFT
) < 0)
381 info
.fMask
|= CMIC_MASK_SHIFT_DOWN
;
384 info
.lpVerb
= MAKEINTRESOURCEA(id
- 1);
385 info
.nShow
= SW_SHOWNORMAL
;
387 picm
->InvokeCommand((LPCMINVOKECOMMANDINFO
)&info
);
408 STDMETHODIMP
CISFBand::IsWindowOwner(HWND hWnd
)
410 return (hWnd
== m_hWnd
) ? S_OK
: S_FALSE
;
413 /*****************************************************************************/
414 // *** IOleCommandTarget methods ***
415 STDMETHODIMP
CISFBand::QueryStatus(const GUID
*pguidCmdGroup
, ULONG cCmds
, OLECMD prgCmds
[], OLECMDTEXT
*pCmdText
)
422 STDMETHODIMP
CISFBand::Exec(const GUID
*pguidCmdGroup
, DWORD nCmdID
, DWORD nCmdexecopt
, VARIANT
*pvaIn
, VARIANT
*pvaOut
)
424 if (IsEqualIID(*pguidCmdGroup
, IID_IBandSite
))
429 if (IsEqualIID(*pguidCmdGroup
, IID_IDeskBand
))
439 /*****************************************************************************/
440 // *** IShellFolderBand ***
441 STDMETHODIMP
CISFBand::GetBandInfoSFB(PBANDINFOSFB pbi
)
446 STDMETHODIMP
CISFBand::InitializeSFB(IShellFolder
*psf
, PCIDLIST_ABSOLUTE pidl
)
448 if (_ILIsDesktop(pidl
))
451 m_pidl
= ILClone(pidl
);
455 psf
->BindToObject(pidl
, 0, IID_PPV_ARG(IShellFolder
, &m_pISF
));
456 m_pidl
= ILClone(pidl
);
462 STDMETHODIMP
CISFBand::SetBandInfoSFB( PBANDINFOSFB pbi
)
467 /*****************************************************************************/
468 // *** IContextMenu ***
469 STDMETHODIMP
CISFBand::GetCommandString(UINT_PTR idCmd
, UINT uFlags
, UINT
*pwReserved
, LPSTR pszName
, UINT cchMax
)
471 /*HRESULT hr = E_INVALIDARG;
473 if (idCmd == IDM_DISPLAY)
478 // Only useful for pre-Vista versions of Windows that
479 // have a Status bar.
480 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
482 L"Display File Name");
486 // GCS_VERBW is an optional feature that enables a caller
487 // to discover the canonical name for the verb that is passed in
488 // through idCommand.
489 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
500 STDMETHODIMP
CISFBand::InvokeCommand(LPCMINVOKECOMMANDINFO pici
)
502 if (!HIWORD(pici
->lpVerb
))
504 switch (LOWORD(pici
->lpVerb
))
506 case IDM_LARGE_ICONS
:
510 HIMAGELIST
* piml
= (HIMAGELIST
*) SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
511 HRESULT hr
= SHGetImageList(SHIL_LARGE
, IID_IImageList
, (void**)&piml
);
512 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
513 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
514 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
515 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
518 case IDM_SMALL_ICONS
:
522 HIMAGELIST
* piml
= (HIMAGELIST
*)SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
523 HRESULT hr
= SHGetImageList(SHIL_SMALL
, IID_IImageList
, (void**)&piml
);
524 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
525 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
526 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
527 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
535 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, TBSTYLE_EX_MIXEDBUTTONS
);
536 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
537 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
542 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, 0);
543 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
544 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
556 STDMETHODIMP
CISFBand::QueryContextMenu(HMENU hmenu
, UINT indexMenu
, UINT idCmdFirst
, UINT idCmdLast
, UINT uFlags
)
558 HMENU qMenu
= LoadMenu(_AtlBaseModule
.GetResourceInstance(), MAKEINTRESOURCE(IDM_POPUPMENU
));
560 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_CHECKED
);
562 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_UNCHECKED
);
566 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_CHECKED
);
567 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_UNCHECKED
);
571 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_CHECKED
);
572 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_UNCHECKED
);
575 UINT idMax
= Shell_MergeMenus(hmenu
, GetSubMenu(qMenu
, 0), indexMenu
, idCmdFirst
, idCmdLast
, MM_SUBMENUSHAVEIDS
);
577 return MAKE_HRESULT(SEVERITY_SUCCESS
, 0, USHORT(idMax
- idCmdFirst
+1));
580 /*****************************************************************************/
583 HRESULT WINAPI
CISFBand_CreateInstance(REFIID riid
, void** ppv
)
585 return ShellObjectCreator
<CISFBand
>(riid
, ppv
);