321 lines
9.8 KiB
C++
321 lines
9.8 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2018-2024 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 2
|
|
* 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, you may find one here:
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include <widgets/grid_icon_text_helpers.h>
|
|
|
|
#include <wx/artprov.h>
|
|
#include <wx/defs.h>
|
|
#include <wx/textctrl.h>
|
|
#include <wx/dc.h>
|
|
|
|
#include <bitmaps.h>
|
|
|
|
|
|
GRID_CELL_ICON_TEXT_RENDERER::GRID_CELL_ICON_TEXT_RENDERER( const std::vector<BITMAPS>& icons,
|
|
const wxArrayString& names ) :
|
|
m_icons( icons ),
|
|
m_names( names )
|
|
{
|
|
}
|
|
|
|
|
|
void GRID_CELL_ICON_TEXT_RENDERER::Draw( wxGrid& aGrid, wxGridCellAttr& aAttr, wxDC& aDC,
|
|
const wxRect& aRect, int aRow, int aCol, bool isSelected )
|
|
{
|
|
wxString value = aGrid.GetCellValue( aRow, aCol );
|
|
|
|
wxRect rect = aRect;
|
|
rect.Inflate( -1 );
|
|
|
|
// erase background
|
|
wxGridCellRenderer::Draw( aGrid, aAttr, aDC, aRect, aRow, aCol, isSelected );
|
|
|
|
// draw the icon
|
|
// note that the set of icons might be smaller than the set of labels if the last
|
|
// label is <...>.
|
|
int position = m_names.Index( value );
|
|
int leftCut = aDC.FromDIP( 4 );
|
|
|
|
if( position < (int) m_icons.size() && position != wxNOT_FOUND )
|
|
{
|
|
wxBitmapBundle bundle = KiBitmapBundle( m_icons[position] );
|
|
|
|
wxBitmap bitmap = bundle.GetBitmap(
|
|
bundle.GetPreferredBitmapSizeAtScale( aDC.GetContentScaleFactor() ) );
|
|
|
|
aDC.DrawBitmap( bitmap,
|
|
rect.GetLeft() + leftCut,
|
|
rect.GetTop() + ( rect.GetHeight() - bitmap.GetLogicalHeight() ) / 2,
|
|
true );
|
|
|
|
leftCut += bitmap.GetLogicalWidth();
|
|
}
|
|
else // still need a bitmap to fetch the width
|
|
{
|
|
wxBitmapBundle bundle = KiBitmapBundle( m_icons[0] );
|
|
|
|
wxBitmap bitmap = bundle.GetBitmap(
|
|
bundle.GetPreferredBitmapSizeAtScale( aDC.GetContentScaleFactor() ) );
|
|
|
|
leftCut += bitmap.GetLogicalWidth();
|
|
}
|
|
|
|
leftCut += aDC.FromDIP( 4 );
|
|
|
|
rect.x += leftCut;
|
|
rect.width -= leftCut;
|
|
|
|
// draw the text
|
|
SetTextColoursAndFont( aGrid, aAttr, aDC, isSelected );
|
|
aGrid.DrawTextRectangle( aDC, value, rect, wxALIGN_LEFT, wxALIGN_CENTRE );
|
|
}
|
|
|
|
wxSize GRID_CELL_ICON_TEXT_RENDERER::GetBestSize( wxGrid& grid, wxGridCellAttr& attr, wxDC& dc,
|
|
int row, int col )
|
|
{
|
|
int bmpIdx = ( row < (int) m_icons.size() && row >= 0 ) ? row : 0;
|
|
wxBitmapBundle bundle = KiBitmapBundle( m_icons[bmpIdx] );
|
|
|
|
wxBitmap bitmap =
|
|
bundle.GetBitmap( bundle.GetPreferredBitmapSizeAtScale( dc.GetContentScaleFactor() ) );
|
|
|
|
wxString text = grid.GetCellValue( row, col );
|
|
wxSize size = wxGridCellStringRenderer::DoGetBestSize( attr, dc, text );
|
|
|
|
size.x += bitmap.GetLogicalWidth() + dc.FromDIP( 8 );
|
|
|
|
return size;
|
|
}
|
|
|
|
|
|
GRID_CELL_ICON_RENDERER::GRID_CELL_ICON_RENDERER( const wxBitmap& icon ) :
|
|
m_icon( icon )
|
|
{
|
|
}
|
|
|
|
|
|
void GRID_CELL_ICON_RENDERER::Draw( wxGrid& aGrid, wxGridCellAttr& aAttr, wxDC& aDC,
|
|
const wxRect& aRect, int aRow, int aCol, bool isSelected )
|
|
{
|
|
wxRect rect = aRect;
|
|
rect.Inflate( -1 );
|
|
|
|
// erase background
|
|
wxGridCellRenderer::Draw( aGrid, aAttr, aDC, aRect, aRow, aCol, isSelected );
|
|
|
|
// Draw icon
|
|
if( m_icon.IsOk() )
|
|
{
|
|
aDC.DrawBitmap( m_icon,
|
|
rect.GetLeft() + ( rect.GetWidth() - m_icon.GetWidth() ) / 2,
|
|
rect.GetTop() + ( rect.GetHeight() - m_icon.GetHeight() ) / 2,
|
|
true );
|
|
}
|
|
}
|
|
|
|
|
|
wxSize GRID_CELL_ICON_RENDERER::GetBestSize( wxGrid& grid, wxGridCellAttr& attr, wxDC& dc,
|
|
int row, int col )
|
|
{
|
|
return wxSize( m_icon.GetWidth() + 6, m_icon.GetHeight() + 4 );
|
|
}
|
|
|
|
|
|
wxGridCellRenderer* GRID_CELL_ICON_RENDERER::Clone() const
|
|
{
|
|
return new GRID_CELL_ICON_RENDERER( m_icon );
|
|
}
|
|
|
|
|
|
//---- Grid helpers: custom wxGridCellRenderer that renders just an icon ----------------
|
|
//
|
|
// Note: this renderer is supposed to be used with read only cells
|
|
|
|
GRID_CELL_STATUS_ICON_RENDERER::GRID_CELL_STATUS_ICON_RENDERER( int aStatus ) :
|
|
m_status( aStatus )
|
|
{
|
|
if( m_status != 0 )
|
|
{
|
|
m_bitmap = wxArtProvider::GetBitmap( wxArtProvider::GetMessageBoxIconId( m_status ),
|
|
wxART_BUTTON );
|
|
}
|
|
else
|
|
{
|
|
// Dummy bitmap for size
|
|
m_bitmap = wxArtProvider::GetBitmap( wxArtProvider::GetMessageBoxIconId( wxICON_INFORMATION ),
|
|
wxART_BUTTON );
|
|
}
|
|
}
|
|
|
|
|
|
void GRID_CELL_STATUS_ICON_RENDERER::Draw( wxGrid& aGrid, wxGridCellAttr& aAttr, wxDC& aDC,
|
|
const wxRect& aRect, int aRow, int aCol,
|
|
bool isSelected )
|
|
{
|
|
wxRect rect = aRect;
|
|
rect.Inflate( -1 );
|
|
|
|
// erase background
|
|
wxGridCellRenderer::Draw( aGrid, aAttr, aDC, aRect, aRow, aCol, isSelected );
|
|
|
|
// Draw icon
|
|
if( ( m_status != 0 ) && m_bitmap.IsOk() )
|
|
{
|
|
aDC.DrawBitmap( m_bitmap,
|
|
rect.GetLeft() + ( rect.GetWidth() - m_bitmap.GetWidth() ) / 2,
|
|
rect.GetTop() + ( rect.GetHeight() - m_bitmap.GetHeight() ) / 2,
|
|
true );
|
|
}
|
|
}
|
|
|
|
|
|
wxSize GRID_CELL_STATUS_ICON_RENDERER::GetBestSize( wxGrid& grid, wxGridCellAttr& attr, wxDC& dc,
|
|
int row, int col )
|
|
{
|
|
return wxSize( m_bitmap.GetWidth() + 6, m_bitmap.GetHeight() + 4 );
|
|
}
|
|
|
|
|
|
wxGridCellRenderer* GRID_CELL_STATUS_ICON_RENDERER::Clone() const
|
|
{
|
|
return new GRID_CELL_STATUS_ICON_RENDERER( m_status );
|
|
}
|
|
|
|
|
|
GRID_CELL_ICON_TEXT_POPUP::GRID_CELL_ICON_TEXT_POPUP( const std::vector<BITMAPS>& icons,
|
|
const wxArrayString& names ) :
|
|
m_icons( icons ),
|
|
m_names( names )
|
|
{
|
|
}
|
|
|
|
|
|
wxGridCellEditor* GRID_CELL_ICON_TEXT_POPUP::Clone() const
|
|
{
|
|
return new GRID_CELL_ICON_TEXT_POPUP( m_icons, m_names );
|
|
}
|
|
|
|
|
|
void GRID_CELL_ICON_TEXT_POPUP::Create( wxWindow* aParent, wxWindowID aId,
|
|
wxEvtHandler* aEventHandler )
|
|
{
|
|
m_control = new wxBitmapComboBox( aParent, aId, wxEmptyString, wxDefaultPosition,
|
|
wxDefaultSize, 0, nullptr,
|
|
wxCB_READONLY | wxTE_PROCESS_ENTER | wxTE_PROCESS_TAB | wxBORDER_NONE );
|
|
|
|
for( unsigned i = 0; i < m_names.size(); ++i )
|
|
{
|
|
// note that the set of icons might be smaller than the set of labels if
|
|
// the last label is <...>.
|
|
if( i < m_icons.size() && m_icons[ i ] != BITMAPS::INVALID_BITMAP )
|
|
Combo()->Append( m_names[ i ], KiBitmapBundle( m_icons[ i ] ) );
|
|
else
|
|
Combo()->Append( m_names[ i ] );
|
|
}
|
|
|
|
wxGridCellEditor::Create(aParent, aId, aEventHandler);
|
|
}
|
|
|
|
|
|
wxString GRID_CELL_ICON_TEXT_POPUP::GetValue() const
|
|
{
|
|
return Combo()->GetValue();
|
|
}
|
|
|
|
|
|
void GRID_CELL_ICON_TEXT_POPUP::SetSize( const wxRect& aRect )
|
|
{
|
|
wxRect rect( aRect );
|
|
rect.Inflate( -1 );
|
|
|
|
#if !defined( __WXMSW__ ) && !defined( __WXGTK__ )
|
|
// Only implemented in generic wxBitmapComboBox; MSW and GTK use native controls
|
|
Combo()->SetButtonPosition( 0, 0, wxRIGHT, 2 );
|
|
#endif
|
|
|
|
#if defined( __WXMAC__ )
|
|
rect.Inflate( 3 ); // no FOCUS_RING, even on Mac
|
|
#endif
|
|
|
|
Combo()->SetSize( rect, wxSIZE_ALLOW_MINUS_ONE );
|
|
}
|
|
|
|
|
|
void GRID_CELL_ICON_TEXT_POPUP::BeginEdit( int aRow, int aCol, wxGrid* aGrid )
|
|
{
|
|
auto evtHandler = static_cast<wxGridCellEditorEvtHandler*>( m_control->GetEventHandler() );
|
|
|
|
// Don't immediately end if we get a kill focus event within BeginEdit
|
|
evtHandler->SetInSetFocus( true );
|
|
|
|
m_value = aGrid->GetTable()->GetValue( aRow, aCol );
|
|
|
|
Combo()->SetSelection( Combo()->FindString( m_value ) );
|
|
Combo()->SetFocus();
|
|
|
|
#ifdef __WXOSX_COCOA__
|
|
// This is a work around for the combobox being simply dismissed when a
|
|
// choice is made in it under OS X. The bug is almost certainly due to a
|
|
// problem in focus events generation logic but it's not obvious to fix and
|
|
// for now this at least allows one to use wxGrid.
|
|
Combo()->Popup();
|
|
#endif
|
|
|
|
// When dropping down the menu, a kill focus event
|
|
// happens after this point, so we can't reset the flag yet.
|
|
#if !defined(__WXGTK__)
|
|
evtHandler->SetInSetFocus( false );
|
|
#endif
|
|
}
|
|
|
|
|
|
bool GRID_CELL_ICON_TEXT_POPUP::EndEdit( int , int , const wxGrid* , const wxString& ,
|
|
wxString *aNewVal )
|
|
{
|
|
const wxString value = Combo()->GetValue();
|
|
|
|
if( value == m_value )
|
|
return false;
|
|
|
|
m_value = value;
|
|
|
|
if( aNewVal )
|
|
*aNewVal = value;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void GRID_CELL_ICON_TEXT_POPUP::ApplyEdit( int aRow, int aCol, wxGrid* aGrid )
|
|
{
|
|
aGrid->GetTable()->SetValue( aRow, aCol, m_value );
|
|
}
|
|
|
|
|
|
void GRID_CELL_ICON_TEXT_POPUP::Reset()
|
|
{
|
|
Combo()->SetSelection( Combo()->FindString( m_value ) );
|
|
}
|
|
|
|
|