1 /*
2  * This program source code file is part of KiCad, a free EDA CAD application.
3  *
4  * Copyright (C) 2018 KiCad Developers, see AUTHORS.txt for contributors.
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License
8  * as published by the Free Software Foundation; either version 2
9  * of the License, or (at your option) any later version.
10  *
11  * This program 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
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, you may find one here:
18  * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19  * or you may search the http://www.gnu.org website for the version 2 license,
20  * or you may write to the Free Software Foundation, Inc.,
21  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
22  */
23 
24 #include <widgets/grid_combobox.h>
25 
26 
27 //-------- Renderer ---------------------------------------------------------------------
28 // None required; just render as normal text.
29 
30 
31 
32 //-------- Editor -----------------------------------------------------------------------
33 //
34 // Note: this implementation is an adaptation of wxGridCellChoiceEditor
35 
36 
GRID_CELL_COMBOBOX(const wxArrayString & names)37 GRID_CELL_COMBOBOX::GRID_CELL_COMBOBOX( const wxArrayString& names ) :
38         m_names( names )
39 {
40 }
41 
42 
Clone() const43 wxGridCellEditor* GRID_CELL_COMBOBOX::Clone() const
44 {
45     return new GRID_CELL_COMBOBOX( m_names );
46 }
47 
48 
Create(wxWindow * aParent,wxWindowID aId,wxEvtHandler * aEventHandler)49 void GRID_CELL_COMBOBOX::Create( wxWindow* aParent, wxWindowID aId, wxEvtHandler* aEventHandler )
50 {
51     m_control = new wxComboBox( aParent, wxID_ANY, wxEmptyString,
52                                 wxDefaultPosition, wxDefaultSize, m_names );
53 
54     wxGridCellEditor::Create(aParent, aId, aEventHandler);
55 }
56 
GetValue() const57 wxString GRID_CELL_COMBOBOX::GetValue() const
58 {
59     return Combo()->GetValue();
60 }
61 
SetSize(const wxRect & aRect)62 void GRID_CELL_COMBOBOX::SetSize( const wxRect& aRect )
63 {
64     wxRect rect( aRect );
65     rect.Inflate( -1 );
66 
67 #if defined( __WXMAC__ )
68     rect.Inflate( 3 );      // no FOCUS_RING, even on Mac
69 #endif
70 
71     Combo()->SetSize( rect, wxSIZE_ALLOW_MINUS_ONE );
72 }
73 
74 
BeginEdit(int aRow,int aCol,wxGrid * aGrid)75 void GRID_CELL_COMBOBOX::BeginEdit( int aRow, int aCol, wxGrid* aGrid )
76 {
77     auto evtHandler = static_cast<wxGridCellEditorEvtHandler*>( m_control->GetEventHandler() );
78 
79     // Don't immediately end if we get a kill focus event within BeginEdit
80     evtHandler->SetInSetFocus( true );
81 
82     // These event handlers are needed to properly dismiss the editor when the popup is closed
83     m_control->Bind(wxEVT_COMBOBOX_DROPDOWN, &GRID_CELL_COMBOBOX::onComboDropDown, this);
84     m_control->Bind(wxEVT_COMBOBOX_CLOSEUP, &GRID_CELL_COMBOBOX::onComboCloseUp, this);
85 
86     m_value = aGrid->GetTable()->GetValue( aRow, aCol );
87 
88     Combo()->SetFocus();
89 
90     // Work around a wxWidgets bug where the drop-down is the wrong width on the first drop.
91     Combo()->Set( Combo()->GetStrings() );
92 
93 #ifdef __WXOSX_COCOA__
94     // This is a work around for the combobox being simply dismissed when a
95     // choice is made in it under OS X. The bug is almost certainly due to a
96     // problem in focus events generation logic but it's not obvious to fix and
97     // for now this at least allows one to use wxGrid.
98     Combo()->Popup();
99 #endif
100 
101     // When dropping down the menu, a kill focus event
102     // happens after this point, so we can't reset the flag yet.
103 #if !defined(__WXGTK20__)
104     evtHandler->SetInSetFocus( false );
105 #endif
106 }
107 
108 
EndEdit(int,int,const wxGrid *,const wxString &,wxString * aNewVal)109 bool GRID_CELL_COMBOBOX::EndEdit( int , int , const wxGrid* , const wxString& , wxString *aNewVal )
110 {
111     const wxString value = Combo()->GetValue();
112 
113     if( value == m_value )
114         return false;
115 
116     m_value = value;
117 
118     if( aNewVal )
119         *aNewVal = value;
120 
121     return true;
122 }
123 
124 
ApplyEdit(int aRow,int aCol,wxGrid * aGrid)125 void GRID_CELL_COMBOBOX::ApplyEdit( int aRow, int aCol, wxGrid* aGrid )
126 {
127     aGrid->GetTable()->SetValue( aRow, aCol, m_value );
128 }
129 
130 
Reset()131 void GRID_CELL_COMBOBOX::Reset()
132 {
133 }
134 
135 
onComboDropDown(wxCommandEvent & aEvent)136 void GRID_CELL_COMBOBOX::onComboDropDown( wxCommandEvent& aEvent )
137 {
138     auto evtHandler = static_cast<wxGridCellEditorEvtHandler*>( m_control->GetEventHandler() );
139 
140     // Once the combobox is dropped, reset the flag to allow the focus-loss handler
141     // to function and close the editor.
142     evtHandler->SetInSetFocus( false );
143 }
144 
145 
onComboCloseUp(wxCommandEvent & aEvent)146 void GRID_CELL_COMBOBOX::onComboCloseUp( wxCommandEvent& aEvent )
147 {
148     auto evtHandler = static_cast<wxGridCellEditorEvtHandler*>( m_control->GetEventHandler() );
149 
150     // Forward the combobox close up event to the cell event handler as a focus kill event
151     // so that the grid editor is dismissed when the combox closes, otherwise it leaves the
152     // dropdown arrow visible in the cell.
153     wxFocusEvent event( wxEVT_KILL_FOCUS, m_control->GetId() );
154     event.SetEventObject( m_control );
155     evtHandler->ProcessEvent( event );
156 }
157