- if (!cb->HasVMT()) return;
- if (g_blockEventsOnDrag) return;
-
- wxCommandEvent event(wxEVT_COMMAND_CHECKBOX_CLICKED, cb->GetId());
- event.SetInt( cb->GetValue() );
- event.SetEventObject(cb);
- cb->GetEventHandler()->ProcessEvent(event);
+ if (g_isIdle) wxapp_install_idle_handler();
+
+ if (!cb->m_hasVMT) return;
+
+ if (g_blockEventsOnDrag) return;
+
+ if (cb->m_blockEvent) return;
+
+#ifdef __WXGTK20__
+ // Transitions for 3state checkbox must be done manually, GTK's checkbox
+ // is 2state with additional "undetermined state" flag which isn't
+ // changed automatically:
+ if (cb->Is3State())
+ {
+ GtkToggleButton *toggle = GTK_TOGGLE_BUTTON(widget);
+
+ if (cb->Is3rdStateAllowedForUser())
+ {
+ // The 3 states cycle like this when clicked:
+ // checked -> undetermined -> unchecked -> checked -> ...
+ bool active = gtk_toggle_button_get_active(toggle);
+ bool inconsistent = gtk_toggle_button_get_inconsistent(toggle);
+
+ cb->m_blockEvent = true;
+
+ if (!active && !inconsistent)
+ {
+ // checked -> undetermined
+ gtk_toggle_button_set_active(toggle, true);
+ gtk_toggle_button_set_inconsistent(toggle, true);
+ }
+ else if (!active && inconsistent)
+ {
+ // undetermined -> unchecked
+ gtk_toggle_button_set_inconsistent(toggle, false);
+ }
+ else if (active && !inconsistent)
+ {
+ // unchecked -> checked
+ // nothing to do
+ }
+ else
+ {
+ wxFAIL_MSG(_T("3state wxCheckBox in unexpected state!"));
+ }
+
+ cb->m_blockEvent = false;
+ }
+ else
+ {
+ // user's action unsets undetermined state:
+ gtk_toggle_button_set_inconsistent(toggle, false);
+ }
+ }
+#endif
+
+ wxCommandEvent event(wxEVT_COMMAND_CHECKBOX_CLICKED, cb->GetId());
+#ifdef __WXGTK20__
+ event.SetInt(cb->Get3StateValue());
+#else
+ event.SetInt(cb->GetValue());
+#endif
+ event.SetEventObject(cb);
+ cb->GetEventHandler()->ProcessEvent(event);