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>
13 #define GET_X_LPARAM(lp) ((int)(short)LOWORD(lp))
14 #define GET_Y_LPARAM(lp) ((int)(short)HIWORD(lp))
20 * Checks whether the given PIDL is of Desktop folder or not.
25 * @return True if PIDL is of Desktop, otherwise false.
28 static BOOL
_ILIsDesktop(LPCITEMIDLIST pidl
)
30 return (pidl
== NULL
|| pidl
->mkid
.cb
== 0);
33 //*****************************************************************************************
36 CISFBand::CISFBand() :
51 * @name CreateSimpleToolbar
53 * Creates a toolbar and fills it up with buttons for enumerated objects.
56 * Handle to the parent window, which receives the appropriate messages from child toolbar.
58 * @return The error code.
61 HRESULT
CISFBand::CreateSimpleToolbar(HWND hWndParent
)
63 // Declare and initialize local constants.
64 const DWORD buttonStyles
= BTNS_AUTOSIZE
;
66 // Create the toolbar.
67 m_hWnd
= CreateWindowEx(0, TOOLBARCLASSNAME
, NULL
,
68 WS_CHILD
| TBSTYLE_FLAT
| TBSTYLE_LIST
| CCS_NORESIZE
| CCS_NODIVIDER
, CW_USEDEFAULT
, CW_USEDEFAULT
, 0, 0,
69 hWndParent
, NULL
, 0, NULL
);
73 // Set the image list.
75 HRESULT hr
= SHGetImageList(SHIL_SMALL
, IID_IImageList
, (void**)&piml
);
76 if (FAILED_UNEXPECTEDLY(hr
))
81 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
84 CComPtr
<IEnumIDList
> pEndl
;
87 hr
= m_pISF
->EnumObjects(0, SHCONTF_FOLDERS
, &pEndl
);
88 if (FAILED_UNEXPECTEDLY(hr
))
94 for (int i
=0; pEndl
->Next(1, &pidl
, NULL
) != S_FALSE
; i
++)
97 int index
= SHMapPIDLToSystemImageListIndex(m_pISF
, pidl
, NULL
);
98 hr
= m_pISF
->GetDisplayNameOf(pidl
, SHGDN_NORMAL
, &stret
);
99 if (FAILED_UNEXPECTEDLY(hr
))
101 StringCchCopyW(sz
, MAX_PATH
, L
"<Unknown-Name>");
104 StrRetToBuf(&stret
, pidl
, sz
, _countof(sz
));
106 TBBUTTON tb
= { MAKELONG(index
, 0), i
, TBSTATE_ENABLED
, buttonStyles
,{ 0 }, (DWORD_PTR
)pidl
, (INT_PTR
)sz
};
107 SendMessage(m_hWnd
, TB_INSERTBUTTONW
, i
, (LPARAM
)&tb
);
110 // Resize the toolbar, and then show it.
111 SendMessage(m_hWnd
, TB_AUTOSIZE
, 0, 0);
116 /*****************************************************************************/
118 // *** IObjectWithSite ***
119 STDMETHODIMP
CISFBand::SetSite(IUnknown
*pUnkSite
)
124 TRACE("CISFBand::SetSite(0x%p)\n", pUnkSite
);
126 hr
= IUnknown_GetWindow(pUnkSite
, &hwndParent
);
129 TRACE("Querying site window failed: 0x%x\n", hr
);
134 hr
= CreateSimpleToolbar(hwndParent
);
135 if (FAILED_UNEXPECTEDLY(hr
))
141 STDMETHODIMP
CISFBand::GetSite(IN REFIID riid
, OUT VOID
**ppvSite
)
143 TRACE("CISFBand::GetSite(0x%p,0x%p)\n", riid
, ppvSite
);
148 hr
= m_Site
->QueryInterface(riid
, ppvSite
);
149 if (FAILED(hr
)) return hr
;
156 /*****************************************************************************/
158 STDMETHODIMP
CISFBand::GetWindow(OUT HWND
*phwnd
)
169 STDMETHODIMP
CISFBand::ContextSensitiveHelp(IN BOOL fEnterMode
)
171 /* FIXME: Implement */
175 STDMETHODIMP
CISFBand::ShowDW(IN BOOL bShow
)
179 ShowWindow(bShow
? SW_SHOW
: SW_HIDE
);
186 STDMETHODIMP
CISFBand::CloseDW(IN DWORD dwReserved
)
193 for (int i
= 0; SendMessage(m_hWnd
, TB_GETBUTTON
, i
, (LPARAM
)&tb
); i
++)
195 CoTaskMemFree((LPITEMIDLIST
)tb
.dwData
);
206 STDMETHODIMP
CISFBand::ResizeBorderDW(LPCRECT prcBorder
, IUnknown
*punkToolbarSite
, BOOL fReserved
)
208 /* No need to implement this method */
213 STDMETHODIMP
CISFBand::GetBandInfo(IN DWORD dwBandID
, IN DWORD dwViewMode
, IN OUT DESKBANDINFO
*pdbi
)
215 TRACE("CTaskBand::GetBandInfo(0x%x,0x%x,0x%p) hWnd=0x%p\n", dwBandID
, dwViewMode
, pdbi
, m_hWnd
);
227 GetWindowRect(&actualRect
);
228 actualSize
.x
= actualRect
.right
- actualRect
.left
;
229 actualSize
.y
= actualRect
.bottom
- actualRect
.top
;
231 // Obtain the ideal size, to be used as min and max
232 SendMessageW(m_hWnd
, TB_AUTOSIZE
, 0, 0);
233 SendMessageW(m_hWnd
, TB_GETMAXSIZE
, 0, reinterpret_cast<LPARAM
>(&maxSize
));
236 SendMessageW(m_hWnd
, TB_GETIDEALSIZE
, FALSE
, reinterpret_cast<LPARAM
>(&idealSize
));
238 // Obtain the button size, to be used as the integral size
239 DWORD size
= SendMessageW(m_hWnd
, TB_GETBUTTONSIZE
, 0, 0);
240 itemSize
.x
= GET_X_LPARAM(size
);
241 itemSize
.y
= GET_Y_LPARAM(size
);
243 if (pdbi
->dwMask
& DBIM_MINSIZE
)
245 pdbi
->ptMinSize
.x
= -1;
246 pdbi
->ptMinSize
.y
= idealSize
.y
;
248 if (pdbi
->dwMask
& DBIM_MAXSIZE
)
250 pdbi
->ptMaxSize
= maxSize
;
252 if (pdbi
->dwMask
& DBIM_INTEGRAL
)
254 pdbi
->ptIntegral
= itemSize
;
256 if (pdbi
->dwMask
& DBIM_ACTUAL
)
258 pdbi
->ptActual
= actualSize
;
260 if (pdbi
->dwMask
& DBIM_TITLE
)
261 wcscpy(pdbi
->wszTitle
, L
"Quick Launch");
262 if (pdbi
->dwMask
& DBIM_MODEFLAGS
)
264 pdbi
->dwModeFlags
= DBIMF_NORMAL
| DBIMF_VARIABLEHEIGHT
| DBIMF_USECHEVRON
| DBIMF_NOMARGINS
| DBIMF_BKCOLOR
| DBIMF_ADDTOFRONT
;
266 if (pdbi
->dwMask
& DBIM_BKCOLOR
)
267 pdbi
->dwMask
&= ~DBIM_BKCOLOR
;
275 /*****************************************************************************/
276 // *** IPersistStream ***
277 STDMETHODIMP
CISFBand::GetClassID(OUT CLSID
*pClassID
)
279 *pClassID
= CLSID_ISFBand
;
284 STDMETHODIMP
CISFBand::IsDirty()
286 /* The object hasn't changed since the last save! */
291 STDMETHODIMP
CISFBand::Load(IN IStream
*pStm
)
293 TRACE("CISFBand::Load called\n");
299 STDMETHODIMP
CISFBand::Save(IN IStream
*pStm
, IN BOOL fClearDirty
)
306 STDMETHODIMP
CISFBand::GetSizeMax(OUT ULARGE_INTEGER
*pcbSize
)
308 TRACE("CISFBand::GetSizeMax called\n");
313 /*****************************************************************************/
314 // *** IWinEventHandler ***
315 STDMETHODIMP
CISFBand::ContainsWindow(IN HWND hWnd
)
317 if (hWnd
== m_hWnd
|| IsChild(hWnd
))
319 TRACE("CISFBand::ContainsWindow(0x%p) returns S_OK\n", hWnd
);
326 STDMETHODIMP
CISFBand::OnWinEvent(HWND hWnd
, UINT uMsg
, WPARAM wParam
, LPARAM lParam
, LRESULT
*theResult
)
333 bool chk
= SendMessage(m_hWnd
, TB_GETBUTTON
, LOWORD(wParam
), (LPARAM
)&tb
);
335 SHInvokeDefaultCommand(m_hWnd
, m_pISF
, (LPITEMIDLIST
)tb
.dwData
);
342 switch (((LPNMHDR
)lParam
)->code
)
347 POINT pt
= ((LPNMMOUSE
)lParam
)->pt
;
348 CComPtr
<IContextMenu
> picm
;
349 HMENU fmenu
= CreatePopupMenu();
352 bool chk
= SendMessage(m_hWnd
, TB_GETBUTTON
, ((LPNMMOUSE
)lParam
)->dwItemSpec
, (LPARAM
)&tb
);
353 LPITEMIDLIST pidl
= (LPITEMIDLIST
)tb
.dwData
;
358 hr
= m_pISF
->GetUIObjectOf(m_hWnd
, 1, &pidl
, IID_NULL_PPV_ARG(IContextMenu
, &picm
));
359 if (FAILED_UNEXPECTEDLY(hr
))
362 hr
= picm
->QueryContextMenu(fmenu
, 0, 1, 0x7FFF, CMF_DEFAULTONLY
);
363 if (FAILED_UNEXPECTEDLY(hr
))
366 int id
= TrackPopupMenuEx(fmenu
, TPM_LEFTALIGN
| TPM_BOTTOMALIGN
| TPM_RETURNCMD
, pt
.x
, pt
.y
, m_hWnd
, 0);
369 CMINVOKECOMMANDINFOEX info
= { 0 };
370 info
.cbSize
= sizeof(info
);
371 info
.fMask
= CMIC_MASK_PTINVOKE
;
372 if (GetKeyState(VK_CONTROL
) < 0)
374 info
.fMask
|= CMIC_MASK_CONTROL_DOWN
;
376 if (GetKeyState(VK_SHIFT
) < 0)
378 info
.fMask
|= CMIC_MASK_SHIFT_DOWN
;
381 info
.lpVerb
= MAKEINTRESOURCEA(id
- 1);
382 info
.nShow
= SW_SHOWNORMAL
;
384 picm
->InvokeCommand((LPCMINVOKECOMMANDINFO
)&info
);
405 STDMETHODIMP
CISFBand::IsWindowOwner(HWND hWnd
)
407 return (hWnd
== m_hWnd
) ? S_OK
: S_FALSE
;
410 /*****************************************************************************/
411 // *** IOleCommandTarget methods ***
412 STDMETHODIMP
CISFBand::QueryStatus(const GUID
*pguidCmdGroup
, ULONG cCmds
, OLECMD prgCmds
[], OLECMDTEXT
*pCmdText
)
419 STDMETHODIMP
CISFBand::Exec(const GUID
*pguidCmdGroup
, DWORD nCmdID
, DWORD nCmdexecopt
, VARIANT
*pvaIn
, VARIANT
*pvaOut
)
421 if (IsEqualIID(*pguidCmdGroup
, IID_IBandSite
))
426 if (IsEqualIID(*pguidCmdGroup
, IID_IDeskBand
))
436 /*****************************************************************************/
437 // *** IShellFolderBand ***
438 STDMETHODIMP
CISFBand::GetBandInfoSFB(PBANDINFOSFB pbi
)
443 STDMETHODIMP
CISFBand::InitializeSFB(IShellFolder
*psf
, PCIDLIST_ABSOLUTE pidl
)
445 if (_ILIsDesktop(pidl
))
448 m_pidl
= ILClone(pidl
);
452 psf
->BindToObject(pidl
, 0, IID_PPV_ARG(IShellFolder
, &m_pISF
));
453 m_pidl
= ILClone(pidl
);
459 STDMETHODIMP
CISFBand::SetBandInfoSFB( PBANDINFOSFB pbi
)
464 /*****************************************************************************/
465 // *** IContextMenu ***
466 STDMETHODIMP
CISFBand::GetCommandString(UINT_PTR idCmd
, UINT uFlags
, UINT
*pwReserved
, LPSTR pszName
, UINT cchMax
)
468 /*HRESULT hr = E_INVALIDARG;
470 if (idCmd == IDM_DISPLAY)
475 // Only useful for pre-Vista versions of Windows that
476 // have a Status bar.
477 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
479 L"Display File Name");
483 // GCS_VERBW is an optional feature that enables a caller
484 // to discover the canonical name for the verb that is passed in
485 // through idCommand.
486 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
497 STDMETHODIMP
CISFBand::InvokeCommand(LPCMINVOKECOMMANDINFO pici
)
499 if (!HIWORD(pici
->lpVerb
))
501 switch (LOWORD(pici
->lpVerb
))
503 case IDM_LARGE_ICONS
:
507 HIMAGELIST
* piml
= (HIMAGELIST
*) SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
508 HRESULT hr
= SHGetImageList(SHIL_LARGE
, IID_IImageList
, (void**)&piml
);
509 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
510 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
511 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
512 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
515 case IDM_SMALL_ICONS
:
519 HIMAGELIST
* piml
= (HIMAGELIST
*)SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
520 HRESULT hr
= SHGetImageList(SHIL_SMALL
, IID_IImageList
, (void**)&piml
);
521 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
522 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
523 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
524 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
532 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, TBSTYLE_EX_MIXEDBUTTONS
);
533 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
534 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
539 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, 0);
540 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
541 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
553 STDMETHODIMP
CISFBand::QueryContextMenu(HMENU hmenu
, UINT indexMenu
, UINT idCmdFirst
, UINT idCmdLast
, UINT uFlags
)
555 HMENU qMenu
= LoadMenu(GetModuleHandleW(L
"browseui.dll"), MAKEINTRESOURCE(IDM_POPUPMENU
));
558 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_CHECKED
);
560 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_UNCHECKED
);
564 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_CHECKED
);
565 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_UNCHECKED
);
569 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_CHECKED
);
570 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_UNCHECKED
);
573 UINT idMax
= Shell_MergeMenus(hmenu
, GetSubMenu(qMenu
, 0), indexMenu
, idCmdFirst
, idCmdLast
, MM_SUBMENUSHAVEIDS
);
575 return MAKE_HRESULT(SEVERITY_SUCCESS
, 0, USHORT(idMax
- idCmdFirst
+1));
578 /*****************************************************************************/
581 HRESULT WINAPI
CISFBand_CreateInstance(REFIID riid
, void** ppv
)
583 return ShellObjectCreator
<CISFBand
>(riid
, ppv
);