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() :
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
|SHCONTF_NONFOLDERS
, &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 StringCchCopyW(sz
, MAX_PATH
, 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
)
263 if (!ILGetDisplayNameEx(NULL
, m_pidl
, pdbi
->wszTitle
, ILGDN_INFOLDER
))
265 pdbi
->dwMask
&= ~DBIM_TITLE
;
268 if (pdbi
->dwMask
& DBIM_MODEFLAGS
)
270 pdbi
->dwModeFlags
= DBIMF_NORMAL
| DBIMF_VARIABLEHEIGHT
| DBIMF_USECHEVRON
| DBIMF_NOMARGINS
| DBIMF_BKCOLOR
| DBIMF_ADDTOFRONT
;
272 if (pdbi
->dwMask
& DBIM_BKCOLOR
)
273 pdbi
->dwMask
&= ~DBIM_BKCOLOR
;
281 /*****************************************************************************/
282 // *** IPersistStream ***
283 STDMETHODIMP
CISFBand::GetClassID(OUT CLSID
*pClassID
)
285 *pClassID
= CLSID_ISFBand
;
290 STDMETHODIMP
CISFBand::IsDirty()
292 /* The object hasn't changed since the last save! */
297 STDMETHODIMP
CISFBand::Load(IN IStream
*pStm
)
299 TRACE("CISFBand::Load called\n");
305 STDMETHODIMP
CISFBand::Save(IN IStream
*pStm
, IN BOOL fClearDirty
)
312 STDMETHODIMP
CISFBand::GetSizeMax(OUT ULARGE_INTEGER
*pcbSize
)
314 TRACE("CISFBand::GetSizeMax called\n");
319 /*****************************************************************************/
320 // *** IWinEventHandler ***
321 STDMETHODIMP
CISFBand::ContainsWindow(IN HWND hWnd
)
323 if (hWnd
== m_hWnd
|| IsChild(hWnd
))
325 TRACE("CISFBand::ContainsWindow(0x%p) returns S_OK\n", hWnd
);
332 STDMETHODIMP
CISFBand::OnWinEvent(HWND hWnd
, UINT uMsg
, WPARAM wParam
, LPARAM lParam
, LRESULT
*theResult
)
339 bool chk
= SendMessage(m_hWnd
, TB_GETBUTTON
, LOWORD(wParam
), (LPARAM
)&tb
);
341 SHInvokeDefaultCommand(m_hWnd
, m_pISF
, (LPITEMIDLIST
)tb
.dwData
);
348 switch (((LPNMHDR
)lParam
)->code
)
353 POINT pt
= ((LPNMMOUSE
)lParam
)->pt
;
354 CComPtr
<IContextMenu
> picm
;
355 HMENU fmenu
= CreatePopupMenu();
358 bool chk
= SendMessage(m_hWnd
, TB_GETBUTTON
, ((LPNMMOUSE
)lParam
)->dwItemSpec
, (LPARAM
)&tb
);
359 LPITEMIDLIST pidl
= (LPITEMIDLIST
)tb
.dwData
;
364 hr
= m_pISF
->GetUIObjectOf(m_hWnd
, 1, &pidl
, IID_NULL_PPV_ARG(IContextMenu
, &picm
));
365 if (FAILED_UNEXPECTEDLY(hr
))
368 hr
= picm
->QueryContextMenu(fmenu
, 0, 1, 0x7FFF, CMF_DEFAULTONLY
);
369 if (FAILED_UNEXPECTEDLY(hr
))
372 int id
= TrackPopupMenuEx(fmenu
, TPM_LEFTALIGN
| TPM_BOTTOMALIGN
| TPM_RETURNCMD
, pt
.x
, pt
.y
, m_hWnd
, 0);
375 CMINVOKECOMMANDINFOEX info
= { 0 };
376 info
.cbSize
= sizeof(info
);
377 info
.fMask
= CMIC_MASK_PTINVOKE
;
378 if (GetKeyState(VK_CONTROL
) < 0)
380 info
.fMask
|= CMIC_MASK_CONTROL_DOWN
;
382 if (GetKeyState(VK_SHIFT
) < 0)
384 info
.fMask
|= CMIC_MASK_SHIFT_DOWN
;
387 info
.lpVerb
= MAKEINTRESOURCEA(id
- 1);
388 info
.nShow
= SW_SHOWNORMAL
;
390 picm
->InvokeCommand((LPCMINVOKECOMMANDINFO
)&info
);
411 STDMETHODIMP
CISFBand::IsWindowOwner(HWND hWnd
)
413 return (hWnd
== m_hWnd
) ? S_OK
: S_FALSE
;
416 /*****************************************************************************/
417 // *** IOleCommandTarget methods ***
418 STDMETHODIMP
CISFBand::QueryStatus(const GUID
*pguidCmdGroup
, ULONG cCmds
, OLECMD prgCmds
[], OLECMDTEXT
*pCmdText
)
425 STDMETHODIMP
CISFBand::Exec(const GUID
*pguidCmdGroup
, DWORD nCmdID
, DWORD nCmdexecopt
, VARIANT
*pvaIn
, VARIANT
*pvaOut
)
427 if (IsEqualIID(*pguidCmdGroup
, IID_IBandSite
))
432 if (IsEqualIID(*pguidCmdGroup
, IID_IDeskBand
))
442 /*****************************************************************************/
443 // *** IShellFolderBand ***
444 STDMETHODIMP
CISFBand::GetBandInfoSFB(PBANDINFOSFB pbi
)
446 if (pbi
->dwMask
== ISFB_MASK_IDLIST
)
448 pbi
->pidl
= ILClone(m_pidl
);
450 return E_OUTOFMEMORY
;
457 STDMETHODIMP
CISFBand::InitializeSFB(IShellFolder
*psf
, PCIDLIST_ABSOLUTE pidl
)
469 CComPtr
<IShellFolder
> psfDesktop
;
470 hr
= SHGetDesktopFolder(&psfDesktop
);
471 if (FAILED_UNEXPECTEDLY(hr
))
474 if (_ILIsDesktop(pidl
))
480 hr
= psfDesktop
->BindToObject(pidl
, NULL
, IID_PPV_ARG(IShellFolder
, &m_pISF
));
481 if (FAILED_UNEXPECTEDLY(hr
))
485 m_pidl
= ILClone(pidl
);
490 CComPtr
<IPersistFolder2
> ppf2
;
491 hr
= psf
->QueryInterface(IID_PPV_ARG(IPersistFolder2
, &ppf2
));
492 if (FAILED_UNEXPECTEDLY(hr
))
495 hr
= ppf2
->GetCurFolder(&m_pidl
);
496 if (FAILED_UNEXPECTEDLY(hr
))
505 STDMETHODIMP
CISFBand::SetBandInfoSFB( PBANDINFOSFB pbi
)
507 if ((pbi
->dwMask
& ISFB_MASK_STATE
) &&
508 (pbi
->dwState
& ISFB_STATE_QLINKSMODE
) &&
509 (pbi
->dwStateMask
& ISFB_STATE_QLINKSMODE
))
517 /*****************************************************************************/
518 // *** IContextMenu ***
519 STDMETHODIMP
CISFBand::GetCommandString(UINT_PTR idCmd
, UINT uFlags
, UINT
*pwReserved
, LPSTR pszName
, UINT cchMax
)
521 /*HRESULT hr = E_INVALIDARG;
523 if (idCmd == IDM_DISPLAY)
528 // Only useful for pre-Vista versions of Windows that
529 // have a Status bar.
530 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
532 L"Display File Name");
536 // GCS_VERBW is an optional feature that enables a caller
537 // to discover the canonical name for the verb that is passed in
538 // through idCommand.
539 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
550 STDMETHODIMP
CISFBand::InvokeCommand(LPCMINVOKECOMMANDINFO pici
)
552 if (!HIWORD(pici
->lpVerb
))
554 switch (LOWORD(pici
->lpVerb
))
556 case IDM_LARGE_ICONS
:
560 HIMAGELIST
* piml
= (HIMAGELIST
*) SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
561 HRESULT hr
= SHGetImageList(SHIL_LARGE
, IID_IImageList
, (void**)&piml
);
562 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
563 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
564 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
565 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
568 case IDM_SMALL_ICONS
:
572 HIMAGELIST
* piml
= (HIMAGELIST
*)SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
573 HRESULT hr
= SHGetImageList(SHIL_SMALL
, IID_IImageList
, (void**)&piml
);
574 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
575 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
576 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
577 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
585 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, TBSTYLE_EX_MIXEDBUTTONS
);
586 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
587 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
592 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, 0);
593 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
594 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
606 STDMETHODIMP
CISFBand::QueryContextMenu(HMENU hmenu
, UINT indexMenu
, UINT idCmdFirst
, UINT idCmdLast
, UINT uFlags
)
608 HMENU qMenu
= LoadMenu(GetModuleHandleW(L
"browseui.dll"), MAKEINTRESOURCE(IDM_POPUPMENU
));
611 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_CHECKED
);
613 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_UNCHECKED
);
617 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_CHECKED
);
618 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_UNCHECKED
);
622 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_CHECKED
);
623 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_UNCHECKED
);
626 UINT idMax
= Shell_MergeMenus(hmenu
, GetSubMenu(qMenu
, 0), indexMenu
, idCmdFirst
, idCmdLast
, MM_SUBMENUSHAVEIDS
);
628 return MAKE_HRESULT(SEVERITY_SUCCESS
, 0, USHORT(idMax
- idCmdFirst
+1));
631 /*****************************************************************************/
634 HRESULT WINAPI
RSHELL_CISFBand_CreateInstance(REFIID riid
, void** ppv
)
636 return ShellObjectCreator
<CISFBand
>(riid
, ppv
);