KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
dialog_find.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) 2012 Marco Mattila <marcom99@gmail.com>
5 * Copyright (C) 2018 Jean-Pierre Charras jp.charras at wanadoo.fr
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, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26#include <pcb_edit_frame.h> // Keep this include at top to avoid compil issue on MSYS2
27#include <board.h>
28#include <pcb_marker.h>
29#include <footprint.h>
30#include <pcb_text.h>
31#include <zone.h>
32#include <dialog_find.h>
33#include <string_utils.h>
34#include <hotkeys_basic.h>
35#include <string>
36#include <tool/tool_manager.h>
37#include <tools/pcb_actions.h>
38#include <wx/fdrepdlg.h>
39
40
41//Defined as global because these values have to survive the destructor
42
43bool FindOptionCase = false;
44bool FindOptionWords = false;
46bool FindOptionWrap = true;
47
48bool FindIncludeTexts = true;
52bool FindIncludeNets = true;
53
54
56 DIALOG_FIND_BASE( aFrame, wxID_ANY, _( "Find" ) ),
57 m_frame( aFrame )
58{
59 GetSizer()->SetSizeHints( this );
60
62
63 while( m_searchCombo->GetCount() > 10 )
64 {
65 m_frame->GetFindHistoryList().pop_back();
66 m_searchCombo->Delete( 9 );
67 }
68
69 if( m_searchCombo->GetCount() )
70 {
71 m_searchCombo->SetSelection( 0 );
72 m_searchCombo->SelectAll();
73 }
74
75 m_matchCase->SetValue( FindOptionCase );
76 m_matchWords->SetValue( FindOptionWords );
78 m_wrap->SetValue( FindOptionWrap );
79
84 m_includeNets->SetValue( FindIncludeNets );
85
86 m_status->SetLabel( wxEmptyString);
87 m_upToDate = false;
88
89 m_hitList.clear();
90 m_it = m_hitList.begin();
91
92 if( int hotkey = ACTIONS::showSearch.GetHotKey() )
93 {
94 wxString hotkeyHint = wxString::Format( wxT( " (%s)" ), KeyNameFromKeyCode( hotkey ) );
95 m_searchPanelLink->SetLabel( m_searchPanelLink->GetLabel() + hotkeyHint );
96 }
97
98 m_findNext->SetDefault();
100
101 Center();
102}
103
104
105void DIALOG_FIND::Preload( const wxString& aFindString )
106{
107 if( !aFindString.IsEmpty() )
108 {
109 m_searchCombo->SetValue( aFindString );
110 m_searchCombo->SelectAll();
111 }
112}
113
114
115void DIALOG_FIND::onTextEnter( wxCommandEvent& aEvent )
116{
117 search( true );
118}
119
120
121void DIALOG_FIND::onFindNextClick( wxCommandEvent& aEvent )
122{
123 search( true );
124}
125
126
127void DIALOG_FIND::onFindPreviousClick( wxCommandEvent& aEvent )
128{
129 search( false );
130}
131
132
133void DIALOG_FIND::onSearchAgainClick( wxCommandEvent& aEvent )
134{
135 m_upToDate = false;
136 search( true );
137}
138
139
140void DIALOG_FIND::onShowSearchPanel( wxHyperlinkEvent& event )
141{
143 {
144 EndModal( wxID_CANCEL );
145
146 CallAfter(
147 []()
148 {
149 if( wxWindow* frame = wxWindow::FindWindowByName( PCB_EDIT_FRAME_NAME ) )
150 static_cast<PCB_EDIT_FRAME*>( frame )->FocusSearch();
151 } );
152 }
153 else
154 {
156 }
157}
158
159
160void DIALOG_FIND::search( bool aDirection )
161{
162 PCB_SCREEN* screen = m_frame->GetScreen();
163 int index;
164 wxString msg;
165 wxString searchString;
166 bool endIsReached = false;
167 bool isFirstSearch = false;
168
169 searchString = m_searchCombo->GetValue();
170
171 if( searchString.IsEmpty() )
172 {
173 Show();
174 return;
175 }
176
177 // Add/move the search string to the top of the list if it isn't already there
178 index = m_searchCombo->FindString( searchString, true );
179
180 if( index == wxNOT_FOUND )
181 {
182 m_searchCombo->Insert( searchString, 0 );
183 m_searchCombo->SetSelection( 0 );
184 m_upToDate = false;
185 m_frame->GetFindHistoryList().Insert( searchString, 0 );
186
187 if( m_searchCombo->GetCount() > 10 )
188 {
189 m_frame->GetFindHistoryList().pop_back();
190 m_searchCombo->Delete( 10 );
191 }
192 }
193 else if( index != 0 )
194 {
195 m_searchCombo->Delete( index );
196 m_searchCombo->Insert( searchString, 0 );
197 m_searchCombo->SetSelection( 0 );
198 m_upToDate = false;
199
200 if( m_frame->GetFindHistoryList().Index( searchString ) )
201 m_frame->GetFindHistoryList().Remove( searchString );
202
203 m_frame->GetFindHistoryList().Insert( searchString, 0 );
204 }
205
206 if( FindOptionCase != m_matchCase->GetValue() )
207 {
208 FindOptionCase = m_matchCase->GetValue();
209 m_upToDate = false;
210 }
211
212 if( FindOptionWords != m_matchWords->GetValue() )
213 {
214 FindOptionWords = m_matchWords->GetValue();
215 m_upToDate = false;
216 }
217
218 if( FindOptionWildcards != m_wildcards->GetValue() )
219 {
220 FindOptionWildcards = m_wildcards->GetValue();
221 m_upToDate = false;
222 }
223
224 FindOptionWrap = m_wrap->GetValue();
225
226 if( FindIncludeTexts != m_includeTexts->GetValue() )
227 {
228 FindIncludeTexts = m_includeTexts->GetValue();
229 m_upToDate = false;
230 }
231
232 if( FindIncludeValues != m_includeValues->GetValue() )
233 {
235 m_upToDate = false;
236 }
237
238 if( FindIncludeReferences != m_includeReferences->GetValue() )
239 {
241 m_upToDate = false;
242 }
243
244 if( FindIncludeMarkers != m_includeMarkers->GetValue() )
245 {
247 m_upToDate = false;
248 }
249
250 if( FindIncludeNets != m_includeNets->GetValue() )
251 {
252 FindIncludeNets = m_includeNets->GetValue();
253 m_upToDate = false;
254 }
255
256 if( FindOptionCase )
258
259 if( FindOptionWords )
260 m_frame->GetFindReplaceData().matchMode = EDA_SEARCH_MATCH_MODE::WHOLEWORD;
261 else if( FindOptionWildcards )
262 m_frame->GetFindReplaceData().matchMode = EDA_SEARCH_MATCH_MODE::WILDCARD;
263 else
264 m_frame->GetFindReplaceData().matchMode = EDA_SEARCH_MATCH_MODE::PLAIN;
265
266 // Search parameters
267 m_frame->GetFindReplaceData().findString = searchString;
268
270 m_frame->GetCanvas()->GetViewStart( &screen->m_StartVisu.x, &screen->m_StartVisu.y );
271
272 BOARD* board = m_frame->GetBoard();
273
274 // Refresh the list of results
275 if( !m_upToDate )
276 {
277 m_status->SetLabel( _( "Searching..." ) );
278 m_hitList.clear();
279
281 {
282 for( FOOTPRINT* fp : board->Footprints() )
283 {
284 if( ( fp->Reference().Matches( m_frame->GetFindReplaceData(), nullptr )
286 || ( fp->Value().Matches( m_frame->GetFindReplaceData(), nullptr )
287 && FindIncludeValues ) )
288 {
289 m_hitList.push_back( fp );
290 }
291
292 if( m_includeTexts->GetValue() )
293 {
294 for( BOARD_ITEM* item : fp->GraphicalItems() )
295 {
296 if( item->Type() == PCB_TEXT_T )
297 {
298 PCB_TEXT* text = static_cast<PCB_TEXT*>( item );
299
300 if( text && text->Matches( m_frame->GetFindReplaceData(), nullptr ) )
301 m_hitList.push_back( fp );
302 }
303 }
304
305 for( PCB_FIELD* field : fp->GetFields() )
306 {
307 if( field->Matches( m_frame->GetFindReplaceData(), nullptr ) )
308 m_hitList.push_back( fp );
309 }
310 }
311 }
312
313 if( FindIncludeTexts )
314 {
315 for( BOARD_ITEM* item : board->Drawings() )
316 {
317 if( item->Type() == PCB_TEXT_T )
318 {
319 PCB_TEXT* text = static_cast<PCB_TEXT*>( item );
320
321 if( text && text->Matches( m_frame->GetFindReplaceData(), nullptr ) )
322 m_hitList.push_back( text );
323 }
324 }
325
326 for( BOARD_ITEM* item : board->Zones() )
327 {
328 ZONE* zone = static_cast<ZONE*>( item );
329
330 if( zone->Matches( m_frame->GetFindReplaceData(), nullptr ) )
331 m_hitList.push_back( zone );
332 }
333 }
334 }
335
337 {
338 for( PCB_MARKER* marker : board->Markers() )
339 {
340 if( marker->Matches( m_frame->GetFindReplaceData(), nullptr ) )
341 m_hitList.push_back( marker );
342 }
343 }
344
345 if( FindIncludeNets )
346 {
347 for( NETINFO_ITEM* net : board->GetNetInfo() )
348 {
349 if( net && net->Matches( m_frame->GetFindReplaceData(), nullptr ) )
350 m_hitList.push_back( net );
351 }
352 }
353
354 m_upToDate = true;
355 isFirstSearch = true;
356
357 if( aDirection )
358 m_it = m_hitList.begin();
359 else
360 m_it = m_hitList.end();
361 }
362
363 // Do we want a sorting algorithm ? If so, implement it here.
364
365 // Get the item to display
366 if( m_hitList.empty() )
367 {
368 m_frame->SetStatusText( wxEmptyString );
369 }
370 else
371 {
372 if( aDirection )
373 {
374 if( m_it != m_hitList.end() && !isFirstSearch )
375 m_it++;
376
377 if( m_it == m_hitList.end() )
378 {
379 if( m_wrap->GetValue() )
380 {
381 m_it = m_hitList.begin();
382 }
383 else
384 {
385 endIsReached = true;
386 m_it--; // point to the last REAL result
387 }
388 }
389 }
390 else
391 {
392 if( m_it == m_hitList.begin() )
393 {
394 if( m_wrap->GetValue() )
395 m_it = m_hitList.end();
396 else
397 endIsReached = true;
398 }
399
400 if( m_it != m_hitList.begin() )
401 m_it--;
402 }
403 }
404
405 // Display the item
406 if( m_hitList.empty() )
407 {
408 m_frame->SetStatusText( wxEmptyString );
409 msg.Printf( _( "'%s' not found" ), searchString );
410 m_frame->ShowInfoBarMsg( msg );
411
412 m_status->SetLabel( msg );
413 }
414 else if( endIsReached )
415 {
416 m_frame->SetStatusText( wxEmptyString );
417 m_frame->ShowInfoBarMsg( _( "No more items to show" ) );
418
419 m_status->SetLabel( _( "No hits" ) );
420 }
421 else
422 {
424
425 msg.Printf( _( "'%s' found" ), searchString );
426 m_frame->SetStatusText( msg );
427
428 msg.Printf( _( "Hit(s): %d / %zu" ),
429 (int)std::distance( m_hitList.begin(), m_it ) + 1,
430 m_hitList.size() );
431 m_status->SetLabel( msg );
432 }
433
436}
437
438
439void DIALOG_FIND::OnCloseButtonClick( wxCommandEvent& aEvent )
440{
441 wxCloseEvent tmp;
442
443 OnClose( tmp );
444
445 aEvent.Skip();
446}
447
448bool DIALOG_FIND::Show( bool show )
449{
450 bool ret = DIALOG_FIND_BASE::Show( show );
451
452 if( show )
453 m_searchCombo->SetFocus();
454
455 return ret;
456}
457
458
459void DIALOG_FIND::OnClose( wxCloseEvent& aEvent )
460{
461 FindOptionCase = m_matchCase->GetValue();
462 FindOptionWords = m_matchWords->GetValue();
463 FindOptionWildcards = m_wildcards->GetValue();
464 FindOptionWrap = m_wrap->GetValue();
465
466 FindIncludeTexts = m_includeTexts->GetValue();
470 FindIncludeNets = m_includeNets->GetValue();
471
472 aEvent.Skip();
473}
static TOOL_ACTION selectItem
Select an item (specified as the event parameter).
Definition: actions.h:223
static TOOL_ACTION showSearch
Definition: actions.h:115
static TOOL_ACTION selectionClear
Clear the current selection.
Definition: actions.h:220
VECTOR2I m_StartVisu
Coordinates in drawing units of the current view position (upper left corner of device)
Definition: base_screen.h:93
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
const NETINFO_LIST & GetNetInfo() const
Definition: board.h:897
const ZONES & Zones() const
Definition: board.h:342
const MARKERS & Markers() const
Definition: board.h:346
const FOOTPRINTS & Footprints() const
Definition: board.h:338
const DRAWINGS & Drawings() const
Definition: board.h:340
Class DIALOG_FIND_BASE.
wxCheckBox * m_matchCase
wxCheckBox * m_includeTexts
wxCheckBox * m_includeValues
wxHyperlinkCtrl * m_searchPanelLink
wxButton * m_findNext
wxComboBox * m_searchCombo
wxCheckBox * m_includeReferences
wxCheckBox * m_matchWords
wxCheckBox * m_includeNets
wxStaticText * m_status
wxCheckBox * m_includeMarkers
wxCheckBox * m_wildcards
wxCheckBox * m_wrap
PCB_EDIT_FRAME * m_frame
Definition: dialog_find.h:95
void onTextEnter(wxCommandEvent &event) override
DIALOG_FIND(PCB_EDIT_FRAME *aParent)
Definition: dialog_find.cpp:55
bool Show(bool show=true) override
The Show method is overridden to make the search combobox focused by default.
std::deque< BOARD_ITEM * >::iterator m_it
Definition: dialog_find.h:97
void Preload(const wxString &aFindString)
void onFindNextClick(wxCommandEvent &event) override
void search(bool direction)
bool m_upToDate
Definition: dialog_find.h:98
void onSearchAgainClick(wxCommandEvent &event) override
BOARD_ITEM * GetItem() const
Return the currently found item or nullptr in the case of no items found.
Definition: dialog_find.h:50
void onShowSearchPanel(wxHyperlinkEvent &event) override
void onFindPreviousClick(wxCommandEvent &event) override
void OnClose(wxCloseEvent &event) override
std::function< void(BOARD_ITEM *)> m_highlightCallback
Definition: dialog_find.h:100
void OnCloseButtonClick(wxCommandEvent &aEvent) override
std::deque< BOARD_ITEM * > m_hitList
Definition: dialog_find.h:96
bool Show(bool show) override
void SetInitialFocus(wxWindow *aWindow)
Sets the window (usually a wxTextCtrl) that should be focused when the dialog is shown.
Definition: dialog_shim.h:66
void ShowInfoBarMsg(const wxString &aMsg, bool aShowCloseButton=false)
Show the WX_INFOBAR displayed on the top of the canvas with a message and an info icon on the left of...
wxArrayString & GetFindHistoryList()
EDA_SEARCH_DATA & GetFindReplaceData()
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
Handle the data for a net.
Definition: netinfo.h:56
PCB_DRAW_PANEL_GAL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
PCB_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
BOARD * GetBoard() const
The main frame for Pcbnew.
bool IsSearchPaneShown()
TOOL_MANAGER * GetToolManager() const
Return the MVC controller.
Definition: tools_holder.h:55
bool RunAction(const std::string &aActionName, T aParam)
Run the specified action immediately, pausing the current action to run the new one.
Definition: tool_manager.h:150
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
bool Matches(const EDA_SEARCH_DATA &aSearchData, void *aAuxData) const override
Compare the item against the search criteria in aSearchData.
Definition: zone.h:166
bool FindIncludeMarkers
Definition: dialog_find.cpp:51
bool FindOptionCase
Definition: dialog_find.cpp:43
bool FindIncludeValues
Definition: dialog_find.cpp:49
bool FindOptionWords
Definition: dialog_find.cpp:44
bool FindOptionWrap
Definition: dialog_find.cpp:46
bool FindIncludeReferences
Definition: dialog_find.cpp:50
bool FindOptionWildcards
Definition: dialog_find.cpp:45
bool FindIncludeTexts
Definition: dialog_find.cpp:48
bool FindIncludeNets
Definition: dialog_find.cpp:52
#define _(s)
#define PCB_EDIT_FRAME_NAME
wxString KeyNameFromKeyCode(int aKeycode, bool *aIsFound)
Return the key name from the key code.
EDA_SEARCH_MATCH_MODE matchMode
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
Definition: typeinfo.h:92