263 lines
7.5 KiB
C++
263 lines
7.5 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2017 KiCad Developers, see AUTHORS.txt for contributors.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation, either version 3 of the License, or (at your
|
|
* option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <functional>
|
|
#include <widgets/footprint_choice.h>
|
|
#include <wx/dc.h>
|
|
#include <wx/pen.h>
|
|
|
|
wxDEFINE_EVENT( EVT_INTERACTIVE_CHOICE, wxCommandEvent );
|
|
|
|
|
|
wxColour FOOTPRINT_CHOICE::m_grey( 0x808080 );
|
|
|
|
|
|
FOOTPRINT_CHOICE::FOOTPRINT_CHOICE( wxWindow* aParent, int aId )
|
|
: wxOwnerDrawnComboBox( aParent, aId, wxEmptyString, wxDefaultPosition, wxDefaultSize,
|
|
/* n */ 0, /* choices */ nullptr, wxCB_READONLY ),
|
|
m_last_selection( 0 )
|
|
{
|
|
}
|
|
|
|
|
|
FOOTPRINT_CHOICE::~FOOTPRINT_CHOICE()
|
|
{
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::DoSetPopupControl( wxComboPopup* aPopup )
|
|
{
|
|
using namespace std::placeholders;
|
|
wxOwnerDrawnComboBox::DoSetPopupControl( aPopup );
|
|
|
|
// Bind events to intercept selections, so the separator can be made nonselectable.
|
|
|
|
GetVListBoxComboPopup()->Bind( wxEVT_MOTION, &FOOTPRINT_CHOICE::TryVetoMouse, this );
|
|
GetVListBoxComboPopup()->Bind( wxEVT_LEFT_DOWN, &FOOTPRINT_CHOICE::TryVetoMouse, this );
|
|
GetVListBoxComboPopup()->Bind( wxEVT_LEFT_UP, &FOOTPRINT_CHOICE::TryVetoMouse, this );
|
|
GetVListBoxComboPopup()->Bind( wxEVT_LEFT_UP, &FOOTPRINT_CHOICE::OnMouseUp, this );
|
|
GetVListBoxComboPopup()->Bind( wxEVT_LEFT_DCLICK, &FOOTPRINT_CHOICE::TryVetoMouse, this );
|
|
GetVListBoxComboPopup()->Bind(
|
|
wxEVT_LISTBOX, std::bind( &FOOTPRINT_CHOICE::TryVetoSelect, this, _1, true ) );
|
|
Bind( wxEVT_COMBOBOX, std::bind( &FOOTPRINT_CHOICE::TryVetoSelect, this, _1, false ) );
|
|
GetVListBoxComboPopup()->Bind(
|
|
wxEVT_CHAR_HOOK, std::bind( &FOOTPRINT_CHOICE::TrySkipSeparator, this, _1, true ) );
|
|
GetVListBoxComboPopup()->Bind( wxEVT_CHAR_HOOK, &FOOTPRINT_CHOICE::OnKeyUp, this );
|
|
Bind( wxEVT_KEY_DOWN, std::bind( &FOOTPRINT_CHOICE::TrySkipSeparator, this, _1, false ) );
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::OnDrawItem( wxDC& aDC, wxRect const& aRect, int aItem, int aFlags ) const
|
|
{
|
|
wxString text = SafeGetString( aItem );
|
|
|
|
if( text == wxEmptyString )
|
|
{
|
|
wxPen pen( m_grey, 1, wxPENSTYLE_SOLID );
|
|
|
|
aDC.SetPen( pen );
|
|
aDC.DrawLine( aRect.x, aRect.y + aRect.height / 2, aRect.x + aRect.width,
|
|
aRect.y + aRect.height / 2 );
|
|
}
|
|
else
|
|
{
|
|
wxCoord x, y;
|
|
|
|
if( aFlags & wxODCB_PAINTING_CONTROL )
|
|
{
|
|
x = aRect.x + GetMargins().x;
|
|
y = ( aRect.height - aDC.GetCharHeight() ) / 2 + aRect.y;
|
|
}
|
|
else
|
|
{
|
|
x = aRect.x + 2;
|
|
y = aRect.y;
|
|
}
|
|
|
|
// If this item has a footprint and that footprint has a ":" delimiter, find the
|
|
// library component, then find that in the display string and grey it out.
|
|
|
|
size_t start_grey = 0;
|
|
size_t end_grey = 0;
|
|
|
|
wxString lib = static_cast<wxStringClientData*>( GetClientObject( aItem ) )->GetData();
|
|
size_t colon_index = lib.rfind( ':' );
|
|
|
|
if( colon_index != wxString::npos )
|
|
{
|
|
wxString library_part = lib.SubString( 0, colon_index );
|
|
size_t library_index = text.rfind( library_part );
|
|
|
|
if( library_index != wxString::npos )
|
|
{
|
|
start_grey = library_index;
|
|
end_grey = start_grey + library_part.Length();
|
|
}
|
|
}
|
|
|
|
if( start_grey != end_grey && !( aFlags & wxODCB_PAINTING_SELECTED ) )
|
|
{
|
|
x = DrawTextFragment( aDC, x, y, text.SubString( 0, start_grey - 1 ) );
|
|
|
|
wxColour standard_color = aDC.GetTextForeground();
|
|
|
|
aDC.SetTextForeground( m_grey );
|
|
x = DrawTextFragment( aDC, x, y, text.SubString( start_grey, end_grey - 1 ) );
|
|
|
|
aDC.SetTextForeground( standard_color );
|
|
x = DrawTextFragment( aDC, x, y, text.SubString( end_grey, text.Length() - 1 ) );
|
|
}
|
|
else
|
|
{
|
|
aDC.DrawText( text, x, y );
|
|
}
|
|
}
|
|
}
|
|
|
|
wxCoord FOOTPRINT_CHOICE::OnMeasureItem( size_t aItem ) const
|
|
{
|
|
if( SafeGetString( aItem ) == "" )
|
|
return 11;
|
|
else
|
|
return wxOwnerDrawnComboBox::OnMeasureItem( aItem );
|
|
}
|
|
|
|
|
|
wxCoord FOOTPRINT_CHOICE::OnMeasureItemWidth( size_t aItem ) const
|
|
{
|
|
if( SafeGetString( aItem ) == "" )
|
|
return GetTextRect().GetWidth() - 2;
|
|
else
|
|
return wxOwnerDrawnComboBox::OnMeasureItemWidth( aItem );
|
|
}
|
|
|
|
|
|
wxCoord FOOTPRINT_CHOICE::DrawTextFragment( wxDC& aDC, wxCoord x, wxCoord y, wxString const& aText )
|
|
{
|
|
aDC.DrawText( aText, x, y );
|
|
return x + aDC.GetTextExtent( aText ).GetWidth();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::TryVetoMouse( wxMouseEvent& aEvent )
|
|
{
|
|
int item = GetVListBoxComboPopup()->VirtualHitTest( aEvent.GetPosition().y );
|
|
|
|
if( SafeGetString( item ) != "" )
|
|
aEvent.Skip();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::OnMouseUp( wxMouseEvent& aEvent )
|
|
{
|
|
int item = GetVListBoxComboPopup()->VirtualHitTest( aEvent.GetPosition().y );
|
|
|
|
wxCommandEvent evt( EVT_INTERACTIVE_CHOICE );
|
|
evt.SetInt( item );
|
|
wxPostEvent( this, evt );
|
|
|
|
aEvent.Skip();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::OnKeyUp( wxKeyEvent& aEvent )
|
|
{
|
|
int item = GetSelectionEither( true );
|
|
|
|
if( aEvent.GetKeyCode() == WXK_RETURN )
|
|
{
|
|
wxCommandEvent evt( EVT_INTERACTIVE_CHOICE );
|
|
evt.SetInt( item );
|
|
wxPostEvent( this, evt );
|
|
}
|
|
|
|
aEvent.Skip();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::TryVetoSelect( wxCommandEvent& aEvent, bool aInner )
|
|
{
|
|
int sel = GetSelectionEither( aInner );
|
|
|
|
if( sel >= 0 && sel < (int) GetCount() )
|
|
{
|
|
wxString text = SafeGetString( sel );
|
|
|
|
if( text == "" )
|
|
SetSelectionEither( aInner, m_last_selection );
|
|
else
|
|
{
|
|
m_last_selection = sel;
|
|
aEvent.Skip();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::TrySkipSeparator( wxKeyEvent& aEvent, bool aInner )
|
|
{
|
|
int key = aEvent.GetKeyCode();
|
|
int sel = GetSelectionEither( aInner );
|
|
int new_sel = sel;
|
|
|
|
if( key == WXK_UP && SafeGetString( sel - 1 ) == wxEmptyString )
|
|
{
|
|
new_sel = sel - 2;
|
|
}
|
|
else if( key == WXK_DOWN && SafeGetString( sel + 1 ) == wxEmptyString )
|
|
{
|
|
new_sel = sel + 2;
|
|
}
|
|
|
|
if( new_sel != sel )
|
|
SetSelectionEither( aInner, new_sel );
|
|
else
|
|
aEvent.Skip();
|
|
}
|
|
|
|
|
|
wxString FOOTPRINT_CHOICE::SafeGetString( int aItem ) const
|
|
{
|
|
if( aItem >= 0 && aItem < (int) GetCount() )
|
|
return GetVListBoxComboPopup()->GetString( aItem );
|
|
else
|
|
return wxEmptyString;
|
|
}
|
|
|
|
|
|
int FOOTPRINT_CHOICE::GetSelectionEither( bool aInner ) const
|
|
{
|
|
if( aInner )
|
|
return GetVListBoxComboPopup()->wxVListBox::GetSelection();
|
|
else
|
|
return GetSelection();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_CHOICE::SetSelectionEither( bool aInner, int aSel )
|
|
{
|
|
if( aSel >= 0 && aSel < (int) GetCount() )
|
|
{
|
|
if( aInner )
|
|
return GetVListBoxComboPopup()->wxVListBox::SetSelection( aSel );
|
|
else
|
|
return SetSelection( aSel );
|
|
}
|
|
}
|