2023-09-26 17:31:45 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2023-11-08 12:29:14 +00:00
|
|
|
* Copyright (C) 2023 CERN
|
2023-09-26 17:31:45 +00:00
|
|
|
* Copyright (C) 2023 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 <pgm_base.h>
|
2023-09-29 16:02:57 +00:00
|
|
|
#include <kiface_base.h>
|
2023-09-26 17:31:45 +00:00
|
|
|
#include <kiway.h>
|
|
|
|
#include <kiway_express.h>
|
|
|
|
#include <wx/button.h>
|
2023-09-29 16:02:57 +00:00
|
|
|
#include <wx/checkbox.h>
|
2023-09-26 17:31:45 +00:00
|
|
|
#include <kiplatform/ui.h>
|
|
|
|
#include <widgets/panel_footprint_chooser.h>
|
|
|
|
#include <settings/settings_manager.h>
|
|
|
|
#include <footprint_editor_settings.h>
|
|
|
|
#include <footprint_chooser_frame.h>
|
2023-11-03 10:34:00 +00:00
|
|
|
#include "wx/display.h"
|
2023-12-08 15:40:19 +00:00
|
|
|
#include <3d_canvas/eda_3d_canvas.h>
|
|
|
|
#include <project_pcb.h>
|
|
|
|
#include <widgets/bitmap_button.h>
|
2023-09-26 17:31:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
static wxArrayString s_FootprintHistoryList;
|
|
|
|
static unsigned s_FootprintHistoryMaxCount = 8;
|
|
|
|
|
|
|
|
static void AddFootprintToHistory( const wxString& aName )
|
|
|
|
{
|
|
|
|
// Remove duplicates
|
2023-09-29 16:02:57 +00:00
|
|
|
for( int ii = (int) s_FootprintHistoryList.GetCount() - 1; ii >= 0; --ii )
|
2023-09-26 17:31:45 +00:00
|
|
|
{
|
|
|
|
if( s_FootprintHistoryList[ ii ] == aName )
|
2023-09-28 13:09:45 +00:00
|
|
|
s_FootprintHistoryList.RemoveAt( (size_t) ii );
|
2023-09-26 17:31:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add the new name at the beginning of the history list
|
|
|
|
s_FootprintHistoryList.Insert( aName, 0 );
|
|
|
|
|
|
|
|
// Remove extra names
|
|
|
|
while( s_FootprintHistoryList.GetCount() >= s_FootprintHistoryMaxCount )
|
|
|
|
s_FootprintHistoryList.RemoveAt( s_FootprintHistoryList.GetCount() - 1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BEGIN_EVENT_TABLE( FOOTPRINT_CHOOSER_FRAME, PCB_BASE_FRAME )
|
2023-09-29 16:02:57 +00:00
|
|
|
EVT_MENU( wxID_CLOSE, FOOTPRINT_CHOOSER_FRAME::closeFootprintChooser )
|
2023-09-26 17:31:45 +00:00
|
|
|
EVT_BUTTON( wxID_OK, FOOTPRINT_CHOOSER_FRAME::OnOK )
|
2023-09-29 16:02:57 +00:00
|
|
|
EVT_BUTTON( wxID_CANCEL, FOOTPRINT_CHOOSER_FRAME::closeFootprintChooser )
|
2023-09-26 17:31:45 +00:00
|
|
|
EVT_PAINT( FOOTPRINT_CHOOSER_FRAME::OnPaint )
|
|
|
|
END_EVENT_TABLE()
|
|
|
|
|
|
|
|
|
2023-10-03 13:48:27 +00:00
|
|
|
#define MODAL_FRAME ( wxRESIZE_BORDER | wxSYSTEM_MENU | wxCAPTION | wxCLOSE_BOX | wxCLIP_CHILDREN \
|
2023-09-26 17:31:45 +00:00
|
|
|
| wxWANTS_CHARS | wxFRAME_NO_TASKBAR | wxSTAY_ON_TOP )
|
|
|
|
|
|
|
|
|
|
|
|
FOOTPRINT_CHOOSER_FRAME::FOOTPRINT_CHOOSER_FRAME( KIWAY* aKiway, wxWindow* aParent ) :
|
2023-09-29 16:02:57 +00:00
|
|
|
PCB_BASE_FRAME( aKiway, aParent, FRAME_FOOTPRINT_CHOOSER, _( "Footprint Chooser" ),
|
2023-10-03 13:48:27 +00:00
|
|
|
wxDefaultPosition, wxDefaultSize, MODAL_FRAME,
|
2023-09-29 16:02:57 +00:00
|
|
|
FOOTPRINT_CHOOSER_FRAME_NAME ),
|
|
|
|
m_filterByPinCount( nullptr ),
|
|
|
|
m_filterByFPFilters( nullptr ),
|
2023-12-08 15:40:19 +00:00
|
|
|
m_boardAdapter(),
|
|
|
|
m_currentCamera( m_trackBallCamera ),
|
|
|
|
m_trackBallCamera( 2 * RANGE_SCALE_3D ),
|
2023-09-30 16:17:09 +00:00
|
|
|
m_pinCount( 0 ),
|
|
|
|
m_firstPaintEvent( true )
|
2023-09-26 17:31:45 +00:00
|
|
|
{
|
|
|
|
SetModal( true );
|
|
|
|
|
2023-12-08 15:40:19 +00:00
|
|
|
m_showFpMode = true;
|
2024-02-16 20:15:59 +00:00
|
|
|
m_messagePanel->Hide();
|
|
|
|
|
2024-02-16 20:18:11 +00:00
|
|
|
wxPanel* bottomPanel = new wxPanel( this );
|
|
|
|
wxBoxSizer* bottomSizer = new wxBoxSizer( wxVERTICAL );
|
|
|
|
|
|
|
|
m_filterByFPFilters = new wxCheckBox( bottomPanel, wxID_ANY, _( "Apply footprint filters" ) );
|
|
|
|
m_filterByPinCount = new wxCheckBox( bottomPanel, wxID_ANY, _( "Filter by pin count" ) );
|
2023-11-08 12:29:14 +00:00
|
|
|
|
|
|
|
m_filterByFPFilters->Show( false );
|
|
|
|
m_filterByPinCount->Show( false );
|
|
|
|
|
|
|
|
if( PCBNEW_SETTINGS* cfg = dynamic_cast<PCBNEW_SETTINGS*>( Kiface().KifaceSettings() ) )
|
|
|
|
{
|
|
|
|
m_filterByFPFilters->SetValue( cfg->m_FootprintChooser.use_fp_filters );
|
|
|
|
m_filterByPinCount->SetValue( cfg->m_FootprintChooser.filter_on_pin_count );
|
|
|
|
}
|
|
|
|
|
2024-02-16 20:18:11 +00:00
|
|
|
wxBoxSizer* frameSizer = new wxBoxSizer( wxVERTICAL );
|
|
|
|
|
2023-09-26 17:31:45 +00:00
|
|
|
m_chooserPanel = new PANEL_FOOTPRINT_CHOOSER( this, this, s_FootprintHistoryList,
|
2023-09-29 16:02:57 +00:00
|
|
|
// Filter
|
2023-11-08 12:29:14 +00:00
|
|
|
[this]( LIB_TREE_NODE& aNode ) -> bool
|
2023-09-29 16:02:57 +00:00
|
|
|
{
|
|
|
|
return filterFootprint( aNode );
|
|
|
|
},
|
2023-12-10 18:11:17 +00:00
|
|
|
// Accept handler
|
2023-09-29 16:02:57 +00:00
|
|
|
[this]()
|
|
|
|
{
|
|
|
|
wxCommandEvent dummy;
|
|
|
|
OnOK( dummy );
|
2023-12-10 18:11:17 +00:00
|
|
|
},
|
|
|
|
// Escape handler
|
|
|
|
[this]()
|
|
|
|
{
|
|
|
|
DismissModal( false );
|
2023-09-29 16:02:57 +00:00
|
|
|
} );
|
2023-09-26 17:31:45 +00:00
|
|
|
|
2024-02-16 20:18:11 +00:00
|
|
|
frameSizer->Add( m_chooserPanel, 1, wxEXPAND );
|
2023-09-26 17:31:45 +00:00
|
|
|
|
2023-12-08 15:40:19 +00:00
|
|
|
build3DCanvas(); // must be called after creating m_chooserPanel
|
|
|
|
m_preview3DCanvas->Show( !m_showFpMode );
|
|
|
|
|
2023-09-29 16:02:57 +00:00
|
|
|
wxBoxSizer* fpFilterSizer = new wxBoxSizer( wxVERTICAL );
|
2023-12-08 15:40:19 +00:00
|
|
|
fpFilterSizer->Add( m_filterByFPFilters, 0, wxLEFT | wxTOP | wxEXPAND, 5 );
|
2023-09-29 16:02:57 +00:00
|
|
|
|
|
|
|
wxBoxSizer* buttonsSizer = new wxBoxSizer( wxHORIZONTAL );
|
2023-12-08 15:40:19 +00:00
|
|
|
fpFilterSizer->Add( m_filterByPinCount, 0, wxLEFT | wxTOP | wxBOTTOM, 5 );
|
|
|
|
buttonsSizer->Add( fpFilterSizer, 0, wxEXPAND | wxLEFT, 10 );
|
|
|
|
|
|
|
|
wxBoxSizer* grbuttSizer = new wxBoxSizer( wxHORIZONTAL );
|
|
|
|
m_grButton3DView = new BITMAP_BUTTON( bottomPanel, wxID_ANY,
|
|
|
|
wxNullBitmap, wxDefaultPosition,
|
|
|
|
wxDefaultSize/*, wxBU_AUTODRAW|wxBORDER_NONE*/ );
|
|
|
|
m_grButton3DView->SetIsRadioButton();
|
|
|
|
m_grButton3DView->SetBitmap( KiBitmapBundle( BITMAPS::shape_3d ) );
|
|
|
|
m_grButton3DView->Check( !m_showFpMode );
|
|
|
|
grbuttSizer->Add( m_grButton3DView, 0, wxALIGN_CENTER_VERTICAL, 5 );
|
|
|
|
|
|
|
|
m_grButtonFpView = new BITMAP_BUTTON( bottomPanel, wxID_ANY,
|
|
|
|
wxNullBitmap, wxDefaultPosition,
|
|
|
|
wxDefaultSize/*, wxBU_AUTODRAW|wxBORDER_NONE*/ );
|
|
|
|
m_grButtonFpView->SetIsRadioButton();
|
|
|
|
m_grButtonFpView->SetBitmap( KiBitmapBundle( BITMAPS::module ) );
|
|
|
|
m_grButtonFpView->Check( m_showFpMode );
|
|
|
|
grbuttSizer->Add( m_grButtonFpView, 0, wxALIGN_CENTER_VERTICAL, 5 );
|
2023-09-29 16:02:57 +00:00
|
|
|
|
2023-09-26 17:31:45 +00:00
|
|
|
wxStdDialogButtonSizer* sdbSizer = new wxStdDialogButtonSizer();
|
2024-02-16 20:18:11 +00:00
|
|
|
wxButton* okButton = new wxButton( bottomPanel, wxID_OK );
|
|
|
|
wxButton* cancelButton = new wxButton( bottomPanel, wxID_CANCEL );
|
2023-09-29 16:02:57 +00:00
|
|
|
|
2023-09-26 17:31:45 +00:00
|
|
|
sdbSizer->AddButton( okButton );
|
|
|
|
sdbSizer->AddButton( cancelButton );
|
|
|
|
sdbSizer->Realize();
|
|
|
|
|
2023-12-08 15:40:19 +00:00
|
|
|
buttonsSizer->Add( grbuttSizer, 1, wxALL, 5 );
|
2023-09-29 16:02:57 +00:00
|
|
|
buttonsSizer->Add( sdbSizer, 1, wxALL, 5 );
|
|
|
|
|
2024-02-16 20:18:11 +00:00
|
|
|
bottomSizer->Add( buttonsSizer, 0, wxEXPAND | wxLEFT, 5 );
|
|
|
|
|
|
|
|
bottomPanel->SetSizer( bottomSizer );
|
|
|
|
frameSizer->Add( bottomPanel, 0, wxEXPAND );
|
|
|
|
|
|
|
|
SetSizer( frameSizer );
|
2023-09-26 17:31:45 +00:00
|
|
|
|
|
|
|
SetTitle( GetTitle() + wxString::Format( _( " (%d items loaded)" ),
|
|
|
|
m_chooserPanel->GetItemCount() ) );
|
|
|
|
|
|
|
|
Layout();
|
|
|
|
m_chooserPanel->FinishSetup();
|
2023-09-29 16:02:57 +00:00
|
|
|
|
|
|
|
m_filterByPinCount->Bind( wxEVT_CHECKBOX,
|
|
|
|
[&]( wxCommandEvent& evt )
|
|
|
|
{
|
|
|
|
m_chooserPanel->Regenerate();
|
|
|
|
} );
|
|
|
|
|
|
|
|
m_filterByFPFilters->Bind( wxEVT_CHECKBOX,
|
|
|
|
[&]( wxCommandEvent& evt )
|
|
|
|
{
|
|
|
|
m_chooserPanel->Regenerate();
|
|
|
|
} );
|
2023-12-08 15:40:19 +00:00
|
|
|
|
|
|
|
// Connect Events
|
|
|
|
m_grButton3DView->Connect( wxEVT_COMMAND_BUTTON_CLICKED ,
|
|
|
|
wxCommandEventHandler( FOOTPRINT_CHOOSER_FRAME::on3DviewReq ),
|
|
|
|
NULL, this );
|
|
|
|
m_grButtonFpView->Connect( wxEVT_COMMAND_BUTTON_CLICKED ,
|
|
|
|
wxCommandEventHandler( FOOTPRINT_CHOOSER_FRAME::onFpViewReq ),
|
|
|
|
NULL, this );
|
|
|
|
|
|
|
|
Connect( FP_SELECTION_EVENT, // custom event fired by a PANEL_FOOTPRINT_CHOOSER
|
|
|
|
wxCommandEventHandler( FOOTPRINT_CHOOSER_FRAME::onFpChanged ), NULL, this );
|
2023-09-29 16:02:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FOOTPRINT_CHOOSER_FRAME::~FOOTPRINT_CHOOSER_FRAME()
|
|
|
|
{
|
2023-12-08 15:40:19 +00:00
|
|
|
// Disconnect Events
|
|
|
|
m_grButton3DView->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED,
|
|
|
|
wxCommandEventHandler( FOOTPRINT_CHOOSER_FRAME::on3DviewReq ), NULL, this );
|
|
|
|
m_grButtonFpView->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED,
|
|
|
|
wxCommandEventHandler( FOOTPRINT_CHOOSER_FRAME::onFpViewReq ), NULL, this );
|
|
|
|
|
|
|
|
Disconnect( FP_SELECTION_EVENT,
|
|
|
|
wxCommandEventHandler( FOOTPRINT_CHOOSER_FRAME::onFpChanged ), NULL, this );
|
|
|
|
|
2023-09-29 16:02:57 +00:00
|
|
|
if( PCBNEW_SETTINGS* cfg = dynamic_cast<PCBNEW_SETTINGS*>( Kiface().KifaceSettings() ) )
|
|
|
|
{
|
|
|
|
cfg->m_FootprintChooser.use_fp_filters = m_filterByFPFilters->GetValue();
|
|
|
|
cfg->m_FootprintChooser.filter_on_pin_count = m_filterByPinCount->GetValue();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-08 12:29:14 +00:00
|
|
|
bool FOOTPRINT_CHOOSER_FRAME::filterFootprint( LIB_TREE_NODE& aNode )
|
2023-09-29 16:02:57 +00:00
|
|
|
{
|
2023-11-08 17:34:14 +00:00
|
|
|
if( aNode.m_Type == LIB_TREE_NODE::TYPE::LIBRARY )
|
2023-11-08 12:29:14 +00:00
|
|
|
{
|
|
|
|
// Normally lib nodes get scored by the max of their children's scores. However, if a
|
|
|
|
// lib node *has* no children then the scorer will call the filter on the lib node itself,
|
|
|
|
// and we just want to return true if we're not filtering at all.
|
|
|
|
return !m_filterByPinCount->GetValue() && !m_filterByFPFilters->GetValue();
|
|
|
|
}
|
2023-10-02 15:24:31 +00:00
|
|
|
|
|
|
|
auto patternMatch =
|
|
|
|
[]( LIB_ID& id, std::vector<std::unique_ptr<EDA_PATTERN_MATCH>>& filters ) -> bool
|
|
|
|
{
|
|
|
|
// The matching is case insensitive
|
|
|
|
wxString name;
|
|
|
|
|
|
|
|
for( const std::unique_ptr<EDA_PATTERN_MATCH>& filter : filters )
|
|
|
|
{
|
|
|
|
name.Empty();
|
|
|
|
|
|
|
|
// If the filter contains a ':' then include the library name in the pattern
|
|
|
|
if( filter->GetPattern().Contains( wxS( ":" ) ) )
|
|
|
|
name = id.GetUniStringLibNickname().Lower() + wxS( ":" );
|
|
|
|
|
|
|
|
name += id.GetUniStringLibItemName().Lower();
|
|
|
|
|
|
|
|
if( filter->Find( name ) )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2023-09-29 16:02:57 +00:00
|
|
|
if( m_pinCount > 0 && m_filterByPinCount->GetValue() )
|
|
|
|
{
|
|
|
|
if( aNode.m_PinCount != m_pinCount )
|
2023-11-08 12:29:14 +00:00
|
|
|
return false;
|
2023-09-29 16:02:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if( !m_fpFilters.empty() && m_filterByFPFilters->GetValue() )
|
|
|
|
{
|
2023-10-02 15:24:31 +00:00
|
|
|
if( !patternMatch( aNode.m_LibId, m_fpFilters ) )
|
2023-11-08 12:29:14 +00:00
|
|
|
return false;
|
2023-09-29 16:02:57 +00:00
|
|
|
}
|
|
|
|
|
2023-11-08 12:29:14 +00:00
|
|
|
return true;
|
2023-09-26 17:31:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FOOTPRINT_CHOOSER_FRAME::doCloseWindow()
|
|
|
|
{
|
|
|
|
// Only dismiss a modal frame once, so that the return values set by
|
|
|
|
// the prior DismissModal() are not bashed for ShowModal().
|
|
|
|
if( !IsDismissed() )
|
|
|
|
DismissModal( false );
|
|
|
|
|
|
|
|
// window to be destroyed by the caller of KIWAY_PLAYER::ShowModal()
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
WINDOW_SETTINGS* FOOTPRINT_CHOOSER_FRAME::GetWindowSettings( APP_SETTINGS_BASE* aCfg )
|
|
|
|
{
|
|
|
|
PCBNEW_SETTINGS* cfg = dynamic_cast<PCBNEW_SETTINGS*>( aCfg );
|
|
|
|
wxCHECK_MSG( cfg, nullptr, wxT( "config not existing" ) );
|
|
|
|
|
|
|
|
return &cfg->m_FootprintViewer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
COLOR_SETTINGS* FOOTPRINT_CHOOSER_FRAME::GetColorSettings( bool aForceRefresh ) const
|
|
|
|
{
|
|
|
|
auto* settings = Pgm().GetSettingsManager().GetAppSettings<FOOTPRINT_EDITOR_SETTINGS>();
|
|
|
|
|
|
|
|
if( settings )
|
|
|
|
return Pgm().GetSettingsManager().GetColorSettings( settings->m_ColorTheme );
|
|
|
|
else
|
|
|
|
return Pgm().GetSettingsManager().GetColorSettings();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FOOTPRINT_CHOOSER_FRAME::KiwayMailIn( KIWAY_EXPRESS& mail )
|
|
|
|
{
|
|
|
|
const std::string& payload = mail.GetPayload();
|
|
|
|
|
|
|
|
switch( mail.Command() )
|
|
|
|
{
|
|
|
|
case MAIL_SYMBOL_NETLIST:
|
|
|
|
{
|
2023-09-29 16:02:57 +00:00
|
|
|
m_pinCount = 0;
|
|
|
|
m_fpFilters.clear();
|
|
|
|
|
2023-09-26 17:31:45 +00:00
|
|
|
/*
|
|
|
|
* Symbol netlist format:
|
2023-09-29 16:02:57 +00:00
|
|
|
* pinCount
|
|
|
|
* fpFilters
|
2023-09-26 17:31:45 +00:00
|
|
|
*/
|
|
|
|
std::vector<std::string> strings = split( payload, "\r" );
|
|
|
|
|
2023-09-29 16:02:57 +00:00
|
|
|
if( strings.size() >= 1 )
|
2023-09-26 17:31:45 +00:00
|
|
|
{
|
2023-09-29 16:02:57 +00:00
|
|
|
wxString pinCountStr( strings[0] );
|
|
|
|
pinCountStr.ToInt( &m_pinCount );
|
2023-09-26 17:31:45 +00:00
|
|
|
|
2023-09-29 16:02:57 +00:00
|
|
|
if( m_pinCount > 0 )
|
|
|
|
{
|
|
|
|
m_filterByPinCount->SetLabel( m_filterByPinCount->GetLabel()
|
|
|
|
+ wxString::Format( wxS( " (%d)" ), m_pinCount ) );
|
|
|
|
m_filterByPinCount->Show( true );
|
|
|
|
}
|
|
|
|
}
|
2023-09-26 17:31:45 +00:00
|
|
|
|
2023-09-29 16:02:57 +00:00
|
|
|
if( strings.size() >= 2 && !strings[1].empty() )
|
|
|
|
{
|
|
|
|
for( const wxString& filter : wxSplit( strings[1], ' ' ) )
|
2023-09-26 17:31:45 +00:00
|
|
|
{
|
2023-09-29 16:02:57 +00:00
|
|
|
m_fpFilters.push_back( std::make_unique<EDA_PATTERN_MATCH_WILDCARD_ANCHORED>() );
|
|
|
|
m_fpFilters.back()->SetPattern( filter.Lower() );
|
2023-09-26 17:31:45 +00:00
|
|
|
}
|
2023-09-29 16:02:57 +00:00
|
|
|
|
|
|
|
m_filterByFPFilters->SetLabel( m_filterByFPFilters->GetLabel()
|
|
|
|
+ wxString::Format( wxS( " (%s)" ), strings[1] ) );
|
|
|
|
m_filterByFPFilters->Show( true );
|
2023-09-26 17:31:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool FOOTPRINT_CHOOSER_FRAME::ShowModal( wxString* aFootprint, wxWindow* aParent )
|
|
|
|
{
|
|
|
|
if( aFootprint && !aFootprint->IsEmpty() )
|
|
|
|
{
|
|
|
|
LIB_ID fpid;
|
|
|
|
|
|
|
|
fpid.Parse( *aFootprint, true );
|
|
|
|
|
|
|
|
if( fpid.IsValid() )
|
|
|
|
m_chooserPanel->SetPreselect( fpid );
|
|
|
|
}
|
|
|
|
|
|
|
|
return KIWAY_PLAYER::ShowModal( aFootprint, aParent );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-11-03 10:34:00 +00:00
|
|
|
static wxRect s_dialogRect( 0, 0, 0, 0 );
|
|
|
|
|
|
|
|
|
|
|
|
void FOOTPRINT_CHOOSER_FRAME::SetPosition( const wxPoint& aNewPosition )
|
|
|
|
{
|
|
|
|
PCB_BASE_FRAME::SetPosition( aNewPosition );
|
|
|
|
|
|
|
|
s_dialogRect.SetPosition( aNewPosition );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool FOOTPRINT_CHOOSER_FRAME::Show( bool show )
|
|
|
|
{
|
|
|
|
bool ret;
|
|
|
|
|
|
|
|
// Show or hide the window. If hiding, save current position and size.
|
|
|
|
// If showing, use previous position and size.
|
|
|
|
if( show )
|
|
|
|
{
|
|
|
|
#ifndef __WINDOWS__
|
|
|
|
PCB_BASE_FRAME::Raise(); // Needed on OS X and some other window managers (i.e. Unity)
|
|
|
|
#endif
|
|
|
|
ret = PCB_BASE_FRAME::Show( show );
|
|
|
|
|
|
|
|
// returns a zeroed-out default wxRect if none existed before.
|
|
|
|
wxRect savedDialogRect = s_dialogRect;
|
|
|
|
|
|
|
|
if( savedDialogRect.GetSize().x != 0 && savedDialogRect.GetSize().y != 0 )
|
|
|
|
{
|
|
|
|
SetSize( savedDialogRect.GetPosition().x, savedDialogRect.GetPosition().y,
|
|
|
|
std::max( wxWindow::GetSize().x, savedDialogRect.GetSize().x ),
|
|
|
|
std::max( wxWindow::GetSize().y, savedDialogRect.GetSize().y ),
|
|
|
|
0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Be sure that the dialog appears in a visible area
|
|
|
|
// (the dialog position might have been stored at the time when it was
|
|
|
|
// shown on another display)
|
|
|
|
if( wxDisplay::GetFromWindow( this ) == wxNOT_FOUND )
|
|
|
|
Centre();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s_dialogRect = wxRect( wxWindow::GetPosition(), wxWindow::GetSize() );
|
|
|
|
ret = PCB_BASE_FRAME::Show( show );
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-09-26 17:31:45 +00:00
|
|
|
void FOOTPRINT_CHOOSER_FRAME::OnPaint( wxPaintEvent& aEvent )
|
|
|
|
{
|
|
|
|
if( m_firstPaintEvent )
|
|
|
|
{
|
|
|
|
KIPLATFORM::UI::FixupCancelButtonCmdKeyCollision( this );
|
|
|
|
KIPLATFORM::UI::ForceFocus( m_chooserPanel->GetFocusTarget() );
|
|
|
|
|
|
|
|
m_firstPaintEvent = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
aEvent.Skip();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FOOTPRINT_CHOOSER_FRAME::OnOK( wxCommandEvent& aEvent )
|
|
|
|
{
|
|
|
|
LIB_ID fpID = m_chooserPanel->GetSelectedLibId();
|
|
|
|
|
|
|
|
if( fpID.IsValid() )
|
|
|
|
{
|
|
|
|
wxString footprint = fpID.Format();
|
|
|
|
|
|
|
|
AddFootprintToHistory( footprint );
|
|
|
|
DismissModal( true, footprint );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DismissModal( false );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-09-29 16:02:57 +00:00
|
|
|
void FOOTPRINT_CHOOSER_FRAME::closeFootprintChooser( wxCommandEvent& aEvent )
|
2023-09-26 17:31:45 +00:00
|
|
|
{
|
|
|
|
Close( false );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-12-08 15:40:19 +00:00
|
|
|
void FOOTPRINT_CHOOSER_FRAME::onFpChanged( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
if( m_showFpMode ) // the 3D viewer is not activated
|
|
|
|
return;
|
|
|
|
|
|
|
|
on3DviewReq( event );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FOOTPRINT_CHOOSER_FRAME::build3DCanvas()
|
|
|
|
{
|
|
|
|
// Create the dummy board used by the 3D canvas
|
|
|
|
m_dummyBoard = new BOARD();
|
|
|
|
m_dummyBoard->SetProject( &Prj(), true );
|
|
|
|
|
|
|
|
// This board will only be used to hold a footprint for viewing
|
|
|
|
m_dummyBoard->SetBoardUse( BOARD_USE::FPHOLDER );
|
|
|
|
|
|
|
|
m_boardAdapter.SetBoard( m_dummyBoard );
|
|
|
|
m_boardAdapter.m_IsBoardView = false;
|
|
|
|
m_boardAdapter.m_IsPreviewer = true; // Force display 3D models, regardless the 3D viewer options
|
|
|
|
|
|
|
|
EDA_3D_VIEWER_SETTINGS* cfg = Pgm().GetSettingsManager().GetAppSettings<EDA_3D_VIEWER_SETTINGS>();
|
|
|
|
m_boardAdapter.m_Cfg = cfg;
|
|
|
|
|
|
|
|
// Build the 3D canvas
|
|
|
|
m_preview3DCanvas = new EDA_3D_CANVAS( m_chooserPanel->m_RightPanel,
|
|
|
|
OGL_ATT_LIST::GetAttributesList( ANTIALIASING_MODE::AA_8X ),
|
|
|
|
m_boardAdapter, m_currentCamera,
|
|
|
|
PROJECT_PCB::Get3DCacheManager( &Prj() ) );
|
|
|
|
|
|
|
|
m_chooserPanel->m_RightPanelSizer->Add( m_preview3DCanvas, 1, wxEXPAND, 5 );
|
|
|
|
m_chooserPanel->m_RightPanel->Layout();
|
|
|
|
|
|
|
|
BOARD_DESIGN_SETTINGS& dummy_bds = m_dummyBoard->GetDesignSettings();
|
|
|
|
dummy_bds.SetBoardThickness( pcbIUScale.mmToIU( 1.6 ) );
|
|
|
|
dummy_bds.SetEnabledLayers( LSET::FrontMask() | LSET::BackMask() );
|
|
|
|
BOARD_STACKUP& dummy_board_stackup = m_dummyBoard->GetDesignSettings().GetStackupDescriptor();
|
|
|
|
dummy_board_stackup.RemoveAll();
|
|
|
|
dummy_board_stackup.BuildDefaultStackupList( &dummy_bds, 2 );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void FOOTPRINT_CHOOSER_FRAME::on3DviewReq( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
m_showFpMode = false;
|
|
|
|
|
|
|
|
m_grButtonFpView->Check( m_showFpMode );
|
|
|
|
m_grButton3DView->Check( !m_showFpMode );
|
|
|
|
|
|
|
|
FOOTPRINT_PREVIEW_WIDGET* viewFpPanel = m_chooserPanel->GetViewerPanel();
|
|
|
|
viewFpPanel->Show( m_showFpMode );
|
|
|
|
m_preview3DCanvas->Show( !m_showFpMode );
|
|
|
|
m_dummyBoard->DeleteAllFootprints();
|
|
|
|
|
|
|
|
if( m_chooserPanel->m_CurrFootprint )
|
|
|
|
m_dummyBoard->Add( (FOOTPRINT*)m_chooserPanel->m_CurrFootprint->Clone() );
|
|
|
|
|
|
|
|
m_preview3DCanvas->ReloadRequest();
|
|
|
|
m_preview3DCanvas->Request_refresh();
|
|
|
|
m_chooserPanel->m_RightPanel->Layout();
|
|
|
|
m_chooserPanel->m_RightPanel->Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FOOTPRINT_CHOOSER_FRAME::onFpViewReq( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
m_showFpMode = true;
|
|
|
|
|
|
|
|
m_grButtonFpView->Check( m_showFpMode );
|
|
|
|
m_grButton3DView->Check( !m_showFpMode );
|
|
|
|
|
|
|
|
FOOTPRINT_PREVIEW_WIDGET* viewFpPanel = m_chooserPanel->GetViewerPanel();
|
|
|
|
viewFpPanel->Show( m_showFpMode );
|
|
|
|
m_preview3DCanvas->Show( !m_showFpMode );
|
|
|
|
m_chooserPanel->m_RightPanel->Layout();
|
|
|
|
m_chooserPanel->m_RightPanel->Refresh();
|
|
|
|
}
|
|
|
|
|