[BROWSEUI]
[reactos.git] / dll / win32 / browseui / toolsband.cpp
1 /*
2 * ReactOS Explorer
3 *
4 * Copyright 2009 Andrew Hill <ash77 at domain reactos.org>
5 *
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.
10 *
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.
15 *
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 */
20
21 /*
22 Implements the toolbar band of a cabinet window
23 */
24
25 #include "precomp.h"
26
27 /* FIXME, I can't include windowsx because it conflicts with some #defines */
28 #define GET_X_LPARAM(lp) ((int)(short)LOWORD(lp))
29 #define GET_Y_LPARAM(lp) ((int)(short)HIWORD(lp))
30
31 class CToolsBand :
32 public CWindowImpl<CToolsBand, CWindow, CControlWinTraits>,
33 public CComObjectRootEx<CComMultiThreadModelNoCS>,
34 public IDeskBand,
35 public IObjectWithSite,
36 public IInputObject,
37 public IPersistStream
38 {
39 private:
40 IDockingWindowSite *fDockSite;
41 GUID fExecCommandCategory;
42 CComPtr<IOleCommandTarget> fExecCommandTarget;
43 public:
44 CToolsBand();
45 ~CToolsBand();
46 public:
47 // *** IDeskBand methods ***
48 virtual HRESULT STDMETHODCALLTYPE GetBandInfo(DWORD dwBandID, DWORD dwViewMode, DESKBANDINFO* pdbi);
49
50 // *** IObjectWithSite methods ***
51 virtual HRESULT STDMETHODCALLTYPE SetSite(IUnknown* pUnkSite);
52 virtual HRESULT STDMETHODCALLTYPE GetSite(REFIID riid, void **ppvSite);
53
54 // *** IOleWindow methods ***
55 virtual HRESULT STDMETHODCALLTYPE GetWindow(HWND *lphwnd);
56 virtual HRESULT STDMETHODCALLTYPE ContextSensitiveHelp(BOOL fEnterMode);
57
58 // *** IDockingWindow methods ***
59 virtual HRESULT STDMETHODCALLTYPE CloseDW(DWORD dwReserved);
60 virtual HRESULT STDMETHODCALLTYPE ResizeBorderDW(const RECT* prcBorder, IUnknown* punkToolbarSite, BOOL fReserved);
61 virtual HRESULT STDMETHODCALLTYPE ShowDW(BOOL fShow);
62
63 // *** IInputObject methods ***
64 virtual HRESULT STDMETHODCALLTYPE HasFocusIO();
65 virtual HRESULT STDMETHODCALLTYPE TranslateAcceleratorIO(LPMSG lpMsg);
66 virtual HRESULT STDMETHODCALLTYPE UIActivateIO(BOOL fActivate, LPMSG lpMsg);
67
68 // *** IPersist methods ***
69 virtual HRESULT STDMETHODCALLTYPE GetClassID(CLSID *pClassID);
70
71 // *** IPersistStream methods ***
72 virtual HRESULT STDMETHODCALLTYPE IsDirty();
73 virtual HRESULT STDMETHODCALLTYPE Load(IStream *pStm);
74 virtual HRESULT STDMETHODCALLTYPE Save(IStream *pStm, BOOL fClearDirty);
75 virtual HRESULT STDMETHODCALLTYPE GetSizeMax(ULARGE_INTEGER *pcbSize);
76
77 // message handlers
78 LRESULT OnGetButtonInfo(UINT idControl, NMHDR *pNMHDR, BOOL &bHandled);
79
80 BEGIN_MSG_MAP(CToolsBand)
81 // MESSAGE_HANDLER(WM_NOTIFY, OnNotify)
82 // MESSAGE_HANDLER(WM_KILLFOCUS, OnKillFocus)
83 NOTIFY_HANDLER(0, TBN_GETBUTTONINFOW, OnGetButtonInfo)
84 END_MSG_MAP()
85
86 BEGIN_COM_MAP(CToolsBand)
87 COM_INTERFACE_ENTRY_IID(IID_IDeskBand, IDeskBand)
88 COM_INTERFACE_ENTRY_IID(IID_IObjectWithSite, IObjectWithSite)
89 COM_INTERFACE_ENTRY_IID(IID_IOleWindow, IOleWindow)
90 COM_INTERFACE_ENTRY_IID(IID_IDockingWindow, IDockingWindow)
91 COM_INTERFACE_ENTRY_IID(IID_IInputObject, IInputObject)
92 COM_INTERFACE_ENTRY_IID(IID_IPersist, IPersist)
93 COM_INTERFACE_ENTRY_IID(IID_IPersistStream, IPersistStream)
94 END_COM_MAP()
95 };
96
97 CToolsBand::CToolsBand()
98 {
99 fDockSite = NULL;
100 }
101
102 CToolsBand::~CToolsBand()
103 {
104 if (fDockSite)
105 fDockSite->Release();
106 }
107
108 HRESULT STDMETHODCALLTYPE CToolsBand::GetBandInfo(DWORD dwBandID, DWORD dwViewMode, DESKBANDINFO* pdbi)
109 {
110 RECT actualRect;
111 POINTL actualSize;
112 POINTL idealSize;
113 POINTL maxSize;
114 POINTL itemSize;
115
116 ::GetWindowRect(m_hWnd, &actualRect);
117 actualSize.x = actualRect.right - actualRect.left;
118 actualSize.y = actualRect.bottom - actualRect.top;
119
120 /* Obtain the ideal size, to be used as min and max */
121 SendMessageW(m_hWnd, TB_AUTOSIZE, 0, 0);
122 SendMessageW(m_hWnd, TB_GETMAXSIZE, 0, reinterpret_cast<LPARAM>(&maxSize));
123
124 idealSize = maxSize;
125 SendMessageW(m_hWnd, TB_GETIDEALSIZE, FALSE, reinterpret_cast<LPARAM>(&idealSize));
126
127 /* Obtain the button size, to be used as the integral size */
128 DWORD size = SendMessageW(m_hWnd, TB_GETBUTTONSIZE, 0, 0);
129 itemSize.x = GET_X_LPARAM(size);
130 itemSize.y = GET_Y_LPARAM(size);
131
132 if (pdbi->dwMask & DBIM_MINSIZE)
133 {
134 pdbi->ptMinSize = idealSize;
135 }
136 if (pdbi->dwMask & DBIM_MAXSIZE)
137 {
138 pdbi->ptMaxSize = maxSize;
139 }
140 if (pdbi->dwMask & DBIM_INTEGRAL)
141 {
142 pdbi->ptIntegral = itemSize;
143 }
144 if (pdbi->dwMask & DBIM_ACTUAL)
145 {
146 pdbi->ptActual = actualSize;
147 }
148 if (pdbi->dwMask & DBIM_TITLE)
149 wcscpy(pdbi->wszTitle, L"");
150 if (pdbi->dwMask & DBIM_MODEFLAGS)
151 pdbi->dwModeFlags = DBIMF_UNDELETEABLE;
152 if (pdbi->dwMask & DBIM_BKCOLOR)
153 pdbi->crBkgnd = 0;
154 return S_OK;
155 }
156
157 static const int backImageIndex = 0;
158 static const int forwardImageIndex = 1;
159 static const int favoritesImageIndex = 2;
160 // 3
161 // 4
162 static const int cutImageIndex = 5;
163 static const int copyImageIndex = 6;
164 static const int pasteImageIndex = 7;
165 static const int undoImageIndex = 8;
166 static const int redoImageIndex = 9;
167 static const int deleteImageIndex = 10;
168 // 11
169 // 12
170 // 13
171 // 14
172 static const int propertiesImageIndex = 15;
173 // 16
174 static const int searchImageIndex = 17;
175 // 18
176 // 19
177 // 20
178 // 21
179 static const int viewsImageIndex = 22;
180 // 23
181 // 24
182 // 25
183 // 26
184 // 27
185 static const int upImageIndex = 28;
186 static const int mapDriveImageIndex = 29;
187 static const int disconnectImageIndex = 30;
188 // 31
189 static const int viewsAltImageIndex = 32; // same image as viewsImageIndex
190 // 33
191 // 34
192 // 35
193 // 36
194 // 37
195 static const int viewsAlt2ImageIndex = 38; // same image as viewsAltImageIndex & viewsImageIndex
196 // 39
197 // 40
198 // 41
199 // 42
200 static const int foldersImageIndex = 43;
201 static const int moveToImageIndex = 44;
202 static const int copyToImageIndex = 45;
203 static const int folderOptionsImageIndex = 46;
204
205 enum StandardToolbarButtons {
206 BtnIdx_Back = 0,
207 BtnIdx_Forward,
208 BtnIdx_Up,
209 BtnIdx_Search,
210 BtnIdx_Folders,
211 BtnIdx_MoveTo,
212 BtnIdx_CopyTo,
213 BtnIdx_Delete,
214 BtnIdx_Undo,
215 BtnIdx_Views,
216 BtnIdx_Stop,
217 BtnIdx_Refresh,
218 BtnIdx_Home,
219 BtnIdx_MapDrive,
220 BtnIdx_Disconnect,
221 BtnIdx_Favorites,
222 BtnIdx_History,
223 BtnIdx_FullScreen,
224 BtnIdx_Properties,
225 BtnIdx_Cut,
226 BtnIdx_Copy,
227 BtnIdx_Paste,
228 BtnIdx_FolderOptions,
229 };
230
231 const int numShownButtons = 13;
232 const int numHiddenButtons = 13;
233 TBBUTTON tbButtonsAdd[numShownButtons + numHiddenButtons] =
234 {
235 { backImageIndex, IDM_GOTO_BACK, TBSTATE_ENABLED, BTNS_DROPDOWN | BTNS_SHOWTEXT, { 0 }, 0, BtnIdx_Back },
236 { forwardImageIndex, IDM_GOTO_FORWARD, TBSTATE_ENABLED, BTNS_DROPDOWN, { 0 }, 0, BtnIdx_Forward },
237 { upImageIndex, IDM_GOTO_UPONELEVEL, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Up },
238 { 6, -1, TBSTATE_ENABLED, BTNS_SEP },
239 { searchImageIndex, gSearchCommandID, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_SHOWTEXT, { 0 }, 0, BtnIdx_Search },
240 { foldersImageIndex, gFoldersCommandID, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_SHOWTEXT, { 0 }, 0, BtnIdx_Folders },
241 { 6, -1, TBSTATE_ENABLED, BTNS_SEP },
242 { moveToImageIndex, gMoveToCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_MoveTo },
243 { copyToImageIndex, gCopyToCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_CopyTo },
244 { deleteImageIndex, gDeleteCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Delete },
245 { undoImageIndex, gUndoCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Undo },
246 { 6, -1, TBSTATE_ENABLED, BTNS_SEP },
247 { viewsImageIndex, gViewsCommandID, TBSTATE_ENABLED, BTNS_WHOLEDROPDOWN, { 0 }, 0, BtnIdx_Views },
248
249 { 0, gStopCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Stop },
250 { 0, IDM_VIEW_REFRESH, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Refresh },
251 { 0, gHomeCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Home },
252 { mapDriveImageIndex, IDM_TOOLS_MAPNETWORKDRIVE, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_MapDrive },
253 { disconnectImageIndex, IDM_TOOLS_DISCONNECTNETWORKDRIVE, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Disconnect },
254 { favoritesImageIndex, gFavoritesCommandID, TBSTATE_ENABLED, BTNS_BUTTON | BTNS_SHOWTEXT, { 0 }, 0, BtnIdx_Favorites },
255 { 0, gHistoryCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_History },
256 { 0, gFullScreenCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_FullScreen },
257 { propertiesImageIndex, gPropertiesCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Properties },
258 { cutImageIndex, gCutCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Cut },
259 { copyImageIndex, gCopyCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Copy },
260 { pasteImageIndex, gPasteCommandID, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_Paste },
261 { folderOptionsImageIndex, IDM_TOOLS_FOLDEROPTIONS, TBSTATE_ENABLED, BTNS_BUTTON, { 0 }, 0, BtnIdx_FolderOptions },
262 };
263
264 HRESULT STDMETHODCALLTYPE CToolsBand::SetSite(IUnknown* pUnkSite){
265 HWND parentWindow;
266 IOleWindow *oleWindow;
267 HWND toolbar;
268 HRESULT hResult;
269
270 if (fDockSite != NULL)
271 fDockSite->Release();
272 if (pUnkSite == NULL)
273 return S_OK;
274 hResult = pUnkSite->QueryInterface(IID_PPV_ARG(IDockingWindowSite, &fDockSite));
275 if (FAILED(hResult))
276 return hResult;
277 parentWindow = NULL;
278 hResult = pUnkSite->QueryInterface(IID_PPV_ARG(IOleWindow, &oleWindow));
279 if (SUCCEEDED(hResult))
280 {
281 oleWindow->GetWindow(&parentWindow);
282 oleWindow->Release();
283 }
284 if (!::IsWindow(parentWindow))
285 return E_FAIL;
286
287 toolbar = CreateWindowEx(
288 TBSTYLE_EX_DOUBLEBUFFER,
289 TOOLBARCLASSNAMEW, NULL,
290 WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN |
291 TBSTYLE_TOOLTIPS | TBSTYLE_TRANSPARENT | TBSTYLE_REGISTERDROP | TBSTYLE_LIST | TBSTYLE_FLAT |
292 CCS_NODIVIDER | CCS_NOPARENTALIGN | CCS_NORESIZE | CCS_TOP,
293 0, 0, 500, 20, parentWindow, NULL, _AtlBaseModule.GetModuleInstance(), 0);
294 if (toolbar == NULL)
295 return E_FAIL;
296 SubclassWindow(toolbar);
297 SendMessage(TB_ADDSTRINGW, (WPARAM) GetModuleHandle(L"browseui.dll"), IDS_STANDARD_TOOLBAR);
298
299 SendMessage(WM_USER + 100, GetSystemMetrics(SM_CXEDGE) / 2, 0);
300 SendMessage(TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0);
301 SendMessage(TB_SETMAXTEXTROWS, 1, 0);
302 SendMessage(TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_MIXEDBUTTONS | TBSTYLE_EX_DRAWDDARROWS);
303
304 HINSTANCE shell32Instance = GetModuleHandle(_T("shell32.dll"));
305 HBITMAP imgNormal = reinterpret_cast<HBITMAP>(
306 LoadImage(shell32Instance, MAKEINTRESOURCE(214),
307 IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE | LR_CREATEDIBSECTION));
308
309 HBITMAP imgHot = reinterpret_cast<HBITMAP>(
310 LoadImage(shell32Instance, MAKEINTRESOURCE(215),
311 IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE | LR_CREATEDIBSECTION));
312
313 if (imgNormal && imgHot)
314 {
315 BITMAP bitmapInfo;
316 GetObjectW(imgNormal, sizeof(bitmapInfo), &bitmapInfo);
317 HIMAGELIST himlNormal = ImageList_Create(bitmapInfo.bmHeight, bitmapInfo.bmHeight, ILC_COLOR32, 4, 4);
318
319 ImageList_Add(himlNormal, imgNormal, NULL);
320 DeleteObject(imgNormal);
321
322 GetObjectW(imgHot, sizeof(bitmapInfo), &bitmapInfo);
323 HIMAGELIST himlHot = ImageList_Create(bitmapInfo.bmHeight, bitmapInfo.bmHeight, ILC_COLOR32, 4, 4);
324
325 ImageList_Add(himlHot, imgHot, NULL);
326 DeleteObject(imgHot);
327
328 SendMessage(TB_SETIMAGELIST, 0, (LPARAM) himlNormal);
329 SendMessage(TB_SETHOTIMAGELIST, 0, (LPARAM) himlHot);
330 }
331
332 SendMessage(TB_ADDBUTTONSW, numShownButtons, (LPARAM)&tbButtonsAdd);
333
334 return hResult;
335 }
336
337 HRESULT STDMETHODCALLTYPE CToolsBand::GetSite(REFIID riid, void **ppvSite)
338 {
339 if (fDockSite == NULL)
340 return E_FAIL;
341 return fDockSite->QueryInterface(riid, ppvSite);
342 }
343
344 HRESULT STDMETHODCALLTYPE CToolsBand::GetWindow(HWND *lphwnd)
345 {
346 if (lphwnd == NULL)
347 return E_POINTER;
348 *lphwnd = m_hWnd;
349 return S_OK;
350 }
351
352 HRESULT STDMETHODCALLTYPE CToolsBand::ContextSensitiveHelp(BOOL fEnterMode)
353 {
354
355 return E_NOTIMPL;
356 }
357
358 HRESULT STDMETHODCALLTYPE CToolsBand::CloseDW(DWORD dwReserved)
359 {
360 ShowDW(FALSE);
361
362 if (IsWindow())
363 DestroyWindow();
364
365 m_hWnd = NULL;
366
367 return S_OK;
368 }
369
370 HRESULT STDMETHODCALLTYPE CToolsBand::ResizeBorderDW(const RECT* prcBorder, IUnknown* punkToolbarSite, BOOL fReserved)
371 {
372 return E_NOTIMPL;
373 }
374
375 HRESULT STDMETHODCALLTYPE CToolsBand::ShowDW(BOOL fShow)
376 {
377 if (m_hWnd)
378 {
379 if (fShow)
380 ShowWindow(SW_SHOW);
381 else
382 ShowWindow(SW_HIDE);
383 }
384 return S_OK;
385 }
386
387 HRESULT STDMETHODCALLTYPE CToolsBand::HasFocusIO()
388 {
389 return E_NOTIMPL;
390 }
391
392 HRESULT STDMETHODCALLTYPE CToolsBand::TranslateAcceleratorIO(LPMSG lpMsg)
393 {
394 return E_NOTIMPL;
395 }
396
397 HRESULT STDMETHODCALLTYPE CToolsBand::UIActivateIO(BOOL fActivate, LPMSG lpMsg)
398 {
399 return E_NOTIMPL;
400 }
401
402 HRESULT STDMETHODCALLTYPE CToolsBand::GetClassID(CLSID *pClassID)
403 {
404 return E_NOTIMPL;
405 }
406
407 HRESULT STDMETHODCALLTYPE CToolsBand::IsDirty()
408 {
409 return E_NOTIMPL;
410 }
411
412 HRESULT STDMETHODCALLTYPE CToolsBand::Load(IStream *pStm)
413 {
414 return E_NOTIMPL;
415 }
416
417 HRESULT STDMETHODCALLTYPE CToolsBand::Save(IStream *pStm, BOOL fClearDirty)
418 {
419 return E_NOTIMPL;
420 }
421
422 HRESULT STDMETHODCALLTYPE CToolsBand::GetSizeMax(ULARGE_INTEGER *pcbSize)
423 {
424 return E_NOTIMPL;
425 }
426
427 LRESULT CToolsBand::OnGetButtonInfo(UINT idControl, NMHDR *pNMHDR, BOOL &bHandled)
428 {
429 TBNOTIFYW *pTBntf = reinterpret_cast<TBNOTIFYW *>(pNMHDR);
430
431 if (pTBntf->iItem >= 0 && pTBntf->iItem < (numShownButtons + numHiddenButtons))
432 {
433 pTBntf->tbButton = tbButtonsAdd[pTBntf->iItem];
434 return TRUE;
435 }
436 else
437 return FALSE;
438 return 0;
439 }
440
441 HRESULT CreateToolsBar(REFIID riid, void **ppv)
442 {
443 CComObject<CToolsBand> *theMenuBar;
444 HRESULT hResult;
445
446 if (ppv == NULL)
447 return E_POINTER;
448 *ppv = NULL;
449 ATLTRY (theMenuBar = new CComObject<CToolsBand>);
450 if (theMenuBar == NULL)
451 return E_OUTOFMEMORY;
452 hResult = theMenuBar->QueryInterface(riid, reinterpret_cast<void **>(ppv));
453 if (FAILED(hResult))
454 {
455 delete theMenuBar;
456 return hResult;
457 }
458 return S_OK;
459 }
460