1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
12 #pragma implementation "choice.h"
19 #include "wx/choice.h"
21 #include "wx/gtk/private.h"
23 //-----------------------------------------------------------------------------
25 //-----------------------------------------------------------------------------
27 extern void wxapp_install_idle_handler();
30 //-----------------------------------------------------------------------------
32 //-----------------------------------------------------------------------------
34 extern bool g_blockEventsOnDrag
;
36 //-----------------------------------------------------------------------------
38 //-----------------------------------------------------------------------------
40 static void gtk_choice_clicked_callback( GtkWidget
*WXUNUSED(widget
), wxChoice
*choice
)
43 wxapp_install_idle_handler();
45 if (!choice
->m_hasVMT
) return;
47 if (g_blockEventsOnDrag
) return;
49 wxCommandEvent
event(wxEVT_COMMAND_CHOICE_SELECTED
, choice
->GetId() );
50 int n
= choice
->GetSelection();
53 event
.SetString( choice
->GetStringSelection() );
54 event
.SetEventObject(choice
);
56 if ( choice
->HasClientObjectData() )
57 event
.SetClientObject( choice
->GetClientObject(n
) );
58 else if ( choice
->HasClientUntypedData() )
59 event
.SetClientData( choice
->GetClientData(n
) );
61 choice
->GetEventHandler()->ProcessEvent(event
);
64 //-----------------------------------------------------------------------------
66 //-----------------------------------------------------------------------------
68 IMPLEMENT_DYNAMIC_CLASS(wxChoice
,wxControl
)
72 m_strings
= (wxSortedArrayString
*)NULL
;
75 bool wxChoice::Create( wxWindow
*parent
, wxWindowID id
,
76 const wxPoint
&pos
, const wxSize
&size
,
77 int n
, const wxString choices
[],
78 long style
, const wxValidator
& validator
, const wxString
&name
)
81 #if (GTK_MINOR_VERSION > 0)
82 m_acceptsFocus
= TRUE
;
85 if (!PreCreation( parent
, pos
, size
) ||
86 !CreateBase( parent
, id
, pos
, size
, style
, validator
, name
))
88 wxFAIL_MSG( wxT("wxChoice creation failed") );
92 m_widget
= gtk_option_menu_new();
94 if ( style
& wxCB_SORT
)
96 // if our m_strings != NULL, DoAppend() will check for it and insert
97 // items in the correct order
98 m_strings
= new wxSortedArrayString
;
101 GtkWidget
*menu
= gtk_menu_new();
103 for (int i
= 0; i
< n
; i
++)
105 GtkAppendHelper(menu
, choices
[i
]);
108 gtk_option_menu_set_menu( GTK_OPTION_MENU(m_widget
), menu
);
110 m_parent
->DoAddChild( this );
114 SetFont( parent
->GetFont() );
118 SetBackgroundColour( parent
->GetBackgroundColour() );
119 SetForegroundColour( parent
->GetForegroundColour() );
126 wxChoice::~wxChoice()
133 int wxChoice::DoAppend( const wxString
&item
)
135 wxCHECK_MSG( m_widget
!= NULL
, -1, wxT("invalid choice control") );
137 GtkWidget
*menu
= gtk_option_menu_get_menu( GTK_OPTION_MENU(m_widget
) );
139 return GtkAppendHelper(menu
, item
);
142 void wxChoice::DoSetItemClientData( int n
, void* clientData
)
144 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid choice control") );
146 wxNode
*node
= m_clientList
.Nth( n
);
147 wxCHECK_RET( node
, wxT("invalid index in wxChoice::DoSetItemClientData") );
149 node
->SetData( (wxObject
*) clientData
);
152 void* wxChoice::DoGetItemClientData( int n
) const
154 wxCHECK_MSG( m_widget
!= NULL
, NULL
, wxT("invalid choice control") );
156 wxNode
*node
= m_clientList
.Nth( n
);
157 wxCHECK_MSG( node
, NULL
, wxT("invalid index in wxChoice::DoGetItemClientData") );
162 void wxChoice::DoSetItemClientObject( int n
, wxClientData
* clientData
)
164 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid choice control") );
166 wxNode
*node
= m_clientList
.Nth( n
);
167 wxCHECK_RET( node
, wxT("invalid index in wxChoice::DoSetItemClientObject") );
169 wxClientData
*cd
= (wxClientData
*) node
->Data();
172 node
->SetData( (wxObject
*) clientData
);
175 wxClientData
* wxChoice::DoGetItemClientObject( int n
) const
177 wxCHECK_MSG( m_widget
!= NULL
, (wxClientData
*) NULL
, wxT("invalid choice control") );
179 wxNode
*node
= m_clientList
.Nth( n
);
180 wxCHECK_MSG( node
, (wxClientData
*)NULL
,
181 wxT("invalid index in wxChoice::DoGetItemClientObject") );
183 return (wxClientData
*) node
->Data();
186 void wxChoice::Clear()
188 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid choice") );
190 gtk_option_menu_remove_menu( GTK_OPTION_MENU(m_widget
) );
191 GtkWidget
*menu
= gtk_menu_new();
192 gtk_option_menu_set_menu( GTK_OPTION_MENU(m_widget
), menu
);
194 if ( HasClientObjectData() )
196 // destroy the data (due to Robert's idea of using wxList<wxObject>
197 // and not wxList<wxClientData> we can't just say
198 // m_clientList.DeleteContents(TRUE) - this would crash!
199 wxNode
*node
= m_clientList
.First();
202 delete (wxClientData
*)node
->Data();
206 m_clientList
.Clear();
212 void wxChoice::Delete( int n
)
214 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid choice") );
216 // VZ: apparently GTK+ doesn't have a built-in function to do it (not even
217 // in 2.0), hence this dump implementation - still better than nothing
221 wxCHECK_RET( n
>= 0 && n
< count
, _T("invalid index in wxChoice::Delete") );
225 for ( i
= 0; i
< count
; i
++ )
228 items
.Add(GetString(i
));
233 for ( i
= 0; i
< count
- 1; i
++ )
239 int wxChoice::FindString( const wxString
&string
) const
241 wxCHECK_MSG( m_widget
!= NULL
, -1, wxT("invalid choice") );
243 // If you read this code once and you think you understand
244 // it, then you are very wrong. Robert Roebling.
246 GtkMenuShell
*menu_shell
= GTK_MENU_SHELL( gtk_option_menu_get_menu( GTK_OPTION_MENU(m_widget
) ) );
248 GList
*child
= menu_shell
->children
;
251 GtkBin
*bin
= GTK_BIN( child
->data
);
252 GtkLabel
*label
= (GtkLabel
*) NULL
;
254 label
= GTK_LABEL(bin
->child
);
256 label
= GTK_LABEL( BUTTON_CHILD(m_widget
) );
258 wxASSERT_MSG( label
!= NULL
, wxT("wxChoice: invalid label") );
261 wxString
tmp( wxGTK_CONV_BACK( gtk_label_get_text( label
) ) );
263 wxString
tmp( label
->label
);
275 int wxChoice::GetSelection() const
277 wxCHECK_MSG( m_widget
!= NULL
, -1, wxT("invalid choice") );
281 return gtk_option_menu_get_history( GTK_OPTION_MENU(m_widget
) );
284 GtkMenuShell
*menu_shell
= GTK_MENU_SHELL( gtk_option_menu_get_menu( GTK_OPTION_MENU(m_widget
) ) );
287 GList
*child
= menu_shell
->children
;
290 GtkBin
*bin
= GTK_BIN( child
->data
);
291 if (!bin
->child
) return count
;
300 void wxChoice::SetString( int WXUNUSED(n
), const wxString
& WXUNUSED(string
) )
302 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid choice") );
304 wxFAIL_MSG(wxT("not implemented"));
307 wxString
wxChoice::GetString( int n
) const
309 wxCHECK_MSG( m_widget
!= NULL
, wxT(""), wxT("invalid choice") );
311 GtkMenuShell
*menu_shell
= GTK_MENU_SHELL( gtk_option_menu_get_menu( GTK_OPTION_MENU(m_widget
) ) );
313 GList
*child
= menu_shell
->children
;
316 GtkBin
*bin
= GTK_BIN( child
->data
);
319 GtkLabel
*label
= (GtkLabel
*) NULL
;
321 label
= GTK_LABEL(bin
->child
);
323 label
= GTK_LABEL( BUTTON_CHILD(m_widget
) );
325 wxASSERT_MSG( label
!= NULL
, wxT("wxChoice: invalid label") );
328 return wxString( wxGTK_CONV_BACK( gtk_label_get_text( label
) ) );
330 return wxString( label
->label
);
337 wxFAIL_MSG( wxT("wxChoice: invalid index in GetString()") );
342 int wxChoice::GetCount() const
344 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid choice") );
346 GtkMenuShell
*menu_shell
= GTK_MENU_SHELL( gtk_option_menu_get_menu( GTK_OPTION_MENU(m_widget
) ) );
348 GList
*child
= menu_shell
->children
;
357 void wxChoice::SetSelection( int n
)
359 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid choice") );
362 gtk_option_menu_set_history( GTK_OPTION_MENU(m_widget
), (gint
)tmp
);
365 void wxChoice::ApplyWidgetStyle()
369 GtkMenuShell
*menu_shell
= GTK_MENU_SHELL( gtk_option_menu_get_menu( GTK_OPTION_MENU(m_widget
) ) );
371 gtk_widget_set_style( m_widget
, m_widgetStyle
);
372 gtk_widget_set_style( GTK_WIDGET( menu_shell
), m_widgetStyle
);
374 GList
*child
= menu_shell
->children
;
377 gtk_widget_set_style( GTK_WIDGET( child
->data
), m_widgetStyle
);
379 GtkBin
*bin
= GTK_BIN( child
->data
);
380 GtkWidget
*label
= (GtkWidget
*) NULL
;
384 label
= BUTTON_CHILD(m_widget
);
386 gtk_widget_set_style( label
, m_widgetStyle
);
392 size_t wxChoice::GtkAppendHelper(GtkWidget
*menu
, const wxString
& item
)
394 GtkWidget
*menu_item
= gtk_menu_item_new_with_label( wxGTK_CONV( item
) );
399 // sorted control, need to insert at the correct index
400 index
= m_strings
->Add(item
);
402 gtk_menu_insert( GTK_MENU(menu
), menu_item
, index
);
406 m_clientList
.Insert( m_clientList
.Item(index
- 1),
411 m_clientList
.Insert( (wxObject
*) NULL
);
416 // normal control, just append
417 gtk_menu_append( GTK_MENU(menu
), menu_item
);
419 m_clientList
.Append( (wxObject
*) NULL
);
421 // don't call wxChoice::GetCount() from here because it doesn't work
422 // if we're called from ctor (and GtkMenuShell is still NULL)
423 index
= m_clientList
.GetCount() - 1;
426 if (GTK_WIDGET_REALIZED(m_widget
))
428 gtk_widget_realize( menu_item
);
429 gtk_widget_realize( GTK_BIN(menu_item
)->child
);
431 if (m_widgetStyle
) ApplyWidgetStyle();
434 gtk_signal_connect( GTK_OBJECT( menu_item
), "activate",
435 GTK_SIGNAL_FUNC(gtk_choice_clicked_callback
), (gpointer
*)this );
437 gtk_widget_show( menu_item
);
439 // return the index of the item in the control
443 wxSize
wxChoice::DoGetBestSize() const
445 wxSize
ret( wxControl::DoGetBestSize() );
447 // we know better our horizontal extent: it depends on the longest string
452 GdkFont
*font
= m_font
.GetInternalFont();
455 size_t count
= GetCount();
456 for ( size_t n
= 0; n
< count
; n
++ )
459 width
= (wxCoord
)gdk_string_width(font
, wxGTK_CONV( GetString(n
) ) );
464 // add extra for the choice "=" button
466 // VZ: I don't know how to get the right value, it seems to be in
467 // GtkOptionMenuProps struct from gtkoptionmenu.c but we can't get
468 // to it - maybe we can use gtk_option_menu_size_request() for this
471 // This default value works only for the default GTK+ theme (i.e.
472 // no theme at all) (FIXME)
473 static const int widthChoiceIndicator
= 35;
474 ret
.x
+= widthChoiceIndicator
;
477 // but not less than the minimal width
481 ret
.y
= 16 + gdk_char_height(GET_STYLE_FONT( m_widget
->style
), 'H' );
486 #endif // wxUSE_CHOICE