KiCad PCB EDA Suite
Loading...
Searching...
No Matches
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 <[email protected]>
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, see <https://www.gnu.org/licenses/>.
20 */
21
22#include <pcb_edit_frame.h> // Keep this include at top to avoid compil issue on MSYS2
23#include <board.h>
24#include <pcb_marker.h>
25#include <footprint.h>
26#include <pcb_text.h>
27#include <zone.h>
28#include <dialog_find.h>
29#include <string_utils.h>
30#include <hotkeys_basic.h>
31#include <string>
32#include <tool/tool_manager.h>
33#include <tools/pcb_actions.h>
34#include <wx/fdrepdlg.h>
35
36
38 DIALOG_FIND_BASE( aFrame, wxID_ANY, _( "Find" ) ),
39 m_frame( aFrame )
40{
41 GetSizer()->SetSizeHints( this );
42
44
45 m_searchCombo->Append( m_frame->GetFindHistoryList() );
46
47 while( m_searchCombo->GetCount() > 10 )
48 {
49 m_frame->GetFindHistoryList().pop_back();
50 m_searchCombo->Delete( 9 );
51 }
52
53 if( m_searchCombo->GetCount() )
54 {
55 m_searchCombo->SetSelection( 0 );
56 m_searchCombo->SelectAll();
57 }
58
59 m_status->SetLabel( wxEmptyString);
60 m_upToDate = false;
61
62 m_hitList.clear();
63 m_it = m_hitList.begin();
64
65 m_board = m_frame->GetBoard();
66
67 if( m_board )
68 m_board->AddListener( this );
69
70 m_frame->Bind( EDA_EVT_BOARD_CHANGED, &DIALOG_FIND::OnBoardChanged, this );
71
72 if( int hotkey = ACTIONS::showSearch.GetHotKey() )
73 {
74 wxString hotkeyHint = wxString::Format( wxT( " (%s)" ), KeyNameFromKeyCode( hotkey ) );
75 m_searchPanelLink->SetLabel( m_searchPanelLink->GetLabel() + hotkeyHint );
76 }
77
78 m_findNext->SetDefault();
80
81 Center();
82}
83
84
86{
87 if( m_board )
88 m_board->RemoveListener( this );
89
90 m_frame->Unbind( EDA_EVT_BOARD_CHANGED, &DIALOG_FIND::OnBoardChanged, this );
91}
92
93
94void DIALOG_FIND::Preload( const wxString& aFindString )
95{
96 if( !aFindString.IsEmpty() )
97 {
98 m_searchCombo->SetValue( aFindString );
99 m_searchCombo->SelectAll();
100 }
101}
102
103
104void DIALOG_FIND::onTextEnter( wxCommandEvent& aEvent )
105{
106 search( true );
107}
108
109
110void DIALOG_FIND::onFindNextClick( wxCommandEvent& aEvent )
111{
112 search( true );
113}
114
115
116void DIALOG_FIND::onFindPreviousClick( wxCommandEvent& aEvent )
117{
118 search( false );
119}
120
121
122void DIALOG_FIND::onSearchAgainClick( wxCommandEvent& aEvent )
123{
124 m_upToDate = false;
125 search( true );
126}
127
128
129void DIALOG_FIND::onShowSearchPanel( wxHyperlinkEvent& event )
130{
131 m_frame->GetToolManager()->RunAction( ACTIONS::showSearch );
132
133 Show( false );
134
135 CallAfter(
136 []()
137 {
138 if( wxWindow* frame = wxWindow::FindWindowByName( PCB_EDIT_FRAME_NAME ) )
139 static_cast<PCB_EDIT_FRAME*>( frame )->FocusSearch();
140 } );
141}
142
143
144void DIALOG_FIND::search( bool aDirection )
145{
146 PCB_SCREEN* screen = m_frame->GetScreen();
147 int index;
148 wxString msg;
149 wxString searchString;
150 bool endIsReached = false;
151 bool isFirstSearch = false;
152
153 searchString = m_searchCombo->GetValue();
154
155 if( searchString.IsEmpty() )
156 {
157 Show();
158 return;
159 }
160
161 // Add/move the search string to the top of the list if it isn't already there
162 index = m_searchCombo->FindString( searchString, true );
163
164 if( index == wxNOT_FOUND )
165 {
166 m_searchCombo->Insert( searchString, 0 );
167 m_searchCombo->SetSelection( 0 );
168 m_upToDate = false;
169 m_frame->GetFindHistoryList().Insert( searchString, 0 );
170
171 if( m_searchCombo->GetCount() > 10 )
172 {
173 m_frame->GetFindHistoryList().pop_back();
174 m_searchCombo->Delete( 10 );
175 }
176 }
177 else if( index != 0 )
178 {
179 m_searchCombo->Delete( index );
180 m_searchCombo->Insert( searchString, 0 );
181 m_searchCombo->SetSelection( 0 );
182 m_upToDate = false;
183
184 if( m_frame->GetFindHistoryList().Index( searchString ) )
185 m_frame->GetFindHistoryList().Remove( searchString );
186
187 m_frame->GetFindHistoryList().Insert( searchString, 0 );
188 }
189
190 EDA_SEARCH_DATA& frd = m_frame->GetFindReplaceData();
191
192 if( m_matchCase->GetValue() )
193 frd.matchCase = true;
194
195 if( m_matchWords->GetValue() )
197 else if( m_wildcards->GetValue() )
199 else
201
202 frd.searchAllFields = m_checkAllFields->GetValue();
203
204 // Search parameters
205 frd.findString = searchString;
206
207 m_frame->GetToolManager()->RunAction( ACTIONS::selectionClear );
208 m_frame->GetCanvas()->GetViewStart( &screen->m_StartVisu.x, &screen->m_StartVisu.y );
209
210 BOARD* board = m_frame->GetBoard();
211
212 // Refresh the list of results
213 if( !m_upToDate )
214 {
215 m_status->SetLabel( _( "Searching..." ) );
216 m_hitList.clear();
217
218 if( m_includeTexts->GetValue() || m_includeValues->GetValue() || m_includeReferences->GetValue() )
219 {
220 for( FOOTPRINT* fp : board->Footprints() )
221 {
222 bool found = false;
223
224 if( m_includeReferences->GetValue() && fp->Reference().Matches( frd, nullptr ) )
225 found = true;
226
227 if( !found && m_includeValues->GetValue() && fp->Value().Matches( frd, nullptr ) )
228 found = true;
229
230 if( !found && m_includeTexts->GetValue() )
231 {
232 for( BOARD_ITEM* item : fp->GraphicalItems() )
233 {
234 if( item->Type() == PCB_TEXT_T )
235 {
236 PCB_TEXT* text = static_cast<PCB_TEXT*>( item );
237
238 if( text->Matches( frd, nullptr ) )
239 {
240 found = true;
241 break;
242 }
243 }
244 }
245 }
246
247 if( !found && m_includeTexts->GetValue() )
248 {
249 for( PCB_FIELD* field : fp->GetFields() )
250 {
251 wxCHECK2( field, continue );
252
253 if( field->Matches( frd, nullptr ) )
254 {
255 found = true;
256 break;
257 }
258 }
259 }
260
261 if( found )
262 m_hitList.push_back( fp );
263 }
264
265 if( m_includeTexts->GetValue() )
266 {
267 for( BOARD_ITEM* item : board->Drawings() )
268 {
269 if( item->Type() == PCB_TEXT_T )
270 {
271 PCB_TEXT* text = static_cast<PCB_TEXT*>( item );
272
273 if( text && text->Matches( frd, nullptr ) )
274 m_hitList.push_back( text );
275 }
276 }
277
278 for( BOARD_ITEM* item : board->Zones() )
279 {
280 ZONE* zone = static_cast<ZONE*>( item );
281
282 if( zone->Matches( frd, nullptr ) )
283 m_hitList.push_back( zone );
284 }
285 }
286 }
287
288 if( m_includeMarkers->GetValue() )
289 {
290 for( PCB_MARKER* marker : board->Markers() )
291 {
292 if( marker->Matches( frd, nullptr ) )
293 m_hitList.push_back( marker );
294 }
295 }
296
297 if( m_includeNets->GetValue() )
298 {
299 for( NETINFO_ITEM* net : board->GetNetInfo() )
300 {
301 if( net && net->Matches( frd, nullptr ) )
302 m_hitList.push_back( net );
303 }
304 }
305
306 m_upToDate = true;
307 isFirstSearch = true;
308
309 if( aDirection )
310 m_it = m_hitList.begin();
311 else
312 m_it = m_hitList.end();
313 }
314
315 // Do we want a sorting algorithm ? If so, implement it here.
316
317 // Get the item to display
318 if( m_hitList.empty() )
319 {
320 m_frame->SetStatusText( wxEmptyString );
321 }
322 else
323 {
324 if( aDirection )
325 {
326 if( m_it != m_hitList.end() && !isFirstSearch )
327 m_it++;
328
329 if( m_it == m_hitList.end() )
330 {
331 if( m_wrap->GetValue() )
332 {
333 m_it = m_hitList.begin();
334 }
335 else
336 {
337 endIsReached = true;
338 m_it--; // point to the last REAL result
339 }
340 }
341 }
342 else
343 {
344 if( m_it == m_hitList.begin() )
345 {
346 if( m_wrap->GetValue() )
347 m_it = m_hitList.end();
348 else
349 endIsReached = true;
350 }
351
352 if( m_it != m_hitList.begin() )
353 m_it--;
354 }
355 }
356
357 // Display the item
358 if( m_hitList.empty() )
359 {
360 m_frame->SetStatusText( wxEmptyString );
361 msg.Printf( _( "'%s' not found" ), searchString );
362 m_frame->ShowInfoBarMsg( msg );
363
364 m_status->SetLabel( msg );
365 }
366 else if( endIsReached || m_it == m_hitList.end() )
367 {
368 m_frame->SetStatusText( wxEmptyString );
369 m_frame->ShowInfoBarMsg( _( "No more items to show" ) );
370
371 m_status->SetLabel( _( "No hits" ) );
372 }
373 else
374 {
375 m_frame->GetToolManager()->RunAction<EDA_ITEM*>( ACTIONS::selectItem, *m_it );
376
377 msg.Printf( _( "'%s' found" ), searchString );
378 m_frame->SetStatusText( msg );
379
380 msg.Printf( _( "Hit(s): %d / %zu" ),
381 (int)std::distance( m_hitList.begin(), m_it ) + 1,
382 m_hitList.size() );
383 m_status->SetLabel( msg );
384 }
385
388}
389
390
391bool DIALOG_FIND::Show( bool show )
392{
393 bool ret = DIALOG_FIND_BASE::Show( show );
394
395 if( show )
396 m_searchCombo->SetFocus();
397
398 return ret;
399}
400
401
402void DIALOG_FIND::OnBoardChanged( wxCommandEvent& event )
403{
404 m_board = m_frame->GetBoard();
405
406 if( m_board )
407 m_board->AddListener( this );
408
409 m_upToDate = false;
410
411 event.Skip();
412}
int index
static TOOL_ACTION selectItem
Select an item (specified as the event parameter).
Definition actions.h:223
static TOOL_ACTION showSearch
Definition actions.h:112
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:89
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition board_item.h:81
Information pertinent to a Pcbnew printed circuit board.
Definition board.h:372
const NETINFO_LIST & GetNetInfo() const
Definition board.h:1086
const ZONES & Zones() const
Definition board.h:424
const MARKERS & Markers() const
Definition board.h:432
const FOOTPRINTS & Footprints() const
Definition board.h:420
const DRAWINGS & Drawings() const
Definition board.h:422
wxCheckBox * m_checkAllFields
wxCheckBox * m_matchCase
wxCheckBox * m_includeTexts
wxCheckBox * m_includeValues
wxHyperlinkCtrl * m_searchPanelLink
wxComboBox * m_searchCombo
wxCheckBox * m_includeReferences
wxCheckBox * m_matchWords
wxCheckBox * m_includeNets
wxStaticText * m_status
wxCheckBox * m_includeMarkers
wxCheckBox * m_wildcards
wxCheckBox * m_wrap
DIALOG_FIND_BASE(wxWindow *parent, wxWindowID id=wxID_ANY, const wxString &title=wxEmptyString, const wxPoint &pos=wxDefaultPosition, const wxSize &size=wxDefaultSize, long style=wxDEFAULT_DIALOG_STYLE)
PCB_EDIT_FRAME * m_frame
~DIALOG_FIND() override
void onTextEnter(wxCommandEvent &event) override
DIALOG_FIND(PCB_EDIT_FRAME *aParent)
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
void Preload(const wxString &aFindString)
void onFindNextClick(wxCommandEvent &event) override
void search(bool direction)
void onSearchAgainClick(wxCommandEvent &event) override
BOARD * m_board
BOARD_ITEM * GetItem() const
Return the currently found item or nullptr in the case of no items found.
Definition dialog_find.h:48
void onShowSearchPanel(wxHyperlinkEvent &event) override
void onFindPreviousClick(wxCommandEvent &event) override
std::function< void(BOARD_ITEM *)> m_highlightCallback
std::deque< BOARD_ITEM * > m_hitList
void OnBoardChanged(wxCommandEvent &event)
bool Show(bool show) override
void OptOut(wxWindow *aWindow)
Opt out of control state saving.
void SetInitialFocus(wxWindow *aWindow)
Sets the window (usually a wxTextCtrl) that should be focused when the dialog is shown.
Definition dialog_shim.h:79
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:46
The main frame for Pcbnew.
Handle a list of polygons defining a copper zone.
Definition zone.h:70
bool Matches(const EDA_SEARCH_DATA &aSearchData, void *aAuxData) const override
Compare the item against the search criteria in aSearchData.
Definition zone.h:163
#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:85