KiCad PCB EDA Suite
Loading...
Searching...
No Matches
panel_footprint_chooser.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2014 Henner Zeller <[email protected]>
5 * Copyright (C) 2023 CERN
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program. If not, see <https://www.gnu.org/licenses/>.
20 */
21
23#include <wx/button.h>
24#include <wx/clipbrd.h>
25#include <wx/log.h>
26#include <wx/panel.h>
27#include <wx/sizer.h>
28#include <wx/splitter.h>
29#include <wx/timer.h>
30#include <wx/wxhtml.h>
31#include <pcb_base_frame.h>
32#include <pcbnew_settings.h>
33#include <pgm_base.h>
36#include <widgets/lib_tree.h>
38#include <footprint.h>
39#include <project_pcb.h>
40#include <kiface_base.h>
41#include <tool/actions.h>
42#include <tool/tool_manager.h>
43#include <widgets/kistatusbar.h>
44
45// When a new footprint is selected, a custom event is sent, for instance to update
46// 3D viewer. So define a FP_SELECTION_EVENT event
47wxDEFINE_EVENT( FP_SELECTION_EVENT, wxCommandEvent );
48
50 const wxArrayString& aFootprintHistoryList,
51 std::function<bool( LIB_TREE_NODE& )> aFilter,
52 std::function<void()> aAcceptHandler,
53 std::function<void()> aEscapeHandler ) :
54 wxPanel( aParent, wxID_ANY, wxDefaultPosition, wxDefaultSize ),
55 m_hsplitter( nullptr ),
56 m_vsplitter( nullptr ),
57 m_frame( aFrame ),
58 m_filter( std::move( aFilter ) ),
59 m_acceptHandler( std::move( aAcceptHandler ) ),
60 m_escapeHandler( std::move( aEscapeHandler ) )
61{
62 m_CurrFootprint = nullptr;
64
65 // Ensure libraries are loaded before building the tree. This is necessary when the footprint
66 // chooser is opened from contexts that don't preload libraries (e.g., from Eeschema).
67 footprints->AsyncLoad();
68 footprints->BlockUntilLoaded();
69
70 m_adapter = FP_TREE_MODEL_ADAPTER::Create( aFrame, footprints );
71 FP_TREE_MODEL_ADAPTER* adapter = static_cast<FP_TREE_MODEL_ADAPTER*>( m_adapter.get() );
72
73 std::vector<LIB_TREE_ITEM*> historyInfos;
74
75 for( const wxString& item : aFootprintHistoryList )
76 {
77 LIB_ID fpid;
78
79 if( fpid.Parse( item ) >= 0 )
80 continue;
81
82 FOOTPRINT* fp = footprints->LoadFootprint( fpid, false );
83
84 if( fp != nullptr )
85 {
86 historyInfos.push_back( fp );
87 m_historyFootprints.push_back( std::unique_ptr<FOOTPRINT>( fp ) );
88 }
89 }
90
91 adapter->DoAddLibrary( wxT( "-- " ) + _( "Recently Used" ) + wxT( " --" ), wxEmptyString,
92 historyInfos, false, true )
94
95 if( historyInfos.size() )
96 adapter->SetPreselectNode( historyInfos[0]->GetLIB_ID(), 0 );
97
98 adapter->SetFilter( &m_filter );
99 adapter->AddLibraries( m_frame );
100
101 // -------------------------------------------------------------------------------------
102 // Construct the actual panel
103 //
104
105 wxBoxSizer* sizer = new wxBoxSizer( wxVERTICAL );
106
107 m_vsplitter = new wxSplitterWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize,
108 wxSP_LIVE_UPDATE | wxSP_NOBORDER | wxSP_3DSASH );
109
110 m_hsplitter = new wxSplitterWindow( m_vsplitter, wxID_ANY, wxDefaultPosition, wxDefaultSize,
111 wxSP_LIVE_UPDATE | wxSP_NOBORDER | wxSP_3DSASH );
112
113 // Avoid the splitter window being assigned as the Parent to additional windows
114 m_vsplitter->SetExtraStyle( wxWS_EX_TRANSIENT );
115 m_hsplitter->SetExtraStyle( wxWS_EX_TRANSIENT );
116
117 m_detailsPanel = new wxPanel( m_vsplitter );
118 auto detailsSizer = new wxBoxSizer( wxVERTICAL );
119 m_detailsPanel->SetSizer( detailsSizer );
120
121 m_details = new HTML_WINDOW( m_detailsPanel, wxID_ANY, wxDefaultPosition, wxDefaultSize );
122 detailsSizer->Add( m_details, 1, wxEXPAND, 5 );
123 m_detailsPanel->Layout();
124 detailsSizer->Fit( m_detailsPanel );
125
126 m_vsplitter->SetSashGravity( 0.5 );
127
128 // Ensure splitted areas are always shown (i.e. 0 size not allowed) when m_detailsPanel is shown
129 m_vsplitter->SetMinimumPaneSize( 80 ); // arbitrary value but reasonable min size
130 m_vsplitter->SplitHorizontally( m_hsplitter, m_detailsPanel );
131
132 sizer->Add( m_vsplitter, 1, wxEXPAND, 5 );
133
134 m_tree = new LIB_TREE( m_hsplitter, wxT( "footprints" ), m_adapter,
136
137 m_hsplitter->SetSashGravity( 0.8 );
138 m_hsplitter->SetMinimumPaneSize( 20 );
139
140 m_RightPanel = new wxPanel( m_hsplitter );
141 m_RightPanelSizer = new wxBoxSizer( wxVERTICAL );
142
144 m_preview_ctrl->SetUserUnits( m_frame->GetUserUnits() );
145 m_RightPanelSizer->Add( m_preview_ctrl, 1, wxEXPAND, 5 );
146
147 m_RightPanel->SetSizer( m_RightPanelSizer );
148 m_RightPanel->Layout();
150
151 m_hsplitter->SplitVertically( m_tree, m_RightPanel );
152
153 m_dbl_click_timer = new wxTimer( this );
154 m_open_libs_timer = new wxTimer( this );
155
156 SetSizer( sizer );
157
158 m_adapter->FinishTreeInitialization();
159
160 Bind( wxEVT_TIMER, &PANEL_FOOTPRINT_CHOOSER::onCloseTimer, this, m_dbl_click_timer->GetId() );
161 Bind( wxEVT_TIMER, &PANEL_FOOTPRINT_CHOOSER::onOpenLibsTimer, this, m_open_libs_timer->GetId() );
162 Bind( EVT_LIBITEM_SELECTED, &PANEL_FOOTPRINT_CHOOSER::onFootprintSelected, this );
163 Bind( EVT_LIBITEM_CHOSEN, &PANEL_FOOTPRINT_CHOOSER::onFootprintChosen, this );
164 m_frame->Bind( wxEVT_MENU_OPEN, &PANEL_FOOTPRINT_CHOOSER::onMenuOpen, this );
165 m_frame->Bind( wxEVT_MENU_CLOSE, &PANEL_FOOTPRINT_CHOOSER::onMenuClose, this );
166
167 m_details->Connect( wxEVT_CHAR_HOOK,
169 nullptr, this );
170
171 Layout();
172
173 // Open the user's previously opened libraries on timer expiration.
174 // This is done on a timer because we need a gross hack to keep GTK from garbling the
175 // display. Must be longer than the search debounce timer.
176 m_open_libs_timer->StartOnce( 300 );
177}
178
179
181{
182 m_frame->Unbind( wxEVT_MENU_OPEN, &PANEL_FOOTPRINT_CHOOSER::onMenuOpen, this );
183 m_frame->Unbind( wxEVT_MENU_CLOSE, &PANEL_FOOTPRINT_CHOOSER::onMenuClose, this );
184 Unbind( wxEVT_TIMER, &PANEL_FOOTPRINT_CHOOSER::onCloseTimer, this );
185 Unbind( EVT_LIBITEM_SELECTED, &PANEL_FOOTPRINT_CHOOSER::onFootprintSelected, this );
186 Unbind( EVT_LIBITEM_CHOSEN, &PANEL_FOOTPRINT_CHOOSER::onFootprintChosen, this );
187
188 m_details->Disconnect( wxEVT_CHAR_HOOK, wxKeyEventHandler( PANEL_FOOTPRINT_CHOOSER::OnDetailsCharHook ),
189 nullptr, this );
190
191 // I am not sure the following two lines are necessary, but they will not hurt anyone
192 m_dbl_click_timer->Stop();
193 m_open_libs_timer->Stop();
194 delete m_dbl_click_timer;
195 delete m_open_libs_timer;
196
197 if( PCBNEW_SETTINGS* cfg = GetAppSettings<PCBNEW_SETTINGS>( "pcbnew" ) )
198 {
199 // Save any changes to column widths, etc.
200 m_adapter->SaveSettings();
201
202 cfg->m_FootprintChooser.width = GetParent()->GetSize().x;
203 cfg->m_FootprintChooser.height = GetParent()->GetSize().y;
204 cfg->m_FootprintChooser.sash_h = m_hsplitter->GetSashPosition();
205
206 if( m_vsplitter )
207 cfg->m_FootprintChooser.sash_v = m_vsplitter->GetSashPosition();
208
209 cfg->m_FootprintChooser.sort_mode = m_tree->GetSortMode();
210 }
211}
212
213
214void PANEL_FOOTPRINT_CHOOSER::OnChar( wxKeyEvent& aEvent )
215{
216 if( aEvent.GetKeyCode() == WXK_ESCAPE )
217 {
218 // First escape clears search string, second escape closes
219 if( !m_tree->GetSearchString().IsEmpty() )
220 {
221 m_tree->SetSearchString( wxEmptyString );
222 GetFocusTarget()->SetFocus();
223 return;
224 }
225
227 }
228 else
229 {
230 aEvent.Skip();
231 }
232}
233
234
235void PANEL_FOOTPRINT_CHOOSER::onMenuOpen( wxMenuEvent& aEvent )
236{
237 m_tree->BlockPreview( true );
238 aEvent.Skip();
239}
240
241
242void PANEL_FOOTPRINT_CHOOSER::onMenuClose( wxMenuEvent& aEvent )
243{
244 m_tree->BlockPreview( false );
245 aEvent.Skip();
246}
247
248
250{
251 if( PCBNEW_SETTINGS* settings = GetAppSettings<PCBNEW_SETTINGS>( "pcbnew" ) )
252 {
253 auto horizPixelsFromDU =
254 [&]( int x ) -> int
255 {
256 wxSize sz( x, 0 );
257 return GetParent()->ConvertDialogToPixels( sz ).x;
258 };
259
260 PCBNEW_SETTINGS::FOOTPRINT_CHOOSER& cfg = settings->m_FootprintChooser;
261
262 int w = cfg.width < 40 ? horizPixelsFromDU( 440 ) : cfg.width;
263 int h = cfg.height < 40 ? horizPixelsFromDU( 340 ) : cfg.height;
264
265 GetParent()->SetSize( wxSize( w, h ) );
266 GetParent()->Layout();
267
268 // We specify the width of the right window (m_symbol_view_panel), because specify
269 // the width of the left window does not work as expected when SetSashGravity() is called
270 if( cfg.sash_h < 0 )
271 cfg.sash_h = horizPixelsFromDU( 220 );
272
273 m_hsplitter->SetSashPosition( cfg.sash_h );
274
275 if( cfg.sash_v < 0 )
276 cfg.sash_v = horizPixelsFromDU( 230 );
277
278 if( m_vsplitter )
279 m_vsplitter->SetSashPosition( cfg.sash_v );
280
282 }
283}
284
285
287{
288 m_preselect = aPreselect;
289 m_adapter->SetPreselectNode( aPreselect, 0 );
290
291 if( m_tree && aPreselect.IsValid() )
292 m_tree->SelectLibId( aPreselect );
293}
294
295
297{
298 return m_tree->GetSelectedLibId();
299}
300
301
302void PANEL_FOOTPRINT_CHOOSER::onCloseTimer( wxTimerEvent& aEvent )
303{
304 // Hack because of eaten MouseUp event. See PANEL_FOOTPRINT_CHOOSER::onFootprintChosen
305 // for the beginning of this spaghetti noodle.
306
307 auto state = wxGetMouseState();
308
309 if( state.LeftIsDown() )
310 {
311 // Mouse hasn't been raised yet, so fire the timer again. Otherwise the
312 // purpose of this timer is defeated.
314 }
315 else
316 {
318 }
319}
320
321
323
325{
326 // Freeze across both steps so the expand and the re-scroll paint as a single final
327 // state, otherwise the selected row visibly shifts and snaps back.
328 m_tree->Freeze();
329
330 if( PCBNEW_SETTINGS* cfg = dynamic_cast<PCBNEW_SETTINGS*>( Kiface().KifaceSettings() ) )
331 m_adapter->OpenLibs( cfg->m_LibTree.open_libs );
332
333 // OpenLibs reshuffles the tree, so re-assert the preselected footprint.
334 if( m_preselect.IsValid() )
335 m_tree->SelectLibId( m_preselect );
336
337 m_tree->Thaw();
338}
339
340
342{
343 if( !m_preview_ctrl || !m_preview_ctrl->IsInitialized() )
344 return;
345
346 LIB_ID lib_id = m_tree->GetSelectedLibId();
347
348 if( !lib_id.IsValid() )
349 {
350 m_preview_ctrl->SetStatusText( _( "No footprint selected" ) );
351 }
352 else
353 {
354 m_preview_ctrl->ClearStatus();
355 m_preview_ctrl->DisplayFootprint( lib_id );
356 }
357
358 m_CurrFootprint = static_cast<FOOTPRINT_PREVIEW_PANEL*>( m_preview_ctrl->GetPreviewPanel() )->GetCurrentFootprint();
359
360 // Send a FP_SELECTION_EVENT event after a footprint change
361 wxCommandEvent event( FP_SELECTION_EVENT, GetId() );
362 event.SetEventObject( this );
363
364 ProcessWindowEvent( event );
365}
366
367
368void PANEL_FOOTPRINT_CHOOSER::onFootprintChosen( wxCommandEvent& aEvent )
369{
370 if( m_tree->GetSelectedLibId().IsValid() )
371 {
372 // Got a selection. We can't just end the modal dialog here, because wx leaks some
373 // events back to the parent window (in particular, the MouseUp following a double click).
374 //
375 // NOW, here's where it gets really fun. wxTreeListCtrl eats MouseUp. This isn't really
376 // feasible to bypass without a fully custom wxDataViewCtrl implementation, and even then
377 // might not be fully possible (docs are vague). To get around this, we use a one-shot
378 // timer to schedule the dialog close.
379 //
380 // See PANEL_FOOTPRINT_CHOOSER::onCloseTimer for the other end of this spaghetti noodle.
382 }
383}
384
385
387{
388 if( m_details && e.GetKeyCode() == 'C' && e.ControlDown() &&
389 !e.AltDown() && !e.ShiftDown() && !e.MetaDown() )
390 {
391 wxString txt = m_details->SelectionToText();
392 wxLogNull doNotLog; // disable logging of failed clipboard actions
393
394 if( wxTheClipboard->Open() )
395 {
396 wxTheClipboard->SetData( new wxTextDataObject( txt ) );
397 wxTheClipboard->Flush(); // Allow data to be available after closing KiCad
398 wxTheClipboard->Close();
399 }
400 }
401 else
402 {
403 e.Skip();
404 }
405}
KIFACE_BASE & Kiface()
Global KIFACE_BASE "get" accessor.
An interface to the global shared library manager that is schematic-specific and linked to one projec...
FOOTPRINT * LoadFootprint(const wxString &aNickname, const wxString &aName, bool aKeepUUID)
Load a FOOTPRINT having aName from the library given by aNickname.
Panel that renders a single footprint via Cairo GAL, meant to be exported through Kiface.
FOOTPRINT * GetCurrentFootprint() const
void AddLibraries(EDA_BASE_FRAME *aParent)
static wxObjectDataPtr< LIB_TREE_MODEL_ADAPTER > Create(PCB_BASE_FRAME *aParent, FOOTPRINT_LIBRARY_ADAPTER *aLibs)
Factory function: create a model adapter in a reference-counting container.
Add dark theme support to wxHtmlWindow.
Definition html_window.h:31
PROJECT & Prj() const
Return a reference to the PROJECT associated with this KIWAY.
void AsyncLoad()
Loads all available libraries for this adapter type in the background.
A logical library item identifier and consists of various portions much like a URI.
Definition lib_id.h:45
int Parse(const UTF8 &aId, bool aFix=false)
Parse LIB_ID with the information from aId.
Definition lib_id.cpp:48
bool IsValid() const
Check if this LID_ID is valid.
Definition lib_id.h:168
void SetPreselectNode(const LIB_ID &aLibId, int aUnit)
Set the symbol name to be selected if there are no search results.
LIB_TREE_NODE_LIBRARY & DoAddLibrary(const wxString &aNodeName, const wxString &aDesc, const std::vector< LIB_TREE_ITEM * > &aItemList, bool pinned, bool presorted)
Add the given list of symbols by alias.
void SetFilter(std::function< bool(LIB_TREE_NODE &aNode)> *aFilter)
Set the filter.
Model class in the component selector Model-View-Adapter (mediated MVC) architecture.
Widget displaying a tree of symbols with optional search text control and description panel.
Definition lib_tree.h:46
@ ALL_WIDGETS
Definition lib_tree.h:55
void onFootprintSelected(wxCommandEvent &aEvent)
wxObjectDataPtr< LIB_TREE_MODEL_ADAPTER > m_adapter
void onCloseTimer(wxTimerEvent &aEvent)
std::function< void()> m_acceptHandler
FOOTPRINT_PREVIEW_WIDGET * m_preview_ctrl
void OnChar(wxKeyEvent &aEvent)
std::function< void()> m_escapeHandler
void onMenuOpen(wxMenuEvent &aEvent)
Handle parent frame menu events to block tree preview.
static constexpr int DblClickDelay
std::function< bool(LIB_TREE_NODE &)> m_filter
void OnDetailsCharHook(wxKeyEvent &aEvt)
void onOpenLibsTimer(wxTimerEvent &aEvent)
void SetPreselect(const LIB_ID &aPreselect)
PANEL_FOOTPRINT_CHOOSER(PCB_BASE_FRAME *aFrame, wxTopLevelWindow *aParent, const wxArrayString &aFootprintHistoryList, std::function< bool(LIB_TREE_NODE &)> aFilter, std::function< void()> aAcceptHandler, std::function< void()> aEscapeHandler)
Create dialog to choose component.
std::vector< std::unique_ptr< FOOTPRINT > > m_historyFootprints
void onMenuClose(wxMenuEvent &aEvent)
void onFootprintChosen(wxCommandEvent &aEvent)
Handle the selection of an item.
LIB_ID GetSelectedLibId() const
To be called after this dialog returns from ShowModal().
Base PCB main window class for Pcbnew, Gerbview, and CvPcb footprint viewer.
static FOOTPRINT_LIBRARY_ADAPTER * FootprintLibAdapter(PROJECT *aProject)
#define _(s)
STL namespace.
wxDEFINE_EVENT(FP_SELECTION_EVENT, wxCommandEvent)
see class PGM_BASE
T * GetAppSettings(const char *aFilename)