KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
sch_find_replace_tool.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) 2019 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
25#include <sch_commit.h>
26#include <sch_sheet_pin.h>
27#include <schematic.h>
29#include <sch_sheet_path.h>
30#include "sch_actions.h"
31
32
34{
36 return UpdateFind( aEvent );
37}
38
39
41{
43 SCH_SEARCH_DATA* schSearchData = dynamic_cast<SCH_SEARCH_DATA*>( &data );
44 bool selectedOnly = schSearchData ? schSearchData->searchSelectedOnly : false;
45
46 auto visit =
47 [&]( EDA_ITEM* aItem, SCH_SHEET_PATH* aSheet )
48 {
49 // We may get triggered when the dialog is not opened due to binding
50 // SelectedItemsModified we also get triggered when the find dialog is
51 // closed....so we need to double check the dialog is open.
52 if( m_frame->m_findReplaceDialog != nullptr
53 && !data.findString.IsEmpty()
54 && aItem->Matches( data, aSheet )
55 && ( !selectedOnly || aItem->IsSelected() ) )
56 {
57 aItem->SetForceVisible( true );
60 }
61 else if( aItem->IsBrightened() )
62 {
63 aItem->SetForceVisible( false );
65 }
66 };
67
68 auto visitAll =
69 [&]()
70 {
71 for( SCH_ITEM* item : m_frame->GetScreen()->Items() )
72 {
73 visit( item, &m_frame->GetCurrentSheet() );
74
75 item->RunOnChildren(
76 [&]( SCH_ITEM* aChild )
77 {
78 visit( aChild, &m_frame->GetCurrentSheet() );
79 },
80 RECURSE_MODE::NO_RECURSE );
81 }
82 };
83
84 if( aEvent.IsAction( &ACTIONS::find ) || aEvent.IsAction( &ACTIONS::findAndReplace )
85 || aEvent.IsAction( &ACTIONS::updateFind ) )
86 {
88 visitAll();
89 }
90 else if( aEvent.Matches( EVENTS::SelectedItemsModified ) )
91 {
92 for( EDA_ITEM* item : m_selectionTool->GetSelection() )
93 visit( item, &m_frame->GetCurrentSheet() );
94 }
95 else if( aEvent.Matches( EVENTS::PointSelectedEvent )
98 || aEvent.Matches( EVENTS::ClearedEvent ) )
99 {
101 {
103 {
105 visitAll();
106 }
107 }
108 else if( selectedOnly )
109 {
110 // Normal find modifies the selection, but selection-based find does not, so we want
111 // to start over in the items we are searching through when the selection changes
112 m_afterItem = nullptr;
113 visitAll();
114 }
115 }
116 else if( m_foundItemHighlighted )
117 {
119 visitAll();
120 }
121
122 getView()->UpdateItems();
125
126 return 0;
127}
128
129
131 SCH_ITEM* aAfter, EDA_SEARCH_DATA& aData,
132 bool reversed )
133{
134 SCH_SEARCH_DATA* schSearchData = dynamic_cast<SCH_SEARCH_DATA*>( &aData );
135 bool selectedOnly = schSearchData ? schSearchData->searchSelectedOnly : false;
136 bool past_item = !aAfter;
137 std::vector<SCH_ITEM*> sorted_items;
138
139 auto addItem =
140 [&](SCH_ITEM* item)
141 {
142 sorted_items.push_back( item );
143
144 if( item->Type() == SCH_SYMBOL_T )
145 {
146 SCH_SYMBOL* cmp = static_cast<SCH_SYMBOL*>( item );
147
148 for( SCH_FIELD& field : cmp->GetFields() )
149 sorted_items.push_back( &field );
150
151 for( SCH_PIN* pin : cmp->GetPins() )
152 sorted_items.push_back( pin );
153 }
154 else if( item->Type() == SCH_SHEET_T )
155 {
156 SCH_SHEET* sheet = static_cast<SCH_SHEET*>( item );
157
158 for( SCH_FIELD& field : sheet->GetFields() )
159 sorted_items.push_back( &field );
160
161 for( SCH_SHEET_PIN* pin : sheet->GetPins() )
162 sorted_items.push_back( pin );
163 }
164 else if( item->IsType( { SCH_LABEL_LOCATE_ANY_T } ) )
165 {
166 SCH_LABEL_BASE* label = static_cast<SCH_LABEL_BASE*>( item );
167
168 for( SCH_FIELD& field : label->GetFields() )
169 sorted_items.push_back( &field );
170 }
171 };
172
173 if( selectedOnly )
174 {
175 for( EDA_ITEM* item : m_selectionTool->GetSelection() )
176 addItem( static_cast<SCH_ITEM*>( item ) );
177 }
178 else
179 {
180 for( SCH_ITEM* item : aScreen->Items() )
181 addItem( item );
182 }
183
184 std::sort( sorted_items.begin(), sorted_items.end(),
185 [&]( SCH_ITEM* a, SCH_ITEM* b )
186 {
187 if( a->GetPosition().x == b->GetPosition().x )
188 {
189 // Ensure deterministic sort
190 if( a->GetPosition().y == b->GetPosition().y )
191 return a->m_Uuid < b->m_Uuid;
192
193 return a->GetPosition().y < b->GetPosition().y;
194 }
195 else
196 return a->GetPosition().x < b->GetPosition().x;
197 } );
198
199 if( reversed )
200 std::reverse( sorted_items.begin(), sorted_items.end() );
201
202 for( SCH_ITEM* item : sorted_items )
203 {
204 if( item == aAfter )
205 {
206 past_item = true;
207 }
208 else if( past_item )
209 {
210 if( aData.markersOnly && item->Type() == SCH_MARKER_T )
211 return item;
212
213 if( item->Matches( aData, aSheet ) )
214 return item;
215 }
216 }
217
218 return nullptr;
219}
220
221
223{
225 bool searchAllSheets = false;
226 bool selectedOnly = false;
227 bool isReversed = aEvent.IsAction( &ACTIONS::findPrevious );
228 SCH_ITEM* item = nullptr;
229 SCH_SHEET_PATH* afterSheet = &m_frame->GetCurrentSheet();
230
231 try
232 {
233 const SCH_SEARCH_DATA& schSearchData = dynamic_cast<const SCH_SEARCH_DATA&>( data );
234 searchAllSheets = !( schSearchData.searchCurrentSheetOnly );
235 selectedOnly = schSearchData.searchSelectedOnly;
236 }
237 catch( const std::bad_cast& )
238 {
239 }
240
241 if( aEvent.IsAction( &ACTIONS::findNextMarker ) )
242 data.markersOnly = true;
243 else if( data.findString.IsEmpty() )
244 return FindAndReplace( ACTIONS::find.MakeEvent() );
245
246 if( m_wrapAroundTimer.IsRunning() )
247 {
248 afterSheet = nullptr;
249 m_afterItem = nullptr;
250 m_wrapAroundTimer.Stop();
252 }
253
254 if( afterSheet || !searchAllSheets )
255 {
257 isReversed );
258 }
259
260 if( !item && searchAllSheets )
261 {
262 SCH_SCREENS screens( m_frame->Schematic().Root() );
263 SCH_SHEET_LIST paths;
264
265 screens.BuildClientSheetPathList();
266
267 for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() )
268 {
269 for( SCH_SHEET_PATH& sheet : screen->GetClientSheetPaths() )
270 paths.push_back( sheet );
271 }
272
273 paths.SortByPageNumbers( false );
274
275 if( isReversed )
276 std::reverse( paths.begin(), paths.end() );
277
278 for( SCH_SHEET_PATH& sheet : paths )
279 {
280 if( afterSheet )
281 {
282 if( afterSheet->GetCurrentHash() == sheet.GetCurrentHash() )
283 afterSheet = nullptr;
284
285 continue;
286 }
287
288 item = nextMatch( sheet.LastScreen(), &sheet, nullptr, data, isReversed );
289
290 if( item )
291 {
292 if( m_frame->Schematic().CurrentSheet() != sheet )
293 {
295 &sheet );
296 }
297
298 break;
299 }
300 }
301 }
302
303 if( item )
304 {
305 m_afterItem = item;
306
307 if( !selectedOnly )
308 {
311 }
312
313 if( !item->IsBrightened() )
314 {
315 // Clear any previous brightening
316 UpdateFind( aEvent );
317
318 // Brighten (and show) found object
319 item->SetForceVisible( true );
322 }
323
326 }
327 else
328 {
329 wxString msg = searchAllSheets ? _( "Reached end of schematic." )
330 : _( "Reached end of sheet." );
331
332 // Show the popup during the time period the user can wrap the search
333 m_frame->ShowFindReplaceStatus( msg + wxS( " " ) +
334 _( "Find again to wrap around to the start." ), 4000 );
335 m_wrapAroundTimer.StartOnce( 4000 );
336 }
337
338 return 0;
339}
340
342{
344 SCH_SEARCH_DATA* schSearchData = dynamic_cast<SCH_SEARCH_DATA*>( &data );
345 bool selectedOnly = schSearchData ? schSearchData->searchSelectedOnly : false;
346
347 return selectedOnly ? m_afterItem : m_selectionTool->GetSelection().Front();
348}
349
351{
353 EDA_ITEM* match = getCurrentMatch();
354
355 return match && match->Matches( data, &m_frame->GetCurrentSheet() );
356}
357
358
360{
362 EDA_ITEM* item = getCurrentMatch();
364
365 if( data.findString.IsEmpty() )
366 return FindAndReplace( ACTIONS::find.MakeEvent() );
367
368 if( item && HasMatch() )
369 {
370 SCH_COMMIT commit( m_frame );
371 SCH_ITEM* sch_item = static_cast<SCH_ITEM*>( item );
372
373 commit.Modify( sch_item, sheet->LastScreen() );
374
375 if( item->Replace( data, sheet ) )
376 {
378 commit.Push( wxS( "Find and Replace" ) );
379 }
380
381 FindNext( ACTIONS::findNext.MakeEvent() );
382 }
383
384 return 0;
385}
386
387
389{
391 bool currentSheetOnly = false;
392 bool selectedOnly = false;
393
394 try
395 {
396 const SCH_SEARCH_DATA& schSearchData = dynamic_cast<const SCH_SEARCH_DATA&>( data );
397 currentSheetOnly = schSearchData.searchCurrentSheetOnly;
398 selectedOnly = schSearchData.searchSelectedOnly;
399 }
400 catch( const std::bad_cast& )
401 {
402 }
403
404 SCH_COMMIT commit( m_frame );
405 bool modified = false; // TODO: move to SCH_COMMIT....
406
407 if( data.findString.IsEmpty() )
408 return FindAndReplace( ACTIONS::find.MakeEvent() );
409
410 auto doReplace =
411 [&]( SCH_ITEM* aItem, SCH_SHEET_PATH* aSheet, EDA_SEARCH_DATA& aData )
412 {
413 commit.Modify( aItem, aSheet->LastScreen() );
414
415 if( aItem->Replace( aData, aSheet ) )
416 {
417 m_frame->UpdateItem( aItem, false, true );
418 modified = true;
419 }
420 };
421
422 if( currentSheetOnly || selectedOnly )
423 {
424 SCH_SHEET_PATH* currentSheet = &m_frame->GetCurrentSheet();
425
426 SCH_ITEM* item = nextMatch( m_frame->GetScreen(), currentSheet, nullptr, data, false );
427
428 while( item )
429 {
430 if( !selectedOnly || item->IsSelected() )
431 doReplace( item, currentSheet, data );
432
433 item = nextMatch( m_frame->GetScreen(), currentSheet, item, data, false );
434 }
435 }
436 else
437 {
438 SCH_SHEET_LIST allSheets = m_frame->Schematic().Hierarchy();
439 SCH_SCREENS screens( m_frame->Schematic().Root() );
440
441 for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() )
442 {
443 SCH_SHEET_LIST sheets = allSheets.FindAllSheetsForScreen( screen );
444
445 for( unsigned ii = 0; ii < sheets.size(); ++ii )
446 {
447 SCH_ITEM* item = nextMatch( screen, &sheets[ii], nullptr, data, false );
448
449 while( item )
450 {
451 if( ii == 0 )
452 {
453 doReplace( item, &sheets[0], data );
454 }
455 else if( item->Type() == SCH_FIELD_T )
456 {
457 SCH_FIELD* field = static_cast<SCH_FIELD*>( item );
458
459 // References must be handled for each distinct sheet
460 if( field->GetId() == FIELD_T::REFERENCE )
461 doReplace( field, &sheets[ii], data );
462 }
463
464 item = nextMatch( screen, &sheets[ii], item, data, false );
465 }
466 }
467 }
468 }
469
470 if( modified )
471 {
472 commit.Push( wxS( "Find and Replace All" ) );
474 }
475
476 return 0;
477}
478
479
481{
495}
static TOOL_ACTION replaceAll
Definition: actions.h:122
static TOOL_ACTION updateFind
Definition: actions.h:123
static TOOL_ACTION findPrevious
Definition: actions.h:119
static TOOL_ACTION findAndReplace
Definition: actions.h:117
static TOOL_ACTION replaceAndFindNext
Definition: actions.h:121
static TOOL_ACTION findNext
Definition: actions.h:118
static TOOL_ACTION findNextMarker
Definition: actions.h:120
static TOOL_ACTION find
Definition: actions.h:116
constexpr const Vec GetCenter() const
Definition: box2.h:230
COMMIT & Modify(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Modify a given item in the model.
Definition: commit.h:108
void FocusOnLocation(const VECTOR2I &aPos)
Useful to focus on a particular location, in find functions.
EDA_SEARCH_DATA & GetFindReplaceData()
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
virtual VECTOR2I GetPosition() const
Definition: eda_item.h:252
virtual const BOX2I GetBoundingBox() const
Return the orthogonal bounding box of this object for display purposes.
Definition: eda_item.cpp:84
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
void SetForceVisible(bool aEnable)
Set and clear force visible flag used to force the item to be drawn even if it's draw attribute is se...
Definition: eda_item.h:202
virtual bool Matches(const EDA_SEARCH_DATA &aSearchData, void *aAuxData) const
Compare the item against the search criteria in aSearchData.
Definition: eda_item.h:386
bool IsSelected() const
Definition: eda_item.h:120
bool IsBrightened() const
Definition: eda_item.h:122
static bool Replace(const EDA_SEARCH_DATA &aSearchData, wxString &aText)
Perform a text replace on aText using the find and replace criteria in aSearchData on items that supp...
Definition: eda_item.cpp:211
static const TOOL_EVENT ClearedEvent
Definition: actions.h:338
static const TOOL_EVENT SelectedEvent
Definition: actions.h:336
static const TOOL_EVENT SelectedItemsModified
Selected items were moved, this can be very high frequency on the canvas, use with care.
Definition: actions.h:343
static const TOOL_EVENT PointSelectedEvent
Definition: actions.h:335
static const TOOL_EVENT UnselectedEvent
Definition: actions.h:337
void UpdateItems()
Iterate through the list of items that asked for updating and updates them.
Definition: view.cpp:1456
SCH_SHEET_LIST Hierarchy() const
Return the full schematic flattened hierarchical sheet list.
Definition: schematic.cpp:208
SCH_SHEET & Root() const
Definition: schematic.h:117
SCH_SHEET_PATH & CurrentSheet() const
Definition: schematic.h:148
static TOOL_ACTION changeSheet
Definition: sch_actions.h:216
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
virtual void Push(const wxString &aMessage=wxT("A commit"), int aCommitFlags=0) override
Execute the changes.
Definition: sch_commit.cpp:435
SCH_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
void ShowFindReplaceDialog(bool aReplace)
Run the Find or Find & Replace dialog.
void ClearFindReplaceStatus()
SCH_SHEET_PATH & GetCurrentSheet() const
SCHEMATIC & Schematic() const
void updateTitle()
Set the main window title bar text.
void ShowFindReplaceStatus(const wxString &aMsg, int aStatusTime)
void UpdateItem(EDA_ITEM *aItem, bool isAddOrDelete=false, bool aUpdateRtree=false) override
Mark an item for refresh.
DIALOG_SCH_FIND * m_findReplaceDialog
FIELD_T GetId() const
Definition: sch_field.h:124
void setTransitions() override
< Set up handlers for various events.
int UpdateFind(const TOOL_EVENT &aEvent)
int FindNext(const TOOL_EVENT &aEvent)
SCH_ITEM * nextMatch(SCH_SCREEN *aScreen, SCH_SHEET_PATH *aSheet, SCH_ITEM *aAfter, EDA_SEARCH_DATA &aData, bool reverse)
Advance the search and returns the next matching item after aAfter.
int ReplaceAll(const TOOL_EVENT &aEvent)
int FindAndReplace(const TOOL_EVENT &aEvent)
int ReplaceAndFindNext(const TOOL_EVENT &aEvent)
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
std::vector< SCH_FIELD > & GetFields()
Definition: sch_label.h:205
Container class that holds multiple SCH_SCREEN objects in a hierarchy.
Definition: sch_screen.h:734
SCH_SCREEN * GetNext()
SCH_SCREEN * GetFirst()
void BuildClientSheetPathList()
Build the list of sheet paths sharing a screen for each screen in use.
EE_RTREE & Items()
Get the full RTree, usually for iterating.
Definition: sch_screen.h:112
int ClearSelection(const TOOL_EVENT &aEvent)
Select all visible items in sheet.
SCH_SELECTION & GetSelection()
A container for handling SCH_SHEET_PATH objects in a flattened hierarchy.
void SortByPageNumbers(bool aUpdateVirtualPageNums=true)
Sort the list of sheets by page number.
SCH_SHEET_LIST FindAllSheetsForScreen(const SCH_SCREEN *aScreen) const
Return a SCH_SHEET_LIST with a copy of all the SCH_SHEET_PATH using a particular screen.
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
size_t GetCurrentHash() const
void UpdateAllScreenReferences() const
Update all the symbol references for this sheet path.
SCH_SCREEN * LastScreen()
Define a sheet pin (label) used in sheets to create hierarchical schematics.
Definition: sch_sheet_pin.h:66
Sheet symbol placed in a schematic, and is the entry point for a sub schematic.
Definition: sch_sheet.h:47
std::vector< SCH_FIELD > & GetFields()
Return a reference to the vector holding the sheet's fields.
Definition: sch_sheet.h:87
std::vector< SCH_SHEET_PIN * > & GetPins()
Definition: sch_sheet.h:187
Schematic symbol object.
Definition: sch_symbol.h:75
void GetFields(std::vector< SCH_FIELD * > &aVector, bool aVisibleOnly) const override
Populate a std::vector with SCH_FIELDs, sorted in ordinal order.
Definition: sch_symbol.cpp:841
std::vector< SCH_PIN * > GetPins(const SCH_SHEET_PATH *aSheet) const
Retrieve a list of the SCH_PINs for the given sheet path.
SCH_SELECTION_TOOL * m_selectionTool
void BrightenItem(EDA_ITEM *aItem)
int AddItemToSel(const TOOL_EVENT &aEvent)
void UnbrightenItem(EDA_ITEM *aItem)
EDA_ITEM * Front() const
Definition: selection.h:172
TOOL_MANAGER * GetToolManager() const
Return the MVC controller.
Definition: tools_holder.h:55
KIGFX::VIEW * getView() const
Returns the instance of #VIEW object used in the application.
Definition: tool_base.cpp:38
Generic, UI-independent tool event.
Definition: tool_event.h:168
bool Matches(const TOOL_EVENT &aEvent) const
Test whether two events match in terms of category & action or command.
Definition: tool_event.h:389
bool IsAction(const TOOL_ACTION *aAction) const
Test if the event contains an action issued upon activation of the given TOOL_ACTION.
Definition: tool_event.cpp:82
void Go(int(T::*aStateFunc)(const TOOL_EVENT &), const TOOL_EVENT_LIST &aConditions=TOOL_EVENT(TC_ANY, TA_ANY))
Define which state (aStateFunc) to go when a certain event arrives (aConditions).
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
#define _(s)
Definition of the SCH_SHEET_PATH and SCH_SHEET_LIST classes for Eeschema.
@ SCH_SYMBOL_T
Definition: typeinfo.h:172
@ SCH_FIELD_T
Definition: typeinfo.h:150
@ SCH_SHEET_T
Definition: typeinfo.h:175
@ SCH_MARKER_T
Definition: typeinfo.h:158
@ SCH_LABEL_LOCATE_ANY_T
Definition: typeinfo.h:191