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>
11 #include <commoncontrols.h>
16 ** drag and drop support
18 ** handle change notifications
19 ** Fix position of the items context menu
20 ** Implement responding to theme change
23 //*****************************************************************************************
26 CISFBand::CISFBand() :
42 * @name CreateSimpleToolbar
44 * Creates a toolbar and fills it up with buttons for enumerated objects.
47 * Handle to the parent window, which receives the appropriate messages from child toolbar.
49 * @return The error code.
52 HRESULT
CISFBand::CreateSimpleToolbar(HWND hWndParent
)
54 // Declare and initialize local constants.
55 const DWORD buttonStyles
= BTNS_AUTOSIZE
;
57 // Create the toolbar.
58 m_hWnd
= CreateWindowEx(0, TOOLBARCLASSNAME
, NULL
,
59 WS_CHILD
| TBSTYLE_FLAT
| TBSTYLE_LIST
| CCS_NORESIZE
| CCS_NODIVIDER
, CW_USEDEFAULT
, CW_USEDEFAULT
, 0, 0,
60 hWndParent
, NULL
, 0, NULL
);
65 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, TBSTYLE_EX_MIXEDBUTTONS
);
67 // Set the image list.
69 HRESULT hr
= SHGetImageList(SHIL_SMALL
, IID_IImageList
, (void**)&piml
);
70 if (FAILED_UNEXPECTEDLY(hr
))
75 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
78 CComPtr
<IEnumIDList
> pEndl
;
81 hr
= m_pISF
->EnumObjects(0, SHCONTF_FOLDERS
|SHCONTF_NONFOLDERS
, &pEndl
);
82 if (FAILED_UNEXPECTEDLY(hr
))
88 for (int i
=0; pEndl
->Next(1, &pidl
, NULL
) != S_FALSE
; i
++)
91 int index
= SHMapPIDLToSystemImageListIndex(m_pISF
, pidl
, NULL
);
92 hr
= m_pISF
->GetDisplayNameOf(pidl
, SHGDN_NORMAL
, &stret
);
93 if (FAILED_UNEXPECTEDLY(hr
))
95 StringCchCopyW(sz
, MAX_PATH
, L
"<Unknown-Name>");
98 StrRetToBuf(&stret
, pidl
, sz
, _countof(sz
));
100 TBBUTTON tb
= { MAKELONG(index
, 0), i
, TBSTATE_ENABLED
, buttonStyles
,{ 0 }, (DWORD_PTR
)pidl
, (INT_PTR
)sz
};
101 SendMessage(m_hWnd
, TB_INSERTBUTTONW
, i
, (LPARAM
)&tb
);
104 // Resize the toolbar, and then show it.
105 SendMessage(m_hWnd
, TB_AUTOSIZE
, 0, 0);
110 /*****************************************************************************/
112 // *** IObjectWithSite ***
113 STDMETHODIMP
CISFBand::SetSite(IUnknown
*pUnkSite
)
118 TRACE("CISFBand::SetSite(0x%p)\n", pUnkSite
);
120 hr
= IUnknown_GetWindow(pUnkSite
, &hwndParent
);
123 TRACE("Querying site window failed: 0x%x\n", hr
);
128 hr
= CreateSimpleToolbar(hwndParent
);
129 if (FAILED_UNEXPECTEDLY(hr
))
135 STDMETHODIMP
CISFBand::GetSite(IN REFIID riid
, OUT VOID
**ppvSite
)
137 TRACE("CISFBand::GetSite(0x%p,0x%p)\n", riid
, ppvSite
);
142 hr
= m_Site
->QueryInterface(riid
, ppvSite
);
143 if (FAILED(hr
)) return hr
;
150 /*****************************************************************************/
152 STDMETHODIMP
CISFBand::GetWindow(OUT HWND
*phwnd
)
163 STDMETHODIMP
CISFBand::ContextSensitiveHelp(IN BOOL fEnterMode
)
165 /* FIXME: Implement */
169 STDMETHODIMP
CISFBand::ShowDW(IN BOOL bShow
)
173 ShowWindow(bShow
? SW_SHOW
: SW_HIDE
);
180 STDMETHODIMP
CISFBand::CloseDW(IN DWORD dwReserved
)
187 for (int i
= 0; SendMessage(m_hWnd
, TB_GETBUTTON
, i
, (LPARAM
)&tb
); i
++)
189 CoTaskMemFree((LPITEMIDLIST
)tb
.dwData
);
200 STDMETHODIMP
CISFBand::ResizeBorderDW(LPCRECT prcBorder
, IUnknown
*punkToolbarSite
, BOOL fReserved
)
202 /* No need to implement this method */
207 STDMETHODIMP
CISFBand::GetBandInfo(IN DWORD dwBandID
, IN DWORD dwViewMode
, IN OUT DESKBANDINFO
*pdbi
)
209 TRACE("CTaskBand::GetBandInfo(0x%x,0x%x,0x%p) hWnd=0x%p\n", dwBandID
, dwViewMode
, pdbi
, m_hWnd
);
221 GetWindowRect(&actualRect
);
222 actualSize
.x
= actualRect
.right
- actualRect
.left
;
223 actualSize
.y
= actualRect
.bottom
- actualRect
.top
;
225 // Obtain the ideal size, to be used as min and max
226 SendMessageW(m_hWnd
, TB_AUTOSIZE
, 0, 0);
227 SendMessageW(m_hWnd
, TB_GETMAXSIZE
, 0, reinterpret_cast<LPARAM
>(&maxSize
));
230 SendMessageW(m_hWnd
, TB_GETIDEALSIZE
, FALSE
, reinterpret_cast<LPARAM
>(&idealSize
));
232 // Obtain the button size, to be used as the integral size
233 DWORD size
= SendMessageW(m_hWnd
, TB_GETBUTTONSIZE
, 0, 0);
234 itemSize
.x
= GET_X_LPARAM(size
);
235 itemSize
.y
= GET_Y_LPARAM(size
);
237 if (pdbi
->dwMask
& DBIM_MINSIZE
)
240 pdbi
->ptMinSize
.x
= idealSize
.x
;
242 pdbi
->ptMinSize
.x
= -1;
243 pdbi
->ptMinSize
.y
= idealSize
.y
;
245 if (pdbi
->dwMask
& DBIM_MAXSIZE
)
247 pdbi
->ptMaxSize
= maxSize
;
249 if (pdbi
->dwMask
& DBIM_INTEGRAL
)
251 pdbi
->ptIntegral
= itemSize
;
253 if (pdbi
->dwMask
& DBIM_ACTUAL
)
255 pdbi
->ptActual
= actualSize
;
257 if (pdbi
->dwMask
& DBIM_TITLE
)
259 if (m_QLaunch
|| !ILGetDisplayNameEx(NULL
, m_pidl
, pdbi
->wszTitle
, ILGDN_INFOLDER
))
261 pdbi
->dwMask
&= ~DBIM_TITLE
;
264 if (pdbi
->dwMask
& DBIM_MODEFLAGS
)
266 pdbi
->dwModeFlags
= DBIMF_NORMAL
| DBIMF_VARIABLEHEIGHT
| DBIMF_USECHEVRON
| DBIMF_NOMARGINS
| DBIMF_BKCOLOR
;
269 pdbi
->dwModeFlags
|= 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
))
514 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, TBSTYLE_EX_MIXEDBUTTONS
);
520 /*****************************************************************************/
521 // *** IContextMenu ***
522 STDMETHODIMP
CISFBand::GetCommandString(UINT_PTR idCmd
, UINT uFlags
, UINT
*pwReserved
, LPSTR pszName
, UINT cchMax
)
524 /*HRESULT hr = E_INVALIDARG;
526 if (idCmd == IDM_DISPLAY)
531 // Only useful for pre-Vista versions of Windows that
532 // have a Status bar.
533 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
535 L"Display File Name");
539 // GCS_VERBW is an optional feature that enables a caller
540 // to discover the canonical name for the verb that is passed in
541 // through idCommand.
542 hr = StringCchCopyW(reinterpret_cast<PWSTR>(pszName),
553 STDMETHODIMP
CISFBand::InvokeCommand(LPCMINVOKECOMMANDINFO pici
)
555 if (!HIWORD(pici
->lpVerb
))
557 switch (LOWORD(pici
->lpVerb
))
559 case IDM_LARGE_ICONS
:
563 HIMAGELIST
* piml
= (HIMAGELIST
*) SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
564 HRESULT hr
= SHGetImageList(SHIL_LARGE
, IID_IImageList
, (void**)&piml
);
565 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
566 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
567 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
568 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
571 case IDM_SMALL_ICONS
:
575 HIMAGELIST
* piml
= (HIMAGELIST
*)SendMessage(m_hWnd
, TB_GETIMAGELIST
, 0, 0);
576 HRESULT hr
= SHGetImageList(SHIL_SMALL
, IID_IImageList
, (void**)&piml
);
577 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
578 SendMessage(m_hWnd
, TB_SETIMAGELIST
, 0, (LPARAM
)piml
);
579 hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
580 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
583 case IDM_OPEN_FOLDER
:
585 SHELLEXECUTEINFO shexinfo
;
587 memset(&shexinfo
, 0x0, sizeof(shexinfo
));
589 shexinfo
.cbSize
= sizeof(shexinfo
);
590 shexinfo
.fMask
= SEE_MASK_IDLIST
;
591 shexinfo
.lpVerb
= _T("open");
592 shexinfo
.lpIDList
= m_pidl
;
593 shexinfo
.nShow
= SW_SHOW
;
595 if (!ShellExecuteEx(&shexinfo
))
605 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, TBSTYLE_EX_MIXEDBUTTONS
);
606 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
607 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
612 SendMessage(m_hWnd
, TB_SETEXTENDEDSTYLE
, 0, 0);
613 HRESULT hr
= IUnknown_Exec(m_Site
, IID_IDeskBand
, DBID_BANDINFOCHANGED
, 0, NULL
, NULL
);
614 if (FAILED_UNEXPECTEDLY(hr
)) return hr
;
626 STDMETHODIMP
CISFBand::QueryContextMenu(HMENU hmenu
, UINT indexMenu
, UINT idCmdFirst
, UINT idCmdLast
, UINT uFlags
)
628 HMENU qMenu
= LoadMenu(GetModuleHandleW(L
"browseui.dll"), MAKEINTRESOURCE(IDM_POPUPMENU
));
631 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_CHECKED
);
633 CheckMenuItem(qMenu
, IDM_SHOW_TEXT
, MF_UNCHECKED
);
637 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_CHECKED
);
638 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_UNCHECKED
);
642 CheckMenuItem(qMenu
, IDM_LARGE_ICONS
, MF_CHECKED
);
643 CheckMenuItem(qMenu
, IDM_SMALL_ICONS
, MF_UNCHECKED
);
646 if (_ILIsDesktop(m_pidl
))
647 DeleteMenu(qMenu
, IDM_OPEN_FOLDER
, MF_BYCOMMAND
);
649 UINT idMax
= Shell_MergeMenus(hmenu
, GetSubMenu(qMenu
, 0), indexMenu
, idCmdFirst
, idCmdLast
, MM_SUBMENUSHAVEIDS
);
651 return MAKE_HRESULT(SEVERITY_SUCCESS
, 0, USHORT(idMax
- idCmdFirst
+1));
654 /*****************************************************************************/
657 HRESULT WINAPI
RSHELL_CISFBand_CreateInstance(REFIID riid
, void** ppv
)
659 return ShellObjectCreator
<CISFBand
>(riid
, ppv
);