]> git.saurik.com Git - wxWidgets.git/blame - src/msw/tbar95.cpp
*** empty log message ***
[wxWidgets.git] / src / msw / tbar95.cpp
CommitLineData
2bda0e17 1/////////////////////////////////////////////////////////////////////////////
1c383dba 2// Name: msw/tbar95.cpp
8a0681f9 3// Purpose: wxToolBar
2bda0e17
KB
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart and Markus Holzem
89b892a2 9// Licence: wxWindows license
2bda0e17
KB
10/////////////////////////////////////////////////////////////////////////////
11
1c383dba
VZ
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
2bda0e17 20#ifdef __GNUG__
1c383dba 21 #pragma implementation "tbar95.h"
2bda0e17
KB
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#ifdef __BORLANDC__
1c383dba 28 #pragma hdrstop
2bda0e17
KB
29#endif
30
31#ifndef WX_PRECOMP
8a0681f9 32 #include "wx/frame.h"
1c383dba
VZ
33 #include "wx/log.h"
34 #include "wx/intl.h"
35 #include "wx/dynarray.h"
4e15f6c5 36 #include "wx/settings.h"
381dd4bf 37 #include "wx/bitmap.h"
2bda0e17
KB
38#endif
39
8a0681f9
VZ
40#if wxUSE_TOOLBAR && defined(__WIN95__) && wxUSE_TOOLBAR_NATIVE
41
42#include "wx/toolbar.h"
2bda0e17 43
ce3ed50d 44#if !defined(__GNUWIN32__) && !defined(__SALFORDC__)
1c383dba 45 #include "malloc.h"
2bda0e17
KB
46#endif
47
1c383dba 48#include "wx/msw/private.h"
2bda0e17 49
57c208c5 50#ifndef __TWIN32__
1c383dba
VZ
51
52#ifdef __GNUWIN32_OLD__
53 #include "wx/msw/gnuwin32/extra.h"
54#else
55 #include <commctrl.h>
65fd5cb0 56#endif
2bda0e17 57
1c383dba
VZ
58#endif // __TWIN32__
59
2bda0e17 60#include "wx/msw/dib.h"
1c383dba
VZ
61#include "wx/app.h" // for GetComCtl32Version
62
63// ----------------------------------------------------------------------------
64// constants
65// ----------------------------------------------------------------------------
66
67// these standard constants are not always defined in compilers headers
2bda0e17 68
6a23cbce 69// Styles
bb6290e3 70#ifndef TBSTYLE_FLAT
1c383dba
VZ
71 #define TBSTYLE_LIST 0x1000
72 #define TBSTYLE_FLAT 0x0800
73 #define TBSTYLE_TRANSPARENT 0x8000
bb6290e3
JS
74#endif
75 // use TBSTYLE_TRANSPARENT if you use TBSTYLE_FLAT
76
6a23cbce
JS
77// Messages
78#ifndef TB_GETSTYLE
1c383dba 79 #define TB_SETSTYLE (WM_USER + 56)
8a0681f9
VZ
80 #define TB_GETSTYLE (WM_USER + 57)
81#endif
82
83#ifndef TB_HITTEST
84 #define TB_HITTEST (WM_USER + 69)
6a23cbce
JS
85#endif
86
1c383dba 87// these values correspond to those used by comctl32.dll
81d66cf3
JS
88#define DEFAULTBITMAPX 16
89#define DEFAULTBITMAPY 15
90#define DEFAULTBUTTONX 24
91#define DEFAULTBUTTONY 24
92#define DEFAULTBARHEIGHT 27
93
1c383dba 94// ----------------------------------------------------------------------------
8a0681f9 95// private function prototypes
1c383dba
VZ
96// ----------------------------------------------------------------------------
97
98static void wxMapBitmap(HBITMAP hBitmap, int width, int height);
99
100// ----------------------------------------------------------------------------
101// wxWin macros
102// ----------------------------------------------------------------------------
103
8a0681f9 104IMPLEMENT_DYNAMIC_CLASS(wxToolBar, wxControl)
2bda0e17 105
8a0681f9
VZ
106BEGIN_EVENT_TABLE(wxToolBar, wxToolBarBase)
107 EVT_MOUSE_EVENTS(wxToolBar::OnMouseEvent)
108 EVT_SYS_COLOUR_CHANGED(wxToolBar::OnSysColourChanged)
2bda0e17 109END_EVENT_TABLE()
2bda0e17 110
8a0681f9
VZ
111// ----------------------------------------------------------------------------
112// private classes
113// ----------------------------------------------------------------------------
114
115class wxToolBarTool : public wxToolBarToolBase
116{
117public:
118 wxToolBarTool(wxToolBar *tbar,
119 int id,
120 const wxBitmap& bitmap1,
121 const wxBitmap& bitmap2,
122 bool toggle,
123 wxObject *clientData,
124 const wxString& shortHelpString,
125 const wxString& longHelpString)
126 : wxToolBarToolBase(tbar, id, bitmap1, bitmap2, toggle,
127 clientData, shortHelpString, longHelpString)
128 {
129 m_nSepCount = 0;
130 }
131
132 wxToolBarTool(wxToolBar *tbar, wxControl *control)
133 : wxToolBarToolBase(tbar, control)
134 {
135 m_nSepCount = 1;
136 }
137
138 // set/get the number of separators which we use to cover the space used by
139 // a control in the toolbar
140 void SetSeparatorsCount(size_t count) { m_nSepCount = count; }
141 size_t GetSeparatorsCount() const { return m_nSepCount; }
142
143private:
144 size_t m_nSepCount;
145};
146
147
1c383dba
VZ
148// ============================================================================
149// implementation
150// ============================================================================
2bda0e17 151
1c383dba 152// ----------------------------------------------------------------------------
8a0681f9 153// wxToolBarTool
1c383dba
VZ
154// ----------------------------------------------------------------------------
155
8a0681f9
VZ
156wxToolBarToolBase *wxToolBar::CreateTool(int id,
157 const wxBitmap& bitmap1,
158 const wxBitmap& bitmap2,
159 bool toggle,
160 wxObject *clientData,
161 const wxString& shortHelpString,
162 const wxString& longHelpString)
163{
164 return new wxToolBarTool(this, id, bitmap1, bitmap2, toggle,
165 clientData, shortHelpString, longHelpString);
166}
167
168wxToolBarToolBase *wxToolBar::CreateTool(wxControl *control)
169{
170 return new wxToolBarTool(this, control);
171}
172
173// ----------------------------------------------------------------------------
174// wxToolBar construction
175// ----------------------------------------------------------------------------
176
177void wxToolBar::Init()
2bda0e17 178{
1c383dba 179 m_hBitmap = 0;
8a0681f9
VZ
180
181 m_nButtons = 0;
182
1c383dba
VZ
183 m_defaultWidth = DEFAULTBITMAPX;
184 m_defaultHeight = DEFAULTBITMAPY;
2bda0e17
KB
185}
186
8a0681f9
VZ
187bool wxToolBar::Create(wxWindow *parent,
188 wxWindowID id,
189 const wxPoint& pos,
190 const wxSize& size,
191 long style,
192 const wxString& name)
2bda0e17 193{
1c383dba
VZ
194 // common initialisation
195 if ( !CreateControl(parent, id, pos, size, style, name) )
196 return FALSE;
89b892a2 197
1c383dba
VZ
198 // prepare flags
199 DWORD msflags = 0; // WS_VISIBLE | WS_CHILD always included
200 if (style & wxBORDER)
201 msflags |= WS_BORDER;
202 msflags |= TBSTYLE_TOOLTIPS;
2bda0e17 203
1c383dba
VZ
204 if (style & wxTB_FLAT)
205 {
206 if (wxTheApp->GetComCtl32Version() > 400)
c8f1f088 207 msflags |= TBSTYLE_FLAT;
1c383dba 208 }
2bda0e17 209
1c383dba
VZ
210 // MSW-specific initialisation
211 if ( !wxControl::MSWCreateControl(TOOLBARCLASSNAME, msflags) )
212 return FALSE;
2bda0e17 213
c8f1f088 214 // toolbar-specific post initialisation
1c383dba 215 ::SendMessage(GetHwnd(), TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0);
89b892a2 216
c8f1f088
VZ
217 // set up the colors and fonts
218 wxRGBToColour(m_backgroundColour, GetSysColor(COLOR_BTNFACE));
219 m_foregroundColour = *wxBLACK;
220
221 SetFont(wxSystemSettings::GetSystemFont(wxSYS_DEFAULT_GUI_FONT));
222
1c383dba
VZ
223 // position it
224 int x = pos.x;
225 int y = pos.y;
226 int width = size.x;
227 int height = size.y;
2bda0e17 228
1c383dba
VZ
229 if (width <= 0)
230 width = 100;
231 if (height <= 0)
232 height = m_defaultHeight;
233 if (x < 0)
234 x = 0;
235 if (y < 0)
236 y = 0;
bb6290e3 237
1c383dba 238 SetSize(x, y, width, height);
2bda0e17 239
1c383dba 240 return TRUE;
2bda0e17
KB
241}
242
8a0681f9 243wxToolBar::~wxToolBar()
2bda0e17 244{
1c383dba
VZ
245 if (m_hBitmap)
246 {
247 ::DeleteObject((HBITMAP) m_hBitmap);
248 }
249}
250
bdc72a22 251// ----------------------------------------------------------------------------
8a0681f9 252// adding/removing tools
bdc72a22
VZ
253// ----------------------------------------------------------------------------
254
8a0681f9
VZ
255bool wxToolBar::DoInsertTool(size_t WXUNUSED(pos),
256 wxToolBarToolBase *tool)
1c383dba 257{
8a0681f9
VZ
258 // nothing special to do here - we really create the toolbar buttons in
259 // Realize() later
260 tool->Attach(this);
261
262 return TRUE;
1c383dba
VZ
263}
264
8a0681f9 265bool wxToolBar::DoDeleteTool(size_t pos, wxToolBarToolBase *tool)
bdc72a22 266{
8a0681f9
VZ
267 // normally, we only delete one button, but we use several separators to
268 // cover the space used by one control sometimes (with old comctl32.dll)
269 size_t nButtonsToDelete = 1;
bdc72a22 270
8a0681f9
VZ
271 // get the size of the button we're going to delete
272 RECT r;
273 if ( !::SendMessage(GetHwnd(), TB_GETITEMRECT, pos, (LPARAM)&r) )
bdc72a22 274 {
8a0681f9 275 wxLogLastError(_T("TB_GETITEMRECT"));
bdc72a22
VZ
276 }
277
8a0681f9 278 int width = r.right - r.left;
bdc72a22 279
8a0681f9
VZ
280 if ( tool->IsControl() )
281 {
282 nButtonsToDelete = ((wxToolBarTool *)tool)->GetSeparatorsCount();
bdc72a22 283
8a0681f9
VZ
284 width *= nButtonsToDelete;
285 }
bdc72a22 286
8a0681f9
VZ
287 while ( nButtonsToDelete-- > 0 )
288 {
289 if ( !::SendMessage(GetHwnd(), TB_DELETEBUTTON, pos, 0) )
290 {
291 wxLogLastError("TB_DELETEBUTTON");
1c383dba 292
8a0681f9
VZ
293 return FALSE;
294 }
295 }
1c383dba 296
8a0681f9 297 tool->Detach();
1c383dba 298
8a0681f9
VZ
299 m_nButtons -= nButtonsToDelete;
300
301 // reposition all the controls after this button
302 wxToolBarToolsList::Node *node = m_tools.Item(pos);
303 for ( node = node->GetNext(); node; node = node->GetNext() )
304 {
305 wxToolBarToolBase *tool2 = node->GetData();
306 if ( tool2->IsControl() )
307 {
308 int x;
309 wxControl *control = tool2->GetControl();
310 control->GetPosition(&x, NULL);
311 control->Move(x - width, -1);
312 }
313 }
1c383dba
VZ
314
315 return TRUE;
316}
317
8a0681f9 318bool wxToolBar::Realize()
1c383dba 319{
8a0681f9
VZ
320 size_t nTools = GetToolsCount();
321 if ( nTools == 0 )
322 {
323 // nothing to do
324 return TRUE;
325 }
1c383dba 326
8a0681f9 327 bool isVertical = (GetWindowStyle() & wxTB_VERTICAL) != 0;
2bda0e17 328
8a0681f9
VZ
329 // First, add the bitmap: we use one bitmap for all toolbar buttons
330 // ----------------------------------------------------------------
2bda0e17 331
8a0681f9
VZ
332 // if we already have a bitmap, we'll replace the existing one - otherwise
333 // we'll install a new one
334 HBITMAP oldToolBarBitmap = (HBITMAP)m_hBitmap;
89b892a2 335
1c383dba
VZ
336 int totalBitmapWidth = (int)(m_defaultWidth * nTools);
337 int totalBitmapHeight = (int)m_defaultHeight;
2bda0e17 338
1c383dba 339 // Create a bitmap for all the tool bitmaps
8a0681f9
VZ
340 HBITMAP hBitmap = ::CreateCompatibleBitmap(ScreenHDC(),
341 totalBitmapWidth,
342 totalBitmapHeight);
343 if ( !hBitmap )
344 {
345 wxLogLastError(_T("CreateCompatibleBitmap"));
346
347 return FALSE;
348 }
349
350 m_hBitmap = (WXHBITMAP)hBitmap;
89b892a2 351
1c383dba
VZ
352 // Now blit all the tools onto this bitmap
353 HDC memoryDC = ::CreateCompatibleDC(NULL);
8a0681f9 354 HBITMAP oldBitmap = (HBITMAP) ::SelectObject(memoryDC, hBitmap);
2bda0e17 355
1c383dba 356 HDC memoryDC2 = ::CreateCompatibleDC(NULL);
2bda0e17 357
1c383dba
VZ
358 // the button position
359 wxCoord x = 0;
2bda0e17 360
1c383dba 361 // the number of buttons (not separators)
8a0681f9 362 int nButtons = 0;
1c383dba 363
8a0681f9
VZ
364 wxToolBarToolsList::Node *node = m_tools.GetFirst();
365 while ( node )
051205e6 366 {
8a0681f9
VZ
367 wxToolBarToolBase *tool = node->GetData();
368 if ( tool->IsButton() )
1c383dba 369 {
8a0681f9 370 HBITMAP hbmp = GetHbitmapOf(tool->GetBitmap1());
1c383dba
VZ
371 if ( hbmp )
372 {
373 HBITMAP oldBitmap2 = (HBITMAP)::SelectObject(memoryDC2, hbmp);
374 if ( !BitBlt(memoryDC, x, 0, m_defaultWidth, m_defaultHeight,
375 memoryDC2, 0, 0, SRCCOPY) )
376 {
377 wxLogLastError("BitBlt");
378 }
379
380 ::SelectObject(memoryDC2, oldBitmap2);
1c383dba 381 }
8a0681f9
VZ
382 else
383 {
384 wxFAIL_MSG( _T("invalid tool button bitmap") );
385 }
386
387 // still inc width and number of buttons because otherwise the
388 // subsequent buttons will all be shifted which is rather confusing
389 // (and like this you'd see immediately which bitmap was bad)
390 x += m_defaultWidth;
391 nButtons++;
1c383dba 392 }
8a0681f9
VZ
393
394 node = node->GetNext();
051205e6 395 }
2bda0e17 396
1c383dba
VZ
397 ::SelectObject(memoryDC, oldBitmap);
398 ::DeleteDC(memoryDC);
399 ::DeleteDC(memoryDC2);
2bda0e17 400
1c383dba 401 // Map to system colours
8a0681f9 402 wxMapBitmap(hBitmap, totalBitmapWidth, totalBitmapHeight);
1c383dba
VZ
403
404 if ( oldToolBarBitmap )
2bda0e17 405 {
1c383dba
VZ
406 TBREPLACEBITMAP replaceBitmap;
407 replaceBitmap.hInstOld = NULL;
408 replaceBitmap.hInstNew = NULL;
409 replaceBitmap.nIDOld = (UINT) oldToolBarBitmap;
8a0681f9
VZ
410 replaceBitmap.nIDNew = (UINT) hBitmap;
411 replaceBitmap.nButtons = nButtons;
412 if ( !::SendMessage(GetHwnd(), TB_REPLACEBITMAP,
413 0, (LPARAM) &replaceBitmap) )
1c383dba
VZ
414 {
415 wxFAIL_MSG(wxT("Could not add bitmap to toolbar"));
416 }
417
8a0681f9 418 ::DeleteObject(oldToolBarBitmap);
1c383dba
VZ
419
420 // Now delete all the buttons
8a0681f9 421 for ( size_t pos = 0; pos < m_nButtons; pos++ )
1c383dba 422 {
8a0681f9
VZ
423 if ( !::SendMessage(GetHwnd(), TB_DELETEBUTTON, 0, 0) )
424 {
425 wxLogLastError("TB_DELETEBUTTON");
426 }
1c383dba 427 }
2bda0e17 428 }
8a0681f9 429 else // no old bitmap
051205e6 430 {
1c383dba
VZ
431 TBADDBITMAP addBitmap;
432 addBitmap.hInst = 0;
8a0681f9 433 addBitmap.nID = (UINT) hBitmap;
1c383dba 434 if ( ::SendMessage(GetHwnd(), TB_ADDBITMAP,
8a0681f9 435 (WPARAM) nButtons, (LPARAM)&addBitmap) == -1 )
1c383dba
VZ
436 {
437 wxFAIL_MSG(wxT("Could not add bitmap to toolbar"));
438 }
051205e6 439 }
2bda0e17 440
8a0681f9
VZ
441 // Next add the buttons and separators
442 // -----------------------------------
443
1c383dba 444 TBBUTTON *buttons = new TBBUTTON[nTools];
2bda0e17 445
8a0681f9 446 // this array will hold the indices of all controls in the toolbar
1c383dba
VZ
447 wxArrayInt controlIds;
448
449 int i = 0;
450 int bitmapId = 0;
451
8a0681f9 452 for ( node = m_tools.GetFirst(); node; node = node->GetNext() )
2bda0e17 453 {
8a0681f9
VZ
454 wxToolBarToolBase *tool = node->GetData();
455
456 // don't add separators to the vertical toolbar - looks ugly
457 if ( isVertical && tool->IsSeparator() )
458 continue;
459
1c383dba
VZ
460 TBBUTTON& button = buttons[i];
461
462 wxZeroMemory(button);
463
8a0681f9 464 switch ( tool->GetStyle() )
1c383dba
VZ
465 {
466 case wxTOOL_STYLE_CONTROL:
8a0681f9 467 button.idCommand = tool->GetId();
1c383dba
VZ
468 // fall through: create just a separator too
469
470 case wxTOOL_STYLE_SEPARATOR:
471 button.fsState = TBSTATE_ENABLED;
472 button.fsStyle = TBSTYLE_SEP;
473 break;
474
475 case wxTOOL_STYLE_BUTTON:
476 button.iBitmap = bitmapId;
8a0681f9 477 button.idCommand = tool->GetId();
1c383dba 478
8a0681f9 479 if ( tool->IsEnabled() )
1c383dba 480 button.fsState |= TBSTATE_ENABLED;
8a0681f9 481 if ( tool->IsToggled() )
1c383dba 482 button.fsState |= TBSTATE_CHECKED;
8a0681f9
VZ
483
484 button.fsStyle = tool->CanBeToggled() ? TBSTYLE_CHECK
485 : TBSTYLE_BUTTON;
1c383dba
VZ
486
487 bitmapId++;
488 break;
489 }
2bda0e17 490
1c383dba 491 i++;
2bda0e17 492 }
1c383dba
VZ
493
494 if ( !::SendMessage(GetHwnd(), TB_ADDBUTTONS,
495 (WPARAM)i, (LPARAM)buttons) )
2bda0e17 496 {
1c383dba 497 wxLogLastError("TB_ADDBUTTONS");
2bda0e17 498 }
89b892a2 499
1c383dba 500 delete [] buttons;
2bda0e17 501
8a0681f9
VZ
502 // Deal with the controls finally
503 // ------------------------------
504
1c383dba 505 // adjust the controls size to fit nicely in the toolbar
8a0681f9
VZ
506 size_t index = 0;
507 for ( node = m_tools.GetFirst(); node; node = node->GetNext(), index++ )
1c383dba 508 {
8a0681f9
VZ
509 wxToolBarToolBase *tool = node->GetData();
510 if ( !tool->IsControl() )
511 continue;
512
1c383dba
VZ
513 wxControl *control = tool->GetControl();
514
515 wxSize size = control->GetSize();
516
bdc72a22
VZ
517 // the position of the leftmost controls corner
518 int left = -1;
519
8a0681f9
VZ
520 // note that we use TB_GETITEMRECT and not TB_GETRECT because the
521 // latter only appeared in v4.70 of comctl32.dll
522 RECT r;
523 if ( !SendMessage(GetHwnd(), TB_GETITEMRECT,
524 index, (LPARAM)(LPRECT)&r) )
525 {
526 wxLogLastError("TB_GETITEMRECT");
527 }
528
bdc72a22
VZ
529 // TB_SETBUTTONINFO message is only supported by comctl32.dll 4.71+
530 #if defined(_WIN32_IE) && (_WIN32_IE >= 0x400 )
531 // available in headers, now check whether it is available now
532 // (during run-time)
533 if ( wxTheApp->GetComCtl32Version() >= 471 )
534 {
535 // set the (underlying) separators width to be that of the
536 // control
537 TBBUTTONINFO tbbi;
538 tbbi.cbSize = sizeof(tbbi);
539 tbbi.dwMask = TBIF_SIZE;
540 tbbi.cx = size.x;
541 if ( !SendMessage(GetHwnd(), TB_SETBUTTONINFO,
8a0681f9 542 tool->GetId(), (LPARAM)&tbbi) )
bdc72a22 543 {
8a0681f9 544 // the id is probably invalid?
bdc72a22
VZ
545 wxLogLastError("TB_SETBUTTONINFO");
546 }
547
548 }
549 else
550 #endif // comctl32.dll 4.71
551 // TB_SETBUTTONINFO unavailable
552 {
bdc72a22 553 // try adding several separators to fit the controls width
bdc72a22
VZ
554 int widthSep = r.right - r.left;
555 left = r.left;
556
557 TBBUTTON tbb;
558 wxZeroMemory(tbb);
559 tbb.idCommand = 0;
560 tbb.fsState = TBSTATE_ENABLED;
561 tbb.fsStyle = TBSTYLE_SEP;
562
563 size_t nSeparators = size.x / widthSep;
564 for ( size_t nSep = 0; nSep < nSeparators; nSep++ )
565 {
bdc72a22
VZ
566 if ( !SendMessage(GetHwnd(), TB_INSERTBUTTON,
567 index, (LPARAM)&tbb) )
568 {
569 wxLogLastError("TB_INSERTBUTTON");
570 }
8a0681f9
VZ
571
572 index++;
bdc72a22
VZ
573 }
574
8a0681f9
VZ
575 // remember the number of separators we used - we'd have to
576 // delete all of them later
577 ((wxToolBarTool *)tool)->SetSeparatorsCount(nSeparators);
578
bdc72a22
VZ
579 // adjust the controls width to exactly cover the separators
580 control->SetSize((nSeparators + 1)*widthSep, -1);
581 }
1c383dba
VZ
582
583 // and position the control itself correctly vertically
1c383dba
VZ
584 int height = r.bottom - r.top;
585 int diff = height - size.y;
586 if ( diff < 0 )
587 {
588 // the control is too high, resize to fit
589 control->SetSize(-1, height - 2);
590
591 diff = 2;
592 }
2bda0e17 593
8a0681f9 594 control->Move(left == -1 ? r.left : left, r.top + (diff + 1) / 2);
1c383dba 595 }
89b892a2 596
8a0681f9
VZ
597 // the max index is the "real" number of buttons - i.e. counting even the
598 // separators which we added just for aligning the controls
599 m_nButtons = index;
89b892a2 600
8a0681f9
VZ
601 if ( !isVertical )
602 {
603 if ( m_maxRows == 0 )
604 {
605 // if not set yet, only one row
606 SetRows(1);
607 }
608 }
609 else if ( m_nButtons > 0 ) // vertical non empty toolbar
610 {
611 if ( m_maxRows == 0 )
612 {
613 // if not set yet, have one column
614 SetRows(m_nButtons);
615 }
616 }
2bda0e17 617
1c383dba 618 return TRUE;
2bda0e17
KB
619}
620
1c383dba
VZ
621// ----------------------------------------------------------------------------
622// message handlers
623// ----------------------------------------------------------------------------
624
8a0681f9 625bool wxToolBar::MSWCommand(WXUINT cmd, WXWORD id)
2bda0e17 626{
8a0681f9
VZ
627 wxToolBarToolBase *tool = FindById((int)id);
628 if ( !tool )
1c383dba
VZ
629 return FALSE;
630
8a0681f9 631 if ( tool->CanBeToggled() )
1c383dba
VZ
632 {
633 LRESULT state = ::SendMessage(GetHwnd(), TB_GETSTATE, id, 0);
8a0681f9 634 tool->SetToggle((state & TBSTATE_CHECKED) != 0);
1c383dba
VZ
635 }
636
8a0681f9
VZ
637 bool toggled = tool->IsToggled();
638
639 // OnLeftClick() can veto the button state change - for buttons which may
640 // be toggled only, of couse
641 if ( !OnLeftClick((int)id, toggled) && tool->CanBeToggled() )
1c383dba 642 {
8a0681f9
VZ
643 // revert back
644 toggled = !toggled;
645 tool->SetToggle(toggled);
646
647 ::SendMessage(GetHwnd(), TB_CHECKBUTTON, id, MAKELONG(toggled, 0));
1c383dba
VZ
648 }
649
650 return TRUE;
2bda0e17
KB
651}
652
8a0681f9 653bool wxToolBar::MSWOnNotify(int WXUNUSED(idCtrl),
fd3f686c
VZ
654 WXLPARAM lParam,
655 WXLPARAM *result)
2bda0e17 656{
89b892a2 657 // First check if this applies to us
2bda0e17 658 NMHDR *hdr = (NMHDR *)lParam;
2bda0e17 659
1c383dba
VZ
660 // the tooltips control created by the toolbar is sometimes Unicode, even
661 // in an ANSI application - this seems to be a bug in comctl32.dll v5
a17e237f
VZ
662 int code = (int)hdr->code;
663 if ( (code != TTN_NEEDTEXTA) && (code != TTN_NEEDTEXTW) )
89b892a2 664 return FALSE;
2bda0e17 665
89b892a2
VZ
666 HWND toolTipWnd = (HWND)::SendMessage((HWND)GetHWND(), TB_GETTOOLTIPS, 0, 0);
667 if ( toolTipWnd != hdr->hwndFrom )
668 return FALSE;
2bda0e17 669
89b892a2
VZ
670 LPTOOLTIPTEXT ttText = (LPTOOLTIPTEXT)lParam;
671 int id = (int)ttText->hdr.idFrom;
89b892a2 672
8a0681f9
VZ
673 wxToolBarToolBase *tool = FindById(id);
674 if ( !tool )
675 return FALSE;
2bda0e17 676
8a0681f9 677 const wxString& help = tool->GetShortHelp();
8df13671
VZ
678
679 if ( !help.IsEmpty() )
89b892a2 680 {
a17e237f 681 if ( code == TTN_NEEDTEXTA )
89b892a2 682 {
837e5743 683 ttText->lpszText = (wxChar *)help.c_str();
89b892a2
VZ
684 }
685#if (_WIN32_IE >= 0x0300)
686 else
687 {
688 // FIXME this is a temp hack only until I understand better what
689 // must be done in both ANSI and Unicode builds
8df13671
VZ
690
691 size_t lenAnsi = help.Len();
b2cce0c4 692 #ifdef __MWERKS__
8df13671
VZ
693 // MetroWerks doesn't like calling mbstowcs with NULL argument
694 size_t lenUnicode = 2*lenAnsi;
b2cce0c4 695 #else
8df13671 696 size_t lenUnicode = mbstowcs(NULL, help, lenAnsi);
b2cce0c4 697 #endif
8df13671
VZ
698
699 // using the pointer of right type avoids us doing all sorts of
700 // pointer arithmetics ourselves
701 wchar_t *dst = (wchar_t *)ttText->szText,
702 *pwz = new wchar_t[lenUnicode + 1];
703 mbstowcs(pwz, help, lenAnsi + 1);
704 memcpy(dst, pwz, lenUnicode*sizeof(wchar_t));
705
706 // put the terminating _wide_ NUL
707 dst[lenUnicode] = 0;
89b892a2
VZ
708
709 delete [] pwz;
710 }
711#endif // _WIN32_IE >= 0x0300
2bda0e17 712 }
89b892a2
VZ
713
714 // For backward compatibility...
8a0681f9 715 OnMouseEnter(tool->GetId());
89b892a2
VZ
716
717 return TRUE;
2bda0e17
KB
718}
719
1c383dba 720// ----------------------------------------------------------------------------
8a0681f9 721// toolbar geometry
1c383dba
VZ
722// ----------------------------------------------------------------------------
723
8a0681f9 724void wxToolBar::SetToolBitmapSize(const wxSize& size)
2bda0e17 725{
1c383dba
VZ
726 wxToolBarBase::SetToolBitmapSize(size);
727
728 ::SendMessage(GetHwnd(), TB_SETBITMAPSIZE, 0, MAKELONG(size.x, size.y));
2bda0e17
KB
729}
730
8a0681f9 731void wxToolBar::SetRows(int nRows)
2bda0e17 732{
8a0681f9
VZ
733 if ( nRows == m_maxRows )
734 {
735 // avoid resizing the frame uselessly
736 return;
737 }
738
739 // TRUE in wParam means to create at least as many rows, FALSE -
740 // at most as many
1c383dba
VZ
741 RECT rect;
742 ::SendMessage(GetHwnd(), TB_SETROWS,
8a0681f9
VZ
743 MAKEWPARAM(nRows, !(GetWindowStyle() & wxTB_VERTICAL)),
744 (LPARAM) &rect);
1c383dba
VZ
745
746 m_maxRows = nRows;
8a0681f9
VZ
747
748 UpdateSize();
749}
750
751// The button size is bigger than the bitmap size
752wxSize wxToolBar::GetToolSize() const
753{
754 // TB_GETBUTTONSIZE is supported from version 4.70
755#if defined(_WIN32_IE) && (_WIN32_IE >= 0x300 )
756 if ( wxTheApp->GetComCtl32Version() >= 470 )
757 {
758 DWORD dw = ::SendMessage(GetHwnd(), TB_GETBUTTONSIZE, 0, 0);
759
760 return wxSize(LOWORD(dw), HIWORD(dw));
761 }
762 else
763#endif // comctl32.dll 4.70+
764 {
765 // defaults
766 return wxSize(m_defaultWidth + 8, m_defaultHeight + 7);
767 }
2bda0e17
KB
768}
769
8a0681f9 770wxToolBarToolBase *wxToolBar::FindToolForPosition(wxCoord x, wxCoord y) const
2bda0e17 771{
8a0681f9
VZ
772 POINT pt;
773 pt.x = x;
774 pt.y = y;
775 int index = (int)::SendMessage(GetHwnd(), TB_HITTEST, 0, (LPARAM)&pt);
776 if ( index < 0 )
1c383dba 777 {
8a0681f9
VZ
778 // it's a separator or there is no tool at all there
779 return (wxToolBarToolBase *)NULL;
1c383dba
VZ
780 }
781
8a0681f9 782 return m_tools.Item((size_t)index)->GetData();
2bda0e17
KB
783}
784
8a0681f9 785void wxToolBar::UpdateSize()
2bda0e17 786{
8a0681f9
VZ
787 // we must refresh the frame after the toolbar size (possibly) changed
788 wxFrame *frame = wxDynamicCast(GetParent(), wxFrame);
789 if ( frame )
790 {
791 // don't change the size, we just need to generate a WM_SIZE
792 RECT r;
793 if ( !GetWindowRect(GetHwndOf(frame), &r) )
794 {
795 wxLogLastError(_T("GetWindowRect"));
796 }
797
798 (void)::SendMessage(GetHwndOf(frame), WM_SIZE, SIZE_RESTORED,
799 MAKELPARAM(r.right - r.left, r.bottom - r.top));
800 }
801
2bda0e17
KB
802}
803
1c383dba
VZ
804// ----------------------------------------------------------------------------
805// tool state
806// ----------------------------------------------------------------------------
807
8a0681f9 808void wxToolBar::DoEnableTool(wxToolBarToolBase *tool, bool enable)
2bda0e17 809{
8a0681f9
VZ
810 ::SendMessage(GetHwnd(), TB_ENABLEBUTTON,
811 (WPARAM)tool->GetId(), (LPARAM)MAKELONG(enable, 0));
2bda0e17
KB
812}
813
8a0681f9 814void wxToolBar::DoToggleTool(wxToolBarToolBase *tool, bool toggle)
2bda0e17 815{
8a0681f9
VZ
816 ::SendMessage(GetHwnd(), TB_CHECKBUTTON,
817 (WPARAM)tool->GetId(), (LPARAM)MAKELONG(toggle, 0));
2bda0e17
KB
818}
819
8a0681f9 820void wxToolBar::DoSetToggle(wxToolBarToolBase *tool, bool toggle)
088a95f5 821{
8a0681f9
VZ
822 // VZ: AFAIK, the button has to be created either with TBSTYLE_CHECK or
823 // without, so we really need to delete the button and recreate it here
824 wxFAIL_MSG( _T("not implemented") );
2bda0e17
KB
825}
826
1c383dba
VZ
827// ----------------------------------------------------------------------------
828// event handlers
829// ----------------------------------------------------------------------------
2bda0e17
KB
830
831// Responds to colour changes, and passes event on to children.
8a0681f9 832void wxToolBar::OnSysColourChanged(wxSysColourChangedEvent& event)
2bda0e17
KB
833{
834 m_backgroundColour = wxColour(GetRValue(GetSysColor(COLOR_BTNFACE)),
89b892a2 835 GetGValue(GetSysColor(COLOR_BTNFACE)), GetBValue(GetSysColor(COLOR_BTNFACE)));
2bda0e17
KB
836
837 // Remap the buttons
8a0681f9 838 Realize();
2bda0e17 839
2bda0e17
KB
840 Refresh();
841
842 // Propagate the event to the non-top-level children
843 wxWindow::OnSysColourChanged(event);
844}
845
8a0681f9 846void wxToolBar::OnMouseEvent(wxMouseEvent& event)
e6460682
JS
847{
848 if (event.RightDown())
849 {
850 // For now, we don't have an id. Later we could
851 // try finding the tool.
852 OnRightClick((int)-1, event.GetX(), event.GetY());
853 }
854 else
855 {
42e69d6b 856 event.Skip();
e6460682
JS
857 }
858}
859
8a0681f9 860long wxToolBar::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam)
bdc72a22 861{
8a0681f9 862 if ( nMsg == WM_SIZE )
bdc72a22 863 {
8a0681f9
VZ
864 // calculate our minor dimenstion ourselves - we're confusing the
865 // standard logic (TB_AUTOSIZE) with our horizontal toolbars and other
866 // hacks
867 RECT r;
868 if ( ::SendMessage(GetHwnd(), TB_GETITEMRECT, 0, (LPARAM)&r) )
869 {
870 int w, h;
871
872 if ( GetWindowStyle() & wxTB_VERTICAL )
873 {
874 w = r.right - r.left;
875 if ( m_maxRows )
876 {
877 w *= (m_nButtons + m_maxRows - 1)/m_maxRows;
878 }
879 h = HIWORD(lParam);
880 }
881 else
882 {
883 w = LOWORD(lParam);
884 h = r.bottom - r.top;
885 if ( m_maxRows )
886 {
887 h += 6; // FIXME: this is the separator line height...
888 h *= m_maxRows;
889 }
890 }
891
892 if ( MAKELPARAM(w, h) != lParam )
893 {
894 // size really changed
895 SetSize(w, h);
896 }
897
898 // message processed
899 return 0;
900 }
bdc72a22
VZ
901 }
902
8a0681f9 903 return wxControl::MSWWindowProc(nMsg, wParam, lParam);
bdc72a22
VZ
904}
905
1c383dba
VZ
906// ----------------------------------------------------------------------------
907// private functions
908// ----------------------------------------------------------------------------
909
bdc72a22
VZ
910// These are the default colors used to map the bitmap colors to the current
911// system colors. Note that they are in BGR format because this is what Windows
912// wants (and not RGB)
2bda0e17
KB
913
914#define BGR_BUTTONTEXT (RGB(000,000,000)) // black
915#define BGR_BUTTONSHADOW (RGB(128,128,128)) // dark grey
916#define BGR_BUTTONFACE (RGB(192,192,192)) // bright grey
917#define BGR_BUTTONHILIGHT (RGB(255,255,255)) // white
8a0681f9 918#define BGR_BACKGROUNDSEL (RGB(255,000,000)) // blue
2bda0e17
KB
919#define BGR_BACKGROUND (RGB(255,000,255)) // magenta
920
921void wxMapBitmap(HBITMAP hBitmap, int width, int height)
922{
1c383dba
VZ
923 COLORMAP ColorMap[] =
924 {
2bda0e17
KB
925 {BGR_BUTTONTEXT, COLOR_BTNTEXT}, // black
926 {BGR_BUTTONSHADOW, COLOR_BTNSHADOW}, // dark grey
927 {BGR_BUTTONFACE, COLOR_BTNFACE}, // bright grey
928 {BGR_BUTTONHILIGHT, COLOR_BTNHIGHLIGHT},// white
929 {BGR_BACKGROUNDSEL, COLOR_HIGHLIGHT}, // blue
930 {BGR_BACKGROUND, COLOR_WINDOW} // magenta
931 };
932
933 int NUM_MAPS = (sizeof(ColorMap)/sizeof(COLORMAP));
934 int n;
935 for ( n = 0; n < NUM_MAPS; n++)
936 {
937 ColorMap[n].to = ::GetSysColor(ColorMap[n].to);
938 }
939
940 HBITMAP hbmOld;
941 HDC hdcMem = CreateCompatibleDC(NULL);
942
943 if (hdcMem)
944 {
c4e7c2aa 945 hbmOld = (HBITMAP) SelectObject(hdcMem, hBitmap);
2bda0e17
KB
946
947 int i, j, k;
948 for ( i = 0; i < width; i++)
949 {
950 for ( j = 0; j < height; j++)
951 {
952 COLORREF pixel = ::GetPixel(hdcMem, i, j);
953/*
954 BYTE red = GetRValue(pixel);
955 BYTE green = GetGValue(pixel);
956 BYTE blue = GetBValue(pixel);
957*/
958
959 for ( k = 0; k < NUM_MAPS; k ++)
960 {
961 if ( ColorMap[k].from == pixel )
962 {
963 /* COLORREF actualPixel = */ ::SetPixel(hdcMem, i, j, ColorMap[k].to);
964 break;
965 }
966 }
967 }
968 }
969
970
971 SelectObject(hdcMem, hbmOld);
972 DeleteObject(hdcMem);
973 }
974
975}
976
977// Some experiments...
978#if 0
979 // What we want to do is create another bitmap which has a depth of 4,
980 // and set the bits. So probably we want to convert this HBITMAP into a
981 // DIB, then call SetDIBits.
982 // AAAGH. The stupid thing is that if newBitmap has a depth of 4 (less than that of
983 // the screen), then SetDIBits fails.
984 HBITMAP newBitmap = ::CreateBitmap(totalBitmapWidth, totalBitmapHeight, 1, 4, NULL);
985 HANDLE newDIB = ::BitmapToDIB((HBITMAP) m_hBitmap, NULL);
986 LPBITMAPINFOHEADER lpbmi = (LPBITMAPINFOHEADER) GlobalLock(newDIB);
987
988 dc = ::GetDC(NULL);
989// LPBITMAPINFOHEADER lpbmi = (LPBITMAPINFOHEADER) newDIB;
990
991 int result = ::SetDIBits(dc, newBitmap, 0, lpbmi->biHeight, FindDIBBits((LPSTR)lpbmi), (LPBITMAPINFO)lpbmi,
992 DIB_PAL_COLORS);
993 DWORD err = GetLastError();
994
995 ::ReleaseDC(NULL, dc);
996
997 // Delete the DIB
998 GlobalUnlock (newDIB);
999 GlobalFree (newDIB);
1000
1001// WXHBITMAP hBitmap2 = wxCreateMappedBitmap((WXHINSTANCE) wxGetInstance(), (WXHBITMAP) m_hBitmap);
1002 // Substitute our new bitmap for the old one
1003 ::DeleteObject((HBITMAP) m_hBitmap);
1004 m_hBitmap = (WXHBITMAP) newBitmap;
1005#endif
1006
1007
8a0681f9 1008#endif // wxUSE_TOOLBAR && Win95